void wxPopupTransientWindow::Popup(wxWindow *winFocus)
{
+ // If we have a single child, we suppose that it must cover the entire
+ // popup window and hence we give the mouse capture to it instead of
+ // keeping it for ourselves.
+ //
+ // Notice that this works best for combobox-like popups which have a single
+ // control inside them and not so well for popups containing a single
+ // wxPanel with multiple children inside it but OTOH it does no harm in
+ // this case neither and we can't reliably distinguish between them.
const wxWindowList& children = GetChildren();
- if ( children.GetCount() )
+ if ( children.GetCount() == 1 )
{
m_child = children.GetFirst()->GetData();
}
if (IsShown() && m_child)
{
- wxPoint pos = ScreenToClient(wxGetMousePosition());
- wxRect rect(GetSize());
-
- if ( rect.Contains(pos) )
+ // Store the last mouse position to minimize the number of calls to
+ // wxFindWindowAtPoint() which are quite expensive.
+ static wxPoint s_posLast;
+ const wxPoint pos = wxGetMousePosition();
+ if ( pos != s_posLast )
{
- if ( m_child->HasCapture() )
+ s_posLast = pos;
+
+ wxWindow* const winUnderMouse = wxFindWindowAtPoint(pos);
+
+ // We release the mouse capture while the mouse is inside the popup
+ // itself to allow using it normally with the controls inside it.
+ if ( wxGetTopLevelParent(winUnderMouse) == this )
{
- m_child->ReleaseMouse();
+ if ( m_child->HasCapture() )
+ {
+ m_child->ReleaseMouse();
+ }
}
- }
- else
- {
- if ( !m_child->HasCapture() )
+ else // And we reacquire it as soon as the mouse goes outside.
{
- m_child->CaptureMouse();
+ if ( !m_child->HasCapture() )
+ {
+ m_child->CaptureMouse();
+ }
}
}
}