]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/gtk/frame.cpp
GetLabelTop should return the stripped label, for compatibility.
[wxWidgets.git] / src / gtk / frame.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/gtk/frame.cpp
3// Purpose:
4// Author: Robert Roebling
5// Id: $Id$
6// Copyright: (c) 1998 Robert Roebling
7// Licence: wxWindows licence
8/////////////////////////////////////////////////////////////////////////////
9
10// For compilers that support precompilation, includes "wx.h".
11#include "wx/wxprec.h"
12
13#include "wx/frame.h"
14
15#ifndef WX_PRECOMP
16 #include "wx/menu.h"
17 #include "wx/toolbar.h"
18 #include "wx/statusbr.h"
19#endif // WX_PRECOMP
20
21#include <gtk/gtk.h>
22#include "wx/gtk/win_gtk.h"
23
24// ----------------------------------------------------------------------------
25// constants
26// ----------------------------------------------------------------------------
27
28static const int wxSTATUS_HEIGHT = 25;
29static const int wxPLACE_HOLDER = 0;
30
31// ----------------------------------------------------------------------------
32// event tables
33// ----------------------------------------------------------------------------
34
35IMPLEMENT_DYNAMIC_CLASS(wxFrame, wxTopLevelWindow)
36
37// ============================================================================
38// implementation
39// ============================================================================
40
41// ----------------------------------------------------------------------------
42// GTK callbacks
43// ----------------------------------------------------------------------------
44
45#if wxUSE_MENUS_NATIVE
46
47//-----------------------------------------------------------------------------
48// "child_attached" of menu bar
49//-----------------------------------------------------------------------------
50
51extern "C" {
52static void gtk_menu_attached_callback( GtkWidget *WXUNUSED(widget), GtkWidget *WXUNUSED(child), wxFrame *win )
53{
54 if (!win->m_hasVMT) return;
55
56 win->m_menuBarDetached = false;
57 win->GtkUpdateSize();
58}
59}
60
61//-----------------------------------------------------------------------------
62// "child_detached" of menu bar
63//-----------------------------------------------------------------------------
64
65extern "C" {
66static void gtk_menu_detached_callback( GtkWidget *WXUNUSED(widget), GtkWidget *WXUNUSED(child), wxFrame *win )
67{
68 if (!win->m_hasVMT) return;
69
70 // Raise the client area area
71 gdk_window_raise( win->m_wxwindow->window );
72
73 win->m_menuBarDetached = true;
74 win->GtkUpdateSize();
75}
76}
77
78#endif // wxUSE_MENUS_NATIVE
79
80#if wxUSE_TOOLBAR
81//-----------------------------------------------------------------------------
82// "child_attached" of tool bar
83//-----------------------------------------------------------------------------
84
85extern "C" {
86static void gtk_toolbar_attached_callback( GtkWidget *WXUNUSED(widget), GtkWidget *WXUNUSED(child), wxFrame *win )
87{
88 if (!win->m_hasVMT) return;
89
90 win->m_toolBarDetached = false;
91 win->GtkUpdateSize();
92}
93}
94
95//-----------------------------------------------------------------------------
96// "child_detached" of tool bar
97//-----------------------------------------------------------------------------
98
99extern "C" {
100static void gtk_toolbar_detached_callback( GtkWidget *WXUNUSED(widget), GtkWidget *WXUNUSED(child), wxFrame *win )
101{
102 if (!win->m_hasVMT) return;
103
104 // Raise the client area area
105 gdk_window_raise( win->m_wxwindow->window );
106
107 win->m_toolBarDetached = true;
108 win->GtkUpdateSize();
109}
110}
111#endif // wxUSE_TOOLBAR
112
113
114// ----------------------------------------------------------------------------
115// wxFrame itself
116// ----------------------------------------------------------------------------
117
118//-----------------------------------------------------------------------------
119// InsertChild for wxFrame
120//-----------------------------------------------------------------------------
121
122#if wxUSE_TOOLBAR
123
124/* Callback for wxFrame. This very strange beast has to be used because
125 * C++ has no virtual methods in a constructor. We have to emulate a
126 * virtual function here as wxWidgets requires different ways to insert
127 * a child in container classes. */
128
129static void wxInsertChildInFrame(wxWindow* parent, wxWindow* child)
130{
131 wxASSERT( GTK_IS_WIDGET(child->m_widget) );
132
133 // These are outside the client area
134 wxFrame* frame = wx_static_cast(wxFrame*, parent);
135 gtk_pizza_put( GTK_PIZZA(frame->m_mainWidget),
136 child->m_widget,
137 child->m_x,
138 child->m_y,
139 child->m_width,
140 child->m_height );
141
142#if wxUSE_TOOLBAR_NATIVE
143 // We connect to these events for recalculating the client area
144 // space when the toolbar is floating
145 if (wxIS_KIND_OF(child,wxToolBar))
146 {
147 if (child->HasFlag(wxTB_DOCKABLE))
148 {
149 g_signal_connect (child->m_widget, "child_attached",
150 G_CALLBACK (gtk_toolbar_attached_callback),
151 parent);
152 g_signal_connect (child->m_widget, "child_detached",
153 G_CALLBACK (gtk_toolbar_detached_callback),
154 parent);
155 }
156 }
157#endif // wxUSE_TOOLBAR_NATIVE
158}
159
160#endif // wxUSE_TOOLBAR
161
162// ----------------------------------------------------------------------------
163// wxFrame creation
164// ----------------------------------------------------------------------------
165
166void wxFrame::Init()
167{
168 m_menuBarDetached = false;
169 m_toolBarDetached = false;
170 m_menuBarHeight = 2;
171 m_fsSaveFlag = 0;
172}
173
174bool wxFrame::Create( wxWindow *parent,
175 wxWindowID id,
176 const wxString& title,
177 const wxPoint& pos,
178 const wxSize& sizeOrig,
179 long style,
180 const wxString &name )
181{
182 return wxFrameBase::Create(parent, id, title, pos, sizeOrig, style, name);
183}
184
185wxFrame::~wxFrame()
186{
187 m_isBeingDeleted = true;
188 DeleteAllBars();
189}
190
191// ----------------------------------------------------------------------------
192// overridden wxWindow methods
193// ----------------------------------------------------------------------------
194
195void wxFrame::DoGetClientSize( int *width, int *height ) const
196{
197 wxASSERT_MSG( (m_widget != NULL), wxT("invalid frame") );
198
199 wxFrameBase::DoGetClientSize(width, height);
200
201 if (height)
202 {
203#if wxUSE_MENUS_NATIVE
204 // menu bar
205 if (HasVisibleMenubar() && !m_menuBarDetached)
206 {
207 *height -= m_menuBarHeight;
208 }
209#endif // wxUSE_MENUS_NATIVE
210
211#if wxUSE_STATUSBAR
212 // status bar
213 if (m_frameStatusBar && GTK_WIDGET_VISIBLE(m_frameStatusBar->m_widget))
214 *height -= wxSTATUS_HEIGHT;
215#endif // wxUSE_STATUSBAR
216 }
217
218#if wxUSE_TOOLBAR
219 // tool bar
220 if (m_frameToolBar &&
221 GTK_WIDGET_VISIBLE(m_frameToolBar->m_widget) && !m_toolBarDetached)
222 {
223 if (m_frameToolBar->IsVertical())
224 {
225 if (width)
226 *width -= m_frameToolBar->GetSize().x;
227 }
228 else
229 {
230 if (height)
231 *height -= m_frameToolBar->GetSize().y;
232 }
233 }
234#endif // wxUSE_TOOLBAR
235
236 if (width != NULL && *width < 0)
237 *width = 0;
238 if (height != NULL && *height < 0)
239 *height = 0;
240}
241
242bool wxFrame::ShowFullScreen(bool show, long style)
243{
244 if (!wxFrameBase::ShowFullScreen(show, style))
245 return false;
246
247 wxWindow* const bar[] = {
248#if wxUSE_MENUS
249 m_frameMenuBar,
250#else
251 NULL,
252#endif
253#if wxUSE_TOOLBAR
254 m_frameToolBar,
255#else
256 NULL,
257#endif
258#if wxUSE_STATUSBAR
259 m_frameStatusBar,
260#else
261 NULL,
262#endif
263 };
264 const long fsNoBar[] = {
265 wxFULLSCREEN_NOMENUBAR, wxFULLSCREEN_NOTOOLBAR, wxFULLSCREEN_NOSTATUSBAR
266 };
267 for (int i = 0; i < 3; i++)
268 {
269 if (show)
270 {
271 if (bar[i] && (style & fsNoBar[i]))
272 {
273 if (bar[i]->IsShown())
274 bar[i]->Show(false);
275 else
276 style &= ~fsNoBar[i];
277 }
278 }
279 else
280 {
281 if (bar[i] && (m_fsSaveFlag & fsNoBar[i]))
282 bar[i]->Show(true);
283 }
284 }
285 if (show)
286 m_fsSaveFlag = style;
287
288 return true;
289}
290
291void wxFrame::GtkOnSize()
292{
293 // avoid recursions
294 if (m_resizing) return;
295 m_resizing = true;
296
297 // this shouldn't happen: wxFrame, wxMDIParentFrame and wxMDIChildFrame have m_wxwindow
298 wxASSERT_MSG( (m_wxwindow != NULL), wxT("invalid frame") );
299
300 // space occupied by m_frameToolBar and m_frameMenuBar
301 int client_area_x_offset = 0,
302 client_area_y_offset = 0;
303
304 /* wxMDIChildFrame derives from wxFrame but it _is_ a wxWindow as it uses
305 wxWindow::Create to create it's GTK equivalent. m_mainWidget is only
306 set in wxFrame::Create so it is used to check what kind of frame we
307 have here. if m_mainWidget is NULL it is a wxMDIChildFrame and so we
308 skip the part which handles m_frameMenuBar, m_frameToolBar and (most
309 importantly) m_mainWidget */
310
311 ConstrainSize();
312
313 if (m_mainWidget)
314 {
315 // TODO
316 // Rewrite this terrible code to using GtkVBox
317
318 // m_mainWidget holds the menubar, the toolbar and the client
319 // area, which is represented by m_wxwindow.
320
321#if wxUSE_MENUS_NATIVE
322 int menubarHeight = 0;
323#endif
324
325#if wxUSE_MENUS_NATIVE
326 if (HasVisibleMenubar())
327 {
328 int xx = m_miniEdge;
329 int yy = m_miniEdge + m_miniTitle;
330 int ww = m_width - 2*m_miniEdge;
331 if (ww < 0)
332 ww = 0;
333 menubarHeight = m_menuBarHeight;
334 if (m_menuBarDetached) menubarHeight = wxPLACE_HOLDER;
335 m_frameMenuBar->m_x = xx;
336 m_frameMenuBar->m_y = yy;
337 m_frameMenuBar->m_width = ww;
338 m_frameMenuBar->m_height = menubarHeight;
339 gtk_pizza_set_size( GTK_PIZZA(m_mainWidget),
340 m_frameMenuBar->m_widget,
341 xx, yy, ww, menubarHeight);
342 client_area_y_offset += menubarHeight;
343 }
344#endif // wxUSE_MENUS_NATIVE
345
346#if wxUSE_TOOLBAR
347 if ((m_frameToolBar) && m_frameToolBar->IsShown() &&
348 (m_frameToolBar->m_widget->parent == m_mainWidget))
349 {
350 int xx = m_miniEdge;
351 int yy = m_miniEdge + m_miniTitle
352#if wxUSE_MENUS_NATIVE
353 + menubarHeight
354#endif
355 ;
356
357 m_frameToolBar->m_x = xx;
358 m_frameToolBar->m_y = yy;
359
360 // don't change the toolbar's reported height/width
361 int ww, hh;
362 if ( m_frameToolBar->GetWindowStyle() & wxTB_VERTICAL )
363 {
364 ww = m_toolBarDetached ? wxPLACE_HOLDER
365 : m_frameToolBar->m_width;
366 hh = m_height - 2*m_miniEdge;
367
368 client_area_x_offset += ww;
369 }
370 else if( m_frameToolBar->HasFlag(wxTB_RIGHT) )
371 {
372 yy += 2;
373 ww = m_toolBarDetached ? wxPLACE_HOLDER
374 : m_frameToolBar->m_width;
375 xx = GetClientSize().x - 1;
376 hh = m_height - 2*m_miniEdge;
377 if( hh < 0 )
378 hh = 0;
379
380 }
381 else if( m_frameToolBar->GetWindowStyle() & wxTB_BOTTOM )
382 {
383 xx = m_miniEdge;
384 yy = GetClientSize().y;
385#if wxUSE_MENUS_NATIVE
386 yy += m_menuBarHeight;
387#endif // wxUSE_MENUS_NATIVE
388 m_frameToolBar->m_x = xx;
389 m_frameToolBar->m_y = yy;
390 ww = m_width - 2*m_miniEdge;
391 hh = m_toolBarDetached ? wxPLACE_HOLDER
392 : m_frameToolBar->m_height;
393 }
394 else
395 {
396 ww = m_width - 2*m_miniEdge;
397 hh = m_toolBarDetached ? wxPLACE_HOLDER
398 : m_frameToolBar->m_height;
399
400 client_area_y_offset += hh;
401 }
402
403 if (ww < 0)
404 ww = 0;
405 if (hh < 0)
406 hh = 0;
407 gtk_pizza_set_size( GTK_PIZZA(m_mainWidget),
408 m_frameToolBar->m_widget,
409 xx, yy, ww, hh );
410 }
411#endif // wxUSE_TOOLBAR
412
413 int client_x = client_area_x_offset + m_miniEdge;
414 int client_y = client_area_y_offset + m_miniEdge + m_miniTitle;
415 int client_w = m_width - client_area_x_offset - 2*m_miniEdge;
416 int client_h = m_height - client_area_y_offset- 2*m_miniEdge - m_miniTitle;
417 if (client_w < 0)
418 client_w = 0;
419 if (client_h < 0)
420 client_h = 0;
421 gtk_pizza_set_size( GTK_PIZZA(m_mainWidget),
422 m_wxwindow,
423 client_x, client_y, client_w, client_h );
424 }
425 else
426 {
427 // If there is no m_mainWidget between m_widget and m_wxwindow there
428 // is no need to set the size or position of m_wxwindow.
429 }
430
431#if wxUSE_STATUSBAR
432 if (m_frameStatusBar && m_frameStatusBar->IsShown())
433 {
434 int xx = 0 + m_miniEdge;
435 int yy = m_height - wxSTATUS_HEIGHT - m_miniEdge - client_area_y_offset;
436 int ww = m_width - 2*m_miniEdge;
437 if (ww < 0)
438 ww = 0;
439 int hh = wxSTATUS_HEIGHT;
440 m_frameStatusBar->m_x = xx;
441 m_frameStatusBar->m_y = yy;
442 m_frameStatusBar->m_width = ww;
443 m_frameStatusBar->m_height = hh;
444 gtk_pizza_set_size( GTK_PIZZA(m_wxwindow),
445 m_frameStatusBar->m_widget,
446 xx, yy, ww, hh );
447 }
448#endif // wxUSE_STATUSBAR
449
450 m_sizeSet = true;
451
452 // send size event to frame
453 wxSizeEvent event( wxSize(m_width,m_height), GetId() );
454 event.SetEventObject( this );
455 GetEventHandler()->ProcessEvent( event );
456
457#if wxUSE_STATUSBAR
458 // send size event to status bar
459 if (m_frameStatusBar)
460 {
461 wxSizeEvent event2( wxSize(m_frameStatusBar->m_width,m_frameStatusBar->m_height), m_frameStatusBar->GetId() );
462 event2.SetEventObject( m_frameStatusBar );
463 m_frameStatusBar->GetEventHandler()->ProcessEvent( event2 );
464 }
465#endif // wxUSE_STATUSBAR
466
467 m_resizing = false;
468}
469
470void wxFrame::OnInternalIdle()
471{
472 wxFrameBase::OnInternalIdle();
473
474#if wxUSE_MENUS_NATIVE
475 if (m_frameMenuBar) m_frameMenuBar->OnInternalIdle();
476#endif // wxUSE_MENUS_NATIVE
477#if wxUSE_TOOLBAR
478 if (m_frameToolBar) m_frameToolBar->OnInternalIdle();
479#endif
480#if wxUSE_STATUSBAR
481 if (m_frameStatusBar)
482 {
483 m_frameStatusBar->OnInternalIdle();
484
485 // There may be controls in the status bar that
486 // need to be updated
487 for ( wxWindowList::compatibility_iterator node = m_frameStatusBar->GetChildren().GetFirst();
488 node;
489 node = node->GetNext() )
490 {
491 wxWindow *child = node->GetData();
492 child->OnInternalIdle();
493 }
494 }
495#endif
496}
497
498// ----------------------------------------------------------------------------
499// menu/tool/status bar stuff
500// ----------------------------------------------------------------------------
501
502#if wxUSE_MENUS_NATIVE
503
504void wxFrame::DetachMenuBar()
505{
506 wxASSERT_MSG( (m_widget != NULL), wxT("invalid frame") );
507 wxASSERT_MSG( (m_wxwindow != NULL), wxT("invalid frame") );
508
509 if ( m_frameMenuBar )
510 {
511 m_frameMenuBar->UnsetInvokingWindow( this );
512
513 if (m_frameMenuBar->GetWindowStyle() & wxMB_DOCKABLE)
514 {
515 g_signal_handlers_disconnect_by_func (m_frameMenuBar->m_widget,
516 (gpointer) gtk_menu_attached_callback,
517 this);
518
519 g_signal_handlers_disconnect_by_func (m_frameMenuBar->m_widget,
520 (gpointer) gtk_menu_detached_callback,
521 this);
522 }
523
524 gtk_widget_ref( m_frameMenuBar->m_widget );
525
526 gtk_container_remove( GTK_CONTAINER(m_mainWidget), m_frameMenuBar->m_widget );
527 }
528
529 wxFrameBase::DetachMenuBar();
530}
531
532void wxFrame::AttachMenuBar( wxMenuBar *menuBar )
533{
534 wxFrameBase::AttachMenuBar(menuBar);
535
536 if (m_frameMenuBar)
537 {
538 m_frameMenuBar->SetInvokingWindow( this );
539
540 m_frameMenuBar->SetParent(this);
541 gtk_pizza_put( GTK_PIZZA(m_mainWidget),
542 m_frameMenuBar->m_widget,
543 m_frameMenuBar->m_x,
544 m_frameMenuBar->m_y,
545 m_frameMenuBar->m_width,
546 m_frameMenuBar->m_height );
547
548 if (menuBar->GetWindowStyle() & wxMB_DOCKABLE)
549 {
550 g_signal_connect (menuBar->m_widget, "child_attached",
551 G_CALLBACK (gtk_menu_attached_callback),
552 this);
553 g_signal_connect (menuBar->m_widget, "child_detached",
554 G_CALLBACK (gtk_menu_detached_callback),
555 this);
556 }
557
558 gtk_widget_show( m_frameMenuBar->m_widget );
559
560 UpdateMenuBarSize();
561 }
562 else
563 {
564 m_menuBarHeight = 2;
565 GtkUpdateSize(); // resize window in OnInternalIdle
566 }
567}
568
569void wxFrame::UpdateMenuBarSize()
570{
571 m_menuBarHeight = 2;
572
573 // this is called after Remove with a NULL m_frameMenuBar
574 if ( m_frameMenuBar )
575 {
576 GtkRequisition req;
577 gtk_widget_ensure_style(m_frameMenuBar->m_widget);
578 // have to call class method directly because
579 // "size_request" signal is overridden by wx
580 GTK_WIDGET_GET_CLASS(m_frameMenuBar->m_widget)->size_request(
581 m_frameMenuBar->m_widget, &req);
582
583 m_menuBarHeight = req.height;
584 }
585
586 // resize window in OnInternalIdle
587 GtkUpdateSize();
588}
589
590bool wxFrame::HasVisibleMenubar() const
591{
592 return m_frameMenuBar && m_frameMenuBar->IsShown();
593}
594#endif // wxUSE_MENUS_NATIVE
595
596#if wxUSE_TOOLBAR
597
598wxToolBar* wxFrame::CreateToolBar( long style, wxWindowID id, const wxString& name )
599{
600 wxASSERT_MSG( (m_widget != NULL), wxT("invalid frame") );
601
602 InsertChildFunction save = m_insertCallback;
603 m_insertCallback = wxInsertChildInFrame;
604 m_frameToolBar = wxFrameBase::CreateToolBar( style, id, name );
605 m_insertCallback = save;
606
607 GtkUpdateSize();
608
609 return m_frameToolBar;
610}
611
612void wxFrame::SetToolBar(wxToolBar *toolbar)
613{
614 bool hadTbar = m_frameToolBar != NULL;
615
616 wxFrameBase::SetToolBar(toolbar);
617
618 if ( m_frameToolBar )
619 {
620 // insert into toolbar area if not already there
621 if ((m_frameToolBar->m_widget->parent) &&
622 (m_frameToolBar->m_widget->parent != m_mainWidget))
623 {
624 GetChildren().DeleteObject( m_frameToolBar );
625
626 gtk_widget_reparent( m_frameToolBar->m_widget, m_mainWidget );
627 GtkUpdateSize();
628 }
629 }
630 else // toolbar unset
631 {
632 // still need to update size if it had been there before
633 if ( hadTbar )
634 {
635 GtkUpdateSize();
636 }
637 }
638}
639
640#endif // wxUSE_TOOLBAR
641
642#if wxUSE_STATUSBAR
643
644wxStatusBar* wxFrame::CreateStatusBar(int number,
645 long style,
646 wxWindowID id,
647 const wxString& name)
648{
649 wxASSERT_MSG( (m_widget != NULL), wxT("invalid frame") );
650
651 // because it will change when toolbar is added
652 GtkUpdateSize();
653
654 return wxFrameBase::CreateStatusBar( number, style, id, name );
655}
656
657void wxFrame::SetStatusBar(wxStatusBar *statbar)
658{
659 bool hadStatBar = m_frameStatusBar != NULL;
660
661 wxFrameBase::SetStatusBar(statbar);
662
663 if (hadStatBar && !m_frameStatusBar)
664 GtkUpdateSize();
665}
666
667void wxFrame::PositionStatusBar()
668{
669 if ( !m_frameStatusBar )
670 return;
671
672 GtkUpdateSize();
673}
674#endif // wxUSE_STATUSBAR