Add wxNavigationEnabled<> helper for implementing TAB navigation.
Provide a clean, public and documented way to implement proper TAB navigation for subwindows of a composite control: instead of using various ugly and never documented WX_XXX_CONTROL_CONTAINER macros it is now enough to simply inherit from wxNavigationEnabled<BaseClass> to do it. No real changes in the code as the new class is not used anywhere yet. git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@68363 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
@@ -443,6 +443,7 @@ All:
|
|||||||
|
|
||||||
All (GUI):
|
All (GUI):
|
||||||
|
|
||||||
|
- Added documented, public wxNavigationEnabled<> class.
|
||||||
- Support float, double and file name values in wxGenericValidator (troelsk).
|
- Support float, double and file name values in wxGenericValidator (troelsk).
|
||||||
- Fix keyboard navigation in wxGrid with hidden columns (ivan_14_32).
|
- Fix keyboard navigation in wxGrid with hidden columns (ivan_14_32).
|
||||||
- Add wxDataViewEvent::IsEditCancelled() (Allonii).
|
- Add wxDataViewEvent::IsEditCancelled() (Allonii).
|
||||||
|
@@ -1,12 +1,11 @@
|
|||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
// Name: wx/containr.h
|
// Name: wx/containr.h
|
||||||
// Purpose: wxControlContainer class declration: a "mix-in" class which
|
// Purpose: wxControlContainer and wxNavigationEnabled declarations
|
||||||
// implements the TAB navigation between the controls
|
|
||||||
// Author: Vadim Zeitlin
|
// Author: Vadim Zeitlin
|
||||||
// Modified by:
|
// Modified by:
|
||||||
// Created: 06.08.01
|
// Created: 06.08.01
|
||||||
// RCS-ID: $Id$
|
// RCS-ID: $Id$
|
||||||
// Copyright: (c) 2001 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
|
// Copyright: (c) 2001, 2011 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
|
||||||
// Licence: wxWindows licence
|
// Licence: wxWindows licence
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
@@ -19,12 +18,11 @@ class WXDLLIMPEXP_FWD_CORE wxWindow;
|
|||||||
class WXDLLIMPEXP_FWD_CORE wxWindowBase;
|
class WXDLLIMPEXP_FWD_CORE wxWindowBase;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Implementation note: wxControlContainer is not a real mix-in but rather
|
This header declares wxControlContainer class however it's not a real
|
||||||
a class meant to be aggregated with (and not inherited from). Although
|
container of controls but rather just a helper used to implement TAB
|
||||||
logically it should be a mix-in, doing it like this has no advantage from
|
navigation among the window children. You should rarely need to use it
|
||||||
the point of view of the existing code but does have some problems (we'd
|
directly, derive from the documented public wxNavigationEnabled<> class to
|
||||||
need to play tricks with event handlers which may be difficult to do
|
implement TAB navigation in a custom composite window.
|
||||||
safely). The price we pay for this simplicity is the ugly macros below.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
@@ -102,6 +100,164 @@ private:
|
|||||||
bool m_inSetFocus;
|
bool m_inSetFocus;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#ifdef wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// wxControlContainer for native TAB navigation
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// this must be a real class as we forward-declare it elsewhere
|
||||||
|
class WXDLLIMPEXP_CORE wxControlContainer : public wxControlContainerBase
|
||||||
|
{
|
||||||
|
protected:
|
||||||
|
// set the focus to the child which had it the last time
|
||||||
|
virtual bool SetFocusToChild();
|
||||||
|
};
|
||||||
|
|
||||||
|
#else // !wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
|
class WXDLLIMPEXP_FWD_CORE wxFocusEvent;
|
||||||
|
class WXDLLIMPEXP_FWD_CORE wxNavigationKeyEvent;
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// wxControlContainer for TAB navigation implemented in wx itself
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
class WXDLLIMPEXP_CORE wxControlContainer : public wxControlContainerBase
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
// default ctor, SetContainerWindow() must be called later
|
||||||
|
wxControlContainer();
|
||||||
|
|
||||||
|
// the methods to be called from the window event handlers
|
||||||
|
void HandleOnNavigationKey(wxNavigationKeyEvent& event);
|
||||||
|
void HandleOnFocus(wxFocusEvent& event);
|
||||||
|
void HandleOnWindowDestroy(wxWindowBase *child);
|
||||||
|
|
||||||
|
// called from OnChildFocus() handler, i.e. when one of our (grand)
|
||||||
|
// children gets the focus
|
||||||
|
void SetLastFocus(wxWindow *win);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
wxDECLARE_NO_COPY_CLASS(wxControlContainer);
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // wxHAS_NATIVE_TAB_TRAVERSAL/!wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
|
// this function is for wxWidgets internal use only
|
||||||
|
extern bool wxSetFocusToChild(wxWindow *win, wxWindow **child);
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// wxNavigationEnabled: Derive from this class to support keyboard navigation
|
||||||
|
// among window children in a wxWindow-derived class. The details of this class
|
||||||
|
// don't matter, you just need to derive from it to make navigation work.
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// The template parameter W must be a wxWindow-derived class.
|
||||||
|
template <class W>
|
||||||
|
class wxNavigationEnabled : public W
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
typedef W BaseWindowClass;
|
||||||
|
|
||||||
|
wxNavigationEnabled()
|
||||||
|
{
|
||||||
|
m_container.SetContainerWindow(this);
|
||||||
|
|
||||||
|
#ifndef wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
Connect(wxEVT_NAVIGATION_KEY,
|
||||||
|
wxNavigationKeyEventHandler(wxNavigationEnabled::OnNavigationKey));
|
||||||
|
|
||||||
|
Connect(wxEVT_SET_FOCUS,
|
||||||
|
wxFocusEventHandler(wxNavigationEnabled::OnFocus));
|
||||||
|
|
||||||
|
Connect(wxEVT_CHILD_FOCUS,
|
||||||
|
wxChildFocusEventHandler(wxNavigationEnabled::OnChildFocus));
|
||||||
|
#endif // !wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual bool AcceptsFocus() const
|
||||||
|
{
|
||||||
|
return m_container.AcceptsFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual bool AcceptsFocusRecursively() const
|
||||||
|
{
|
||||||
|
return m_container.AcceptsFocusRecursively();
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual bool AcceptsFocusFromKeyboard() const
|
||||||
|
{
|
||||||
|
return m_container.AcceptsFocusFromKeyboard();
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void AddChild(wxWindowBase *child)
|
||||||
|
{
|
||||||
|
BaseWindowClass::AddChild(child);
|
||||||
|
|
||||||
|
m_container.UpdateCanFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void RemoveChild(wxWindowBase *child)
|
||||||
|
{
|
||||||
|
#ifndef wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
m_container.HandleOnWindowDestroy(child);
|
||||||
|
#endif // !wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
|
BaseWindowClass::RemoveChild(child);
|
||||||
|
|
||||||
|
m_container.UpdateCanFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void SetFocus()
|
||||||
|
{
|
||||||
|
if ( !m_container.DoSetFocus() )
|
||||||
|
BaseWindowClass::SetFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetFocusIgnoringChildren()
|
||||||
|
{
|
||||||
|
BaseWindowClass::SetFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void AcceptFocus(bool acceptFocus)
|
||||||
|
{
|
||||||
|
m_container.SetCanFocus(acceptFocus);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
#ifndef wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
void OnNavigationKey(wxNavigationKeyEvent& event)
|
||||||
|
{
|
||||||
|
m_container.HandleOnNavigationKey(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
void OnFocus(wxFocusEvent& event)
|
||||||
|
{
|
||||||
|
m_container.HandleOnFocus(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
void OnChildFocus(wxChildFocusEvent& event)
|
||||||
|
{
|
||||||
|
m_container.SetLastFocus(event.GetWindow());
|
||||||
|
event.Skip();
|
||||||
|
}
|
||||||
|
#endif // !wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
|
wxControlContainer m_container;
|
||||||
|
|
||||||
|
|
||||||
|
wxDECLARE_NO_COPY_TEMPLATE_CLASS(wxNavigationEnabled, W);
|
||||||
|
};
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// Compatibility macros from now on, do NOT use them and preferably do not even
|
||||||
|
// look at them.
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
#ifdef WXWIN_COMPATIBILITY_2_8
|
||||||
|
|
||||||
// common part of WX_DECLARE_CONTROL_CONTAINER in the native and generic cases,
|
// common part of WX_DECLARE_CONTROL_CONTAINER in the native and generic cases,
|
||||||
// it should be used in the wxWindow-derived class declaration
|
// it should be used in the wxWindow-derived class declaration
|
||||||
#define WX_DECLARE_CONTROL_CONTAINER_BASE() \
|
#define WX_DECLARE_CONTROL_CONTAINER_BASE() \
|
||||||
@@ -156,20 +312,9 @@ protected: \
|
|||||||
return m_container.AcceptsFocusFromKeyboard(); \
|
return m_container.AcceptsFocusFromKeyboard(); \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#ifdef wxHAS_NATIVE_TAB_TRAVERSAL
|
#ifdef wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
|
||||||
// wxControlContainer for native TAB navigation
|
|
||||||
// ----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
// this must be a real class as we forward-declare it elsewhere
|
|
||||||
class WXDLLIMPEXP_CORE wxControlContainer : public wxControlContainerBase
|
|
||||||
{
|
|
||||||
protected:
|
|
||||||
// set the focus to the child which had it the last time
|
|
||||||
virtual bool SetFocusToChild();
|
|
||||||
};
|
|
||||||
|
|
||||||
#define WX_EVENT_TABLE_CONTROL_CONTAINER(classname)
|
#define WX_EVENT_TABLE_CONTROL_CONTAINER(classname)
|
||||||
|
|
||||||
#define WX_DECLARE_CONTROL_CONTAINER WX_DECLARE_CONTROL_CONTAINER_BASE
|
#define WX_DECLARE_CONTROL_CONTAINER WX_DECLARE_CONTROL_CONTAINER_BASE
|
||||||
@@ -191,38 +336,6 @@ protected:
|
|||||||
|
|
||||||
#else // !wxHAS_NATIVE_TAB_TRAVERSAL
|
#else // !wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
class WXDLLIMPEXP_FWD_CORE wxFocusEvent;
|
|
||||||
class WXDLLIMPEXP_FWD_CORE wxNavigationKeyEvent;
|
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
|
||||||
// wxControlContainer for TAB navigation implemented in wx itself
|
|
||||||
// ----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
class WXDLLIMPEXP_CORE wxControlContainer : public wxControlContainerBase
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
// default ctor, SetContainerWindow() must be called later
|
|
||||||
wxControlContainer();
|
|
||||||
|
|
||||||
// the methods to be called from the window event handlers
|
|
||||||
void HandleOnNavigationKey(wxNavigationKeyEvent& event);
|
|
||||||
void HandleOnFocus(wxFocusEvent& event);
|
|
||||||
void HandleOnWindowDestroy(wxWindowBase *child);
|
|
||||||
|
|
||||||
// called from OnChildFocus() handler, i.e. when one of our (grand)
|
|
||||||
// children gets the focus
|
|
||||||
void SetLastFocus(wxWindow *win);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
|
|
||||||
wxDECLARE_NO_COPY_CLASS(wxControlContainer);
|
|
||||||
};
|
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
|
||||||
// macros which may be used by the classes wishing to implement TAB navigation
|
|
||||||
// among their children
|
|
||||||
// ----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
// declare the methods to be forwarded
|
// declare the methods to be forwarded
|
||||||
#define WX_DECLARE_CONTROL_CONTAINER() \
|
#define WX_DECLARE_CONTROL_CONTAINER() \
|
||||||
WX_DECLARE_CONTROL_CONTAINER_BASE(); \
|
WX_DECLARE_CONTROL_CONTAINER_BASE(); \
|
||||||
@@ -274,7 +387,6 @@ public: \
|
|||||||
|
|
||||||
#endif // wxHAS_NATIVE_TAB_TRAVERSAL/!wxHAS_NATIVE_TAB_TRAVERSAL
|
#endif // wxHAS_NATIVE_TAB_TRAVERSAL/!wxHAS_NATIVE_TAB_TRAVERSAL
|
||||||
|
|
||||||
// this function is for wxWidgets internal use only
|
#endif // WXWIN_COMPATIBILITY_2_8
|
||||||
extern bool wxSetFocusToChild(wxWindow *win, wxWindow **child);
|
|
||||||
|
|
||||||
#endif // _WX_CONTAINR_H_
|
#endif // _WX_CONTAINR_H_
|
||||||
|
68
interface/wx/containr.h
Normal file
68
interface/wx/containr.h
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Name: wx/containr.h
|
||||||
|
// Purpose: documentation of wxNavigationEnabled<>
|
||||||
|
// Author: Vadim Zeitlin
|
||||||
|
// Created: 2011-07-23
|
||||||
|
// RCS-ID: $Id$
|
||||||
|
// Copyright: (c) 2011 Vadim Zeitlin <vadim@wxwidgets.org>
|
||||||
|
// Licence: wxWindows licence
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
/**
|
||||||
|
A helper class implementing TAB navigation among the window children.
|
||||||
|
|
||||||
|
This class contains the functionality needed to correctly implement TAB
|
||||||
|
navigation among the children of the window. Its exact contents is not
|
||||||
|
important and is intentionally not documented as the only way to use this
|
||||||
|
class is to inherit from it instead of inheriting from the usual base class
|
||||||
|
directly. For example, if some class needs to inherit from wxControl but
|
||||||
|
contains multiple sub-windows and needs to support keyboard navigation, it
|
||||||
|
is enough to declare it in the following way:
|
||||||
|
@code
|
||||||
|
class MyControlWithSubChildren :
|
||||||
|
public wxNavigationEnabled<wxControl>
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
// Default constructor is implemented in the same way as always.
|
||||||
|
MyControlWithSubChildren() { }
|
||||||
|
|
||||||
|
// Non-default constructor can't use wxControl ctor any more as
|
||||||
|
// wxControl is not its direct base class, but it can use Create().
|
||||||
|
MyControlWithSubChildren(wxWindow *parent, wxWindowID winid)
|
||||||
|
{
|
||||||
|
wxControl::Create(parent, winid);
|
||||||
|
|
||||||
|
// More creation code...
|
||||||
|
}
|
||||||
|
|
||||||
|
// Everything else as usual ...
|
||||||
|
};
|
||||||
|
@endcode
|
||||||
|
|
||||||
|
@library{wxcore}
|
||||||
|
|
||||||
|
@since 2.9.3
|
||||||
|
*/
|
||||||
|
template <class W>
|
||||||
|
class wxNavigationEnabled : public W
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// The name of the real base window class that this class derives from.
|
||||||
|
typedef W BaseWindowClass;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Default constructor.
|
||||||
|
|
||||||
|
This class provides only the default constructor as it's not possible,
|
||||||
|
in general, to provide all the constructors of the real base class
|
||||||
|
BaseWindowClass.
|
||||||
|
|
||||||
|
This is however not usually a problem for wxWindow-derived classes as,
|
||||||
|
by convention, they always define a Create() method such that calling
|
||||||
|
it on an object initialized using the default constructor is equivalent
|
||||||
|
to using a non-default constructor directly. So the classes inheriting
|
||||||
|
from wxNavigationEnabled<W> should simply call W::Create() in their
|
||||||
|
constructors.
|
||||||
|
*/
|
||||||
|
wxNavigationEnabled();
|
||||||
|
};
|
Reference in New Issue
Block a user