protected:
// event handlers
void OnKillFocus(wxFocusEvent& event);
- void OnKeyUp(wxKeyEvent& event);
+ void OnKeyDown(wxKeyEvent& event);
private:
wxPopupTransientWindow *m_popup;
BEGIN_EVENT_TABLE(wxPopupFocusHandler, wxEvtHandler)
EVT_KILL_FOCUS(wxPopupFocusHandler::OnKillFocus)
- EVT_KEY_UP(wxPopupFocusHandler::OnKeyUp)
+ EVT_KEY_DOWN(wxPopupFocusHandler::OnKeyDown)
END_EVENT_TABLE()
// ============================================================================
{
m_child =
m_focus = (wxWindow *)NULL;
+
+ m_handlerFocus = NULL;
+ m_handlerPopup = NULL;
}
wxPopupTransientWindow::wxPopupTransientWindow(wxWindow *parent, int style)
wxPopupTransientWindow::~wxPopupTransientWindow()
{
PopHandlers();
+
+ delete m_handlerFocus;
+ delete m_handlerPopup;
}
void wxPopupTransientWindow::PopHandlers()
{
if ( m_child )
{
- m_child->PopEventHandler(TRUE /* delete it */);
+ if ( !m_child->RemoveEventHandler(m_handlerPopup) )
+ {
+ // something is very wrong and someone else probably deleted our
+ // handler - so don't risk deleting it second time
+ m_handlerPopup = NULL;
+ }
+
m_child->ReleaseMouse();
m_child = NULL;
}
if ( m_focus )
{
- m_focus->PopEventHandler(TRUE /* delete it */);
+ if ( !m_focus->RemoveEventHandler(m_handlerFocus) )
+ {
+ // see above
+ m_handlerFocus = NULL;
+ }
+
m_focus = NULL;
}
}
// first
Show();
+ delete m_handlerPopup;
+ m_handlerPopup = new wxPopupWindowHandler(this);
+
m_child->CaptureMouse();
- m_child->PushEventHandler(new wxPopupWindowHandler(this));
+ m_child->PushEventHandler(m_handlerPopup);
m_focus = winFocus ? winFocus : this;
m_focus->SetFocus();
#ifdef __WXMSW__
- // FIXME: I don't know why does this happen but sometimes SetFocus() simply
- // refuses to work under MSW - no error happens but the focus is not
- // given to the window, i.e. the assert below is triggered
- //
- // Try work around this as we can...
-
- //wxASSERT_MSG( FindFocus() == m_focus, _T("setting focus failed") );
+ // MSW doesn't allow to set focus to the popup window, but we need to
+ // subclass the window which has the focus, and not winFocus passed in or
+ // otherwise everything else breaks down
m_focus = FindFocus();
if ( m_focus )
#endif // __WXMSW__
{
- m_focus->PushEventHandler(new wxPopupFocusHandler(this));
+ delete m_handlerFocus;
+ m_handlerFocus = new wxPopupFocusHandler(this);
+
+ m_focus->PushEventHandler(m_handlerFocus);
}
}
wxWindow *sbar = NULL;
wxWindow *win = (wxWindow *)event.GetEventObject();
+
switch ( win->HitTest(pos.x, pos.y) )
{
case wxHT_WINDOW_OUTSIDE:
// when we lose focus we always disappear - unless it goes to the popup (in
// which case we don't really lose it)
- if ( event.GetWindow() != m_popup )
- m_popup->DismissAndNotify();
+ wxWindow *win = event.GetWindow();
+ while ( win )
+ {
+ if ( win == m_popup )
+ return;
+ win = win->GetParent();
+ }
+
+ m_popup->DismissAndNotify();
}
-void wxPopupFocusHandler::OnKeyUp(wxKeyEvent& event)
+void wxPopupFocusHandler::OnKeyDown(wxKeyEvent& event)
{
// let the window have it first, it might process the keys
if ( !m_popup->ProcessEvent(event) )