// Utilies needed by the popups or native implementations
//
+ // Returns true if given key combination should toggle the popup.
+ // NB: This is a separate from other keyboard handling because:
+ // 1) Replaceability.
+ // 2) Centralized code (otherwise it'd be split up between
+ // wxComboCtrl key handler and wxVListBoxComboPopup's
+ // key handler).
+ virtual bool IsKeyPopupToggle(const wxKeyEvent& event) const = 0;
+
// Draws focus background (on combo control) in a way typical on platform.
// Unless you plan to paint your own focus indicator, you should always call this
// in your wxComboPopup::PaintComboControl implementation.
void CreateTextCtrl( int extraStyle, const wxValidator& validator );
// Installs standard input handler to combo (and optionally to the textctrl)
- void InstallInputHandlers( bool alsoTextCtrl = true );
+ void InstallInputHandlers();
// Draws dropbutton. Using wxRenderer or bitmaps, as appropriate.
void DrawButton( wxDC& dc, const wxRect& rect, bool paintBg = true );
void OnFocusEvent(wxFocusEvent& event);
void OnTextCtrlEvent(wxCommandEvent& event);
void OnSysColourChanged(wxSysColourChangedEvent& event);
+ void OnKeyEvent(wxKeyEvent& event);
// Set customization flags (directs how wxComboCtrlBase helpers behave)
void Customize( wxUint32 flags ) { m_iFlags |= flags; }
// popup interface
wxComboPopup* m_popupInterface;
- // this is for this control itself
- wxEvtHandler* m_extraEvtHandler;
-
- // this is for text
+ // this is input etc. handler for the text control
wxEvtHandler* m_textEvtHandler;
// this is for the top level window
// ----------------------------------------------------------------------------
//
-// This is pushed to the event handler queue of either combo box
-// or its textctrl (latter if not readonly combo).
+// This is pushed to the event handler queue of the child textctrl.
//
class wxComboBoxExtraInputHandler : public wxEvtHandler
{
void wxComboBoxExtraInputHandler::OnKey(wxKeyEvent& event)
{
- int keycode = event.GetKeyCode();
+ // Let the wxComboCtrl event handler have a go first.
+ wxComboCtrlBase* combo = m_combo;
+ wxObject* prevObj = event.GetEventObject();
- if ( keycode == WXK_TAB && !m_combo->IsPopupShown() )
- {
- wxNavigationKeyEvent evt;
- evt.SetFlags(wxNavigationKeyEvent::FromTab|
- (!event.ShiftDown()?wxNavigationKeyEvent::IsForward:
- wxNavigationKeyEvent::IsBackward));
- evt.SetEventObject(m_combo);
- m_combo->GetParent()->GetEventHandler()->AddPendingEvent(evt);
- return;
- }
-
- if ( m_combo->IsPopupShown() )
- {
- // pass it to the popped up control
- m_combo->GetPopupControl()->GetControl()->AddPendingEvent(event);
- }
- else // no popup
- {
- int comboStyle = m_combo->GetWindowStyle();
- wxComboPopup* popupInterface = m_combo->GetPopupControl();
-
- if ( !popupInterface )
- {
- event.Skip();
- return;
- }
+ event.SetId(combo->GetId());
+ event.SetEventObject(combo);
+ combo->GetEventHandler()->ProcessEvent(event);
- if ( (comboStyle & wxCB_READONLY) ||
- ( keycode != WXK_RIGHT && keycode != WXK_LEFT )
- )
- {
- popupInterface->OnComboKeyEvent(event);
- }
- else
- event.Skip();
- }
+ event.SetId(((wxWindow*)prevObj)->GetId());
+ event.SetEventObject(prevObj);
}
void wxComboBoxExtraInputHandler::OnFocus(wxFocusEvent& event)
m_combo->SetSelection(-1,-1);
}
- if ( event.GetId() != m_combo->GetId() )
- {
- // Add textctrl set focus events as combo set focus events
- // NOTE: Simply changing the event and skipping didn't seem
- // to do the trick.
- wxFocusEvent evt2(wxEVT_SET_FOCUS,m_combo->GetId());
- evt2.SetEventObject(m_combo);
- m_combo->GetEventHandler()->ProcessEvent(evt2);
- }
- else
- event.Skip();
+ // Send focus indication to parent.
+ // NB: This is needed for cases where the textctrl gets focus
+ // instead of its parent. While this may trigger multiple
+ // wxEVT_SET_FOCUSes (since m_text->SetFocus is called
+ // from combo's focus event handler), they should be quite
+ // harmless.
+ wxFocusEvent evt2(wxEVT_SET_FOCUS,m_combo->GetId());
+ evt2.SetEventObject(m_combo);
+ m_combo->GetEventHandler()->ProcessEvent(evt2);
event.Skip();
}
EVT_SET_FOCUS(wxComboCtrlBase::OnFocusEvent)
EVT_KILL_FOCUS(wxComboCtrlBase::OnFocusEvent)
//EVT_BUTTON(wxID_ANY,wxComboCtrlBase::OnButtonClickEvent)
+ EVT_KEY_DOWN(wxComboCtrlBase::OnKeyEvent)
EVT_TEXT_ENTER(wxID_ANY,wxComboCtrlBase::OnTextCtrlEvent)
EVT_SYS_COLOUR_CHANGED(wxComboCtrlBase::OnSysColourChanged)
END_EVENT_TABLE()
m_text = (wxTextCtrl*) NULL;
m_popupInterface = (wxComboPopup*) NULL;
- m_extraEvtHandler = (wxEvtHandler*) NULL;
m_popupExtraHandler = (wxEvtHandler*) NULL;
m_textEvtHandler = (wxEvtHandler*) NULL;
return true;
}
-void wxComboCtrlBase::InstallInputHandlers( bool alsoTextCtrl )
+void wxComboCtrlBase::InstallInputHandlers()
{
- if ( m_text && alsoTextCtrl )
+ if ( m_text )
{
m_textEvtHandler = new wxComboBoxExtraInputHandler(this);
m_text->PushEventHandler(m_textEvtHandler);
}
-
- wxComboBoxExtraInputHandler* inputHandler = new wxComboBoxExtraInputHandler(this);
- PushEventHandler(inputHandler);
- m_extraEvtHandler = inputHandler;
}
void
DestroyPopup();
- RemoveEventHandler(m_extraEvtHandler);
-
if ( m_text )
m_text->RemoveEventHandler(m_textEvtHandler);
delete m_textEvtHandler;
- delete m_extraEvtHandler;
}
event.Skip();
}
-void wxComboCtrlBase::OnFocusEvent( wxFocusEvent& )
+void wxComboCtrlBase::OnKeyEvent(wxKeyEvent& event)
{
- // First click is the first part of double-click
- // Some platforms don't generate down-less mouse up-event
- // (Windows does, GTK+2 doesn't), so that's why we have
- // to do this.
- m_timeLastMouseUp = ::wxGetLocalTimeMillis();
+ if ( IsPopupShown() )
+ {
+ // pass it to the popped up control
+ GetPopupControl()->GetControl()->AddPendingEvent(event);
+ }
+ else // no popup
+ {
+ int keycode = event.GetKeyCode();
- if ( m_text )
+ if ( keycode == WXK_TAB )
+ {
+ wxNavigationKeyEvent evt;
+ evt.SetFlags(wxNavigationKeyEvent::FromTab|
+ (!event.ShiftDown() ? wxNavigationKeyEvent::IsForward
+ : wxNavigationKeyEvent::IsBackward));
+ evt.SetEventObject(this);
+ GetParent()->GetEventHandler()->AddPendingEvent(evt);
+ return;
+ }
+
+ if ( IsKeyPopupToggle(event) )
+ {
+ OnButtonClick();
+ return;
+ }
+
+ int comboStyle = GetWindowStyle();
+ wxComboPopup* popupInterface = GetPopupControl();
+
+ if ( !popupInterface )
+ {
+ event.Skip();
+ return;
+ }
+
+ if ( (comboStyle & wxCB_READONLY) ||
+ (keycode != WXK_RIGHT && keycode != WXK_LEFT) )
+ {
+ popupInterface->OnComboKeyEvent(event);
+ }
+ else
+ event.Skip();
+ }
+}
+
+void wxComboCtrlBase::OnFocusEvent( wxFocusEvent& event )
+{
+ if ( event.GetEventType() == wxEVT_SET_FOCUS )
{
- m_text->SetFocus();
+ // First click is the first part of double-click
+ // Some platforms don't generate down-less mouse up-event
+ // (Windows does, GTK+2 doesn't), so that's why we have
+ // to do this.
+ m_timeLastMouseUp = ::wxGetLocalTimeMillis();
+
+ if ( m_text && m_text != ::wxWindow::FindFocus() )
+ m_text->SetFocus();
}
- else
- // no need to check for m_widthCustomPaint - that
- // area never gets special handling when selected
- // (in writable mode, that is)
- Refresh();
+
+ Refresh();
}
void wxComboCtrlBase::OnSysColourChanged(wxSysColourChangedEvent& WXUNUSED(event))