1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/dlgcmn.cpp
3 // Purpose: common (to all ports) wxDialog functions
4 // Author: Vadim Zeitlin
8 // Copyright: (c) Vadim Zeitlin
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
27 #include "wx/dialog.h"
31 #include "wx/button.h"
32 #include "wx/dcclient.h"
34 #include "wx/settings.h"
35 #include "wx/stattext.h"
37 #include "wx/containr.h"
40 #include "wx/statline.h"
41 #include "wx/sysopt.h"
42 #include "wx/private/stattext.h"
45 // ----------------------------------------------------------------------------
47 // ----------------------------------------------------------------------------
49 BEGIN_EVENT_TABLE(wxDialogBase
, wxTopLevelWindow
)
50 EVT_BUTTON(wxID_ANY
, wxDialogBase::OnButton
)
52 EVT_CLOSE(wxDialogBase::OnCloseWindow
)
54 EVT_CHAR_HOOK(wxDialogBase::OnCharHook
)
57 void wxDialogBase::Init()
60 m_affirmativeId
= wxID_OK
;
61 m_escapeId
= wxID_ANY
;
63 // the dialogs have this flag on by default to prevent the events from the
64 // dialog controls from reaching the parent frame which is usually
65 // undesirable and can lead to unexpected and hard to find bugs
66 SetExtraStyle(GetExtraStyle() | wxWS_EX_BLOCK_EVENTS
);
69 // helper of GetParentForModalDialog()
70 static bool CanBeUsedAsParent(wxWindow
*parent
)
72 extern WXDLLIMPEXP_DATA_CORE(wxList
) wxPendingDelete
;
74 return !parent
->HasExtraStyle(wxWS_EX_TRANSIENT
) &&
75 parent
->IsShownOnScreen() &&
76 !wxPendingDelete
.Member(parent
) &&
77 !parent
->IsBeingDeleted();
80 wxWindow
*wxDialogBase::GetParentForModalDialog(wxWindow
*parent
) const
82 // creating a parent-less modal dialog will result (under e.g. wxGTK2)
83 // in an unfocused dialog, so try to find a valid parent for it:
85 parent
= wxGetTopLevelParent(parent
);
87 if ( !parent
|| !CanBeUsedAsParent(parent
) )
88 parent
= wxTheApp
->GetTopWindow();
90 if ( parent
&& !CanBeUsedAsParent(parent
) )
92 // can't use this one, it's going to disappear
101 class wxTextSizerWrapper
: public wxTextWrapper
104 wxTextSizerWrapper(wxWindow
*win
)
110 wxSizer
*CreateSizer(const wxString
& text
, int widthMax
)
112 m_sizer
= new wxBoxSizer(wxVERTICAL
);
113 Wrap(m_win
, text
, widthMax
);
118 virtual void OnOutputLine(const wxString
& line
)
122 m_sizer
->Add(new wxStaticText(m_win
, wxID_ANY
, line
));
124 else // empty line, no need to create a control for it
127 m_hLine
= m_win
->GetCharHeight();
129 m_sizer
->Add(5, m_hLine
);
139 wxSizer
*wxDialogBase::CreateTextSizer(const wxString
& message
)
141 // I admit that this is complete bogus, but it makes
142 // message boxes work for pda screens temporarily..
144 const bool is_pda
= wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA
;
147 widthMax
= wxSystemSettings::GetMetric( wxSYS_SCREEN_X
) - 25;
150 // '&' is used as accel mnemonic prefix in the wxWidgets controls but in
151 // the static messages created by CreateTextSizer() (used by wxMessageBox,
152 // for example), we don't want this special meaning, so we need to quote it
153 wxString
text(message
);
154 text
.Replace(_T("&"), _T("&&"));
156 wxTextSizerWrapper
wrapper(this);
158 return wrapper
.CreateSizer(text
, widthMax
);
161 #endif // wxUSE_STATTEXT
163 wxSizer
*wxDialogBase::CreateButtonSizer(long flags
)
165 wxSizer
*sizer
= NULL
;
167 #ifdef __SMARTPHONE__
168 wxDialog
* dialog
= (wxDialog
*) this;
170 dialog
->SetLeftMenu(wxID_OK
);
172 if ( flags
& wxCANCEL
)
173 dialog
->SetRightMenu(wxID_CANCEL
);
176 dialog
->SetLeftMenu(wxID_YES
);
179 dialog
->SetRightMenu(wxID_NO
);
180 #else // !__SMARTPHONE__
185 // PocketPC guidelines recommend for Ok/Cancel dialogs to use OK button
186 // located inside caption bar and implement Cancel functionality through
187 // Undo outside dialog. As native behaviour this will be default here but
188 // can be replaced with real wxButtons by setting the option below to 1
189 if ( (flags
& ~(wxCANCEL
|wxNO_DEFAULT
)) != wxOK
||
190 wxSystemOptions::GetOptionInt(wxT("wince.dialog.real-ok-cancel")) )
191 #endif // __POCKETPC__
193 sizer
= CreateStdDialogButtonSizer(flags
);
195 #endif // wxUSE_BUTTON
197 #endif // __SMARTPHONE__/!__SMARTPHONE__
202 wxSizer
*wxDialogBase::CreateSeparatedButtonSizer(long flags
)
204 wxSizer
*sizer
= CreateButtonSizer(flags
);
208 // Mac Human Interface Guidelines recommend not to use static lines as
210 #if wxUSE_STATLINE && !defined(__WXMAC__)
211 wxBoxSizer
*topsizer
= new wxBoxSizer(wxVERTICAL
);
212 topsizer
->Add(new wxStaticLine(this),
213 wxSizerFlags().Expand().DoubleBorder(wxBOTTOM
));
214 topsizer
->Add(sizer
, wxSizerFlags().Expand());
216 #endif // wxUSE_STATLINE
223 wxStdDialogButtonSizer
*wxDialogBase::CreateStdDialogButtonSizer( long flags
)
225 wxStdDialogButtonSizer
*sizer
= new wxStdDialogButtonSizer();
228 wxButton
*yes
= NULL
;
233 ok
= new wxButton(this, wxID_OK
);
234 sizer
->AddButton(ok
);
237 if (flags
& wxCANCEL
)
239 wxButton
*cancel
= new wxButton(this, wxID_CANCEL
);
240 sizer
->AddButton(cancel
);
245 yes
= new wxButton(this, wxID_YES
);
246 sizer
->AddButton(yes
);
251 no
= new wxButton(this, wxID_NO
);
252 sizer
->AddButton(no
);
257 wxButton
*apply
= new wxButton(this, wxID_APPLY
);
258 sizer
->AddButton(apply
);
263 wxButton
*close
= new wxButton(this, wxID_CLOSE
);
264 sizer
->AddButton(close
);
269 wxButton
*help
= new wxButton(this, wxID_HELP
);
270 sizer
->AddButton(help
);
273 if (flags
& wxNO_DEFAULT
)
296 SetAffirmativeId(wxID_OK
);
297 else if (flags
& wxYES
)
298 SetAffirmativeId(wxID_YES
);
305 #endif // wxUSE_BUTTON
307 // ----------------------------------------------------------------------------
308 // standard buttons handling
309 // ----------------------------------------------------------------------------
311 void wxDialogBase::EndDialog(int rc
)
319 void wxDialogBase::AcceptAndClose()
321 if ( Validate() && TransferDataFromWindow() )
323 EndDialog(m_affirmativeId
);
327 void wxDialogBase::SetAffirmativeId(int affirmativeId
)
329 m_affirmativeId
= affirmativeId
;
332 void wxDialogBase::SetEscapeId(int escapeId
)
334 m_escapeId
= escapeId
;
337 bool wxDialogBase::EmulateButtonClickIfPresent(int id
)
340 wxButton
*btn
= wxDynamicCast(FindWindow(id
), wxButton
);
342 if ( !btn
|| !btn
->IsEnabled() || !btn
->IsShown() )
345 wxCommandEvent
event(wxEVT_COMMAND_BUTTON_CLICKED
, id
);
346 event
.SetEventObject(btn
);
347 btn
->GetEventHandler()->ProcessEvent(event
);
350 #else // !wxUSE_BUTTON
353 #endif // wxUSE_BUTTON/!wxUSE_BUTTON
356 bool wxDialogBase::IsEscapeKey(const wxKeyEvent
& event
)
358 // for most platforms, Esc key is used to close the dialogs
359 return event
.GetKeyCode() == WXK_ESCAPE
&&
360 event
.GetModifiers() == wxMOD_NONE
;
363 void wxDialogBase::OnCharHook(wxKeyEvent
& event
)
365 if ( event
.GetKeyCode() == WXK_ESCAPE
)
367 int idCancel
= GetEscapeId();
371 // don't handle Esc specially at all
375 // this value is special: it means translate Esc to wxID_CANCEL
376 // but if there is no such button, then fall back to wxID_OK
377 if ( EmulateButtonClickIfPresent(wxID_CANCEL
) )
379 idCancel
= GetAffirmativeId();
383 // translate Esc to button press for the button with given id
384 if ( EmulateButtonClickIfPresent(idCancel
) )
392 void wxDialogBase::OnButton(wxCommandEvent
& event
)
394 const int id
= event
.GetId();
395 if ( id
== GetAffirmativeId() )
399 else if ( id
== wxID_APPLY
)
402 TransferDataFromWindow();
404 // TODO: disable the Apply button until things change again
406 else if ( id
== GetEscapeId() ||
407 (id
== wxID_CANCEL
&& GetEscapeId() == wxID_ANY
) )
409 EndDialog(wxID_CANCEL
);
411 else // not a standard button
417 // ----------------------------------------------------------------------------
418 // other event handlers
419 // ----------------------------------------------------------------------------
421 void wxDialogBase::OnCloseWindow(wxCloseEvent
& WXUNUSED(event
))
423 // We'll send a Cancel message by default, which may close the dialog.
424 // Check for looping if the Cancel event handler calls Close().
426 // Note that if a cancel button and handler aren't present in the dialog,
427 // nothing will happen when you close the dialog via the window manager, or
428 // via Close(). We wouldn't want to destroy the dialog by default, since
429 // the dialog may have been created on the stack. However, this does mean
430 // that calling dialog->Close() won't delete the dialog unless the handler
431 // for wxID_CANCEL does so. So use Destroy() if you want to be sure to
432 // destroy the dialog. The default OnCancel (above) simply ends a modal
433 // dialog, and hides a modeless dialog.
435 // VZ: this is horrible and MT-unsafe. Can't we reuse some of these global
436 // lists here? don't dare to change it now, but should be done later!
437 static wxList closing
;
439 if ( closing
.Member(this) )
442 closing
.Append(this);
444 wxCommandEvent
cancelEvent(wxEVT_COMMAND_BUTTON_CLICKED
, wxID_CANCEL
);
445 cancelEvent
.SetEventObject( this );
446 GetEventHandler()->ProcessEvent(cancelEvent
); // This may close the dialog
448 closing
.DeleteObject(this);
451 void wxDialogBase::OnSysColourChanged(wxSysColourChangedEvent
& WXUNUSED(event
))
453 SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE
));