changed the handling of the default buttons to be more logical
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@15516 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
parent
7c720ce633
commit
036da5e374
@ -191,6 +191,7 @@ wxMSW:
|
|||||||
- Implemented wxMouseCaptureChangedEvent and made wxGenericDragImage check it
|
- Implemented wxMouseCaptureChangedEvent and made wxGenericDragImage check it
|
||||||
has the capture before release it.
|
has the capture before release it.
|
||||||
- fixed bugs in multiple selection wxCheckListBox
|
- fixed bugs in multiple selection wxCheckListBox
|
||||||
|
- default button handling is now closer to expected
|
||||||
|
|
||||||
wxGTK:
|
wxGTK:
|
||||||
|
|
||||||
|
@ -37,18 +37,29 @@ class WXDLLEXPORT wxWindowBase;
|
|||||||
|
|
||||||
class WXDLLEXPORT wxControlContainer
|
class WXDLLEXPORT wxControlContainer
|
||||||
{
|
{
|
||||||
DECLARE_NO_COPY_CLASS(wxControlContainer)
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// ctors and such
|
// ctors and such
|
||||||
wxControlContainer(wxWindow *winParent = NULL);
|
wxControlContainer(wxWindow *winParent = NULL);
|
||||||
void SetContainerWindow(wxWindow *winParent) { m_winParent = winParent; }
|
void SetContainerWindow(wxWindow *winParent) { m_winParent = winParent; }
|
||||||
|
|
||||||
// default item access
|
// default item access: we have a permanent default item which is the one
|
||||||
wxWindow *GetDefaultItem() const { return m_winDefault; }
|
// set by the user code but we may also have a temporary default item which
|
||||||
|
// would be chosen if the user pressed "Enter" now but the default action
|
||||||
|
// reverts to the "permanent" default as soon as this temporary default
|
||||||
|
// item lsoes focus
|
||||||
|
|
||||||
|
// get the default item, temporary or permanent
|
||||||
|
wxWindow *GetDefaultItem() const
|
||||||
|
{ return m_winTmpDefault ? m_winTmpDefault : m_winDefault; }
|
||||||
|
|
||||||
|
// set the permanent default item, return its old value
|
||||||
wxWindow *SetDefaultItem(wxWindow *win)
|
wxWindow *SetDefaultItem(wxWindow *win)
|
||||||
{ wxWindow *winOld = m_winDefault; m_winDefault = win; return winOld; }
|
{ wxWindow *winOld = m_winDefault; m_winDefault = win; return winOld; }
|
||||||
|
|
||||||
|
// set a temporary default item, SetTmpDefaultItem(NULL) should be called
|
||||||
|
// soon after a call to SetTmpDefaultItem(window)
|
||||||
|
void SetTmpDefaultItem(wxWindow *win) { m_winTmpDefault = win; }
|
||||||
|
|
||||||
// the methods to be called from the window event handlers
|
// the methods to be called from the window event handlers
|
||||||
void HandleOnNavigationKey(wxNavigationKeyEvent& event);
|
void HandleOnNavigationKey(wxNavigationKeyEvent& event);
|
||||||
void HandleOnFocus(wxFocusEvent& event);
|
void HandleOnFocus(wxFocusEvent& event);
|
||||||
@ -72,8 +83,13 @@ protected:
|
|||||||
// the child which had the focus last time this panel was activated
|
// the child which had the focus last time this panel was activated
|
||||||
wxWindow *m_winLastFocused;
|
wxWindow *m_winLastFocused;
|
||||||
|
|
||||||
// a default window (e.g. a button) or NULL
|
// a default window (usually a button) or NULL
|
||||||
wxWindow *m_winDefault;
|
wxWindow *m_winDefault;
|
||||||
|
|
||||||
|
// a temporary override of m_winDefault, use the latter if NULL
|
||||||
|
wxWindow *m_winTmpDefault;
|
||||||
|
|
||||||
|
DECLARE_NO_COPY_CLASS(wxControlContainer)
|
||||||
};
|
};
|
||||||
|
|
||||||
// this function is for wxWindows internal use only
|
// this function is for wxWindows internal use only
|
||||||
@ -94,6 +110,7 @@ public: \
|
|||||||
virtual void RemoveChild(wxWindowBase *child); \
|
virtual void RemoveChild(wxWindowBase *child); \
|
||||||
virtual wxWindow *GetDefaultItem() const; \
|
virtual wxWindow *GetDefaultItem() const; \
|
||||||
virtual wxWindow *SetDefaultItem(wxWindow *child); \
|
virtual wxWindow *SetDefaultItem(wxWindow *child); \
|
||||||
|
virtual void SetTmpDefaultItem(wxWindow *win); \
|
||||||
\
|
\
|
||||||
protected: \
|
protected: \
|
||||||
wxControlContainer m_container
|
wxControlContainer m_container
|
||||||
@ -111,6 +128,11 @@ wxWindow *classname::SetDefaultItem(wxWindow *child) \
|
|||||||
return m_container.SetDefaultItem(child); \
|
return m_container.SetDefaultItem(child); \
|
||||||
} \
|
} \
|
||||||
\
|
\
|
||||||
|
void classname::SetTmpDefaultItem(wxWindow *child) \
|
||||||
|
{ \
|
||||||
|
m_container.SetTmpDefaultItem(child); \
|
||||||
|
} \
|
||||||
|
\
|
||||||
wxWindow *classname::GetDefaultItem() const \
|
wxWindow *classname::GetDefaultItem() const \
|
||||||
{ \
|
{ \
|
||||||
return m_container.GetDefaultItem(); \
|
return m_container.GetDefaultItem(); \
|
||||||
|
@ -69,6 +69,13 @@ protected:
|
|||||||
// send a notification event, return TRUE if processed
|
// send a notification event, return TRUE if processed
|
||||||
bool SendClickEvent();
|
bool SendClickEvent();
|
||||||
|
|
||||||
|
// default button handling
|
||||||
|
void SetTmpDefault();
|
||||||
|
void UnsetTmpDefault();
|
||||||
|
|
||||||
|
static void UpdateDefaultStyle(wxWindow *winDefault,
|
||||||
|
wxWindow *winOldDefault);
|
||||||
|
|
||||||
// usually overridden base class virtuals
|
// usually overridden base class virtuals
|
||||||
virtual wxSize DoGetBestSize() const;
|
virtual wxSize DoGetBestSize() const;
|
||||||
virtual WXDWORD MSWGetStyle(long style, WXDWORD *exstyle) const;
|
virtual WXDWORD MSWGetStyle(long style, WXDWORD *exstyle) const;
|
||||||
|
@ -400,6 +400,9 @@ public:
|
|||||||
virtual wxWindow *SetDefaultItem(wxWindow * WXUNUSED(child))
|
virtual wxWindow *SetDefaultItem(wxWindow * WXUNUSED(child))
|
||||||
{ return NULL; }
|
{ return NULL; }
|
||||||
|
|
||||||
|
// set this child as temporary default
|
||||||
|
virtual void SetTmpDefaultItem(wxWindow * WXUNUSED(win)) { }
|
||||||
|
|
||||||
// parent/children relations
|
// parent/children relations
|
||||||
// -------------------------
|
// -------------------------
|
||||||
|
|
||||||
|
@ -45,6 +45,7 @@ wxControlContainer::wxControlContainer(wxWindow *winParent)
|
|||||||
m_winParent = winParent;
|
m_winParent = winParent;
|
||||||
|
|
||||||
m_winLastFocused =
|
m_winLastFocused =
|
||||||
|
m_winTmpDefault =
|
||||||
m_winDefault = NULL;
|
m_winDefault = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -263,6 +264,9 @@ void wxControlContainer::HandleOnWindowDestroy(wxWindowBase *child)
|
|||||||
|
|
||||||
if ( child == m_winDefault )
|
if ( child == m_winDefault )
|
||||||
m_winDefault = NULL;
|
m_winDefault = NULL;
|
||||||
|
|
||||||
|
if ( child == m_winTmpDefault )
|
||||||
|
m_winTmpDefault = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
|
@ -180,36 +180,71 @@ wxSize wxButtonBase::GetDefaultSize()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
// set this button as the default one in its panel
|
// default button handling
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// set this button as the (permanently) default one in its panel
|
||||||
void wxButton::SetDefault()
|
void wxButton::SetDefault()
|
||||||
{
|
{
|
||||||
wxWindow *parent = GetParent();
|
wxWindow *parent = GetParent();
|
||||||
wxButton *btnOldDefault;
|
|
||||||
if ( parent )
|
|
||||||
{
|
|
||||||
wxWindow *winOldDefault = parent->SetDefaultItem(this);
|
|
||||||
btnOldDefault = wxDynamicCast(winOldDefault, wxButton);
|
|
||||||
|
|
||||||
::SendMessage(GetWinHwnd(parent), DM_SETDEFID, m_windowId, 0L);
|
wxCHECK_RET( parent, _T("button without parent?") );
|
||||||
}
|
|
||||||
else // is a button without parent really normal?
|
|
||||||
{
|
|
||||||
btnOldDefault = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( btnOldDefault && btnOldDefault != this )
|
// set this one as the default button both for wxWindows and Windows
|
||||||
|
wxWindow *winOldDefault = parent->SetDefaultItem(this);
|
||||||
|
::SendMessage(GetWinHwnd(parent), DM_SETDEFID, m_windowId, 0L);
|
||||||
|
|
||||||
|
UpdateDefaultStyle(this, winOldDefault);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wxButton::SetTmpDefault()
|
||||||
|
{
|
||||||
|
wxWindow *parent = GetParent();
|
||||||
|
|
||||||
|
wxCHECK_RET( parent, _T("button without parent?") );
|
||||||
|
|
||||||
|
wxWindow *winOldDefault = parent->GetDefaultItem();
|
||||||
|
parent->SetTmpDefaultItem(this);
|
||||||
|
if ( winOldDefault != this )
|
||||||
|
{
|
||||||
|
UpdateDefaultStyle(this, winOldDefault);
|
||||||
|
}
|
||||||
|
//else: no styles to update
|
||||||
|
}
|
||||||
|
|
||||||
|
void wxButton::UnsetTmpDefault()
|
||||||
|
{
|
||||||
|
wxWindow *parent = GetParent();
|
||||||
|
|
||||||
|
wxCHECK_RET( parent, _T("button without parent?") );
|
||||||
|
|
||||||
|
parent->SetTmpDefaultItem(NULL);
|
||||||
|
|
||||||
|
wxWindow *winOldDefault = parent->GetDefaultItem();
|
||||||
|
if ( winOldDefault != this )
|
||||||
|
{
|
||||||
|
UpdateDefaultStyle(winOldDefault, this);
|
||||||
|
}
|
||||||
|
//else: we had been default before anyhow
|
||||||
|
}
|
||||||
|
|
||||||
|
/* static */
|
||||||
|
void
|
||||||
|
wxButton::UpdateDefaultStyle(wxWindow *winDefault, wxWindow *winOldDefault)
|
||||||
|
{
|
||||||
|
// clear the BS_DEFPUSHBUTTON for the old default button
|
||||||
|
wxButton *btnOldDefault = wxDynamicCast(winOldDefault, wxButton);
|
||||||
|
if ( btnOldDefault && btnOldDefault != winDefault )
|
||||||
{
|
{
|
||||||
// remove the BS_DEFPUSHBUTTON style from the other button
|
// remove the BS_DEFPUSHBUTTON style from the other button
|
||||||
long style = GetWindowLong(GetHwndOf(btnOldDefault), GWL_STYLE);
|
long style = ::GetWindowLong(GetHwndOf(btnOldDefault), GWL_STYLE);
|
||||||
|
|
||||||
// don't do it with the owner drawn buttons because it will reset
|
// don't do it with the owner drawn buttons because it will reset
|
||||||
// BS_OWNERDRAW style bit too (BS_OWNERDRAW & BS_DEFPUSHBUTTON != 0)!
|
// BS_OWNERDRAW style bit too (BS_OWNERDRAW & BS_DEFPUSHBUTTON != 0)!
|
||||||
if ( (style & BS_OWNERDRAW) != BS_OWNERDRAW )
|
if ( (style & BS_OWNERDRAW) != BS_OWNERDRAW )
|
||||||
{
|
{
|
||||||
style &= ~BS_DEFPUSHBUTTON;
|
style &= ~BS_DEFPUSHBUTTON;
|
||||||
SendMessage(GetHwndOf(btnOldDefault), BM_SETSTYLE, style, 1L);
|
::SendMessage(GetHwndOf(btnOldDefault), BM_SETSTYLE, style, 1L);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -219,12 +254,20 @@ void wxButton::SetDefault()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// set this button as the default
|
// and set BS_DEFPUSHBUTTON for this button
|
||||||
long style = GetWindowLong(GetHwnd(), GWL_STYLE);
|
wxButton *btnDefault = wxDynamicCast(winDefault, wxButton);
|
||||||
if ( (style & BS_OWNERDRAW) != BS_OWNERDRAW )
|
if ( btnDefault )
|
||||||
{
|
{
|
||||||
style |= BS_DEFPUSHBUTTON;
|
long style = ::GetWindowLong(GetHwndOf(btnDefault), GWL_STYLE);
|
||||||
SendMessage(GetHwnd(), BM_SETSTYLE, style, 1L);
|
if ( (style & BS_OWNERDRAW) != BS_OWNERDRAW )
|
||||||
|
{
|
||||||
|
style |= BS_DEFPUSHBUTTON;
|
||||||
|
::SendMessage(GetHwndOf(btnDefault), BM_SETSTYLE, style, 1L);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
btnDefault->Refresh();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -266,13 +309,18 @@ bool wxButton::MSWCommand(WXUINT param, WXWORD WXUNUSED(id))
|
|||||||
|
|
||||||
long wxButton::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam)
|
long wxButton::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam)
|
||||||
{
|
{
|
||||||
// when we receive focus, we want to become the default button in our
|
// when we receive focus, we want to temporary become the default button in
|
||||||
// parent panel
|
// our parent panel so that pressing "Enter" would activate us -- and when
|
||||||
|
// losing it we should restore the previous default button as well
|
||||||
if ( nMsg == WM_SETFOCUS )
|
if ( nMsg == WM_SETFOCUS )
|
||||||
{
|
{
|
||||||
SetDefault();
|
SetTmpDefault();
|
||||||
|
|
||||||
// let the default processign take place too
|
// let the default processing take place too
|
||||||
|
}
|
||||||
|
else if ( nMsg == WM_KILLFOCUS )
|
||||||
|
{
|
||||||
|
UnsetTmpDefault();
|
||||||
}
|
}
|
||||||
else if ( nMsg == WM_LBUTTONDBLCLK )
|
else if ( nMsg == WM_LBUTTONDBLCLK )
|
||||||
{
|
{
|
||||||
@ -280,7 +328,7 @@ long wxButton::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam)
|
|||||||
// appearance - without this, it won't do it
|
// appearance - without this, it won't do it
|
||||||
(void)wxControl::MSWWindowProc(WM_LBUTTONDOWN, wParam, lParam);
|
(void)wxControl::MSWWindowProc(WM_LBUTTONDOWN, wParam, lParam);
|
||||||
|
|
||||||
// and conitnue with processing the message normally as well
|
// and continue with processing the message normally as well
|
||||||
}
|
}
|
||||||
|
|
||||||
// let the base class do all real processing
|
// let the base class do all real processing
|
||||||
|
@ -1995,14 +1995,9 @@ bool wxWindowMSW::MSWProcessMessage(WXMSG* pMsg)
|
|||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
else // no default button
|
//else: no default button
|
||||||
#endif // wxUSE_BUTTON
|
#endif // wxUSE_BUTTON
|
||||||
{
|
// treat Enter as TAB: pass to the next control
|
||||||
// no special function for enter and don't even
|
|
||||||
// let IsDialogMessage() have it: it seems to
|
|
||||||
// do something really strange with it
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
Loading…
Reference in New Issue
Block a user