]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/motif/mdi.cpp
made wxFFile a bit more safe: don't crash when Tell() and Length() are called on...
[wxWidgets.git] / src / motif / mdi.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: mdi.cpp
3// Purpose: MDI classes
4// Author: Julian Smart
5// Modified by:
6// Created: 17/09/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
13#pragma implementation "mdi.h"
14#endif
15
16#ifdef __VMS
17#define XtDisplay XTDISPLAY
18#define XtWindow XTWINDOW
19#endif
20
21#include "wx/mdi.h"
22#include "wx/menu.h"
23#include "wx/settings.h"
24#include "wx/icon.h"
25
26#ifdef __VMS__
27#pragma message disable nosimpint
28#endif
29#include <Xm/Xm.h>
30#include <Xm/BulletinB.h>
31#include <Xm/Form.h>
32#include <Xm/MainW.h>
33#include <Xm/RowColumn.h>
34#include <Xm/CascadeBG.h>
35#include <Xm/Text.h>
36#include <Xm/PushBG.h>
37#include <Xm/AtomMgr.h>
38#include <Xm/Protocols.h>
39#ifdef __VMS__
40#pragma message enable nosimpint
41#endif
42
43#include "wx/motif/private.h"
44
45extern wxList wxModelessWindows;
46
47// Implemented in frame.cpp
48extern void wxFrameFocusProc(Widget workArea, XtPointer clientData,
49 XmAnyCallbackStruct *cbs);
50
51#define wxID_NOTEBOOK_CLIENT_AREA wxID_HIGHEST + 100
52
53IMPLEMENT_DYNAMIC_CLASS(wxMDIParentFrame, wxFrame)
54IMPLEMENT_DYNAMIC_CLASS(wxMDIChildFrame, wxFrame)
55IMPLEMENT_DYNAMIC_CLASS(wxMDIClientWindow, wxNotebook)
56
57BEGIN_EVENT_TABLE(wxMDIParentFrame, wxFrame)
58 EVT_SIZE(wxMDIParentFrame::OnSize)
59 EVT_ACTIVATE(wxMDIParentFrame::OnActivate)
60 EVT_SYS_COLOUR_CHANGED(wxMDIParentFrame::OnSysColourChanged)
61 EVT_MENU_HIGHLIGHT_ALL(wxMDIParentFrame::OnMenuHighlight)
62END_EVENT_TABLE()
63
64BEGIN_EVENT_TABLE(wxMDIClientWindow, wxNotebook)
65 EVT_SCROLL(wxMDIClientWindow::OnScroll)
66 EVT_NOTEBOOK_PAGE_CHANGED(wxID_NOTEBOOK_CLIENT_AREA, wxMDIClientWindow::OnPageChanged)
67END_EVENT_TABLE()
68
69
70// Parent frame
71
72wxMDIParentFrame::wxMDIParentFrame()
73{
74 m_clientWindow = (wxMDIClientWindow*) NULL;
75 m_activeChild = (wxMDIChildFrame*) NULL;
76 m_activeMenuBar = (wxMenuBar*) NULL;
77}
78
79bool wxMDIParentFrame::Create(wxWindow *parent,
80 wxWindowID id,
81 const wxString& title,
82 const wxPoint& pos,
83 const wxSize& size,
84 long style,
85 const wxString& name)
86{
87 m_clientWindow = (wxMDIClientWindow*) NULL;
88 m_activeChild = (wxMDIChildFrame*) NULL;
89 m_activeMenuBar = (wxMenuBar*) NULL;
90
91 bool success = wxFrame::Create(parent, id, title, pos, size, style, name);
92 if (success)
93 {
94 // TODO: app cannot override OnCreateClient since
95 // wxMDIParentFrame::OnCreateClient will still be called
96 // (we're in the constructor). How to resolve?
97
98 m_clientWindow = OnCreateClient();
99
100 // Uses own style for client style
101 m_clientWindow->CreateClient(this, GetWindowStyleFlag());
102
103 int w, h;
104 GetClientSize(& w, & h);
105 m_clientWindow->SetSize(0, 0, w, h);
106 return TRUE;
107 }
108 else
109 return FALSE;
110}
111
112wxMDIParentFrame::~wxMDIParentFrame()
113{
114 // Make sure we delete the client window last of all
115 RemoveChild(m_clientWindow);
116
117 DestroyChildren();
118
119 delete m_clientWindow;
120 m_clientWindow = NULL;
121}
122
123void wxMDIParentFrame::SetMenuBar(wxMenuBar *menu_bar)
124{
125 m_frameMenuBar = menu_bar;
126
127 SetChildMenuBar((wxMDIChildFrame*) NULL);
128}
129
130void wxMDIParentFrame::OnSize(wxSizeEvent& WXUNUSED(event))
131{
132#if wxUSE_CONSTRAINTS
133 if (GetAutoLayout())
134 Layout();
135#endif
136 int x = 0;
137 int y = 0;
138 int width, height;
139 GetClientSize(&width, &height);
140
141 if ( GetClientWindow() )
142 GetClientWindow()->SetSize(x, y, width, height);
143}
144
145void wxMDIParentFrame::DoGetClientSize(int *width, int *height) const
146{
147 wxFrame::DoGetClientSize(width, height);
148}
149
150void wxMDIParentFrame::OnActivate(wxActivateEvent& WXUNUSED(event))
151{
152 // Do nothing
153}
154
155// Returns the active MDI child window
156wxMDIChildFrame *wxMDIParentFrame::GetActiveChild() const
157{
158 return m_activeChild;
159}
160
161// Create the client window class (don't Create the window,
162// just return a new class)
163wxMDIClientWindow *wxMDIParentFrame::OnCreateClient()
164{
165 return new wxMDIClientWindow ;
166}
167
168// Set the child's menu into the parent frame
169void wxMDIParentFrame::SetChildMenuBar(wxMDIChildFrame* child)
170{
171 wxMenuBar* oldMenuBar = m_activeMenuBar;
172
173 if (child == (wxMDIChildFrame*) NULL) // No child: use parent frame
174 {
175 if (GetMenuBar() && (GetMenuBar() != m_activeMenuBar))
176 {
177 // if (m_activeMenuBar)
178 // m_activeMenuBar->DestroyMenuBar();
179
180 m_activeMenuBar = GetMenuBar();
181 m_activeMenuBar->CreateMenuBar(this);
182 /*
183 if (oldMenuBar && XtIsManaged((Widget) oldMenuBar->GetMainWidget()))
184 XtUnmanageChild((Widget) oldMenuBar->GetMainWidget());
185 */
186 if (oldMenuBar && oldMenuBar->GetMainWidget())
187 XtUnmapWidget((Widget) oldMenuBar->GetMainWidget());
188
189 }
190 }
191 else if (child->GetMenuBar() == (wxMenuBar*) NULL) // No child menu bar: use parent frame
192 {
193 if (GetMenuBar() && (GetMenuBar() != m_activeMenuBar))
194 {
195 // if (m_activeMenuBar)
196 // m_activeMenuBar->DestroyMenuBar();
197 m_activeMenuBar = GetMenuBar();
198 m_activeMenuBar->CreateMenuBar(this);
199 /*
200 if (oldMenuBar && XtIsManaged((Widget) oldMenuBar->GetMainWidget()))
201 XtUnmanageChild((Widget) oldMenuBar->GetMainWidget());
202 */
203 if (oldMenuBar && oldMenuBar->GetMainWidget())
204 XtUnmapWidget((Widget) oldMenuBar->GetMainWidget());
205 }
206 }
207 else // The child has a menubar
208 {
209 if (child->GetMenuBar() != m_activeMenuBar)
210 {
211 // if (m_activeMenuBar)
212 // m_activeMenuBar->DestroyMenuBar();
213
214 m_activeMenuBar = child->GetMenuBar();
215 m_activeMenuBar->CreateMenuBar(this);
216 /*
217 if (oldMenuBar && XtIsManaged((Widget) oldMenuBar->GetMainWidget()))
218 XtUnmanageChild((Widget) oldMenuBar->GetMainWidget());
219 */
220 if (oldMenuBar && oldMenuBar->GetMainWidget())
221 XtUnmapWidget((Widget) oldMenuBar->GetMainWidget());
222 }
223 }
224}
225
226// Redirect events to active child first
227bool wxMDIParentFrame::ProcessEvent(wxEvent& event)
228{
229 // Stops the same event being processed repeatedly
230 static wxEventType inEvent = wxEVT_NULL;
231 if (inEvent == event.GetEventType())
232 return FALSE;
233
234 inEvent = event.GetEventType();
235
236 bool res = FALSE;
237 if (m_activeChild && event.IsKindOf(CLASSINFO(wxCommandEvent)))
238 {
239 res = m_activeChild->GetEventHandler()->ProcessEvent(event);
240 }
241
242 if (!res)
243 res = GetEventHandler()->wxEvtHandler::ProcessEvent(event);
244
245 inEvent = wxEVT_NULL;
246
247 return res;
248}
249
250void wxMDIParentFrame::DoSetSize(int x, int y,
251 int width, int height,
252 int sizeFlags)
253{
254 wxWindow::DoSetSize(x, y, width, height, sizeFlags);
255}
256
257void wxMDIParentFrame::DoSetClientSize(int width, int height)
258{
259 wxWindow::DoSetClientSize(width, height);
260}
261
262// Responds to colour changes, and passes event on to children.
263void wxMDIParentFrame::OnSysColourChanged(wxSysColourChangedEvent& event)
264{
265 // TODO
266
267 // Propagate the event to the non-top-level children
268 wxFrame::OnSysColourChanged(event);
269}
270
271// MDI operations
272void wxMDIParentFrame::Cascade()
273{
274 // TODO
275}
276
277void wxMDIParentFrame::Tile()
278{
279 // TODO
280}
281
282void wxMDIParentFrame::ArrangeIcons()
283{
284 // TODO
285}
286
287void wxMDIParentFrame::ActivateNext()
288{
289 // TODO
290}
291
292void wxMDIParentFrame::ActivatePrevious()
293{
294 // TODO
295}
296
297// Default menu selection behaviour - display a help string
298void wxMDIParentFrame::OnMenuHighlight(wxMenuEvent& event)
299{
300 if (GetStatusBar())
301 {
302 if (event.GetMenuId() == -1)
303 SetStatusText("");
304 else
305 {
306 wxMenuBar *menuBar = (wxMenuBar*) NULL;
307 if (GetActiveChild())
308 menuBar = GetActiveChild()->GetMenuBar();
309 else
310 menuBar = GetMenuBar();
311 if (menuBar)
312 {
313 wxString helpString(menuBar->GetHelpString(event.GetMenuId()));
314 if (helpString != "")
315 SetStatusText(helpString);
316 }
317 }
318 }
319}
320
321// Child frame
322
323wxMDIChildFrame::wxMDIChildFrame()
324{
325 m_mdiParentFrame = (wxMDIParentFrame*) NULL;
326}
327
328bool wxMDIChildFrame::Create(wxMDIParentFrame *parent,
329 wxWindowID id,
330 const wxString& title,
331 const wxPoint& pos,
332 const wxSize& size,
333 long style,
334 const wxString& name)
335{
336 SetName(name);
337 SetWindowStyleFlag(style);
338
339 m_backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE);
340 m_foregroundColour = *wxBLACK;
341 m_font = wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
342
343 if ( id > -1 )
344 m_windowId = id;
345 else
346 m_windowId = (int)NewControlId();
347
348 wxMDIClientWindow* clientWindow = parent->GetClientWindow();
349
350 wxASSERT_MSG( (clientWindow != (wxWindow*) NULL), "Missing MDI client window.");
351
352 if (clientWindow) clientWindow->AddChild(this);
353
354 SetMDIParentFrame(parent);
355
356 int width = size.x;
357 int height = size.y;
358 if (width == -1)
359 width = 200; // TODO: give reasonable default
360 if (height == -1)
361 height = 200; // TODO: give reasonable default
362
363 // We're deactivating the old child
364 wxMDIChildFrame* oldActiveChild = parent->GetActiveChild();
365 if (oldActiveChild)
366 {
367 wxActivateEvent event(wxEVT_ACTIVATE, FALSE, oldActiveChild->GetId());
368 event.SetEventObject( oldActiveChild );
369 oldActiveChild->GetEventHandler()->ProcessEvent(event);
370 }
371
372 // This is the currently active child
373 parent->SetActiveChild((wxMDIChildFrame*) this);
374
375 // This time we'll try a bog-standard bulletin board for
376 // the 'frame'. A main window doesn't seem to work.
377
378 m_mainWidget = (WXWidget) XtVaCreateWidget("client",
379 xmBulletinBoardWidgetClass, (Widget) clientWindow->GetTopWidget(),
380 XmNmarginWidth, 0,
381 XmNmarginHeight, 0,
382 /*
383 XmNrightAttachment, XmATTACH_FORM,
384 XmNleftAttachment, XmATTACH_FORM,
385 XmNtopAttachment, XmATTACH_FORM,
386 XmNbottomAttachment, XmATTACH_FORM,
387 */
388 XmNresizePolicy, XmRESIZE_NONE,
389 NULL);
390
391 XtAddEventHandler((Widget) m_mainWidget, ExposureMask,FALSE,
392 wxUniversalRepaintProc, (XtPointer) this);
393
394 AttachWidget (parent, m_mainWidget, (WXWidget) NULL, pos.x, pos.y, size.x, size.y);
395
396 ChangeBackgroundColour();
397
398 XtManageChild((Widget) m_mainWidget);
399
400 SetTitle(title);
401
402 clientWindow->AddPage(this, title, TRUE);
403 clientWindow->Refresh();
404
405 // Positions the toolbar and status bar -- but we don't have any.
406 // PreResize();
407
408 wxModelessWindows.Append(this);
409 return TRUE;
410}
411
412
413wxMDIChildFrame::~wxMDIChildFrame()
414{
415 if (m_mainWidget)
416 XtRemoveEventHandler((Widget) m_mainWidget, ExposureMask,FALSE,
417 wxUniversalRepaintProc, (XtPointer) this);
418
419 if (GetMDIParentFrame())
420 {
421 wxMDIParentFrame* parentFrame = GetMDIParentFrame();
422
423 if (parentFrame->GetActiveChild() == this)
424 parentFrame->SetActiveChild((wxMDIChildFrame*) NULL);
425 wxMDIClientWindow* clientWindow = parentFrame->GetClientWindow();
426
427 // Remove page if still there
428 {
429 int i = clientWindow->FindPage(this);
430
431 if (i != -1)
432 {
433 clientWindow->RemovePage(i);
434 clientWindow->Refresh();
435 }
436 }
437
438 // Set the selection to the first remaining page
439 if (clientWindow->GetPageCount() > 0)
440 {
441 wxMDIChildFrame* child = (wxMDIChildFrame*) clientWindow->GetPage(0);
442 parentFrame->SetActiveChild(child);
443 parentFrame->SetChildMenuBar(child);
444 }
445 else
446 {
447 parentFrame->SetActiveChild((wxMDIChildFrame*) NULL);
448 parentFrame->SetChildMenuBar((wxMDIChildFrame*) NULL);
449 }
450 }
451}
452
453#if 0
454// Implementation: intercept and act upon raise and lower commands.
455void wxMDIChildFrame::OnRaise()
456{
457 wxMDIParentFrame* parentFrame = (wxMDIParentFrame*) GetParent() ;
458 wxMDIChildFrame* oldActiveChild = parentFrame->GetActiveChild();
459 parentFrame->SetActiveChild(this);
460
461 if (oldActiveChild)
462 {
463 wxActivateEvent event(wxEVT_ACTIVATE, FALSE, oldActiveChild->GetId());
464 event.SetEventObject( oldActiveChild );
465 oldActiveChild->GetEventHandler()->ProcessEvent(event);
466 }
467
468 wxActivateEvent event(wxEVT_ACTIVATE, TRUE, this->GetId());
469 event.SetEventObject( this );
470 this->GetEventHandler()->ProcessEvent(event);
471}
472
473void wxMDIChildFrame::OnLower()
474{
475 wxMDIParentFrame* parentFrame = (wxMDIParentFrame*) GetParent() ;
476 wxMDIChildFrame* oldActiveChild = parentFrame->GetActiveChild();
477
478 if (oldActiveChild == this)
479 {
480 wxActivateEvent event(wxEVT_ACTIVATE, FALSE, oldActiveChild->GetId());
481 event.SetEventObject( oldActiveChild );
482 oldActiveChild->GetEventHandler()->ProcessEvent(event);
483 }
484 // TODO: unfortunately we don't now know which is the top-most child,
485 // so make the active child NULL.
486 parentFrame->SetActiveChild((wxMDIChildFrame*) NULL);
487}
488#endif
489
490// Set the client size (i.e. leave the calculation of borders etc.
491// to wxWindows)
492void wxMDIChildFrame::DoSetClientSize(int width, int height)
493{
494 wxWindow::DoSetClientSize(width, height);
495}
496
497void wxMDIChildFrame::DoGetClientSize(int* width, int* height) const
498{
499 wxWindow::DoGetSize(width, height);
500}
501
502void wxMDIChildFrame::DoSetSize(int x, int y, int width, int height, int sizeFlags)
503{
504 wxWindow::DoSetSize(x, y, width, height, sizeFlags);
505}
506
507void wxMDIChildFrame::DoGetSize(int* width, int* height) const
508{
509 wxWindow::DoGetSize(width, height);
510}
511
512void wxMDIChildFrame::DoGetPosition(int *x, int *y) const
513{
514 wxWindow::DoGetPosition(x, y);
515}
516
517bool wxMDIChildFrame::Show(bool show)
518{
519 SetVisibleStatus( show );
520 return wxWindow::Show(show);
521}
522
523void wxMDIChildFrame::SetMenuBar(wxMenuBar *menuBar)
524{
525 // Don't create the underlying menubar yet; need to recreate
526 // it every time the child is activated.
527 m_frameMenuBar = menuBar;
528
529 // We make the assumption that if you're setting the menubar,
530 // this is the currently active child.
531 GetMDIParentFrame()->SetChildMenuBar(this);
532}
533
534// Set icon
535void wxMDIChildFrame::SetIcon(const wxIcon& icon)
536{
537 m_icons = wxIconBundle( icon );
538
539 if (icon.Ok())
540 {
541 // Not appropriate since there are no icons in
542 // a tabbed window
543 }
544}
545
546void wxMDIChildFrame::SetIcons(const wxIconBundle& icons)
547{
548 m_icons = icons;
549}
550
551void wxMDIChildFrame::SetTitle(const wxString& title)
552{
553 wxTopLevelWindow::SetTitle( title );
554 wxMDIClientWindow* clientWindow = GetMDIParentFrame()->GetClientWindow();
555
556 // Remove page if still there
557 {
558 int i = clientWindow->FindPage(this);
559
560 if (i != -1)
561 clientWindow->SetPageText(i, title);
562 }
563}
564
565// MDI operations
566void wxMDIChildFrame::Maximize()
567{
568 // TODO
569}
570
571void wxMDIChildFrame::Iconize(bool WXUNUSED(iconize))
572{
573 // TODO
574}
575
576bool wxMDIChildFrame::IsIconized() const
577{
578 return FALSE;
579}
580
581// Is it maximized? Always maximized under Motif, using the
582// tabbed MDI implementation.
583bool wxMDIChildFrame::IsMaximized(void) const
584{
585 return TRUE;
586}
587
588void wxMDIChildFrame::Restore()
589{
590 // TODO
591}
592
593void wxMDIChildFrame::Activate()
594{
595 // TODO
596}
597
598void wxMDIChildFrame::CaptureMouse()
599{
600 wxWindow::CaptureMouse();
601}
602
603void wxMDIChildFrame::ReleaseMouse()
604{
605 wxWindow::ReleaseMouse();
606}
607
608void wxMDIChildFrame::Raise()
609{
610 wxWindow::Raise();
611}
612
613void wxMDIChildFrame::Lower(void)
614{
615 wxWindow::Raise();
616}
617
618void wxMDIChildFrame::SetSizeHints(int WXUNUSED(minW), int WXUNUSED(minH), int WXUNUSED(maxW), int WXUNUSED(maxH), int WXUNUSED(incW), int WXUNUSED(incH))
619{
620}
621
622// Client window
623
624wxMDIClientWindow::wxMDIClientWindow()
625{
626}
627
628wxMDIClientWindow::~wxMDIClientWindow()
629{
630 // By the time this destructor is called, the child frames will have been
631 // deleted and removed from the notebook/client window.
632 DestroyChildren();
633
634 m_mainWidget = (WXWidget) 0;
635}
636
637bool wxMDIClientWindow::CreateClient(wxMDIParentFrame *parent, long style)
638{
639 SetWindowStyleFlag(style);
640
641 // m_windowParent = parent;
642 // m_backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE);
643
644 bool success = wxNotebook::Create(parent, wxID_NOTEBOOK_CLIENT_AREA, wxPoint(0, 0), wxSize(100, 100), 0);
645 if (success)
646 {
647 wxFont font(10, wxSWISS, wxNORMAL, wxNORMAL);
648 SetFont(font);
649 return TRUE;
650 }
651 else
652 return FALSE;
653}
654
655int wxMDIClientWindow::FindPage(const wxNotebookPage* page)
656{
657 for (int i = GetPageCount() - 1; i >= 0; --i)
658 {
659 if (GetPage(i) == page)
660 return i;
661 }
662
663 return -1;
664}
665
666void wxMDIClientWindow::DoSetSize(int x, int y, int width, int height, int sizeFlags)
667{
668 wxWindow::DoSetSize(x, y, width, height, sizeFlags);
669}
670
671void wxMDIClientWindow::DoSetClientSize(int width, int height)
672{
673 wxWindow::DoSetClientSize(width, height);
674}
675
676void wxMDIClientWindow::DoGetClientSize(int *width, int *height) const
677{
678 wxWindow::DoGetClientSize(width, height);
679}
680
681void wxMDIClientWindow::DoGetSize(int *width, int *height) const
682{
683 wxWindow::DoGetSize(width, height);
684}
685
686void wxMDIClientWindow::DoGetPosition(int *x, int *y) const
687{
688 wxWindow::DoGetPosition(x, y);
689}
690
691void wxMDIClientWindow::OnScroll(wxScrollEvent& event)
692{
693 // Default(); // Default processing: OBSOLETE FUNCTION
694 event.Skip();
695}
696
697void wxMDIClientWindow::OnPageChanged(wxNotebookEvent& event)
698{
699 // Notify child that it has been activated
700 if (event.GetOldSelection() != -1)
701 {
702 wxMDIChildFrame* oldChild = (wxMDIChildFrame*) GetPage(event.GetOldSelection());
703 if (oldChild)
704 {
705 wxActivateEvent event(wxEVT_ACTIVATE, FALSE, oldChild->GetId());
706 event.SetEventObject( oldChild );
707 oldChild->GetEventHandler()->ProcessEvent(event);
708 }
709 }
710 if (event.GetSelection() != -1)
711 {
712 wxMDIChildFrame* activeChild = (wxMDIChildFrame*) GetPage(event.GetSelection());
713 if (activeChild)
714 {
715 wxActivateEvent event(wxEVT_ACTIVATE, TRUE, activeChild->GetId());
716 event.SetEventObject( activeChild );
717 activeChild->GetEventHandler()->ProcessEvent(event);
718
719 if (activeChild->GetMDIParentFrame())
720 {
721 activeChild->GetMDIParentFrame()->SetActiveChild(activeChild);
722 activeChild->GetMDIParentFrame()->SetChildMenuBar(activeChild);
723 }
724 }
725 }
726 event.Skip();
727}