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