]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/msgdlg.cpp
don't append anything back in DoDeleteOneItem() if the control becomes empty (fixes...
[wxWidgets.git] / src / msw / msgdlg.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/msgdlg.cpp
3// Purpose: wxMessageDialog
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#ifdef __BORLANDC__
16 #pragma hdrstop
17#endif
18
19#if wxUSE_MSGDLG
20
21#include "wx/msgdlg.h"
22
23#ifndef WX_PRECOMP
24 #include "wx/app.h"
25 #include "wx/utils.h"
26 #include "wx/dialog.h"
27 #include "wx/hashmap.h"
28#endif
29
30#include "wx/msw/private.h"
31
32// For MB_TASKMODAL
33#ifdef __WXWINCE__
34#include "wx/msw/wince/missing.h"
35#endif
36
37IMPLEMENT_CLASS(wxMessageDialog, wxDialog)
38
39// there can potentially be one message box per thread so we use a hash map
40// with thread ids as keys and (currently shown) message boxes as values
41//
42// TODO: replace this with wxTLS once it's available
43WX_DECLARE_HASH_MAP(unsigned long, wxMessageDialog *,
44 wxIntegerHash, wxIntegerEqual,
45 wxMessageDialogMap);
46
47namespace
48{
49
50wxMessageDialogMap& HookMap()
51{
52 static wxMessageDialogMap s_Map;
53
54 return s_Map;
55}
56
57} // anonymous namespace
58
59/* static */
60WXLRESULT wxCALLBACK
61wxMessageDialog::HookFunction(int code, WXWPARAM wParam, WXLPARAM lParam)
62{
63 // Find the thread-local instance of wxMessageDialog
64 const DWORD tid = ::GetCurrentThreadId();
65 wxMessageDialogMap::iterator node = HookMap().find(tid);
66 wxCHECK_MSG( node != HookMap().end(), false,
67 wxT("bogus thread id in wxMessageDialog::Hook") );
68
69 wxMessageDialog * const wnd = node->second;
70
71 const HHOOK hhook = (HHOOK)wnd->m_hook;
72 const LRESULT rc = ::CallNextHookEx(hhook, code, wParam, lParam);
73
74 if ( code == HC_ACTION && lParam )
75 {
76 const CWPRETSTRUCT * const s = (CWPRETSTRUCT *)lParam;
77
78 if ( s->message == HCBT_ACTIVATE )
79 {
80 // we won't need this hook any longer
81 ::UnhookWindowsHookEx(hhook);
82 wnd->m_hook = NULL;
83 HookMap().erase(tid);
84
85 if ( wnd->GetMessageDialogStyle() & wxCENTER )
86 {
87 wnd->SetHWND(s->hwnd);
88 wnd->Center(); // center on parent
89 wnd->SetHWND(NULL);
90 }
91 //else: default behaviour, center on screen
92 }
93 }
94
95 return rc;
96}
97
98int wxMessageDialog::ShowModal()
99{
100 if ( !wxTheApp->GetTopWindow() )
101 {
102 // when the message box is shown from wxApp::OnInit() (i.e. before the
103 // message loop is entered), this must be done or the next message box
104 // will never be shown - just try putting 2 calls to wxMessageBox() in
105 // OnInit() to see it
106 while ( wxTheApp->Pending() )
107 wxTheApp->Dispatch();
108 }
109
110 // use the top level window as parent if none specified
111 if ( !m_parent )
112 m_parent = FindSuitableParent();
113 HWND hWnd = m_parent ? GetHwndOf(m_parent) : NULL;
114
115 // translate wx style in MSW
116 unsigned int msStyle = MB_OK;
117 const long wxStyle = GetMessageDialogStyle();
118 if (wxStyle & wxYES_NO)
119 {
120#if !(defined(__SMARTPHONE__) && defined(__WXWINCE__))
121 if (wxStyle & wxCANCEL)
122 msStyle = MB_YESNOCANCEL;
123 else
124#endif // !(__SMARTPHONE__ && __WXWINCE__)
125 msStyle = MB_YESNO;
126
127 if (wxStyle & wxNO_DEFAULT)
128 msStyle |= MB_DEFBUTTON2;
129 }
130
131 if (wxStyle & wxOK)
132 {
133 if (wxStyle & wxCANCEL)
134 msStyle = MB_OKCANCEL;
135 else
136 msStyle = MB_OK;
137 }
138 if (wxStyle & wxICON_EXCLAMATION)
139 msStyle |= MB_ICONEXCLAMATION;
140 else if (wxStyle & wxICON_HAND)
141 msStyle |= MB_ICONHAND;
142 else if (wxStyle & wxICON_INFORMATION)
143 msStyle |= MB_ICONINFORMATION;
144 else if (wxStyle & wxICON_QUESTION)
145 msStyle |= MB_ICONQUESTION;
146
147 if ( wxStyle & wxSTAY_ON_TOP )
148 msStyle |= MB_TOPMOST;
149
150#ifndef __WXWINCE__
151 if ( wxTheApp->GetLayoutDirection() == wxLayout_RightToLeft )
152 msStyle |= MB_RTLREADING | MB_RIGHT;
153#endif
154
155 if (hWnd)
156 msStyle |= MB_APPLMODAL;
157 else
158 msStyle |= MB_TASKMODAL;
159
160 // per MSDN documentation for MessageBox() we can prefix the message with 2
161 // right-to-left mark characters to tell the function to use RTL layout
162 // (unfortunately this only works in Unicode builds)
163 wxString message = GetFullMessage();
164#if wxUSE_UNICODE
165 if ( wxTheApp->GetLayoutDirection() == wxLayout_RightToLeft )
166 {
167 // NB: not all compilers support \u escapes
168 static const wchar_t wchRLM = 0x200f;
169 message.Prepend(wxString(wchRLM, 2));
170 }
171#endif // wxUSE_UNICODE
172
173 // install the hook if we need to position the dialog in a non-default way
174 if ( wxStyle & wxCENTER )
175 {
176 const DWORD tid = ::GetCurrentThreadId();
177 m_hook = ::SetWindowsHookEx(WH_CALLWNDPROCRET,
178 &wxMessageDialog::HookFunction, NULL, tid);
179 HookMap()[tid] = this;
180 }
181
182 // do show the dialog
183 int msAns = MessageBox(hWnd, message.wx_str(), m_caption.wx_str(), msStyle);
184 int ans;
185 switch (msAns)
186 {
187 default:
188 wxFAIL_MSG(_T("unexpected ::MessageBox() return code"));
189 // fall through
190
191 case IDCANCEL:
192 ans = wxID_CANCEL;
193 break;
194 case IDOK:
195 ans = wxID_OK;
196 break;
197 case IDYES:
198 ans = wxID_YES;
199 break;
200 case IDNO:
201 ans = wxID_NO;
202 break;
203 }
204 return ans;
205}
206
207#endif // wxUSE_MSGDLG