1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
11 #pragma implementation "frame.h"
15 #include "wx/dialog.h"
16 #include "wx/control.h"
19 #include "wx/toolbar.h"
20 #include "wx/statusbr.h"
21 #include "wx/dcclient.h"
26 #include "wx/gtk/win_gtk.h"
28 //-----------------------------------------------------------------------------
30 //-----------------------------------------------------------------------------
32 const int wxMENU_HEIGHT
= 27;
33 const int wxSTATUS_HEIGHT
= 25;
34 const int wxPLACE_HOLDER
= 0;
36 //-----------------------------------------------------------------------------
38 //-----------------------------------------------------------------------------
40 extern void wxapp_install_idle_handler();
43 //-----------------------------------------------------------------------------
45 //-----------------------------------------------------------------------------
47 extern wxList wxPendingDelete
;
49 //-----------------------------------------------------------------------------
51 //-----------------------------------------------------------------------------
55 extern void debug_focus_in( GtkWidget
* widget
, const wxChar
* name
, const wxChar
*window
);
59 //-----------------------------------------------------------------------------
61 //-----------------------------------------------------------------------------
63 static void gtk_frame_size_callback( GtkWidget
*WXUNUSED(widget
), GtkAllocation
* alloc
, wxFrame
*win
)
66 wxapp_install_idle_handler();
68 if (!win
->m_hasVMT
) return;
70 if ((win
->m_width
!= alloc
->width
) || (win
->m_height
!= alloc
->height
))
72 win
->m_width
= alloc
->width
;
73 win
->m_height
= alloc
->height
;
78 //-----------------------------------------------------------------------------
80 //-----------------------------------------------------------------------------
82 static gint
gtk_frame_delete_callback( GtkWidget
*WXUNUSED(widget
), GdkEvent
*WXUNUSED(event
), wxFrame
*win
)
85 wxapp_install_idle_handler();
92 //-----------------------------------------------------------------------------
93 // "child_attached" of menu bar
94 //-----------------------------------------------------------------------------
96 static void gtk_menu_attached_callback( GtkWidget
*WXUNUSED(widget
), GtkWidget
*WXUNUSED(child
), wxFrame
*win
)
98 if (!win
->m_hasVMT
) return;
100 win
->m_menuBarDetached
= FALSE
;
104 //-----------------------------------------------------------------------------
105 // "child_detached" of menu bar
106 //-----------------------------------------------------------------------------
108 static void gtk_menu_detached_callback( GtkWidget
*WXUNUSED(widget
), GtkWidget
*WXUNUSED(child
), wxFrame
*win
)
110 if (!win
->m_hasVMT
) return;
112 win
->m_menuBarDetached
= TRUE
;
116 //-----------------------------------------------------------------------------
117 // "child_attached" of tool bar
118 //-----------------------------------------------------------------------------
120 static void gtk_toolbar_attached_callback( GtkWidget
*WXUNUSED(widget
), GtkWidget
*WXUNUSED(child
), wxFrame
*win
)
122 if (!win
->m_hasVMT
) return;
124 win
->m_toolBarDetached
= FALSE
;
129 //-----------------------------------------------------------------------------
130 // "child_detached" of tool bar
131 //-----------------------------------------------------------------------------
133 static void gtk_toolbar_detached_callback( GtkWidget
*widget
, GtkWidget
*WXUNUSED(child
), wxFrame
*win
)
135 if (g_isIdle
) wxapp_install_idle_handler();
137 if (!win
->m_hasVMT
) return;
139 win
->m_toolBarDetached
= TRUE
;
143 //-----------------------------------------------------------------------------
145 //-----------------------------------------------------------------------------
147 static gint
gtk_frame_configure_callback( GtkWidget
*WXUNUSED(widget
), GdkEventConfigure
*event
, wxFrame
*win
)
150 wxapp_install_idle_handler();
152 if (!win
->m_hasVMT
) return FALSE
;
157 wxMoveEvent
mevent( wxPoint(win
->m_x
,win
->m_y
), win
->GetId() );
158 mevent
.SetEventObject( win
);
159 win
->GetEventHandler()->ProcessEvent( mevent
);
164 //-----------------------------------------------------------------------------
165 // "realize" from m_widget
166 //-----------------------------------------------------------------------------
168 /* we cannot MWM hints and icons before the widget has been realized,
169 so we do this directly after realization */
172 gtk_frame_realized_callback( GtkWidget
*widget
, wxFrame
*win
)
175 wxapp_install_idle_handler();
177 /* all this is for Motif Window Manager "hints" and is supposed to be
178 recognized by other WM as well. not tested. */
179 long decor
= (long) GDK_DECOR_BORDER
;
180 long func
= (long) GDK_FUNC_MOVE
;
182 if ((win
->GetWindowStyle() & wxCAPTION
) != 0)
183 decor
|= GDK_DECOR_TITLE
;
184 if ((win
->GetWindowStyle() & wxSYSTEM_MENU
) != 0)
186 decor
|= GDK_DECOR_MENU
;
187 func
|= GDK_FUNC_CLOSE
;
189 if ((win
->GetWindowStyle() & wxMINIMIZE_BOX
) != 0)
191 func
|= GDK_FUNC_MINIMIZE
;
192 decor
|= GDK_DECOR_MINIMIZE
;
194 if ((win
->GetWindowStyle() & wxMAXIMIZE_BOX
) != 0)
196 func
|= GDK_FUNC_MAXIMIZE
;
197 decor
|= GDK_DECOR_MAXIMIZE
;
199 if ((win
->GetWindowStyle() & wxRESIZE_BORDER
) != 0)
201 func
|= GDK_FUNC_RESIZE
;
202 decor
|= GDK_DECOR_RESIZEH
;
206 gdk_window_set_decorations( win
->m_widget
->window
, (GdkWMDecoration
)decor
);
207 gdk_window_set_functions( win
->m_widget
->window
, (GdkWMFunction
)func
);
209 /* GTK's shrinking/growing policy */
210 if ((win
->GetWindowStyle() & wxRESIZE_BORDER
) == 0)
211 gtk_window_set_policy(GTK_WINDOW(win
->m_widget
), 0, 0, 1);
213 gtk_window_set_policy(GTK_WINDOW(win
->m_widget
), 1, 1, 1);
216 if (win
->m_icon
!= wxNullIcon
)
218 wxIcon
icon( win
->m_icon
);
219 win
->m_icon
= wxNullIcon
;
220 win
->SetIcon( icon
);
223 /* we set the focus to the child that accepts the focus. this
224 doesn't really have to be done in "realize" but why not? */
225 wxWindowList::Node
*node
= win
->GetChildren().GetFirst();
228 wxWindow
*child
= node
->GetData();
229 if (child
->AcceptsFocus())
235 node
= node
->GetNext();
241 //-----------------------------------------------------------------------------
242 // InsertChild for wxFrame
243 //-----------------------------------------------------------------------------
245 /* Callback for wxFrame. This very strange beast has to be used because
246 * C++ has no virtual methods in a constructor. We have to emulate a
247 * virtual function here as wxWindows requires different ways to insert
248 * a child in container classes. */
250 static void wxInsertChildInFrame( wxWindow
* parent
, wxWindow
* child
)
252 if (wxIS_KIND_OF(child
,wxToolBar
) || wxIS_KIND_OF(child
,wxMenuBar
))
254 /* actually, menubars are never inserted here, but this
255 may change one day */
257 /* these are outside the client area */
258 wxFrame
* frame
= (wxFrame
*) parent
;
259 gtk_myfixed_put( GTK_MYFIXED(frame
->m_mainWidget
),
260 GTK_WIDGET(child
->m_widget
),
266 /* we connect to these events for recalculating the client area
267 space when the toolbar is floating */
268 if (wxIS_KIND_OF(child
,wxToolBar
))
270 wxToolBar
*toolBar
= (wxToolBar
*) child
;
271 if (toolBar
->GetWindowStyle() & wxTB_DOCKABLE
)
273 gtk_signal_connect( GTK_OBJECT(toolBar
->m_widget
), "child_attached",
274 GTK_SIGNAL_FUNC(gtk_toolbar_attached_callback
), (gpointer
)parent
);
276 gtk_signal_connect( GTK_OBJECT(toolBar
->m_widget
), "child_detached",
277 GTK_SIGNAL_FUNC(gtk_toolbar_detached_callback
), (gpointer
)parent
);
283 /* these are inside the client area */
284 gtk_myfixed_put( GTK_MYFIXED(parent
->m_wxwindow
),
285 GTK_WIDGET(child
->m_widget
),
292 /* resize on OnInternalIdle */
293 parent
->UpdateSize();
296 //-----------------------------------------------------------------------------
298 //-----------------------------------------------------------------------------
300 BEGIN_EVENT_TABLE(wxFrame
, wxWindow
)
301 EVT_SIZE(wxFrame::OnSize
)
302 EVT_CLOSE(wxFrame::OnCloseWindow
)
303 EVT_MENU_HIGHLIGHT_ALL(wxFrame::OnMenuHighlight
)
306 IMPLEMENT_DYNAMIC_CLASS(wxFrame
,wxWindow
)
310 m_frameMenuBar
= (wxMenuBar
*) NULL
;
311 m_frameStatusBar
= (wxStatusBar
*) NULL
;
312 m_frameToolBar
= (wxToolBar
*) NULL
;
316 m_mainWidget
= (GtkWidget
*) NULL
;
317 m_menuBarDetached
= FALSE
;
318 m_toolBarDetached
= FALSE
;
319 m_insertCallback
= wxInsertChildInFrame
;
322 wxFrame::wxFrame( wxWindow
*parent
, wxWindowID id
, const wxString
&title
,
323 const wxPoint
&pos
, const wxSize
&size
,
324 long style
, const wxString
&name
)
326 m_frameMenuBar
= (wxMenuBar
*) NULL
;
327 m_frameStatusBar
= (wxStatusBar
*) NULL
;
328 m_frameToolBar
= (wxToolBar
*) NULL
;
332 m_mainWidget
= (GtkWidget
*) NULL
;
333 m_menuBarDetached
= FALSE
;
334 m_toolBarDetached
= FALSE
;
335 m_insertCallback
= wxInsertChildInFrame
;
336 Create( parent
, id
, title
, pos
, size
, style
, name
);
339 bool wxFrame::Create( wxWindow
*parent
, wxWindowID id
, const wxString
&title
,
340 const wxPoint
&pos
, const wxSize
&size
,
341 long style
, const wxString
&name
)
343 wxTopLevelWindows
.Append( this );
345 m_needParent
= FALSE
;
347 PreCreation( parent
, id
, pos
, size
, style
, name
);
351 m_insertCallback
= wxInsertChildInFrame
;
353 GtkWindowType win_type
= GTK_WINDOW_TOPLEVEL
;
354 if (style
& wxSIMPLE_BORDER
) win_type
= GTK_WINDOW_POPUP
;
356 m_widget
= gtk_window_new( win_type
);
359 gtk_window_set_wmclass( GTK_WINDOW(m_widget
), name
.mb_str(), name
.mb_str() );
362 debug_focus_in( m_widget
, _T("wxFrame::m_widget"), name
);
365 gtk_window_set_title( GTK_WINDOW(m_widget
), title
.mbc_str() );
366 GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_FOCUS
);
368 gtk_signal_connect( GTK_OBJECT(m_widget
), "delete_event",
369 GTK_SIGNAL_FUNC(gtk_frame_delete_callback
), (gpointer
)this );
371 /* m_mainWidget holds the toolbar, the menubar and the client area */
372 m_mainWidget
= gtk_myfixed_new();
373 gtk_widget_show( m_mainWidget
);
374 GTK_WIDGET_UNSET_FLAGS( m_mainWidget
, GTK_CAN_FOCUS
);
375 gtk_container_add( GTK_CONTAINER(m_widget
), m_mainWidget
);
378 debug_focus_in( m_mainWidget
, _T("wxFrame::m_mainWidget"), name
);
381 /* m_wxwindow only represents the client area without toolbar and menubar */
382 m_wxwindow
= gtk_myfixed_new();
383 gtk_widget_show( m_wxwindow
);
384 gtk_container_add( GTK_CONTAINER(m_mainWidget
), m_wxwindow
);
387 debug_focus_in( m_wxwindow
, _T("wxFrame::m_wxwindow"), name
);
390 /* we donm't allow the frame to get the focus as otherwise
391 the frame will grabit at arbitrary fcous changes. */
392 GTK_WIDGET_UNSET_FLAGS( m_wxwindow
, GTK_CAN_FOCUS
);
394 if (m_parent
) m_parent
->AddChild( this );
398 /* we cannot set MWM hints and icons before the widget has
399 been realized, so we do this directly after realization */
400 gtk_signal_connect( GTK_OBJECT(m_widget
), "realize",
401 GTK_SIGNAL_FUNC(gtk_frame_realized_callback
), (gpointer
) this );
403 /* the user resized the frame by dragging etc. */
404 gtk_signal_connect( GTK_OBJECT(m_widget
), "size_allocate",
405 GTK_SIGNAL_FUNC(gtk_frame_size_callback
), (gpointer
)this );
407 /* the only way to get the window size is to connect to this event */
408 gtk_signal_connect( GTK_OBJECT(m_widget
), "configure_event",
409 GTK_SIGNAL_FUNC(gtk_frame_configure_callback
), (gpointer
)this );
416 if (m_frameMenuBar
) delete m_frameMenuBar
;
417 m_frameMenuBar
= (wxMenuBar
*) NULL
;
419 if (m_frameStatusBar
) delete m_frameStatusBar
;
420 m_frameStatusBar
= (wxStatusBar
*) NULL
;
422 if (m_frameToolBar
) delete m_frameToolBar
;
423 m_frameToolBar
= (wxToolBar
*) NULL
;
425 wxTopLevelWindows
.DeleteObject( this );
427 if (wxTheApp
->GetTopWindow() == this)
428 wxTheApp
->SetTopWindow( (wxWindow
*) NULL
);
430 if (wxTopLevelWindows
.Number() == 0)
431 wxTheApp
->ExitMainLoop();
434 bool wxFrame::Show( bool show
)
436 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
438 if (show
&& !m_sizeSet
)
440 /* by calling GtkOnSize here, we don't have to call
441 either after showing the frame, which would entail
442 much ugly flicker or from within the size_allocate
443 handler, because GTK 1.1.X forbids that. */
445 GtkOnSize( m_x
, m_y
, m_width
, m_height
);
448 return wxWindow::Show( show
);
451 bool wxFrame::Destroy()
453 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
455 if (!wxPendingDelete
.Member(this)) wxPendingDelete
.Append(this);
460 void wxFrame::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags
)
462 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
464 /* this shouldn't happen: wxFrame, wxMDIParentFrame and wxMDIChildFrame have m_wxwindow */
465 wxASSERT_MSG( (m_wxwindow
!= NULL
), _T("invalid frame") );
467 /* avoid recursions */
468 if (m_resizing
) return;
473 int old_width
= m_width
;
474 int old_height
= m_height
;
476 if ((sizeFlags
& wxSIZE_USE_EXISTING
) == wxSIZE_USE_EXISTING
)
478 if (x
!= -1) m_x
= x
;
479 if (y
!= -1) m_y
= y
;
480 if (width
!= -1) m_width
= width
;
481 if (height
!= -1) m_height
= height
;
491 if ((sizeFlags
& wxSIZE_AUTO_WIDTH
) == wxSIZE_AUTO_WIDTH
)
493 if (width
== -1) m_width
= 80;
496 if ((sizeFlags
& wxSIZE_AUTO_HEIGHT
) == wxSIZE_AUTO_HEIGHT
)
498 if (height
== -1) m_height
= 26;
501 if ((m_minWidth
!= -1) && (m_width
< m_minWidth
)) m_width
= m_minWidth
;
502 if ((m_minHeight
!= -1) && (m_height
< m_minHeight
)) m_height
= m_minHeight
;
503 if ((m_maxWidth
!= -1) && (m_width
> m_maxWidth
)) m_width
= m_maxWidth
;
504 if ((m_maxHeight
!= -1) && (m_height
> m_maxHeight
)) m_height
= m_maxHeight
;
506 if ((m_x
!= -1) || (m_y
!= -1))
508 if ((m_x
!= old_x
) || (m_y
!= old_y
))
510 /* we set the size here and in gtk_frame_map_callback */
511 gtk_widget_set_uposition( m_widget
, m_x
, m_y
);
515 if ((m_width
!= old_width
) || (m_height
!= old_height
))
517 /* we set the size in GtkOnSize, i.e. mostly the actual resizing is
518 done either directly before the frame is shown or in idle time
519 so that different calls to SetSize() don't lead to flicker. */
526 void wxFrame::Centre( int direction
)
528 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
533 if ((direction
& wxHORIZONTAL
) == wxHORIZONTAL
) x
= (gdk_screen_width () - m_width
) / 2;
534 if ((direction
& wxVERTICAL
) == wxVERTICAL
) y
= (gdk_screen_height () - m_height
) / 2;
539 void wxFrame::GetClientSize( int *width
, int *height
) const
541 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
543 wxWindow::GetClientSize( width
, height
);
549 if (!m_menuBarDetached
)
550 (*height
) -= wxMENU_HEIGHT
;
552 (*height
) -= wxPLACE_HOLDER
;
556 if (m_frameStatusBar
) (*height
) -= wxSTATUS_HEIGHT
;
561 if (!m_toolBarDetached
)
564 m_frameToolBar
->GetSize( (int *) NULL
, &y
);
568 (*height
) -= wxPLACE_HOLDER
;
572 (*height
) -= m_miniEdge
*2 + m_miniTitle
;
576 (*width
) -= m_miniEdge
*2;
580 void wxFrame::DoSetClientSize( int width
, int height
)
582 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
587 if (!m_menuBarDetached
)
588 height
+= wxMENU_HEIGHT
;
590 height
+= wxPLACE_HOLDER
;
594 if (m_frameStatusBar
) height
+= wxSTATUS_HEIGHT
;
599 if (!m_toolBarDetached
)
602 m_frameToolBar
->GetSize( (int *) NULL
, &y
);
606 height
+= wxPLACE_HOLDER
;
609 wxWindow::DoSetClientSize( width
+ m_miniEdge
*2, height
+ m_miniEdge
*2 + m_miniTitle
);
612 void wxFrame::GtkOnSize( int WXUNUSED(x
), int WXUNUSED(y
), int width
, int height
)
614 // due to a bug in gtk, x,y are always 0
618 /* avoid recursions */
619 if (m_resizing
) return;
622 /* this shouldn't happen: wxFrame, wxMDIParentFrame and wxMDIChildFrame have m_wxwindow */
623 wxASSERT_MSG( (m_wxwindow
!= NULL
), _T("invalid frame") );
628 /* space occupied by m_frameToolBar and m_frameMenuBar */
629 int client_area_y_offset
= 0;
631 /* wxMDIChildFrame derives from wxFrame but it _is_ a wxWindow as it uses
632 wxWindow::Create to create it's GTK equivalent. m_mainWidget is only
633 set in wxFrame::Create so it is used to check what kind of frame we
634 have here. if m_mainWidget is NULL it is a wxMDIChildFrame and so we
635 skip the part which handles m_frameMenuBar, m_frameToolBar and (most
636 importantly) m_mainWidget */
640 /* check if size is in legal range */
641 if ((m_minWidth
!= -1) && (m_width
< m_minWidth
)) m_width
= m_minWidth
;
642 if ((m_minHeight
!= -1) && (m_height
< m_minHeight
)) m_height
= m_minHeight
;
643 if ((m_maxWidth
!= -1) && (m_width
> m_maxWidth
)) m_width
= m_maxWidth
;
644 if ((m_maxHeight
!= -1) && (m_height
> m_maxHeight
)) m_height
= m_maxHeight
;
646 /* I revert back to wxGTK's original behaviour. m_mainWidget holds the
647 * menubar, the toolbar and the client area, which is represented by
649 * this hurts in the eye, but I don't want to call SetSize()
650 * because I don't want to call any non-native functions here. */
655 int yy
= m_miniEdge
+ m_miniTitle
;
656 int ww
= m_width
- 2*m_miniEdge
;
657 int hh
= wxMENU_HEIGHT
;
658 if (m_menuBarDetached
) hh
= wxPLACE_HOLDER
;
659 m_frameMenuBar
->m_x
= xx
;
660 m_frameMenuBar
->m_y
= yy
;
661 m_frameMenuBar
->m_width
= ww
;
662 m_frameMenuBar
->m_height
= hh
;
663 gtk_myfixed_set_size( GTK_MYFIXED(m_mainWidget
),
664 m_frameMenuBar
->m_widget
,
666 client_area_y_offset
+= hh
;
672 int yy
= m_miniEdge
+ m_miniTitle
;
675 if (!m_menuBarDetached
)
678 yy
+= wxPLACE_HOLDER
;
680 int ww
= m_width
- 2*m_miniEdge
;
681 int hh
= m_frameToolBar
->m_height
;
682 if (m_toolBarDetached
) hh
= wxPLACE_HOLDER
;
683 m_frameToolBar
->m_x
= xx
;
684 m_frameToolBar
->m_y
= yy
;
685 /* m_frameToolBar->m_height = hh; don't change the toolbar's height */
686 m_frameToolBar
->m_width
= ww
;
687 gtk_myfixed_set_size( GTK_MYFIXED(m_mainWidget
),
688 m_frameToolBar
->m_widget
,
690 client_area_y_offset
+= hh
;
693 int client_x
= m_miniEdge
;
694 int client_y
= client_area_y_offset
+ m_miniEdge
+ m_miniTitle
;
695 int client_w
= m_width
- 2*m_miniEdge
;
696 int client_h
= m_height
- client_area_y_offset
- 2*m_miniEdge
- m_miniTitle
;
697 gtk_myfixed_set_size( GTK_MYFIXED(m_mainWidget
),
699 client_x
, client_y
, client_w
, client_h
);
703 /* if there is no m_mainWidget between m_widget and m_wxwindow there
704 is no need to set the size or position of m_wxwindow. */
707 if (m_frameStatusBar
)
709 int xx
= 0 + m_miniEdge
;
710 int yy
= m_height
- wxSTATUS_HEIGHT
- m_miniEdge
- client_area_y_offset
;
711 int ww
= m_width
- 2*m_miniEdge
;
712 int hh
= wxSTATUS_HEIGHT
;
713 m_frameStatusBar
->m_x
= xx
;
714 m_frameStatusBar
->m_y
= yy
;
715 m_frameStatusBar
->m_width
= ww
;
716 m_frameStatusBar
->m_height
= hh
;
717 gtk_myfixed_set_size( GTK_MYFIXED(m_wxwindow
),
718 m_frameStatusBar
->m_widget
,
722 /* we actually set the size of a frame here and no-where else */
723 gtk_widget_set_usize( m_widget
, m_width
, m_height
);
727 /* send size event to frame */
728 wxSizeEvent
event( wxSize(m_width
,m_height
), GetId() );
729 event
.SetEventObject( this );
730 GetEventHandler()->ProcessEvent( event
);
732 /* send size event to status bar */
733 if (m_frameStatusBar
)
735 wxSizeEvent
event2( wxSize(m_frameStatusBar
->m_width
,m_frameStatusBar
->m_height
), m_frameStatusBar
->GetId() );
736 event2
.SetEventObject( m_frameStatusBar
);
737 m_frameStatusBar
->GetEventHandler()->ProcessEvent( event2
);
743 void wxFrame::OnInternalIdle()
745 if (!m_sizeSet
&& GTK_WIDGET_REALIZED(m_wxwindow
))
746 GtkOnSize( m_x
, m_y
, m_width
, m_height
);
751 void wxFrame::OnCloseWindow( wxCloseEvent
& event
)
756 void wxFrame::OnSize( wxSizeEvent
&WXUNUSED(event
) )
758 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
766 /* do we have exactly one child? */
767 wxWindow
*child
= (wxWindow
*)NULL
;
768 for ( wxNode
*node
= GetChildren().First(); node
; node
= node
->Next() )
770 wxWindow
*win
= (wxWindow
*)node
->Data();
771 if ( !wxIS_KIND_OF(win
,wxFrame
) && !wxIS_KIND_OF(win
,wxDialog
) )
775 /* it's the second one: do nothing */
783 /* no children at all? */
786 /* yes: set it's size to fill all the frame */
787 int client_x
, client_y
;
788 GetClientSize( &client_x
, &client_y
);
789 child
->SetSize( 1, 1, client_x
-2, client_y
-2 );
794 static void SetInvokingWindow( wxMenu
*menu
, wxWindow
*win
)
796 menu
->SetInvokingWindow( win
);
798 #if (GTK_MINOR_VERSION > 0)
799 /* support for native hot keys */
800 gtk_accel_group_attach( menu
->m_accel
, GTK_OBJECT(win
->m_widget
));
803 wxNode
*node
= menu
->GetItems().First();
806 wxMenuItem
*menuitem
= (wxMenuItem
*)node
->Data();
807 if (menuitem
->IsSubMenu())
808 SetInvokingWindow( menuitem
->GetSubMenu(), win
);
813 void wxFrame::SetMenuBar( wxMenuBar
*menuBar
)
815 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
816 wxASSERT_MSG( (m_wxwindow
!= NULL
), _T("invalid frame") );
818 m_frameMenuBar
= menuBar
;
822 #if (GTK_MINOR_VERSION > 0) && (GTK_MICRO_VERSION > 0)
823 /* support for native key accelerators indicated by underscroes */
824 gtk_accel_group_attach( m_frameMenuBar
->m_accel
, GTK_OBJECT(m_widget
));
827 wxNode
*node
= m_frameMenuBar
->GetMenus().First();
830 wxMenu
*menu
= (wxMenu
*)node
->Data();
831 SetInvokingWindow( menu
, this );
835 if (m_frameMenuBar
->GetParent() != this)
837 m_frameMenuBar
->SetParent(this);
838 gtk_myfixed_put( GTK_MYFIXED(m_mainWidget
),
839 m_frameMenuBar
->m_widget
,
842 m_frameMenuBar
->m_width
,
843 m_frameMenuBar
->m_height
);
845 if (menuBar
->GetWindowStyle() & wxMB_DOCKABLE
)
847 gtk_signal_connect( GTK_OBJECT(menuBar
->m_widget
), "child_attached",
848 GTK_SIGNAL_FUNC(gtk_menu_attached_callback
), (gpointer
)this );
850 gtk_signal_connect( GTK_OBJECT(menuBar
->m_widget
), "child_detached",
851 GTK_SIGNAL_FUNC(gtk_menu_detached_callback
), (gpointer
)this );
856 /* resize window in OnInternalIdle */
860 wxMenuBar
*wxFrame::GetMenuBar() const
862 return m_frameMenuBar
;
865 void wxFrame::OnMenuHighlight(wxMenuEvent
& event
)
869 // if no help string found, we will clear the status bar text
872 int menuId
= event
.GetMenuId();
875 wxMenuBar
*menuBar
= GetMenuBar();
878 helpString
= menuBar
->GetHelpString(menuId
);
882 SetStatusText(helpString
);
886 wxToolBar
* wxFrame::CreateToolBar(long style
, wxWindowID id
, const wxString
& name
)
888 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
890 wxCHECK_MSG( m_frameToolBar
== NULL
, FALSE
, _T("recreating toolbar in wxFrame") );
892 m_frameToolBar
= OnCreateToolBar( style
, id
, name
);
894 GetChildren().DeleteObject( m_frameToolBar
);
898 return m_frameToolBar
;
901 wxToolBar
* wxFrame::OnCreateToolBar( long style
, wxWindowID id
, const wxString
& name
)
903 return new wxToolBar( this, id
, wxDefaultPosition
, wxDefaultSize
, style
, name
);
906 wxToolBar
*wxFrame::GetToolBar() const
908 return m_frameToolBar
;
911 wxStatusBar
* wxFrame::CreateStatusBar( int number
, long style
, wxWindowID id
, const wxString
& name
)
913 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
915 wxCHECK_MSG( m_frameStatusBar
== NULL
, FALSE
, _T("recreating status bar in wxFrame") );
917 m_frameStatusBar
= OnCreateStatusBar( number
, style
, id
, name
);
921 return m_frameStatusBar
;
924 wxStatusBar
*wxFrame::OnCreateStatusBar( int number
, long style
, wxWindowID id
, const wxString
& name
)
926 wxStatusBar
*statusBar
= (wxStatusBar
*) NULL
;
928 statusBar
= new wxStatusBar(this, id
, wxPoint(0, 0), wxSize(100, 20), style
, name
);
930 // Set the height according to the font and the border size
931 wxClientDC
dc(statusBar
);
932 dc
.SetFont( statusBar
->GetFont() );
935 dc
.GetTextExtent( "X", &x
, &y
);
937 int height
= (int)( (y
* 1.1) + 2* statusBar
->GetBorderY());
939 statusBar
->SetSize( -1, -1, 100, height
);
941 statusBar
->SetFieldsCount( number
);
945 void wxFrame::Command( int id
)
947 wxCommandEvent
commandEvent(wxEVT_COMMAND_MENU_SELECTED
, id
);
948 commandEvent
.SetInt( id
);
949 commandEvent
.SetEventObject( this );
951 wxMenuBar
*bar
= GetMenuBar();
954 wxMenuItem
*item
= bar
->FindItemForId(id
) ;
955 if (item
&& item
->IsCheckable())
957 bar
->Check(id
,!bar
->Checked(id
)) ;
960 wxEvtHandler
* evtHandler
= GetEventHandler();
962 evtHandler
->ProcessEvent(commandEvent
);
965 void wxFrame::SetStatusText(const wxString
& text
, int number
)
967 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
969 wxCHECK_RET( m_frameStatusBar
!= NULL
, _T("no statusbar to set text for") );
971 m_frameStatusBar
->SetStatusText(text
, number
);
974 void wxFrame::SetStatusWidths(int n
, const int widths_field
[] )
976 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
978 wxCHECK_RET( m_frameStatusBar
!= NULL
, _T("no statusbar to set widths for") );
980 m_frameStatusBar
->SetStatusWidths(n
, widths_field
);
983 wxStatusBar
*wxFrame::GetStatusBar() const
985 return m_frameStatusBar
;
988 void wxFrame::SetTitle( const wxString
&title
)
990 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
993 if (m_title
.IsNull()) m_title
= _T("");
994 gtk_window_set_title( GTK_WINDOW(m_widget
), title
.mbc_str() );
997 void wxFrame::SetIcon( const wxIcon
&icon
)
999 wxASSERT_MSG( (m_widget
!= NULL
), _T("invalid frame") );
1002 if (!icon
.Ok()) return;
1004 if (!m_widget
->window
) return;
1006 wxMask
*mask
= icon
.GetMask();
1007 GdkBitmap
*bm
= (GdkBitmap
*) NULL
;
1008 if (mask
) bm
= mask
->GetBitmap();
1010 gdk_window_set_icon( m_widget
->window
, (GdkWindow
*) NULL
, icon
.GetPixmap(), bm
);