]> git.saurik.com Git - wxWidgets.git/blame - src/mac/dialog.cpp
fixed WM_GETDLGCODE processing to include the bits returned by the DefWindowProc...
[wxWidgets.git] / src / mac / dialog.cpp
CommitLineData
e9576ca5
SC
1/////////////////////////////////////////////////////////////////////////////
2// Name: dialog.cpp
3// Purpose: wxDialog class
4// Author: AUTHOR
5// Modified by:
6// Created: ??/??/98
7// RCS-ID: $Id$
8// Copyright: (c) AUTHOR
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12#ifdef __GNUG__
13#pragma implementation "dialog.h"
14#endif
15
16#include "wx/dialog.h"
17#include "wx/utils.h"
18#include "wx/frame.h"
19#include "wx/app.h"
20#include "wx/settings.h"
21
d497dca4 22#include "wx/mac/uma.h"
519cb848 23
e9576ca5
SC
24// Lists to keep track of windows, so we can disable/enable them
25// for modal dialogs
26wxList wxModalDialogs;
fe08e597 27//wxList wxModelessWindows; // Frames and modeless dialogs
e9576ca5
SC
28extern wxList wxPendingDelete;
29
2f1ae414 30#if !USE_SHARED_LIBRARY
a15eb0a5 31IMPLEMENT_DYNAMIC_CLASS(wxDialog, wxTopLevelWindow)
e9576ca5 32
d491523b 33BEGIN_EVENT_TABLE(wxDialog, wxDialogBase)
e9576ca5
SC
34 EVT_BUTTON(wxID_OK, wxDialog::OnOK)
35 EVT_BUTTON(wxID_APPLY, wxDialog::OnApply)
36 EVT_BUTTON(wxID_CANCEL, wxDialog::OnCancel)
a15eb0a5 37
e9576ca5 38 EVT_CHAR_HOOK(wxDialog::OnCharHook)
a15eb0a5 39
e9576ca5 40 EVT_SYS_COLOUR_CHANGED(wxDialog::OnSysColourChanged)
a15eb0a5 41
e9576ca5
SC
42 EVT_CLOSE(wxDialog::OnCloseWindow)
43END_EVENT_TABLE()
44
2f1ae414 45#endif
e9576ca5
SC
46
47wxDialog::wxDialog()
48{
2f1ae414 49 m_isShown = FALSE;
a756f210 50 SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE));
e9576ca5
SC
51}
52
53bool wxDialog::Create(wxWindow *parent, wxWindowID id,
54 const wxString& title,
55 const wxPoint& pos,
56 const wxSize& size,
57 long style,
58 const wxString& name)
59{
e9576ca5 60
a756f210 61 SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE));
e9576ca5 62
e9576ca5 63
a15eb0a5
SC
64 if ( !wxTopLevelWindow::Create(parent, id, title, pos, size, style, name) )
65 return FALSE;
e9576ca5 66
2f1ae414 67 MacCreateRealWindow( title , pos , size , MacRemoveBordersFromStyle(style) , name ) ;
e9576ca5 68
4159f685 69 m_macWindowBackgroundTheme = kThemeBrushDialogBackgroundActive ;
2240c9ed 70 SetThemeWindowBackground( (WindowRef) m_macWindow , m_macWindowBackgroundTheme , false ) ;
519cb848 71 return TRUE;
e9576ca5
SC
72}
73
74void wxDialog::SetModal(bool flag)
75{
2f1ae414
SC
76 if ( flag )
77 {
78 m_windowStyle |= wxDIALOG_MODAL;
79
80 wxModelessWindows.DeleteObject(this);
81 }
82 else
83 {
84 m_windowStyle &= ~wxDIALOG_MODAL;
85
86 wxModelessWindows.Append(this);
87 }
e9576ca5
SC
88}
89
90wxDialog::~wxDialog()
91{
e7549107 92 m_isBeingDeleted = TRUE ;
a15eb0a5 93 Show(FALSE);
e9576ca5
SC
94}
95
a15eb0a5 96// By default, pressing escape cancels the dialog , on mac command-stop does the same thing
e9576ca5
SC
97void wxDialog::OnCharHook(wxKeyEvent& event)
98{
a15eb0a5
SC
99 if (
100 ( event.m_keyCode == WXK_ESCAPE ||
101 ( event.m_keyCode == '.' && event.MetaDown() ) )
102 && FindWindow(wxID_CANCEL) )
e9576ca5
SC
103 {
104 // Behaviour changed in 2.0: we'll send a Cancel message
105 // to the dialog instead of Close.
106 wxCommandEvent cancelEvent(wxEVT_COMMAND_BUTTON_CLICKED, wxID_CANCEL);
107 cancelEvent.SetEventObject( this );
108 GetEventHandler()->ProcessEvent(cancelEvent);
109
110 return;
111 }
112 // We didn't process this event.
113 event.Skip();
114}
115
2f1ae414 116bool wxDialog::IsModal() const
51abe921 117{
2f1ae414 118 return (GetWindowStyleFlag() & wxDIALOG_MODAL) != 0;
51abe921
SC
119}
120
2f1ae414
SC
121
122bool wxDialog::IsModalShowing() const
51abe921 123{
2f1ae414 124 return wxModalDialogs.Find((wxDialog *)this) != NULL; // const_cast
51abe921
SC
125}
126
519cb848 127extern bool s_macIsInModalLoop ;
e9576ca5
SC
128
129bool wxDialog::Show(bool show)
130{
2f1ae414
SC
131 if ( !wxDialogBase::Show(show) )
132 {
133 // nothing to do
134 return FALSE;
135 }
e9576ca5 136
2f1ae414
SC
137 if ( show )
138 {
139 // usually will result in TransferDataToWindow() being called
140 InitDialog();
141 }
e9576ca5 142
2f1ae414 143 if ( IsModal() )
e7549107 144 {
2f1ae414
SC
145 if ( show )
146 {
147 DoShowModal();
148 }
149 else // end of modal dialog
150 {
151 // this will cause IsModalShowing() return FALSE and our local
152 // message loop will terminate
153 wxModalDialogs.DeleteObject(this);
154 }
155 }
e7549107 156
2f1ae414 157 return TRUE;
e9576ca5
SC
158}
159
2f1ae414 160void wxDialog::DoShowModal()
51abe921 161{
2f1ae414 162 wxCHECK_RET( !IsModalShowing(), _T("DoShowModal() called twice") );
51abe921 163
2f1ae414 164 wxModalDialogs.Append(this);
51abe921 165
2f1ae414
SC
166 wxWindow *parent = GetParent();
167
168 // remember where the focus was
169 wxWindow *winFocus = FindFocus();
170 if ( !winFocus )
171 {
172 winFocus = parent;
173 }
174 if ( !winFocus )
175 {
176 winFocus = wxTheApp->GetTopWindow();
177 }
178 // TODO : test whether parent gets disabled
179
a49afa93 180 bool formerModal = s_macIsInModalLoop ;
2f1ae414 181 s_macIsInModalLoop = true ;
51abe921 182
2f1ae414
SC
183 while ( IsModalShowing() )
184 {
185 while ( !wxTheApp->Pending() && wxTheApp->ProcessIdle() )
186 {
187 }
188 wxTheApp->MacDoOneEvent() ;
189 }
190
a49afa93 191 s_macIsInModalLoop = formerModal ;
51abe921 192
2f1ae414 193 // TODO probably reenable the parent window if any
51abe921 194
2f1ae414
SC
195 // and restore focus
196 if ( winFocus )
197 {
198 winFocus->SetFocus();
199 }
51abe921 200}
519cb848 201
2f1ae414 202
e9576ca5
SC
203// Replacement for Show(TRUE) for modal dialogs - returns return code
204int wxDialog::ShowModal()
205{
519cb848 206 m_windowStyle |= wxDIALOG_MODAL;
51abe921
SC
207 Show(TRUE);
208 return GetReturnCode();
e9576ca5
SC
209}
210
2f1ae414
SC
211// NB: this function (surprizingly) may be called for both modal and modeless
212// dialogs and should work for both of them
e9576ca5
SC
213void wxDialog::EndModal(int retCode)
214{
51abe921
SC
215 SetReturnCode(retCode);
216 Show(FALSE);
e9576ca5
SC
217}
218
219// Standard buttons
220void wxDialog::OnOK(wxCommandEvent& event)
221{
51abe921
SC
222 if ( Validate() && TransferDataFromWindow() )
223 {
2f1ae414 224 EndModal(wxID_OK);
51abe921 225 }
e9576ca5
SC
226}
227
228void wxDialog::OnApply(wxCommandEvent& event)
229{
51abe921
SC
230 if (Validate())
231 TransferDataFromWindow();
232 // TODO probably need to disable the Apply button until things change again
e9576ca5
SC
233}
234
235void wxDialog::OnCancel(wxCommandEvent& event)
236{
e9576ca5 237 EndModal(wxID_CANCEL);
7c74e7fe
SC
238}
239
e3065973 240void wxDialog::OnCloseWindow(wxCloseEvent& event)
e9576ca5 241{
e3065973 242 // We'll send a Cancel message by default,
e9576ca5 243 // which may close the dialog.
e3065973
JS
244 // Check for looping if the Cancel event handler calls Close().
245
246 // Note that if a cancel button and handler aren't present in the dialog,
247 // nothing will happen when you close the dialog via the window manager, or
248 // via Close().
249 // We wouldn't want to destroy the dialog by default, since the dialog may have been
250 // created on the stack.
251 // However, this does mean that calling dialog->Close() won't delete the dialog
252 // unless the handler for wxID_CANCEL does so. So use Destroy() if you want to be
253 // sure to destroy the dialog.
254 // The default OnCancel (above) simply ends a modal dialog, and hides a modeless dialog.
e9576ca5
SC
255
256 static wxList closing;
51abe921 257
e9576ca5 258 if ( closing.Member(this) )
e3065973 259 return;
51abe921 260
e9576ca5 261 closing.Append(this);
51abe921 262
e3065973
JS
263 wxCommandEvent cancelEvent(wxEVT_COMMAND_BUTTON_CLICKED, wxID_CANCEL);
264 cancelEvent.SetEventObject( this );
265 GetEventHandler()->ProcessEvent(cancelEvent); // This may close the dialog
e9576ca5
SC
266
267 closing.DeleteObject(this);
e9576ca5
SC
268}
269
e9576ca5
SC
270void wxDialog::OnSysColourChanged(wxSysColourChangedEvent& event)
271{
a756f210 272 SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE));
e9576ca5
SC
273 Refresh();
274}
275