1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/motif/mdi.cpp
3 // Purpose: MDI classes
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // For compilers that support precompilation, includes "wx.h".
13 #include "wx/wxprec.h"
20 #include "wx/settings.h"
24 #pragma message disable nosimpint
27 #include <Xm/BulletinB.h>
30 #include <Xm/RowColumn.h>
31 #include <Xm/CascadeBG.h>
33 #include <Xm/PushBG.h>
34 #include <Xm/AtomMgr.h>
35 #include <Xm/Protocols.h>
37 #pragma message enable nosimpint
40 #include "wx/motif/private.h"
42 extern wxList wxModelessWindows
;
44 // Implemented in frame.cpp
45 extern void wxFrameFocusProc(Widget workArea
, XtPointer clientData
,
46 XmAnyCallbackStruct
*cbs
);
48 #define wxID_NOTEBOOK_CLIENT_AREA wxID_HIGHEST + 100
50 IMPLEMENT_DYNAMIC_CLASS(wxMDIParentFrame
, wxFrame
)
51 IMPLEMENT_DYNAMIC_CLASS(wxMDIChildFrame
, wxFrame
)
52 IMPLEMENT_DYNAMIC_CLASS(wxMDIClientWindow
, wxNotebook
)
54 BEGIN_EVENT_TABLE(wxMDIParentFrame
, wxFrame
)
55 EVT_SIZE(wxMDIParentFrame::OnSize
)
56 EVT_ACTIVATE(wxMDIParentFrame::OnActivate
)
57 EVT_SYS_COLOUR_CHANGED(wxMDIParentFrame::OnSysColourChanged
)
58 EVT_MENU_HIGHLIGHT_ALL(wxMDIParentFrame::OnMenuHighlight
)
61 BEGIN_EVENT_TABLE(wxMDIClientWindow
, wxNotebook
)
62 EVT_SCROLL(wxMDIClientWindow::OnScroll
)
63 EVT_NOTEBOOK_PAGE_CHANGED(wxID_NOTEBOOK_CLIENT_AREA
, wxMDIClientWindow::OnPageChanged
)
69 wxMDIParentFrame::wxMDIParentFrame()
71 m_clientWindow
= (wxMDIClientWindow
*) NULL
;
72 m_activeChild
= (wxMDIChildFrame
*) NULL
;
73 m_activeMenuBar
= (wxMenuBar
*) NULL
;
76 bool wxMDIParentFrame::Create(wxWindow
*parent
,
78 const wxString
& title
,
84 m_clientWindow
= (wxMDIClientWindow
*) NULL
;
85 m_activeChild
= (wxMDIChildFrame
*) NULL
;
86 m_activeMenuBar
= (wxMenuBar
*) NULL
;
88 bool success
= wxFrame::Create(parent
, id
, title
, pos
, size
, style
, name
);
91 // TODO: app cannot override OnCreateClient since
92 // wxMDIParentFrame::OnCreateClient will still be called
93 // (we're in the constructor). How to resolve?
95 m_clientWindow
= OnCreateClient();
97 // Uses own style for client style
98 m_clientWindow
->CreateClient(this, GetWindowStyleFlag());
101 GetClientSize(& w
, & h
);
102 m_clientWindow
->SetSize(0, 0, w
, h
);
109 wxMDIParentFrame::~wxMDIParentFrame()
111 // Make sure we delete the client window last of all
112 RemoveChild(m_clientWindow
);
116 delete m_clientWindow
;
117 m_clientWindow
= NULL
;
120 void wxMDIParentFrame::SetMenuBar(wxMenuBar
*menu_bar
)
122 m_frameMenuBar
= menu_bar
;
124 SetChildMenuBar((wxMDIChildFrame
*) NULL
);
127 void wxMDIParentFrame::OnSize(wxSizeEvent
& WXUNUSED(event
))
129 #if wxUSE_CONSTRAINTS
136 GetClientSize(&width
, &height
);
138 if ( GetClientWindow() )
139 GetClientWindow()->SetSize(x
, y
, width
, height
);
142 void wxMDIParentFrame::DoGetClientSize(int *width
, int *height
) const
144 wxFrame::DoGetClientSize(width
, height
);
147 void wxMDIParentFrame::OnActivate(wxActivateEvent
& WXUNUSED(event
))
152 // Returns the active MDI child window
153 wxMDIChildFrame
*wxMDIParentFrame::GetActiveChild() const
155 return m_activeChild
;
158 // Create the client window class (don't Create the window,
159 // just return a new class)
160 wxMDIClientWindow
*wxMDIParentFrame::OnCreateClient()
162 return new wxMDIClientWindow
;
165 // Set the child's menu into the parent frame
166 void wxMDIParentFrame::SetChildMenuBar(wxMDIChildFrame
* child
)
168 wxMenuBar
* oldMenuBar
= m_activeMenuBar
;
170 if (child
== (wxMDIChildFrame
*) NULL
) // No child: use parent frame
172 if (GetMenuBar() && (GetMenuBar() != m_activeMenuBar
))
174 // if (m_activeMenuBar)
175 // m_activeMenuBar->DestroyMenuBar();
177 m_activeMenuBar
= GetMenuBar();
178 m_activeMenuBar
->CreateMenuBar(this);
180 if (oldMenuBar && XtIsManaged((Widget) oldMenuBar->GetMainWidget()))
181 XtUnmanageChild((Widget) oldMenuBar->GetMainWidget());
183 if (oldMenuBar
&& oldMenuBar
->GetMainWidget())
184 XtUnmapWidget((Widget
) oldMenuBar
->GetMainWidget());
188 else if (child
->GetMenuBar() == (wxMenuBar
*) NULL
) // No child menu bar: use parent frame
190 if (GetMenuBar() && (GetMenuBar() != m_activeMenuBar
))
192 // if (m_activeMenuBar)
193 // m_activeMenuBar->DestroyMenuBar();
194 m_activeMenuBar
= GetMenuBar();
195 m_activeMenuBar
->CreateMenuBar(this);
197 if (oldMenuBar && XtIsManaged((Widget) oldMenuBar->GetMainWidget()))
198 XtUnmanageChild((Widget) oldMenuBar->GetMainWidget());
200 if (oldMenuBar
&& oldMenuBar
->GetMainWidget())
201 XtUnmapWidget((Widget
) oldMenuBar
->GetMainWidget());
204 else // The child has a menubar
206 if (child
->GetMenuBar() != m_activeMenuBar
)
208 // if (m_activeMenuBar)
209 // m_activeMenuBar->DestroyMenuBar();
211 m_activeMenuBar
= child
->GetMenuBar();
212 m_activeMenuBar
->CreateMenuBar(this);
214 if (oldMenuBar && XtIsManaged((Widget) oldMenuBar->GetMainWidget()))
215 XtUnmanageChild((Widget) oldMenuBar->GetMainWidget());
217 if (oldMenuBar
&& oldMenuBar
->GetMainWidget())
218 XtUnmapWidget((Widget
) oldMenuBar
->GetMainWidget());
223 // Redirect events to active child first
224 bool wxMDIParentFrame::ProcessEvent(wxEvent
& event
)
226 // Stops the same event being processed repeatedly
227 static wxEventType inEvent
= wxEVT_NULL
;
228 if (inEvent
== event
.GetEventType())
231 inEvent
= event
.GetEventType();
234 if (m_activeChild
&& event
.IsKindOf(CLASSINFO(wxCommandEvent
)))
236 res
= m_activeChild
->HandleWindowEvent(event
);
240 res
= GetEventHandler()->wxEvtHandler::ProcessEvent(event
);
242 inEvent
= wxEVT_NULL
;
247 void wxMDIParentFrame::DoSetSize(int x
, int y
,
248 int width
, int height
,
251 wxWindow::DoSetSize(x
, y
, width
, height
, sizeFlags
);
254 void wxMDIParentFrame::DoSetClientSize(int width
, int height
)
256 wxWindow::DoSetClientSize(width
, height
);
259 // Responds to colour changes, and passes event on to children.
260 void wxMDIParentFrame::OnSysColourChanged(wxSysColourChangedEvent
& event
)
264 // Propagate the event to the non-top-level children
265 wxFrame::OnSysColourChanged(event
);
269 void wxMDIParentFrame::Cascade()
274 void wxMDIParentFrame::Tile(wxOrientation
WXUNUSED(orient
))
279 void wxMDIParentFrame::ArrangeIcons()
284 void wxMDIParentFrame::ActivateNext()
289 void wxMDIParentFrame::ActivatePrevious()
294 // Default menu selection behaviour - display a help string
295 void wxMDIParentFrame::OnMenuHighlight(wxMenuEvent
& event
)
299 if (event
.GetMenuId() == -1)
300 SetStatusText(wxEmptyString
);
303 wxMenuBar
*menuBar
= (wxMenuBar
*) NULL
;
304 if (GetActiveChild())
305 menuBar
= GetActiveChild()->GetMenuBar();
307 menuBar
= GetMenuBar();
310 wxString
helpString(menuBar
->GetHelpString(event
.GetMenuId()));
311 if (!helpString
.empty())
312 SetStatusText(helpString
);
320 wxMDIChildFrame::wxMDIChildFrame()
322 m_mdiParentFrame
= (wxMDIParentFrame
*) NULL
;
325 bool wxMDIChildFrame::Create(wxMDIParentFrame
*parent
,
327 const wxString
& title
,
331 const wxString
& name
)
334 SetWindowStyleFlag(style
);
339 m_windowId
= (int)NewControlId();
341 wxMDIClientWindow
* clientWindow
= parent
->GetClientWindow();
343 wxCHECK_MSG( clientWindow
, false, "Missing MDI client window." );
345 clientWindow
->AddChild(this);
347 SetMDIParentFrame(parent
);
353 width
= 200; // TODO: give reasonable default
355 height
= 200; // TODO: give reasonable default
357 // We're deactivating the old child
358 wxMDIChildFrame
* oldActiveChild
= parent
->GetActiveChild();
361 wxActivateEvent
event(wxEVT_ACTIVATE
, false, oldActiveChild
->GetId());
362 event
.SetEventObject( oldActiveChild
);
363 oldActiveChild
->HandleWindowEvent(event
);
366 // This is the currently active child
367 parent
->SetActiveChild((wxMDIChildFrame
*) this);
369 // This time we'll try a bog-standard bulletin board for
370 // the 'frame'. A main window doesn't seem to work.
372 m_mainWidget
= (WXWidget
) XtVaCreateWidget("client",
373 xmBulletinBoardWidgetClass
, (Widget
) clientWindow
->GetTopWidget(),
377 XmNrightAttachment, XmATTACH_FORM,
378 XmNleftAttachment, XmATTACH_FORM,
379 XmNtopAttachment, XmATTACH_FORM,
380 XmNbottomAttachment, XmATTACH_FORM,
382 XmNresizePolicy
, XmRESIZE_NONE
,
385 XtAddEventHandler((Widget
) m_mainWidget
, ExposureMask
,False
,
386 wxUniversalRepaintProc
, (XtPointer
) this);
389 AttachWidget (parent
, m_mainWidget
, (WXWidget
) NULL
, pos
.x
, pos
.y
, size
.x
, size
.y
);
391 XtManageChild((Widget
) m_mainWidget
);
395 clientWindow
->AddPage(this, title
, true);
396 clientWindow
->Refresh();
398 // Positions the toolbar and status bar -- but we don't have any.
401 wxModelessWindows
.Append(this);
406 wxMDIChildFrame::~wxMDIChildFrame()
409 XtRemoveEventHandler((Widget
) m_mainWidget
, ExposureMask
,False
,
410 wxUniversalRepaintProc
, (XtPointer
) this);
412 if (GetMDIParentFrame())
414 wxMDIParentFrame
* parentFrame
= GetMDIParentFrame();
416 if (parentFrame
->GetActiveChild() == this)
417 parentFrame
->SetActiveChild((wxMDIChildFrame
*) NULL
);
418 wxMDIClientWindow
* clientWindow
= parentFrame
->GetClientWindow();
420 // Remove page if still there
422 int i
= clientWindow
->FindPage(this);
426 clientWindow
->RemovePage(i
);
427 clientWindow
->Refresh();
431 // Set the selection to the first remaining page
432 if (clientWindow
->GetPageCount() > 0)
434 wxMDIChildFrame
* child
= (wxMDIChildFrame
*) clientWindow
->GetPage(0);
435 parentFrame
->SetActiveChild(child
);
436 parentFrame
->SetChildMenuBar(child
);
440 parentFrame
->SetActiveChild((wxMDIChildFrame
*) NULL
);
441 parentFrame
->SetChildMenuBar((wxMDIChildFrame
*) NULL
);
447 // Implementation: intercept and act upon raise and lower commands.
448 void wxMDIChildFrame::OnRaise()
450 wxMDIParentFrame
* parentFrame
= (wxMDIParentFrame
*) GetParent() ;
451 wxMDIChildFrame
* oldActiveChild
= parentFrame
->GetActiveChild();
452 parentFrame
->SetActiveChild(this);
456 wxActivateEvent
event(wxEVT_ACTIVATE
, false, oldActiveChild
->GetId());
457 event
.SetEventObject( oldActiveChild
);
458 oldActiveChild
->HandleWindowEvent(event
);
461 wxActivateEvent
event(wxEVT_ACTIVATE
, true, this->GetId());
462 event
.SetEventObject( this );
463 this->HandleWindowEvent(event
);
466 void wxMDIChildFrame::OnLower()
468 wxMDIParentFrame
* parentFrame
= (wxMDIParentFrame
*) GetParent() ;
469 wxMDIChildFrame
* oldActiveChild
= parentFrame
->GetActiveChild();
471 if (oldActiveChild
== this)
473 wxActivateEvent
event(wxEVT_ACTIVATE
, false, oldActiveChild
->GetId());
474 event
.SetEventObject( oldActiveChild
);
475 oldActiveChild
->HandleWindowEvent(event
);
477 // TODO: unfortunately we don't now know which is the top-most child,
478 // so make the active child NULL.
479 parentFrame
->SetActiveChild((wxMDIChildFrame
*) NULL
);
483 // Set the client size (i.e. leave the calculation of borders etc.
485 void wxMDIChildFrame::DoSetClientSize(int width
, int height
)
487 wxWindow::DoSetClientSize(width
, height
);
490 void wxMDIChildFrame::DoGetClientSize(int* width
, int* height
) const
492 wxWindow::DoGetSize(width
, height
);
495 void wxMDIChildFrame::DoSetSize(int x
, int y
, int width
, int height
, int sizeFlags
)
497 wxWindow::DoSetSize(x
, y
, width
, height
, sizeFlags
);
500 void wxMDIChildFrame::DoGetSize(int* width
, int* height
) const
502 wxWindow::DoGetSize(width
, height
);
505 void wxMDIChildFrame::DoGetPosition(int *x
, int *y
) const
507 wxWindow::DoGetPosition(x
, y
);
510 bool wxMDIChildFrame::Show(bool show
)
512 SetVisibleStatus( show
);
513 return wxWindow::Show(show
);
516 void wxMDIChildFrame::SetMenuBar(wxMenuBar
*menuBar
)
518 // Don't create the underlying menubar yet; need to recreate
519 // it every time the child is activated.
520 m_frameMenuBar
= menuBar
;
522 // We make the assumption that if you're setting the menubar,
523 // this is the currently active child.
524 GetMDIParentFrame()->SetChildMenuBar(this);
528 void wxMDIChildFrame::SetIcon(const wxIcon
& icon
)
530 m_icons
= wxIconBundle( icon
);
534 // Not appropriate since there are no icons in
539 void wxMDIChildFrame::SetIcons(const wxIconBundle
& icons
)
544 void wxMDIChildFrame::SetTitle(const wxString
& title
)
546 wxTopLevelWindow::SetTitle( title
);
547 wxMDIClientWindow
* clientWindow
= GetMDIParentFrame()->GetClientWindow();
549 // Remove page if still there
551 int i
= clientWindow
->FindPage(this);
554 clientWindow
->SetPageText(i
, title
);
559 void wxMDIChildFrame::Maximize()
564 void wxMDIChildFrame::Iconize(bool WXUNUSED(iconize
))
569 bool wxMDIChildFrame::IsIconized() const
574 // Is it maximized? Always maximized under Motif, using the
575 // tabbed MDI implementation.
576 bool wxMDIChildFrame::IsMaximized(void) const
581 void wxMDIChildFrame::Restore()
586 void wxMDIChildFrame::Activate()
591 void wxMDIChildFrame::CaptureMouse()
593 wxWindow::CaptureMouse();
596 void wxMDIChildFrame::ReleaseMouse()
598 wxWindow::ReleaseMouse();
601 void wxMDIChildFrame::Raise()
606 void wxMDIChildFrame::Lower(void)
611 void wxMDIChildFrame::DoSetSizeHints(int WXUNUSED(minW
), int WXUNUSED(minH
), int WXUNUSED(maxW
), int WXUNUSED(maxH
), int WXUNUSED(incW
), int WXUNUSED(incH
))
617 wxMDIClientWindow::wxMDIClientWindow()
621 wxMDIClientWindow::~wxMDIClientWindow()
623 // By the time this destructor is called, the child frames will have been
624 // deleted and removed from the notebook/client window.
627 m_mainWidget
= (WXWidget
) 0;
630 bool wxMDIClientWindow::CreateClient(wxMDIParentFrame
*parent
, long style
)
632 SetWindowStyleFlag(style
);
634 bool success
= wxNotebook::Create(parent
, wxID_NOTEBOOK_CLIENT_AREA
, wxPoint(0, 0), wxSize(100, 100), 0);
643 int wxMDIClientWindow::FindPage(const wxNotebookPage
* page
)
645 for (int i
= GetPageCount() - 1; i
>= 0; --i
)
647 if (GetPage(i
) == page
)
654 void wxMDIClientWindow::DoSetSize(int x
, int y
, int width
, int height
, int sizeFlags
)
656 wxWindow::DoSetSize(x
, y
, width
, height
, sizeFlags
);
659 void wxMDIClientWindow::DoSetClientSize(int width
, int height
)
661 wxWindow::DoSetClientSize(width
, height
);
664 void wxMDIClientWindow::DoGetClientSize(int *width
, int *height
) const
666 wxWindow::DoGetClientSize(width
, height
);
669 void wxMDIClientWindow::DoGetSize(int *width
, int *height
) const
671 wxWindow::DoGetSize(width
, height
);
674 void wxMDIClientWindow::DoGetPosition(int *x
, int *y
) const
676 wxWindow::DoGetPosition(x
, y
);
679 void wxMDIClientWindow::OnScroll(wxScrollEvent
& event
)
681 // Default(); // Default processing: OBSOLETE FUNCTION
685 void wxMDIClientWindow::OnPageChanged(wxNotebookEvent
& event
)
687 // Notify child that it has been activated
688 if (event
.GetOldSelection() != -1)
690 wxMDIChildFrame
* oldChild
= (wxMDIChildFrame
*) GetPage(event
.GetOldSelection());
693 wxActivateEvent
event(wxEVT_ACTIVATE
, false, oldChild
->GetId());
694 event
.SetEventObject( oldChild
);
695 oldChild
->HandleWindowEvent(event
);
698 if (event
.GetSelection() != -1)
700 wxMDIChildFrame
* activeChild
= (wxMDIChildFrame
*) GetPage(event
.GetSelection());
703 wxActivateEvent
event(wxEVT_ACTIVATE
, true, activeChild
->GetId());
704 event
.SetEventObject( activeChild
);
705 activeChild
->HandleWindowEvent(event
);
707 if (activeChild
->GetMDIParentFrame())
709 activeChild
->GetMDIParentFrame()->SetActiveChild(activeChild
);
710 activeChild
->GetMDIParentFrame()->SetChildMenuBar(activeChild
);