1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "frame.h"
17 #include "wx/statusbr.h"
18 #include "wx/toolbar.h"
19 #include "wx/menuitem.h"
21 #include "wx/dcclient.h"
22 #include "wx/dialog.h"
23 #include "wx/settings.h"
28 #pragma message disable nosimpint
30 #if defined(__ultrix) || defined(__sgi)
35 #include <X11/Shell.h>
41 #include <Xm/MwmUtil.h>
42 #include <Xm/BulletinB.h>
45 #include <Xm/RowColumn.h>
47 #include <Xm/AtomMgr.h>
48 #include <Xm/LabelG.h>
51 #include <Xm/Protocols.h>
54 #pragma message enable nosimpint
57 #include "wx/motif/private.h"
59 void wxCloseFrameCallback(Widget
, XtPointer
, XmAnyCallbackStruct
*cbs
);
60 void wxFrameFocusProc(Widget workArea
, XtPointer clientData
,
61 XmAnyCallbackStruct
*cbs
);
62 static void wxFrameMapProc(Widget frameShell
, XtPointer clientData
,
63 XCrossingEvent
* event
);
65 extern wxList wxModelessWindows
;
66 extern wxList wxPendingDelete
;
68 // TODO: this should be tidied so that any frame can be the
70 static bool wxTopLevelUsed
= FALSE
;
72 #if !USE_SHARED_LIBRARY
73 BEGIN_EVENT_TABLE(wxFrame
, wxWindow
)
74 EVT_SIZE(wxFrame::OnSize
)
75 EVT_ACTIVATE(wxFrame::OnActivate
)
76 EVT_MENU_HIGHLIGHT_ALL(wxFrame::OnMenuHighlight
)
77 EVT_SYS_COLOUR_CHANGED(wxFrame::OnSysColourChanged
)
78 EVT_IDLE(wxFrame::OnIdle
)
79 EVT_CLOSE(wxFrame::OnCloseWindow
)
82 IMPLEMENT_DYNAMIC_CLASS(wxFrame
, wxWindow
)
85 #if wxUSE_NATIVE_STATUSBAR
86 bool wxFrame::m_useNativeStatusBar
= TRUE
;
88 bool wxFrame::m_useNativeStatusBar
= FALSE
;
94 m_frameToolBar
= NULL
;
95 #endif // wxUSE_TOOLBAR
97 m_frameMenuBar
= NULL
;
98 m_frameStatusBar
= NULL
;
104 m_frameShell
= (WXWidget
) NULL
;
105 m_frameWidget
= (WXWidget
) NULL
;;
106 m_workArea
= (WXWidget
) NULL
;;
107 m_clientArea
= (WXWidget
) NULL
;;
108 m_visibleStatus
= TRUE
;
112 bool wxFrame::Create(wxWindow
*parent
,
114 const wxString
& title
,
118 const wxString
& name
)
121 wxTopLevelWindows
.Append(this);
125 m_windowStyle
= style
;
126 m_frameMenuBar
= NULL
;
128 m_frameToolBar
= NULL
;
129 #endif // wxUSE_TOOLBAR
130 m_frameStatusBar
= NULL
;
133 m_frameShell
= (WXWidget
) NULL
;
134 m_frameWidget
= (WXWidget
) NULL
;;
135 m_workArea
= (WXWidget
) NULL
;;
136 m_clientArea
= (WXWidget
) NULL
;;
137 m_visibleStatus
= TRUE
;
140 m_backgroundColour
= wxSystemSettings::GetSystemColour(wxSYS_COLOUR_APPWORKSPACE
);
141 m_foregroundColour
= *wxBLACK
;
142 m_font
= wxSystemSettings::GetSystemFont(wxSYS_DEFAULT_GUI_FONT
);
147 m_windowId
= (int)NewControlId();
149 if (parent
) parent
->AddChild(this);
151 wxModelessWindows
.Append(this);
153 int x
= pos
.x
, y
= pos
.y
;
154 int width
= size
.x
, height
= size
.y
;
156 // Set reasonable values for position and size if defaults have been
159 // MB TODO: something better than these arbitrary values ?
160 // VZ should use X resources for this...
166 int displayW
, displayH
;
167 wxDisplaySize( &displayW
, &displayH
);
171 x
= (displayW
- width
) / 2;
176 y
= (displayH
- height
) / 2;
182 // Change suggested by Matthew Flatt
183 m_frameShell
= (WXWidget
)XtAppCreateShell
186 wxTheApp
->GetClassName(),
187 topLevelShellWidgetClass
,
188 (Display
*) wxGetDisplay(),
195 m_frameShell
= wxTheApp
->GetTopLevelWidget();
196 wxTopLevelUsed
= TRUE
;
199 XtVaSetValues((Widget
) m_frameShell
,
200 // Allows menu to resize
201 XmNallowShellResize
, True
,
202 XmNdeleteResponse
, XmDO_NOTHING
,
203 XmNmappedWhenManaged
, False
,
204 XmNiconic
, (style
& wxICONIZE
) ? TRUE
: FALSE
,
208 XtVaSetValues((Widget
) m_frameShell
,
209 XmNtitle
, (const char*) title
,
212 m_frameWidget
= (WXWidget
) XtVaCreateManagedWidget("main_window",
213 xmMainWindowWidgetClass
, (Widget
) m_frameShell
,
214 XmNresizePolicy
, XmRESIZE_NONE
,
217 m_workArea
= (WXWidget
) XtVaCreateWidget("form",
218 xmFormWidgetClass
, (Widget
) m_frameWidget
,
219 XmNresizePolicy
, XmRESIZE_NONE
,
222 m_clientArea
= (WXWidget
) XtVaCreateWidget("client",
223 xmBulletinBoardWidgetClass
, (Widget
) m_workArea
,
226 XmNrightAttachment
, XmATTACH_FORM
,
227 XmNleftAttachment
, XmATTACH_FORM
,
228 XmNtopAttachment
, XmATTACH_FORM
,
229 XmNbottomAttachment
, XmATTACH_FORM
,
230 // XmNresizePolicy, XmRESIZE_ANY,
233 wxLogDebug("Created frame (0x%08x) with work area 0x%08x and client "
234 "area 0x%08x", m_frameWidget
, m_workArea
, m_clientArea
);
236 XtAddEventHandler((Widget
) m_clientArea
, ExposureMask
,FALSE
,
237 wxUniversalRepaintProc
, (XtPointer
) this);
239 XtVaSetValues((Widget
) m_frameWidget
,
240 XmNworkWindow
, (Widget
) m_workArea
,
243 XtManageChild((Widget
) m_clientArea
);
244 XtManageChild((Widget
) m_workArea
);
246 wxAddWindowToTable((Widget
) m_workArea
, this);
250 XtOverrideTranslations((Widget
) m_workArea
,
251 ptr
= XtParseTranslationTable("<Configure>: resize()"));
255 XtAddCallback((Widget
) m_workArea
, XmNfocusCallback
,
256 (XtCallbackProc
)wxFrameFocusProc
, (XtPointer
)this);
258 /* Part of show-&-hide fix */
259 XtAddEventHandler((Widget
) m_frameShell
, StructureNotifyMask
,
260 False
, (XtEventHandler
)wxFrameMapProc
,
261 (XtPointer
)m_workArea
);
264 XtVaSetValues((Widget
) m_frameShell
, XmNx
, x
, NULL
);
266 XtVaSetValues((Widget
) m_frameShell
, XmNy
, y
, NULL
);
268 XtVaSetValues((Widget
) m_frameShell
, XmNwidth
, width
, NULL
);
270 XtVaSetValues((Widget
) m_frameShell
, XmNheight
, height
, NULL
);
272 m_mainWidget
= m_frameWidget
;
276 // This patch comes from Torsten Liermann lier@lier1.muc.de
277 if (XmIsMotifWMRunning( (Widget
) m_frameShell
))
280 if (style
& wxRESIZE_BORDER
)
281 decor
|= MWM_DECOR_RESIZEH
;
282 if (style
& wxSYSTEM_MENU
)
283 decor
|= MWM_DECOR_MENU
;
284 if ((style
& wxCAPTION
) ||
285 (style
& wxTINY_CAPTION_HORIZ
) ||
286 (style
& wxTINY_CAPTION_VERT
))
287 decor
|= MWM_DECOR_TITLE
;
288 if (style
& wxTHICK_FRAME
)
289 decor
|= MWM_DECOR_BORDER
;
290 if (style
& wxTHICK_FRAME
)
291 decor
|= MWM_DECOR_BORDER
;
292 if (style
& wxMINIMIZE_BOX
)
293 decor
|= MWM_DECOR_MINIMIZE
;
294 if (style
& wxMAXIMIZE_BOX
)
295 decor
|= MWM_DECOR_MAXIMIZE
;
296 XtVaSetValues((Widget
) m_frameShell
,XmNmwmDecorations
,decor
,NULL
) ;
298 // This allows non-Motif window managers to support at least the
299 // no-decorations case.
303 XtVaSetValues((Widget
) m_frameShell
,XmNoverrideRedirect
,TRUE
,NULL
);
305 XtRealizeWidget((Widget
) m_frameShell
);
307 // Intercept CLOSE messages from the window manager
308 Atom WM_DELETE_WINDOW
= XmInternAtom(XtDisplay((Widget
) m_frameShell
), "WM_DELETE_WINDOW", False
);
309 #if (XmREVISION > 1 || XmVERSION > 1)
310 XmAddWMProtocolCallback((Widget
) m_frameShell
, WM_DELETE_WINDOW
, (XtCallbackProc
) wxCloseFrameCallback
, (XtPointer
)this);
313 XmAddWMProtocolCallback((Widget
) m_frameShell
, WM_DELETE_WINDOW
, (XtCallbackProc
) wxCloseFrameCallback
, (caddr_t
)this);
315 XmAddWMProtocolCallback((Widget
) m_frameShell
, WM_DELETE_WINDOW
, (void (*)())wxCloseFrameCallback
, (caddr_t
)this);
319 ChangeBackgroundColour();
323 wxSizeEvent
sizeEvent(wxSize(width
, height
), GetId());
324 sizeEvent
.SetEventObject(this);
326 GetEventHandler()->ProcessEvent(sizeEvent
);
333 m_isBeingDeleted
= TRUE
;
336 XtRemoveEventHandler((Widget
) m_clientArea
, ExposureMask
, FALSE
,
337 wxUniversalRepaintProc
, (XtPointer
) this);
344 m_frameMenuBar
->DestroyMenuBar();
346 // Hack to stop core dump on Ultrix, OSF, for some strange reason.
347 #if MOTIF_MENUBAR_DELETE_FIX
348 GetMenuBar()->SetMainWidget((WXWidget
) NULL
);
350 delete m_frameMenuBar
;
351 m_frameMenuBar
= NULL
;
354 wxTopLevelWindows
.DeleteObject(this);
355 wxModelessWindows
.DeleteObject(this);
357 if (m_frameStatusBar
)
358 delete m_frameStatusBar
;
364 for (i = 0; i < wxMAX_STATUS; i++)
365 if (statusTextWidget[i])
366 XtDestroyWidget (statusTextWidget[i]);
369 XtDestroyWidget (statusLineForm);
371 if (statusLineWidget)
372 XtDestroyWidget (statusLineWidget);
377 wxDeleteWindowFromTable((Widget
) m_workArea
);
379 XtDestroyWidget ((Widget
) m_workArea
);
384 wxDeleteWindowFromTable((Widget
) m_frameWidget
);
385 XtDestroyWidget ((Widget
) m_frameWidget
);
389 XtDestroyWidget ((Widget
) m_frameShell
);
391 SetMainWidget((WXWidget
) NULL
);
393 /* Check if it's the last top-level window */
395 if (wxTheApp
&& (wxTopLevelWindows
.Number() == 0))
397 wxTheApp
->SetTopWindow(NULL
);
399 if (wxTheApp
->GetExitOnFrameDelete())
401 // Signal to the app that we're going to close
402 wxTheApp
->ExitMainLoop();
408 // Get size *available for subwindows* i.e. excluding menu bar, toolbar etc.
409 void wxFrame::DoGetClientSize(int *x
, int *y
) const
412 XtVaGetValues((Widget
) m_workArea
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
414 if (m_frameStatusBar
)
417 m_frameStatusBar
->GetSize(& sbw
, & sbh
);
424 m_frameToolBar
->GetSize(& tbw
, & tbh
);
425 if (m_frameToolBar
->GetWindowStyleFlag() & wxTB_VERTICAL
)
430 #endif // wxUSE_TOOLBAR
432 if (GetMenuBar() != (wxMenuBar*) NULL)
434 // it seems that if a frame holds a panel, the menu bar size
435 // gets automatically taken care of --- grano@cs.helsinki.fi 4.4.95
436 bool hasSubPanel = FALSE;
437 for(wxNode* node = GetChildren().First(); node; node = node->Next())
439 wxWindow *win = (wxWindow *)node->Data();
440 hasSubPanel = (win->IsKindOf(CLASSINFO(wxPanel)) && !win->IsKindOf(CLASSINFO(wxDialog)));
447 XtVaGetValues((Widget) GetMenuBarWidget(), XmNheight, &ys, NULL);
456 // Set the client size (i.e. leave the calculation of borders etc.
458 void wxFrame::DoSetClientSize(int width
, int height
)
460 // Calculate how large the new main window should be
461 // by finding the difference between the client area and the
462 // main window area, and adding on to the new client area
464 XtVaSetValues((Widget
) m_workArea
, XmNwidth
, width
, NULL
);
468 if (m_frameStatusBar
)
471 m_frameStatusBar
->GetSize(& sbw
, & sbh
);
478 m_frameToolBar
->GetSize(& tbw
, & tbh
);
479 if (m_frameToolBar
->GetWindowStyleFlag() & wxTB_VERTICAL
)
484 #endif // wxUSE_TOOLBAR
486 XtVaSetValues((Widget
) m_workArea
, XmNheight
, height
, NULL
);
490 wxSizeEvent
sizeEvent(wxSize(width
, height
), GetId());
491 sizeEvent
.SetEventObject(this);
493 GetEventHandler()->ProcessEvent(sizeEvent
);
497 void wxFrame::DoGetSize(int *width
, int *height
) const
500 XtVaGetValues((Widget
) m_frameShell
, XmNwidth
, &xx
, XmNheight
, &yy
, NULL
);
501 *width
= xx
; *height
= yy
;
504 void wxFrame::DoGetPosition(int *x
, int *y
) const
506 Window parent_window
= XtWindow((Widget
) m_frameShell
),
507 next_parent
= XtWindow((Widget
) m_frameShell
),
508 root
= RootWindowOfScreen(XtScreen((Widget
) m_frameShell
));
510 // search for the parent that is child of ROOT, because the WM may
511 // reparent twice and notify only the next parent (like FVWM)
512 while (next_parent
!= root
) {
513 Window
*theChildren
; unsigned int n
;
514 parent_window
= next_parent
;
515 XQueryTree(XtDisplay((Widget
) m_frameShell
), parent_window
, &root
,
516 &next_parent
, &theChildren
, &n
);
517 XFree(theChildren
); // not needed
519 int xx
, yy
; unsigned int dummy
;
520 XGetGeometry(XtDisplay((Widget
) m_frameShell
), parent_window
, &root
,
521 &xx
, &yy
, &dummy
, &dummy
, &dummy
, &dummy
);
526 void wxFrame::DoSetSize(int x
, int y
, int width
, int height
, int WXUNUSED(sizeFlags
))
529 XtVaSetValues((Widget
) m_frameShell
, XmNx
, x
, NULL
);
531 XtVaSetValues((Widget
) m_frameShell
, XmNy
, y
, NULL
);
533 XtVaSetValues((Widget
) m_frameWidget
, XmNwidth
, width
, NULL
);
535 XtVaSetValues((Widget
) m_frameWidget
, XmNheight
, height
, NULL
);
537 if (!(height
== -1 && width
== -1))
541 wxSizeEvent
sizeEvent(wxSize(width
, height
), GetId());
542 sizeEvent
.SetEventObject(this);
544 GetEventHandler()->ProcessEvent(sizeEvent
);
548 bool wxFrame::Show(bool show
)
551 return wxWindow::Show(show
);
553 m_visibleStatus
= show
; /* show-&-hide fix */
557 XtMapWidget((Widget
) m_frameShell
);
558 XRaiseWindow(XtDisplay((Widget
) m_frameShell
), XtWindow((Widget
) m_frameShell
));
560 XtUnmapWidget((Widget
) m_frameShell
);
561 // XmUpdateDisplay(wxTheApp->topLevel); // Experimental: may be responsible for crashes
566 void wxFrame::Iconize(bool iconize
)
572 XtVaSetValues((Widget
) m_frameShell
, XmNiconic
, (Boolean
)iconize
, NULL
);
575 // Equivalent to maximize/restore in Windows
576 void wxFrame::Maximize(bool maximize
)
580 if (maximize
&& m_frameShell
)
581 XtVaSetValues((Widget
) m_frameShell
, XmNiconic
, FALSE
, NULL
);
584 bool wxFrame::IsIconized() const
590 XtVaGetValues((Widget
) m_frameShell
, XmNiconic
, &iconic
, NULL
);
595 bool wxFrame::IsMaximized(void) const
597 // No maximizing in Motif (?)
601 void wxFrame::SetTitle(const wxString
& title
)
603 if (title
== m_title
)
609 XtVaSetValues((Widget
) m_frameShell
,
610 XmNtitle
, (const char*) title
,
611 XmNiconName
, (const char*) title
,
615 void wxFrame::SetIcon(const wxIcon
& icon
)
622 if (!icon
.Ok() || !icon
.GetPixmap())
625 XtVaSetValues((Widget
) m_frameShell
, XtNiconPixmap
, icon
.GetPixmap(), NULL
);
628 wxStatusBar
*wxFrame::OnCreateStatusBar(int number
, long style
, wxWindowID id
,
629 const wxString
& name
)
631 wxStatusBar
*statusBar
= NULL
;
633 statusBar
= new wxStatusBar(this, id
, wxPoint(0, 0), wxSize(100, 20),
636 // Set the height according to the font and the border size
637 wxClientDC
dc(statusBar
);
638 dc
.SetFont(statusBar
->GetFont());
641 dc
.GetTextExtent("X", &x
, &y
);
643 int height
= (int)( (y
* 1.1) + 2* statusBar
->GetBorderY());
645 statusBar
->SetSize(-1, -1, 100, height
);
647 statusBar
->SetFieldsCount(number
);
651 wxStatusBar
* wxFrame::CreateStatusBar(int number
, long style
, wxWindowID id
,
652 const wxString
& name
)
654 // Calling CreateStatusBar twice is an error.
655 wxCHECK_MSG( m_frameStatusBar
== NULL
, FALSE
,
656 "recreating status bar in wxFrame" );
658 m_frameStatusBar
= OnCreateStatusBar(number
, style
, id
,
660 if ( m_frameStatusBar
)
663 return m_frameStatusBar
;
669 void wxFrame::SetStatusText(const wxString
& text
, int number
)
671 wxCHECK_RET( m_frameStatusBar
!= NULL
, "no statusbar to set text for" );
673 m_frameStatusBar
->SetStatusText(text
, number
);
676 void wxFrame::SetStatusWidths(int n
, const int widths_field
[])
678 wxCHECK_RET( m_frameStatusBar
!= NULL
, "no statusbar to set widths for" );
680 m_frameStatusBar
->SetStatusWidths(n
, widths_field
);
684 void wxFrame::PositionStatusBar()
686 if (!m_frameStatusBar
)
690 GetClientSize(&w
, &h
);
692 m_frameStatusBar
->GetSize(&sw
, &sh
);
694 // Since we wish the status bar to be directly under the client area,
695 // we use the adjusted sizes without using wxSIZE_NO_ADJUSTMENTS.
696 m_frameStatusBar
->SetSize(0, h
, w
, sh
);
699 WXWidget
wxFrame::GetMenuBarWidget() const
702 return GetMenuBar()->GetMainWidget();
704 return (WXWidget
) NULL
;
707 void wxFrame::SetMenuBar(wxMenuBar
*menuBar
)
711 m_frameMenuBar
= NULL
;
715 // Currently can't set it twice
716 // wxASSERT_MSG( (m_frameMenuBar == (wxMenuBar*) NULL), "Cannot set the menubar more than once");
720 m_frameMenuBar
->DestroyMenuBar();
721 delete m_frameMenuBar
;
724 m_frameMenuBar
= menuBar
;
725 m_frameMenuBar
->CreateMenuBar(this);
730 // Work out max. size
731 wxNode
*node
= GetChildren().First();
736 // Find a child that's a subwindow, but not a dialog box.
737 wxWindow
*win
= (wxWindow
*)node
->Data();
739 if (!win
->IsKindOf(CLASSINFO(wxFrame
)) &&
740 !win
->IsKindOf(CLASSINFO(wxDialog
)))
744 win
->GetSize(&width
, &height
);
745 win
->GetPosition(&x
, &y
);
747 if ((x
+ width
) > max_width
)
748 max_width
= x
+ width
;
749 if ((y
+ height
) > max_height
)
750 max_height
= y
+ height
;
754 SetClientSize(max_width
, max_height
);
757 // Responds to colour changes, and passes event on to children.
758 void wxFrame::OnSysColourChanged(wxSysColourChangedEvent
& event
)
760 SetBackgroundColour(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_APPWORKSPACE
));
763 if ( m_frameStatusBar
)
765 wxSysColourChangedEvent event2
;
766 event2
.SetEventObject( m_frameStatusBar
);
767 m_frameStatusBar
->ProcessEvent(event2
);
770 // Propagate the event to the non-top-level children
771 wxWindow::OnSysColourChanged(event
);
774 // Default resizing behaviour - if only ONE subwindow,
775 // resize to client rectangle size
776 void wxFrame::OnSize(wxSizeEvent
& WXUNUSED(event
))
778 // if we're using constraints - do use them
779 #if wxUSE_CONSTRAINTS
780 if ( GetAutoLayout() ) {
786 // do we have _exactly_ one child?
787 wxWindow
*child
= NULL
;
788 for ( wxNode
*node
= GetChildren().First(); node
; node
= node
->Next() )
790 wxWindow
*win
= (wxWindow
*)node
->Data();
791 if ( !win
->IsKindOf(CLASSINFO(wxFrame
)) &&
792 !win
->IsKindOf(CLASSINFO(wxDialog
)) &&
793 (win
!= GetStatusBar())
795 && (win
!= GetToolBar())
796 #endif // wxUSE_TOOLBAR
800 return; // it's our second subwindow - nothing to do
806 // we have exactly one child - set it's size to fill the whole frame
807 int clientW
, clientH
;
808 GetClientSize(&clientW
, &clientH
);
813 child
->SetSize(x
, y
, clientW
, clientH
);
817 // Default activation behaviour - set the focus for the first child
819 void wxFrame::OnActivate(wxActivateEvent
& event
)
821 if (!event
.GetActive())
824 for(wxNode
*node
= GetChildren().First(); node
; node
= node
->Next())
826 // Find a child that's a subwindow, but not a dialog box.
827 wxWindow
*child
= (wxWindow
*)node
->Data();
828 if (!child
->IsKindOf(CLASSINFO(wxFrame
)) &&
829 !child
->IsKindOf(CLASSINFO(wxDialog
)))
837 // The default implementation for the close window event.
838 // OnClose for backward compatibility.
840 void wxFrame::OnCloseWindow(wxCloseEvent
& WXUNUSED(event
))
846 // I added this function because I got missing symbols when linking
847 // Maybe it should be included on all platforms. But what should it do?
848 void wxFrame::OnIdle(wxIdleEvent
& WXUNUSED(event
))
853 // Destroy the window (delayed, if a managed window)
854 bool wxFrame::Destroy()
856 if (!wxPendingDelete
.Member(this))
857 wxPendingDelete
.Append(this);
861 // Default menu selection behaviour - display a help string
862 void wxFrame::OnMenuHighlight(wxMenuEvent
& event
)
866 if (event
.GetMenuId() == -1)
870 wxMenuBar
*menuBar
= GetMenuBar();
873 wxString
helpString(menuBar
->GetHelpString(event
.GetMenuId()));
874 if (helpString
!= "")
875 SetStatusText(helpString
);
881 wxMenuBar
*wxFrame::GetMenuBar() const
883 return m_frameMenuBar
;
886 void wxFrame::Centre(int direction
)
888 int display_width
, display_height
, width
, height
, x
, y
;
889 wxDisplaySize(&display_width
, &display_height
);
891 GetSize(&width
, &height
);
894 if (direction
& wxHORIZONTAL
)
895 x
= (int)((display_width
- width
)/2);
896 if (direction
& wxVERTICAL
)
897 y
= (int)((display_height
- height
)/2);
899 SetSize(x
, y
, width
, height
);
902 // Call this to simulate a menu command
903 void wxFrame::Command(int id
)
908 void wxFrame::ProcessCommand(int id
)
910 wxCommandEvent
commandEvent(wxEVT_COMMAND_MENU_SELECTED
, id
);
911 commandEvent
.SetInt( id
);
912 commandEvent
.SetEventObject( this );
914 wxMenuBar
*bar
= GetMenuBar() ;
918 /* TODO: check the menu item if required
919 wxMenuItem *item = bar->FindItemForId(id) ;
920 if (item && item->IsCheckable())
922 bar->Check(id,!bar->Checked(id)) ;
926 wxEvtHandler
* evtHandler
= GetEventHandler();
928 evtHandler
->ProcessEvent(commandEvent
);
931 // Checks if there is a toolbar, and returns the first free client position
932 wxPoint
wxFrame::GetClientAreaOrigin() const
939 GetToolBar()->GetSize(& w
, & h
);
941 if (GetToolBar()->GetWindowStyleFlag() & wxTB_VERTICAL
)
950 #endif // wxUSE_TOOLBAR
955 void wxFrame::ScreenToClient(int *x
, int *y
) const
957 wxWindow::ScreenToClient(x
, y
);
959 // We may be faking the client origin.
960 // So a window that's really at (0, 30) may appear
961 // (to wxWin apps) to be at (0, 0).
962 wxPoint
pt(GetClientAreaOrigin());
967 void wxFrame::ClientToScreen(int *x
, int *y
) const
969 // We may be faking the client origin.
970 // So a window that's really at (0, 30) may appear
971 // (to wxWin apps) to be at (0, 0).
972 wxPoint
pt1(GetClientAreaOrigin());
976 wxWindow::ClientToScreen(x
, y
);
980 wxToolBar
* wxFrame::CreateToolBar(long style
, wxWindowID id
, const wxString
& name
)
982 wxCHECK_MSG( m_frameToolBar
== NULL
, FALSE
,
983 "recreating toolbar in wxFrame" );
985 wxToolBar
* toolBar
= OnCreateToolBar(style
, id
, name
);
998 wxToolBar
* wxFrame::OnCreateToolBar(long style
, wxWindowID id
, const wxString
& name
)
1000 return new wxToolBar(this, id
, wxPoint(0, 0), wxSize(100, 24), style
, name
);
1003 void wxFrame::SetToolBar(wxToolBar
*toolbar
)
1005 m_frameToolBar
= toolbar
;
1008 wxToolBar
*wxFrame::GetToolBar() const
1010 return m_frameToolBar
;
1013 void wxFrame::PositionToolBar()
1017 GetClientSize(& cw
, &ch
);
1022 GetToolBar()->GetSize(& tw
, & th
);
1024 if (GetToolBar()->GetWindowStyleFlag() & wxTB_VERTICAL
)
1026 // Use the 'real' position. wxSIZE_NO_ADJUSTMENTS
1027 // means, pretend we don't have toolbar/status bar, so we
1028 // have the original client size.
1029 GetToolBar()->SetSize(0, 0, tw
, ch
+ th
, wxSIZE_NO_ADJUSTMENTS
);
1033 // Use the 'real' position
1034 GetToolBar()->SetSize(0, 0, cw
, th
, wxSIZE_NO_ADJUSTMENTS
);
1038 #endif // wxUSE_TOOLBAR
1040 void wxFrame::CaptureMouse()
1045 if (GetMainWidget())
1046 XtAddGrab((Widget
) m_frameShell
, TRUE
, FALSE
);
1047 m_winCaptured
= TRUE
;
1050 void wxFrame::ReleaseMouse()
1055 if (GetMainWidget())
1056 XtRemoveGrab((Widget
) m_frameShell
);
1057 m_winCaptured
= FALSE
;
1060 void wxFrame::Raise(void)
1062 Window parent_window
= XtWindow((Widget
) m_frameShell
),
1063 next_parent
= XtWindow((Widget
) m_frameShell
),
1064 root
= RootWindowOfScreen(XtScreen((Widget
) m_frameShell
));
1065 // search for the parent that is child of ROOT, because the WM may
1066 // reparent twice and notify only the next parent (like FVWM)
1067 while (next_parent
!= root
) {
1068 Window
*theChildren
; unsigned int n
;
1069 parent_window
= next_parent
;
1070 XQueryTree(XtDisplay((Widget
) m_frameShell
), parent_window
, &root
,
1071 &next_parent
, &theChildren
, &n
);
1072 XFree(theChildren
); // not needed
1074 XRaiseWindow(XtDisplay((Widget
) m_frameShell
), parent_window
);
1077 void wxFrame::Lower(void)
1079 Window parent_window
= XtWindow((Widget
) m_frameShell
),
1080 next_parent
= XtWindow((Widget
) m_frameShell
),
1081 root
= RootWindowOfScreen(XtScreen((Widget
) m_frameShell
));
1082 // search for the parent that is child of ROOT, because the WM may
1083 // reparent twice and notify only the next parent (like FVWM)
1084 while (next_parent
!= root
) {
1085 Window
*theChildren
; unsigned int n
;
1086 parent_window
= next_parent
;
1087 XQueryTree(XtDisplay((Widget
) m_frameShell
), parent_window
, &root
,
1088 &next_parent
, &theChildren
, &n
);
1089 XFree(theChildren
); // not needed
1091 XLowerWindow(XtDisplay((Widget
) m_frameShell
), parent_window
);
1094 void wxFrameFocusProc(Widget
WXUNUSED(workArea
), XtPointer
WXUNUSED(clientData
),
1095 XmAnyCallbackStruct
*WXUNUSED(cbs
))
1097 // wxDebugMsg("focus proc from frame %ld\n",(long)frame);
1099 // wxFrame *frame = (wxFrame *)clientData;
1100 // frame->GetEventHandler()->OnSetFocus();
1103 /* MATTEW: Used to insure that hide-&-show within an event cycle works */
1104 static void wxFrameMapProc(Widget frameShell
, XtPointer clientData
,
1105 XCrossingEvent
* event
)
1107 wxFrame
*frame
= (wxFrame
*)wxGetWindowFromTable((Widget
)clientData
);
1110 XEvent
*e
= (XEvent
*)event
;
1112 if (e
->xany
.type
== MapNotify
)
1115 XtVaSetValues(frameShell
, XmNiconic
, (Boolean
)False
, NULL
);
1116 if (!frame
->GetVisibleStatus())
1118 /* We really wanted this to be hidden! */
1119 XtUnmapWidget((Widget
) frame
->GetShellWidget());
1122 else if (e
->xany
.type
== UnmapNotify
)
1124 XtVaSetValues(frameShell
, XmNiconic
, (Boolean
)True
, NULL
);
1129 bool wxFrame::PreResize()
1133 #endif // wxUSE_TOOLBAR
1134 PositionStatusBar();
1138 WXWidget
wxFrame::GetClientWidget() const
1140 return m_clientArea
;
1143 void wxFrame::ChangeFont(bool WXUNUSED(keepOriginalSize
))
1148 void wxFrame::ChangeBackgroundColour()
1150 if (GetClientWidget())
1151 DoChangeBackgroundColour(GetClientWidget(), m_backgroundColour
);
1154 void wxFrame::ChangeForegroundColour()
1156 if (GetClientWidget())
1157 DoChangeForegroundColour(GetClientWidget(), m_foregroundColour
);
1160 void wxCloseFrameCallback(Widget
WXUNUSED(widget
), XtPointer client_data
, XmAnyCallbackStruct
*WXUNUSED(cbs
))
1162 wxFrame
*frame
= (wxFrame
*)client_data
;
1164 wxCloseEvent
closeEvent(wxEVT_CLOSE_WINDOW
, frame
->GetId());
1165 closeEvent
.SetEventObject(frame
);
1167 // May delete the frame (with delayed deletion)
1168 frame
->GetEventHandler()->ProcessEvent(closeEvent
);