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:
parent
64603764f9
commit
8053a364e1
@ -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() {}
|
||||||
|
Loading…
Reference in New Issue
Block a user