* Copyright 2001-2015, Haiku, Inc.
* Distributed under the terms of the MIT License.
*
* Authors:
* Marc Flerackers, mflerackers@androme.be
* Ingo Weinhold, ingo_weinhold@gmx.de
*/
#include <stdlib.h>
#include <string.h>
#include <Control.h>
#include <PropertyInfo.h>
#include <Window.h>
#include <binary_compatibility/Interface.h>
#include <Icon.h>
static property_info sPropertyList[] = {
{
"Enabled",
{ B_GET_PROPERTY, B_SET_PROPERTY },
{ B_DIRECT_SPECIFIER },
NULL, 0,
{ B_BOOL_TYPE }
},
{
"Label",
{ B_GET_PROPERTY, B_SET_PROPERTY },
{ B_DIRECT_SPECIFIER },
NULL, 0,
{ B_STRING_TYPE }
},
{
"Value",
{ B_GET_PROPERTY, B_SET_PROPERTY },
{ B_DIRECT_SPECIFIER },
NULL, 0,
{ B_INT32_TYPE }
},
{ 0 }
};
BControl::BControl(BRect frame, const char* name, const char* label,
BMessage* message, uint32 resizingMode, uint32 flags)
:
BView(frame, name, resizingMode, flags)
{
InitData(NULL);
SetLabel(label);
SetMessage(message);
}
BControl::BControl(const char* name, const char* label, BMessage* message,
uint32 flags)
:
BView(name, flags)
{
InitData(NULL);
SetLabel(label);
SetMessage(message);
}
BControl::~BControl()
{
free(fLabel);
delete fIcon;
SetMessage(NULL);
}
BControl::BControl(BMessage* data)
:
BView(data)
{
InitData(data);
BMessage message;
if (data->FindMessage("_msg", &message) == B_OK)
SetMessage(new BMessage(message));
const char* label;
if (data->FindString("_label", &label) == B_OK)
SetLabel(label);
int32 value;
if (data->FindInt32("_val", &value) == B_OK)
SetValue(value);
bool toggle;
if (data->FindBool("_disable", &toggle) == B_OK)
SetEnabled(!toggle);
if (data->FindBool("be:wants_nav", &toggle) == B_OK)
fWantsNav = toggle;
}
BArchivable*
BControl::Instantiate(BMessage* data)
{
if (validate_instantiation(data, "BControl"))
return new BControl(data);
return NULL;
}
status_t
BControl::Archive(BMessage* data, bool deep) const
{
status_t status = BView::Archive(data, deep);
if (status == B_OK && Message())
status = data->AddMessage("_msg", Message());
if (status == B_OK && fLabel)
status = data->AddString("_label", fLabel);
if (status == B_OK && fValue != B_CONTROL_OFF)
status = data->AddInt32("_val", fValue);
if (status == B_OK && !fEnabled)
status = data->AddBool("_disable", true);
return status;
}
void
BControl::WindowActivated(bool active)
{
BView::WindowActivated(active);
if (IsFocus())
Invalidate();
}
void
BControl::AttachedToWindow()
{
AdoptParentColors();
if (ViewColor() == B_TRANSPARENT_COLOR
|| Parent() == NULL) {
AdoptSystemColors();
}
if (Parent() != NULL) {
float tint = B_NO_TINT;
color_which which = ViewUIColor(&tint);
if (which != B_NO_COLOR)
SetLowUIColor(which, tint);
else
SetLowColor(ViewColor());
}
if (!Messenger().IsValid())
SetTarget(Window());
BView::AttachedToWindow();
}
void
BControl::DetachedFromWindow()
{
BView::DetachedFromWindow();
}
void
BControl::AllAttached()
{
BView::AllAttached();
}
void
BControl::AllDetached()
{
BView::AllDetached();
}
void
BControl::MessageReceived(BMessage* message)
{
if (message->what == B_GET_PROPERTY || message->what == B_SET_PROPERTY) {
BMessage reply(B_REPLY);
bool handled = false;
BMessage specifier;
int32 index;
int32 form;
const char* property;
if (message->GetCurrentSpecifier(&index, &specifier, &form, &property) == B_OK) {
if (strcmp(property, "Label") == 0) {
if (message->what == B_GET_PROPERTY) {
reply.AddString("result", fLabel);
handled = true;
} else {
const char* label;
if (message->FindString("data", &label) == B_OK) {
SetLabel(label);
reply.AddInt32("error", B_OK);
handled = true;
}
}
} else if (strcmp(property, "Value") == 0) {
if (message->what == B_GET_PROPERTY) {
reply.AddInt32("result", fValue);
handled = true;
} else {
int32 value;
if (message->FindInt32("data", &value) == B_OK) {
SetValue(value);
reply.AddInt32("error", B_OK);
handled = true;
}
}
} else if (strcmp(property, "Enabled") == 0) {
if (message->what == B_GET_PROPERTY) {
reply.AddBool("result", fEnabled);
handled = true;
} else {
bool enabled;
if (message->FindBool("data", &enabled) == B_OK) {
SetEnabled(enabled);
reply.AddInt32("error", B_OK);
handled = true;
}
}
}
}
if (handled) {
message->SendReply(&reply);
return;
}
}
BView::MessageReceived(message);
}
void
BControl::MakeFocus(bool focus)
{
if (focus == IsFocus())
return;
BView::MakeFocus(focus);
if (Window() != NULL) {
fFocusChanging = true;
Invalidate(Bounds());
Flush();
fFocusChanging = false;
}
}
void
BControl::KeyDown(const char* bytes, int32 numBytes)
{
if (*bytes == B_ENTER || *bytes == B_SPACE) {
if (!fEnabled)
return;
SetValue(Value() ? B_CONTROL_OFF : B_CONTROL_ON);
Invoke();
} else
BView::KeyDown(bytes, numBytes);
}
void
BControl::MouseDown(BPoint where)
{
BView::MouseDown(where);
}
void
BControl::MouseUp(BPoint where)
{
BView::MouseUp(where);
}
void
BControl::MouseMoved(BPoint where, uint32 code, const BMessage* dragMessage)
{
BView::MouseMoved(where, code, dragMessage);
}
void
BControl::SetLabel(const char* label)
{
if (label != NULL && !label[0])
label = NULL;
if ((fLabel && label && !strcmp(fLabel, label))
|| ((fLabel == NULL || !fLabel[0]) && label == NULL))
return;
free(fLabel);
fLabel = label ? strdup(label) : NULL;
InvalidateLayout();
Invalidate();
}
const char*
BControl::Label() const
{
return fLabel;
}
void
BControl::SetValue(int32 value)
{
if (value == fValue)
return;
fValue = value;
Invalidate();
}
void
BControl::SetValueNoUpdate(int32 value)
{
fValue = value;
}
int32
BControl::Value() const
{
return fValue;
}
void
BControl::SetEnabled(bool enabled)
{
if (fEnabled == enabled)
return;
fEnabled = enabled;
if (fEnabled && fWantsNav)
SetFlags(Flags() | B_NAVIGABLE);
else if (!fEnabled && (Flags() & B_NAVIGABLE)) {
fWantsNav = true;
SetFlags(Flags() & ~B_NAVIGABLE);
} else
fWantsNav = false;
if (Window()) {
Invalidate(Bounds());
Flush();
}
}
bool
BControl::IsEnabled() const
{
return fEnabled;
}
void
BControl::GetPreferredSize(float* _width, float* _height)
{
BView::GetPreferredSize(_width, _height);
}
void
BControl::ResizeToPreferred()
{
BView::ResizeToPreferred();
}
status_t
BControl::Invoke(BMessage* message)
{
bool notify = false;
uint32 kind = InvokeKind(¬ify);
if (!message && !notify)
message = Message();
BMessage clone(kind);
if (!message) {
if (!IsWatched())
return B_BAD_VALUE;
} else
clone = *message;
clone.AddInt64("when", (int64)system_time());
clone.AddPointer("source", this);
clone.AddInt32("be:value", fValue);
clone.AddMessenger("be:sender", BMessenger(this));
status_t err;
if (message)
err = BInvoker::Invoke(&clone);
else
err = B_BAD_VALUE;
SendNotices(kind, &clone);
return err;
}
BHandler*
BControl::ResolveSpecifier(BMessage* message, int32 index,
BMessage* specifier, int32 what, const char* property)
{
BPropertyInfo propInfo(sPropertyList);
if (propInfo.FindMatch(message, 0, specifier, what, property) >= B_OK)
return this;
return BView::ResolveSpecifier(message, index, specifier, what,
property);
}
status_t
BControl::GetSupportedSuites(BMessage* message)
{
message->AddString("suites", "suite/vnd.Be-control");
BPropertyInfo propInfo(sPropertyList);
message->AddFlat("messages", &propInfo);
return BView::GetSupportedSuites(message);
}
status_t
BControl::Perform(perform_code code, void* _data)
{
switch (code) {
case PERFORM_CODE_MIN_SIZE:
((perform_data_min_size*)_data)->return_value
= BControl::MinSize();
return B_OK;
case PERFORM_CODE_MAX_SIZE:
((perform_data_max_size*)_data)->return_value
= BControl::MaxSize();
return B_OK;
case PERFORM_CODE_PREFERRED_SIZE:
((perform_data_preferred_size*)_data)->return_value
= BControl::PreferredSize();
return B_OK;
case PERFORM_CODE_LAYOUT_ALIGNMENT:
((perform_data_layout_alignment*)_data)->return_value
= BControl::LayoutAlignment();
return B_OK;
case PERFORM_CODE_HAS_HEIGHT_FOR_WIDTH:
((perform_data_has_height_for_width*)_data)->return_value
= BControl::HasHeightForWidth();
return B_OK;
case PERFORM_CODE_GET_HEIGHT_FOR_WIDTH:
{
perform_data_get_height_for_width* data
= (perform_data_get_height_for_width*)_data;
BControl::GetHeightForWidth(data->width, &data->min, &data->max,
&data->preferred);
return B_OK;
}
case PERFORM_CODE_SET_LAYOUT:
{
perform_data_set_layout* data = (perform_data_set_layout*)_data;
BControl::SetLayout(data->layout);
return B_OK;
}
case PERFORM_CODE_LAYOUT_INVALIDATED:
{
perform_data_layout_invalidated* data
= (perform_data_layout_invalidated*)_data;
BControl::LayoutInvalidated(data->descendants);
return B_OK;
}
case PERFORM_CODE_DO_LAYOUT:
{
BControl::DoLayout();
return B_OK;
}
case PERFORM_CODE_SET_ICON:
{
perform_data_set_icon* data = (perform_data_set_icon*)_data;
return BControl::SetIcon(data->icon, data->flags);
}
}
return BView::Perform(code, _data);
}
status_t
BControl::SetIcon(const BBitmap* bitmap, uint32 flags)
{
status_t error = BIcon::UpdateIcon(bitmap, flags, fIcon);
if (error == B_OK) {
InvalidateLayout();
Invalidate();
}
return error;
}
status_t
BControl::SetIconBitmap(const BBitmap* bitmap, uint32 which, uint32 flags)
{
status_t error = BIcon::SetIconBitmap(bitmap, which, flags, fIcon);
if (error != B_OK) {
InvalidateLayout();
Invalidate();
}
return error;
}
const BBitmap*
BControl::IconBitmap(uint32 which) const
{
return fIcon != NULL ? fIcon->Bitmap(which) : NULL;
}
bool
BControl::IsFocusChanging() const
{
return fFocusChanging;
}
bool
BControl::IsTracking() const
{
return fTracking;
}
void
BControl::SetTracking(bool state)
{
fTracking = state;
}
extern "C" status_t
B_IF_GCC_2(_ReservedControl1__8BControl, _ZN8BControl17_ReservedControl1Ev)(
BControl* control, const BBitmap* icon, uint32 flags)
{
perform_data_set_icon data;
data.icon = icon;
data.flags = flags;
return control->Perform(PERFORM_CODE_SET_ICON, &data);
}
void BControl::_ReservedControl2() {}
void BControl::_ReservedControl3() {}
void BControl::_ReservedControl4() {}
BControl &
BControl::operator=(const BControl &)
{
return *this;
}
void
BControl::InitData(BMessage* data)
{
SetViewUIColor(B_PANEL_BACKGROUND_COLOR);
SetLowUIColor(ViewUIColor());
fLabel = NULL;
SetLabel(B_EMPTY_STRING);
fValue = B_CONTROL_OFF;
fEnabled = true;
fFocusChanging = false;
fTracking = false;
fWantsNav = Flags() & B_NAVIGABLE;
fIcon = NULL;
if (data && data->HasString("_fname"))
SetFont(be_plain_font, B_FONT_FAMILY_AND_STYLE);
}