Assynchronous mouse hooks and correct handling of BRadioButtons in BBox-en :)

git-svn-id: file:///srv/svn/repos/haiku/trunk/current@3534 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Marc Flerackers 2003-06-16 06:37:33 +00:00
parent 64603764f9
commit 8053a364e1

View File

@ -31,6 +31,8 @@
// System Includes ------------------------------------------------------------- // System Includes -------------------------------------------------------------
#include <RadioButton.h> #include <RadioButton.h>
#include <Errors.h> #include <Errors.h>
#include <Box.h>
#include <Window.h>
// Project Includes ------------------------------------------------------------ // Project Includes ------------------------------------------------------------
@ -46,8 +48,12 @@ BRadioButton::BRadioButton(BRect frame, const char *name, const char *label,
: BControl(frame, name, label, message, resizMask, flags), : BControl(frame, name, label, message, resizMask, flags),
fOutlined(false) fOutlined(false)
{ {
if (Bounds().Height() < 18.0f) // Resize to minimum height if needed
ResizeTo(Bounds().Width(), 18.0f); font_height fh;
GetFontHeight(&fh);
float minHeight = (float)ceil(6.0f + fh.ascent + fh.descent);
if (Bounds().Height() < minHeight)
ResizeTo(Bounds().Width(), minHeight);
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
BRadioButton::BRadioButton(BMessage *archive) BRadioButton::BRadioButton(BMessage *archive)
@ -100,7 +106,7 @@ void BRadioButton::Draw(BRect updateRect)
darken1 = tint_color(no_tint, B_DARKEN_1_TINT), darken1 = tint_color(no_tint, B_DARKEN_1_TINT),
darken2 = tint_color(no_tint, B_DARKEN_2_TINT), darken2 = tint_color(no_tint, B_DARKEN_2_TINT),
darken3 = tint_color(no_tint, B_DARKEN_3_TINT), darken3 = tint_color(no_tint, B_DARKEN_3_TINT),
darken4 = tint_color(no_tint, B_DARKEN_4_TINT), //darken4 = tint_color(no_tint, B_DARKEN_4_TINT),
darkenmax = tint_color(no_tint, B_DARKEN_MAX_TINT); darkenmax = tint_color(no_tint, B_DARKEN_MAX_TINT);
BRect rect(1.0, 3.0, 13.0, 15.0); BRect rect(1.0, 3.0, 13.0, 15.0);
@ -226,17 +232,52 @@ void BRadioButton::Draw(BRect updateRect)
void BRadioButton::MouseDown(BPoint point) void BRadioButton::MouseDown(BPoint point)
{ {
if (!IsEnabled()) if (!IsEnabled())
{
BControl::MouseDown(point);
return; return;
}
SetMouseEventMask(B_POINTER_EVENTS, B_NO_POINTER_HISTORY |
B_SUSPEND_VIEW_FOCUS);
SetTracking(true);
fOutlined = true; fOutlined = true;
Invalidate(); Draw(Bounds());
Flush();
if (Window()->Flags() & B_ASYNCHRONOUS_CONTROLS)
{
BRect bounds = Bounds();
uint32 buttons;
do
{
snooze(40000);
GetMouse(&point, &buttons, true);
bool inside = bounds.Contains(point);
if (fOutlined != inside)
{
fOutlined = inside;
Draw(Bounds());
Flush();
}
} while (buttons != 0);
if (fOutlined)
{
fOutlined = false;
Draw(Bounds());
Flush();
SetValue(B_CONTROL_ON);
Invoke();
}
else
{
Draw(Bounds());
Flush();
}
}
else
{
SetTracking(true);
SetMouseEventMask(B_POINTER_EVENTS, B_LOCK_WINDOW_FOCUS);
}
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::AttachedToWindow() void BRadioButton::AttachedToWindow()
@ -246,38 +287,81 @@ void BRadioButton::AttachedToWindow()
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::KeyDown(const char *bytes, int32 numBytes) void BRadioButton::KeyDown(const char *bytes, int32 numBytes)
{ {
BControl::KeyDown(bytes, numBytes); // TODO add select_next_button functionality
switch (bytes[0])
{
case ' ':
{
if (IsEnabled() && !Value())
{
SetValue(B_CONTROL_ON);
Invoke();
}
break;
}
default:
BControl::KeyDown(bytes, numBytes);
}
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::SetValue(int32 value) void BRadioButton::SetValue(int32 value)
{ {
if (BControl::Value() == value) BControl::SetValue(value);
if (!value)
return; return;
if (!IsTracking() && value == B_CONTROL_ON) BView *parent = Parent();
BView *child = NULL;
if (parent)
{ {
BView *sibling; // If the parent is a BBox, the group parent is the parent of the BBox
BBox *box = dynamic_cast<BBox*>(parent);
for (sibling = PreviousSibling(); sibling != NULL; if (box && box->LabelView() == this)
sibling = sibling->PreviousSibling()) parent = box->Parent();
if (parent)
{ {
BRadioButton* radio = dynamic_cast<BRadioButton*>(sibling); BBox *box = dynamic_cast<BBox*>(parent);
if (radio != NULL) // If the parent is a BBox, skip the label if there is one
radio->BControl::SetValue(B_CONTROL_OFF); if (box && box->LabelView())
} child = parent->ChildAt(1);
else
for (sibling = NextSibling(); sibling != NULL; child = parent->ChildAt(0);
sibling = sibling->NextSibling())
{
BRadioButton* radio = dynamic_cast<BRadioButton*>(sibling);
if (radio != NULL)
radio->BControl::SetValue(B_CONTROL_OFF);
} }
else
child = Window()->ChildAt(0);
} }
else if (Window())
child = Window()->ChildAt(0);
BControl::SetValue(value); while (child)
{
BRadioButton *radio = dynamic_cast<BRadioButton*>(child);
if (child != this && radio)
radio->SetValue(B_CONTROL_OFF);
else
{
// If the child is a BBox, check if the label is a radiobutton
BBox *box = dynamic_cast<BBox*>(child);
if (box && box->LabelView())
{
radio = dynamic_cast<BRadioButton*>(box->LabelView());
if (radio)
radio->SetValue(B_CONTROL_OFF);
}
}
child = child->NextSibling();
}
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::GetPreferredSize(float *width, float *height) void BRadioButton::GetPreferredSize(float *width, float *height)
@ -285,8 +369,13 @@ void BRadioButton::GetPreferredSize(float *width, float *height)
font_height fh; font_height fh;
GetFontHeight(&fh); GetFontHeight(&fh);
*height = (float)ceil(fh.ascent + fh.descent + fh.leading) + 6.0f; *height = (float)ceil(fh.ascent + fh.descent) + 6.0f;
*width = 22.0f + (float)ceil(StringWidth(Label())); *width = 22.0f; // TODO: check if ascent is included
if (Label())
*width += StringWidth(Label());
*width = (float)ceil(*width);
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::ResizeToPreferred() void BRadioButton::ResizeToPreferred()
@ -311,39 +400,48 @@ void BRadioButton::WindowActivated(bool active)
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::MouseUp(BPoint point) void BRadioButton::MouseUp(BPoint point)
{ {
if (IsEnabled() && IsTracking()) if (!IsTracking())
return;
bool inside = Bounds().Contains(point);
if (fOutlined != inside)
{
fOutlined = inside;
Draw(Bounds());
Flush();
}
if (fOutlined)
{ {
fOutlined = false; fOutlined = false;
SetTracking(false); Draw(Bounds());
Flush();
if (Bounds().Contains(point)) SetValue(B_CONTROL_ON);
{ Invoke();
SetValue(B_CONTROL_ON);
Invoke();
}
} }
else else
{ {
BControl::MouseUp(point); Draw(Bounds());
return; Flush();
} }
Invalidate(); SetTracking(false);
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::MouseMoved(BPoint point, uint32 transit, const BMessage *message) void BRadioButton::MouseMoved(BPoint point, uint32 transit, const BMessage *message)
{ {
if (IsEnabled() && IsTracking()) if (!IsTracking())
{ return;
if (transit == B_EXITED_VIEW)
fOutlined = false;
else if (transit == B_ENTERED_VIEW)
fOutlined = true;
Invalidate(); bool inside = Bounds().Contains(point);
if (fOutlined != inside)
{
fOutlined = inside;
Draw(Bounds());
Flush();
} }
else
BView::MouseMoved(point, transit, message);
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::DetachedFromWindow() void BRadioButton::DetachedFromWindow()
@ -365,7 +463,8 @@ BHandler *BRadioButton::ResolveSpecifier(BMessage *message, int32 index,
BMessage *specifier, int32 what, BMessage *specifier, int32 what,
const char *property) const char *property)
{ {
return ResolveSpecifier(message, index, specifier, what, property); return BControl::ResolveSpecifier(message, index, specifier, what,
property);
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::MakeFocus(bool focused) void BRadioButton::MakeFocus(bool focused)
@ -385,12 +484,12 @@ void BRadioButton::AllDetached()
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
status_t BRadioButton::GetSupportedSuites(BMessage *message) status_t BRadioButton::GetSupportedSuites(BMessage *message)
{ {
return GetSupportedSuites(message); return BControl::GetSupportedSuites(message);
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
status_t BRadioButton::Perform(perform_code d, void *arg) status_t BRadioButton::Perform(perform_code d, void *arg)
{ {
return B_ERROR; return BControl::Perform(d, arg);
} }
//------------------------------------------------------------------------------ //------------------------------------------------------------------------------
void BRadioButton::_ReservedRadioButton1() {} void BRadioButton::_ReservedRadioButton1() {}