]> git.saurik.com Git - wxWidgets.git/blame - src/mac/carbon/mdi.cpp
use less expensive GetLastPosition() instead of GetValue() to implement IsEmpty()
[wxWidgets.git] / src / mac / carbon / mdi.cpp
CommitLineData
e9576ca5 1/////////////////////////////////////////////////////////////////////////////
59cf2e49 2// Name: src/mac/carbon/mdi.cpp
e9576ca5 3// Purpose: MDI classes
a31a5f85 4// Author: Stefan Csomor
e9576ca5 5// Modified by:
a31a5f85 6// Created: 1998-01-01
e9576ca5 7// RCS-ID: $Id$
a31a5f85 8// Copyright: (c) Stefan Csomor
59cf2e49 9// Licence: wxWindows licence
e9576ca5
SC
10/////////////////////////////////////////////////////////////////////////////
11
3d1a4878
SC
12#include "wx/wxprec.h"
13
179e085f
RN
14#if wxUSE_MDI
15
59cf2e49
WS
16#include "wx/mdi.h"
17
38c8fce8 18#ifndef WX_PRECOMP
38c8fce8
DE
19 #include "wx/log.h"
20 #include "wx/menu.h"
21 #include "wx/settings.h"
22 #include "wx/statusbr.h"
23#endif
e9576ca5 24
76a5e5d2 25#include "wx/mac/private.h"
70024cfb 26#include "wx/mac/uma.h"
76a5e5d2 27
fe08e597 28extern wxWindowList wxModelessWindows;
e9576ca5 29
e9576ca5
SC
30IMPLEMENT_DYNAMIC_CLASS(wxMDIParentFrame, wxFrame)
31IMPLEMENT_DYNAMIC_CLASS(wxMDIChildFrame, wxFrame)
32IMPLEMENT_DYNAMIC_CLASS(wxMDIClientWindow, wxWindow)
33
34BEGIN_EVENT_TABLE(wxMDIParentFrame, wxFrame)
59cf2e49
WS
35 EVT_ACTIVATE(wxMDIParentFrame::OnActivate)
36 EVT_SYS_COLOUR_CHANGED(wxMDIParentFrame::OnSysColourChanged)
e9576ca5
SC
37END_EVENT_TABLE()
38
39BEGIN_EVENT_TABLE(wxMDIClientWindow, wxWindow)
59cf2e49 40 EVT_SCROLL(wxMDIClientWindow::OnScroll)
e9576ca5
SC
41END_EVENT_TABLE()
42
fd8278bf
VZ
43static const wxChar *TRACE_MDI = _T("mdi");
44
0a67a93b
SC
45static const int IDM_WINDOWTILEHOR = 4001;
46static const int IDM_WINDOWCASCADE = 4002;
47static const int IDM_WINDOWICONS = 4003;
48static const int IDM_WINDOWNEXT = 4004;
49static const int IDM_WINDOWTILEVERT = 4005;
0a67a93b 50
fb728ebb
SC
51// others
52
53void UMAHighlightAndActivateWindow( WindowRef inWindowRef , bool inActivate )
54{
55#if 1 // TODO REMOVE
56 if ( inWindowRef )
57 {
58// bool isHighlighted = IsWindowHighlited( inWindowRef ) ;
59// if ( inActivate != isHighlighted )
60#ifndef __LP64__
61 GrafPtr port ;
62 GetPort( &port ) ;
63 SetPortWindowPort( inWindowRef ) ;
64#endif
65 HiliteWindow( inWindowRef , inActivate ) ;
66 ControlRef control = NULL ;
67 ::GetRootControl( inWindowRef , &control ) ;
68 if ( control )
69 {
70 if ( inActivate )
71 ::ActivateControl( control ) ;
72 else
73 ::DeactivateControl( control ) ;
74 }
75#ifndef __LP64__
76 SetPort( port ) ;
77#endif
78 }
79#endif
80}
81
a57ac1c4 82// ----------------------------------------------------------------------------
e9576ca5 83// Parent frame
a57ac1c4 84// ----------------------------------------------------------------------------
e9576ca5 85
a57ac1c4 86void wxMDIParentFrame::Init()
e9576ca5 87{
0a67a93b
SC
88 m_clientWindow = NULL;
89 m_currentChild = NULL;
90 m_windowMenu = (wxMenu*) NULL;
a57ac1c4
VZ
91 m_parentFrameActive = true;
92 m_shouldBeShown = false;
e9576ca5
SC
93}
94
95bool wxMDIParentFrame::Create(wxWindow *parent,
5a7d70fe
DS
96 wxWindowID id,
97 const wxString& title,
98 const wxPoint& pos,
99 const wxSize& size,
100 long style,
101 const wxString& name)
e9576ca5 102{
e40298d5
JS
103 // this style can be used to prevent a window from having the standard MDI
104 // "Window" menu
105 if ( style & wxFRAME_NO_WINDOW_MENU )
106 {
107 m_windowMenu = (wxMenu *)NULL;
108 style -= wxFRAME_NO_WINDOW_MENU ;
109 }
110 else // normal case: we have the window menu, so construct it
111 {
112 m_windowMenu = new wxMenu;
5a7d70fe 113
e40298d5
JS
114 m_windowMenu->Append(IDM_WINDOWCASCADE, wxT("&Cascade"));
115 m_windowMenu->Append(IDM_WINDOWTILEHOR, wxT("Tile &Horizontally"));
116 m_windowMenu->Append(IDM_WINDOWTILEVERT, wxT("Tile &Vertically"));
117 m_windowMenu->AppendSeparator();
118 m_windowMenu->Append(IDM_WINDOWICONS, wxT("&Arrange Icons"));
119 m_windowMenu->Append(IDM_WINDOWNEXT, wxT("&Next"));
120 }
5a7d70fe 121
70024cfb 122 wxFrame::Create( parent , id , title , pos , size , style , name ) ;
a57ac1c4 123 m_parentFrameActive = true;
5a7d70fe 124
e40298d5 125 OnCreateClient();
5a7d70fe 126
a57ac1c4 127 return true;
e9576ca5
SC
128}
129
130wxMDIParentFrame::~wxMDIParentFrame()
131{
0a67a93b 132 DestroyChildren();
5a7d70fe 133
ea3cdc4f 134 // already deleted by DestroyChildren()
0a67a93b 135 m_clientWindow = NULL ;
5a7d70fe 136
ea3cdc4f 137 delete m_windowMenu;
e9576ca5
SC
138}
139
e9576ca5
SC
140void wxMDIParentFrame::SetMenuBar(wxMenuBar *menu_bar)
141{
e40298d5 142 wxFrame::SetMenuBar( menu_bar ) ;
e9576ca5
SC
143}
144
0bba37f5
DE
145void wxMDIParentFrame::GetRectForTopLevelChildren(int *x, int *y, int *w, int *h)
146{
5a7d70fe 147 if (x)
0bba37f5 148 *x = 0;
5a7d70fe 149 if (y)
0bba37f5 150 *y = 0;
5a7d70fe
DS
151
152 wxDisplaySize(w, h);
0bba37f5
DE
153}
154
a57ac1c4
VZ
155void wxMDIParentFrame::AddChild(wxWindowBase *child)
156{
9fbc9db0
SC
157 // moved this to front, so that we don't run into unset m_parent problems later
158 wxFrame::AddChild(child);
159
a57ac1c4
VZ
160 if ( !m_currentChild )
161 {
162 m_currentChild = wxDynamicCast(child, wxMDIChildFrame);
163
3ee39f97 164 if ( m_currentChild && IsShown() && !ShouldBeVisible() )
a57ac1c4 165 {
3ee39f97
VZ
166 // we shouldn't remain visible any more
167 wxFrame::Show(false);
168 m_shouldBeShown = true;
a57ac1c4
VZ
169 }
170 }
a57ac1c4
VZ
171}
172
173void wxMDIParentFrame::RemoveChild(wxWindowBase *child)
174{
175 if ( child == m_currentChild )
176 {
177 // the current child isn't active any more, try to find another one
178 m_currentChild = NULL;
179
180 for ( wxWindowList::compatibility_iterator node = GetChildren().GetFirst();
181 node;
182 node = node->GetNext() )
183 {
184 wxMDIChildFrame *
185 childCur = wxDynamicCast(node->GetData(), wxMDIChildFrame);
186 if ( childCur != child )
187 {
188 m_currentChild = childCur;
189 break;
190 }
191 }
192 }
193
194 wxFrame::RemoveChild(child);
195
196 // if there are no more children left we need to show the frame if we
197 // hadn't shown it before because there were active children and it was
198 // useless (note that we have to do it after fully removing the child, i.e.
199 // after calling the base class RemoveChild() as otherwise we risk to touch
200 // pointer to the child being deleted)
201 if ( !m_currentChild && m_shouldBeShown && !IsShown() )
202 {
203 // we have to show it, but at least move it out of sight and make it of
204 // smallest possible size (unfortunately (0, 0) doesn't work so that it
205 // doesn't appear in expose
206 SetSize(-10000, -10000, 1, 1);
207 Show();
208 }
209}
210
70024cfb 211void wxMDIParentFrame::MacActivate(long timestamp, bool activating)
e9576ca5 212{
5a7d70fe
DS
213 wxLogTrace(TRACE_MDI, wxT("MDI PARENT=%p MacActivate(0x%08lx,%s)"), this, timestamp, activating ? wxT("ACTIV") : wxT("deact"));
214
215 if (activating)
e40298d5 216 {
5a7d70fe 217 if (s_macDeactivateWindow && s_macDeactivateWindow->GetParent() == this)
70024cfb 218 {
fd8278bf 219 wxLogTrace(TRACE_MDI, wxT("child had been scheduled for deactivation, rehighlighting"));
5a7d70fe 220
70024cfb 221 UMAHighlightAndActivateWindow((WindowRef)s_macDeactivateWindow->MacGetWindowRef(), true);
5a7d70fe
DS
222
223 wxLogTrace(TRACE_MDI, wxT("finished highliting child"));
224
70024cfb
DE
225 s_macDeactivateWindow = NULL;
226 }
5a7d70fe 227 else if (s_macDeactivateWindow == this)
70024cfb 228 {
fd8278bf 229 wxLogTrace(TRACE_MDI, wxT("Avoided deactivation/activation of this=%p"), this);
5a7d70fe 230
70024cfb
DE
231 s_macDeactivateWindow = NULL;
232 }
233 else // window to deactivate is NULL or is not us or one of our kids
234 {
235 // activate kid instead
5a7d70fe
DS
236 if (m_currentChild)
237 m_currentChild->MacActivate(timestamp, activating);
70024cfb 238 else
5a7d70fe 239 wxFrame::MacActivate(timestamp, activating);
70024cfb 240 }
e40298d5 241 }
70024cfb 242 else
e40298d5 243 {
70024cfb 244 // We were scheduled for deactivation, and now we do it.
5a7d70fe 245 if (s_macDeactivateWindow == this)
e40298d5 246 {
70024cfb 247 s_macDeactivateWindow = NULL;
5a7d70fe
DS
248 if (m_currentChild)
249 m_currentChild->MacActivate(timestamp, activating);
250 wxFrame::MacActivate(timestamp, activating);
70024cfb
DE
251 }
252 else // schedule ourselves for deactivation
253 {
5a7d70fe
DS
254 if (s_macDeactivateWindow)
255 wxLogTrace(TRACE_MDI, wxT("window=%p SHOULD have been deactivated, oh well!"), s_macDeactivateWindow);
fd8278bf 256 wxLogTrace(TRACE_MDI, wxT("Scheduling delayed MDI Parent deactivation"));
5a7d70fe 257
70024cfb 258 s_macDeactivateWindow = this;
e40298d5 259 }
e40298d5 260 }
e9576ca5
SC
261}
262
70024cfb
DE
263void wxMDIParentFrame::OnActivate(wxActivateEvent& event)
264{
265 event.Skip();
266}
267
e9576ca5
SC
268// Returns the active MDI child window
269wxMDIChildFrame *wxMDIParentFrame::GetActiveChild() const
270{
e40298d5 271 return m_currentChild ;
e9576ca5
SC
272}
273
274// Create the client window class (don't Create the window,
275// just return a new class)
276wxMDIClientWindow *wxMDIParentFrame::OnCreateClient()
277{
0a67a93b 278 m_clientWindow = new wxMDIClientWindow( this );
5a7d70fe 279
0a67a93b 280 return m_clientWindow;
e9576ca5
SC
281}
282
283// Responds to colour changes, and passes event on to children.
284void wxMDIParentFrame::OnSysColourChanged(wxSysColourChangedEvent& event)
285{
286 // TODO
5a7d70fe 287
e9576ca5
SC
288 // Propagate the event to the non-top-level children
289 wxFrame::OnSysColourChanged(event);
290}
291
292// MDI operations
293void wxMDIParentFrame::Cascade()
294{
295 // TODO
296}
297
0d97c090 298void wxMDIParentFrame::Tile(wxOrientation WXUNUSED(orient))
e9576ca5
SC
299{
300 // TODO
301}
302
303void wxMDIParentFrame::ArrangeIcons()
304{
305 // TODO
306}
307
308void wxMDIParentFrame::ActivateNext()
309{
310 // TODO
311}
312
313void wxMDIParentFrame::ActivatePrevious()
314{
315 // TODO
316}
317
a57ac1c4
VZ
318bool wxMDIParentFrame::ShouldBeVisible() const
319{
320 for ( wxWindowList::compatibility_iterator node = GetChildren().GetFirst();
321 node;
322 node = node->GetNext() )
323 {
49c48f81
VZ
324 wxWindow *win = node->GetData();
325
326 if ( win->IsShown()
327 && !wxDynamicCast(win, wxMDIChildFrame)
a57ac1c4 328#if wxUSE_STATUSBAR
3a46bcdd 329 && win != (wxWindow*) GetStatusBar()
5a7d70fe
DS
330#endif
331 && win != GetClientWindow() )
a57ac1c4
VZ
332 {
333 // if we have a non-MDI child, do remain visible so that it could
334 // be used
335 return true;
336 }
337 }
338
339 return false;
340}
341
29e92efb
VZ
342bool wxMDIParentFrame::Show( bool show )
343{
a57ac1c4
VZ
344 m_shouldBeShown = false;
345
29e92efb
VZ
346 // don't really show the MDI frame unless it has any children other than
347 // MDI children as it is pretty useless in this case
507ad426 348
29e92efb
VZ
349 if ( show )
350 {
a57ac1c4 351 if ( !ShouldBeVisible() && m_currentChild )
4a60e3f0 352 {
a57ac1c4
VZ
353 // don't make the window visible now but remember that we should
354 // have had done it
355 m_shouldBeShown = true;
5a7d70fe 356
a57ac1c4 357 return false;
4a60e3f0 358 }
29e92efb
VZ
359 }
360
a57ac1c4 361 return wxFrame::Show(show);
29e92efb
VZ
362}
363
a57ac1c4 364// ----------------------------------------------------------------------------
e9576ca5 365// Child frame
a57ac1c4 366// ----------------------------------------------------------------------------
e9576ca5
SC
367
368wxMDIChildFrame::wxMDIChildFrame()
0a67a93b
SC
369{
370 Init() ;
371}
372void wxMDIChildFrame::Init()
e9576ca5
SC
373{
374}
375
376bool wxMDIChildFrame::Create(wxMDIParentFrame *parent,
59cf2e49
WS
377 wxWindowID id,
378 const wxString& title,
379 const wxPoint& pos,
380 const wxSize& size,
381 long style,
382 const wxString& name)
e9576ca5
SC
383{
384 SetName(name);
5a7d70fe 385
59cf2e49 386 if ( id == wxID_ANY )
e9576ca5 387 m_windowId = (int)NewControlId();
59cf2e49
WS
388 else
389 m_windowId = id;
5a7d70fe
DS
390
391 if (parent)
392 parent->AddChild(this);
393
bb64b8bd 394 MacCreateRealWindow( title, pos , size , MacRemoveBordersFromStyle(style) , name ) ;
5a7d70fe 395
facd6764
SC
396 SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE));
397
e9576ca5 398 wxModelessWindows.Append(this);
bb64b8bd 399
923608c3 400 return true;
e9576ca5
SC
401}
402
403wxMDIChildFrame::~wxMDIChildFrame()
404{
0a67a93b 405 DestroyChildren();
e9576ca5
SC
406}
407
408void wxMDIChildFrame::SetMenuBar(wxMenuBar *menu_bar)
409{
e40298d5 410 return wxFrame::SetMenuBar( menu_bar ) ;
e9576ca5
SC
411}
412
70024cfb
DE
413void wxMDIChildFrame::MacActivate(long timestamp, bool activating)
414{
bb64b8bd
DS
415 wxLogTrace(TRACE_MDI, wxT("MDI child=%p MacActivate(0x%08lx,%s)"),this, timestamp, activating ? wxT("ACTIV") : wxT("deact"));
416
70024cfb
DE
417 wxMDIParentFrame *mdiparent = wxDynamicCast(m_parent, wxMDIParentFrame);
418 wxASSERT(mdiparent);
bb64b8bd
DS
419
420 if (activating)
70024cfb 421 {
bb64b8bd 422 if (s_macDeactivateWindow == m_parent)
70024cfb 423 {
fd8278bf 424 wxLogTrace(TRACE_MDI, wxT("parent had been scheduled for deactivation, rehighlighting"));
bb64b8bd 425
70024cfb 426 UMAHighlightAndActivateWindow((WindowRef)s_macDeactivateWindow->MacGetWindowRef(), true);
bb64b8bd
DS
427
428 wxLogTrace(TRACE_MDI, wxT("finished highliting parent"));
429
70024cfb
DE
430 s_macDeactivateWindow = NULL;
431 }
bb64b8bd
DS
432 else if ((mdiparent->m_currentChild == this) || !s_macDeactivateWindow)
433 mdiparent->wxFrame::MacActivate(timestamp, activating);
434
435 if (mdiparent->m_currentChild && mdiparent->m_currentChild != this)
436 mdiparent->m_currentChild->wxFrame::MacActivate(timestamp, false);
70024cfb
DE
437 mdiparent->m_currentChild = this;
438
bb64b8bd 439 if (s_macDeactivateWindow == this)
70024cfb 440 {
bb64b8bd
DS
441 wxLogTrace(TRACE_MDI, wxT("Avoided deactivation/activation of this=%p"), this);
442
443 s_macDeactivateWindow = NULL;
70024cfb
DE
444 }
445 else
446 wxFrame::MacActivate(timestamp, activating);
447 }
448 else
449 {
450 // We were scheduled for deactivation, and now we do it.
bb64b8bd 451 if (s_macDeactivateWindow == this)
70024cfb
DE
452 {
453 s_macDeactivateWindow = NULL;
bb64b8bd
DS
454 wxFrame::MacActivate(timestamp, activating);
455 if (mdiparent->m_currentChild == this)
456 mdiparent->wxFrame::MacActivate(timestamp, activating);
70024cfb
DE
457 }
458 else // schedule ourselves for deactivation
459 {
bb64b8bd
DS
460 if (s_macDeactivateWindow)
461 wxLogTrace(TRACE_MDI, wxT("window=%p SHOULD have been deactivated, oh well!"), s_macDeactivateWindow);
fd8278bf 462 wxLogTrace(TRACE_MDI, wxT("Scheduling delayed deactivation"));
bb64b8bd 463
70024cfb
DE
464 s_macDeactivateWindow = this;
465 }
466 }
467}
468
e9576ca5
SC
469// MDI operations
470void wxMDIChildFrame::Maximize()
471{
0a67a93b 472 wxFrame::Maximize() ;
e9576ca5
SC
473}
474
475void wxMDIChildFrame::Restore()
476{
0a67a93b 477 wxFrame::Restore() ;
e9576ca5
SC
478}
479
480void wxMDIChildFrame::Activate()
481{
e9576ca5
SC
482}
483
0a67a93b
SC
484//-----------------------------------------------------------------------------
485// wxMDIClientWindow
486//-----------------------------------------------------------------------------
e9576ca5
SC
487
488wxMDIClientWindow::wxMDIClientWindow()
489{
490}
491
492wxMDIClientWindow::~wxMDIClientWindow()
493{
0a67a93b 494 DestroyChildren();
e9576ca5
SC
495}
496
497bool wxMDIClientWindow::CreateClient(wxMDIParentFrame *parent, long style)
498{
bb64b8bd 499 if ( !wxWindow::Create(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize, style) )
a57ac1c4 500 return false;
bb64b8bd 501
0a67a93b 502 wxModelessWindows.Append(this);
bb64b8bd 503
a57ac1c4 504 return true;
e9576ca5
SC
505}
506
be7a1013
DE
507// Get size *available for subwindows* i.e. excluding menu bar.
508void wxMDIClientWindow::DoGetClientSize(int *x, int *y) const
509{
510 wxDisplaySize( x , y ) ;
511}
512
e9576ca5 513// Explicitly call default scroll behaviour
89954433 514void wxMDIClientWindow::OnScroll(wxScrollEvent& WXUNUSED(event))
e9576ca5 515{
e9576ca5
SC
516}
517
a57ac1c4 518#endif // wxUSE_MDI