const wxPoint& WXUNUSED(pos) = wxDefaultPosition)
: wxMessageDialogBase(parent, message, caption, style)
{
+ m_hook = NULL;
}
virtual int ShowModal();
-protected:
+private:
+ static WXLRESULT wxCALLBACK HookFunction(int code, WXWPARAM, WXLPARAM);
+
+ WXHANDLE m_hook; // HHOOK used to position the message box
+
DECLARE_DYNAMIC_CLASS(wxMessageDialog)
DECLARE_NO_COPY_CLASS(wxMessageDialog)
};
#if wxUSE_MSGDLG
void MyFrame::MessageBox(wxCommandEvent& WXUNUSED(event) )
{
- wxMessageDialog dialog(NULL,
+ wxMessageDialog dialog(this,
_T("This is a message box\nA long, long string to test out the message box properly"),
_T("Message box text"),
- wxNO_DEFAULT | wxYES_NO | wxCANCEL | wxICON_INFORMATION);
+ wxCENTER | wxNO_DEFAULT | wxYES_NO | wxCANCEL | wxICON_INFORMATION);
if ( dialog.SetYesNoLabels(_T("Answer &Yes"),_T("Answer &No")) )
{
#include "wx/app.h"
#include "wx/utils.h"
#include "wx/dialog.h"
+ #include "wx/hashmap.h"
#endif
#include "wx/msw/private.h"
IMPLEMENT_CLASS(wxMessageDialog, wxDialog)
+// there can potentially be one message box per thread so we use a hash map
+// with thread ids as keys and (currently shown) message boxes as values
+//
+// TODO: replace this with wxTLS once it's available
+WX_DECLARE_HASH_MAP(unsigned long, wxMessageDialog *,
+ wxIntegerHash, wxIntegerEqual,
+ wxMessageDialogMap);
+
+namespace
+{
+
+wxMessageDialogMap& HookMap()
+{
+ static wxMessageDialogMap s_Map;
+
+ return s_Map;
+}
+
+} // anonymous namespace
+
+/* static */
+WXLRESULT wxCALLBACK
+wxMessageDialog::HookFunction(int code, WXWPARAM wParam, WXLPARAM lParam)
+{
+ // Find the thread-local instance of wxMessageDialog
+ const DWORD tid = ::GetCurrentThreadId();
+ wxMessageDialogMap::iterator node = HookMap().find(tid);
+ wxCHECK_MSG( node != HookMap().end(), false,
+ wxT("bogus thread id in wxMessageDialog::Hook") );
+
+ wxMessageDialog * const wnd = node->second;
+
+ const HHOOK hhook = (HHOOK)wnd->m_hook;
+ const LRESULT rc = ::CallNextHookEx(hhook, code, wParam, lParam);
+
+ if ( code == HC_ACTION && lParam )
+ {
+ const CWPRETSTRUCT * const s = (CWPRETSTRUCT *)lParam;
+
+ if ( s->message == HCBT_ACTIVATE )
+ {
+ // we won't need this hook any longer
+ ::UnhookWindowsHookEx(hhook);
+ wnd->m_hook = NULL;
+ HookMap().erase(tid);
+
+ if ( wnd->GetMessageDialogStyle() & wxCENTER )
+ {
+ wnd->SetHWND(s->hwnd);
+ wnd->Center(); // center on parent
+ wnd->SetHWND(NULL);
+ }
+ //else: default behaviour, center on screen
+ }
+ }
+
+ return rc;
+}
+
int wxMessageDialog::ShowModal()
{
if ( !wxTheApp->GetTopWindow() )
}
#endif // wxUSE_UNICODE
+ // install the hook if we need to position the dialog in a non-default way
+ if ( wxStyle & wxCENTER )
+ {
+ const DWORD tid = ::GetCurrentThreadId();
+ m_hook = ::SetWindowsHookEx(WH_CALLWNDPROCRET,
+ &wxMessageDialog::HookFunction, NULL, tid);
+ HookMap()[tid] = this;
+ }
+
// do show the dialog
int msAns = MessageBox(hWnd, message.wx_str(), m_caption.wx_str(), msStyle);
int ans;