wxXmlNode::GetAttribute's pointer argument must not be NULL, check for it
[wxWidgets.git] / src / common / framecmn.cpp
CommitLineData
63fec618 1/////////////////////////////////////////////////////////////////////////////
76b49cf4 2// Name: src/common/framecmn.cpp
63fec618
VZ
3// Purpose: common (for all platforms) wxFrame functions
4// Author: Julian Smart, Vadim Zeitlin
5// Created: 01/02/97
439b3bf1 6// Id: $Id$
55d99c7a 7// Copyright: (c) 1998 Robert Roebling and Julian Smart
65571936 8// Licence: wxWindows licence
63fec618
VZ
9/////////////////////////////////////////////////////////////////////////////
10
7c0ea335
VZ
11// ============================================================================
12// declarations
13// ============================================================================
14
15// ----------------------------------------------------------------------------
16// headers
17// ----------------------------------------------------------------------------
18
f701d7ab
JS
19// For compilers that support precompilation, includes "wx.h".
20#include "wx/wxprec.h"
21
22#ifdef __BORLANDC__
7c0ea335 23 #pragma hdrstop
f701d7ab
JS
24#endif
25
76b49cf4
WS
26#include "wx/frame.h"
27
1e6feb95 28#ifndef WX_PRECOMP
1e6feb95
VZ
29 #include "wx/menu.h"
30 #include "wx/menuitem.h"
31 #include "wx/dcclient.h"
4e3e485b 32 #include "wx/toolbar.h"
7c0ea335 33 #include "wx/statusbr.h"
3304646d 34#endif // WX_PRECOMP
7c0ea335
VZ
35
36// ----------------------------------------------------------------------------
37// event table
38// ----------------------------------------------------------------------------
39
96ac065f
VZ
40#if wxUSE_MENUS && wxUSE_STATUSBAR
41
7d9f12f3 42BEGIN_EVENT_TABLE(wxFrameBase, wxTopLevelWindow)
0b30bb0b 43 EVT_MENU_OPEN(wxFrameBase::OnMenuOpen)
96ac065f
VZ
44 EVT_MENU_CLOSE(wxFrameBase::OnMenuClose)
45
7c0ea335 46 EVT_MENU_HIGHLIGHT_ALL(wxFrameBase::OnMenuHighlight)
7c0ea335
VZ
47END_EVENT_TABLE()
48
96ac065f
VZ
49#endif // wxUSE_MENUS && wxUSE_STATUSBAR
50
7c0ea335
VZ
51// ============================================================================
52// implementation
53// ============================================================================
54
55// ----------------------------------------------------------------------------
56// construction/destruction
57// ----------------------------------------------------------------------------
58
59wxFrameBase::wxFrameBase()
60{
1e6feb95 61#if wxUSE_MENUS
7c0ea335 62 m_frameMenuBar = NULL;
1e6feb95 63#endif // wxUSE_MENUS
7c0ea335
VZ
64
65#if wxUSE_TOOLBAR
66 m_frameToolBar = NULL;
67#endif // wxUSE_TOOLBAR
68
69#if wxUSE_STATUSBAR
70 m_frameStatusBar = NULL;
71#endif // wxUSE_STATUSBAR
1f361cdd
MB
72
73 m_statusBarPane = 0;
7c0ea335
VZ
74}
75
799ea011
GD
76wxFrameBase::~wxFrameBase()
77{
78 // this destructor is required for Darwin
79}
80
7c0ea335
VZ
81wxFrame *wxFrameBase::New(wxWindow *parent,
82 wxWindowID id,
83 const wxString& title,
84 const wxPoint& pos,
85 const wxSize& size,
86 long style,
87 const wxString& name)
88{
89 return new wxFrame(parent, id, title, pos, size, style, name);
90}
91
92void wxFrameBase::DeleteAllBars()
93{
1e6feb95 94#if wxUSE_MENUS
7c0ea335
VZ
95 if ( m_frameMenuBar )
96 {
97 delete m_frameMenuBar;
98 m_frameMenuBar = (wxMenuBar *) NULL;
99 }
1e6feb95 100#endif // wxUSE_MENUS
7c0ea335
VZ
101
102#if wxUSE_STATUSBAR
103 if ( m_frameStatusBar )
104 {
105 delete m_frameStatusBar;
106 m_frameStatusBar = (wxStatusBar *) NULL;
107 }
108#endif // wxUSE_STATUSBAR
109
110#if wxUSE_TOOLBAR
111 if ( m_frameToolBar )
112 {
113 delete m_frameToolBar;
114 m_frameToolBar = (wxToolBar *) NULL;
115 }
116#endif // wxUSE_TOOLBAR
117}
118
1e6feb95
VZ
119bool wxFrameBase::IsOneOfBars(const wxWindow *win) const
120{
121#if wxUSE_MENUS
122 if ( win == GetMenuBar() )
d1b20379 123 return true;
1e6feb95
VZ
124#endif // wxUSE_MENUS
125
126#if wxUSE_STATUSBAR
127 if ( win == GetStatusBar() )
d1b20379 128 return true;
1e6feb95
VZ
129#endif // wxUSE_STATUSBAR
130
131#if wxUSE_TOOLBAR
132 if ( win == GetToolBar() )
d1b20379 133 return true;
1e6feb95
VZ
134#endif // wxUSE_TOOLBAR
135
d1b20379 136 return false;
1e6feb95
VZ
137}
138
1c4f8f8d
VZ
139// ----------------------------------------------------------------------------
140// wxFrame size management: we exclude the areas taken by menu/status/toolbars
141// from the client area, so the client area is what's really available for the
142// frame contents
143// ----------------------------------------------------------------------------
144
145// get the origin of the client area in the client coordinates
146wxPoint wxFrameBase::GetClientAreaOrigin() const
147{
7d9f12f3 148 wxPoint pt = wxTopLevelWindow::GetClientAreaOrigin();
1c4f8f8d 149
a9928e9d 150#if wxUSE_TOOLBAR && !defined(__WXUNIVERSAL__)
d4597e13
VZ
151 wxToolBar *toolbar = GetToolBar();
152 if ( toolbar && toolbar->IsShown() )
1c4f8f8d
VZ
153 {
154 int w, h;
d4597e13 155 toolbar->GetSize(&w, &h);
1c4f8f8d 156
d4597e13 157 if ( toolbar->GetWindowStyleFlag() & wxTB_VERTICAL )
1c4f8f8d
VZ
158 {
159 pt.x += w;
160 }
161 else
162 {
163 pt.y += h;
164 }
165 }
166#endif // wxUSE_TOOLBAR
167
168 return pt;
169}
170
2b022169
RD
171
172void wxFrameBase::SendSizeEvent()
173{
174 wxSizeEvent event( GetSize(), GetId() );
175 event.SetEventObject( this );
176 GetEventHandler()->AddPendingEvent( event );
177}
178
179
7c0ea335
VZ
180// ----------------------------------------------------------------------------
181// misc
182// ----------------------------------------------------------------------------
183
7c0ea335
VZ
184bool wxFrameBase::ProcessCommand(int id)
185{
1e6feb95 186#if wxUSE_MENUS
7c0ea335
VZ
187 wxMenuBar *bar = GetMenuBar();
188 if ( !bar )
d1b20379 189 return false;
7c0ea335 190
3ca6a5f0
BP
191 wxCommandEvent commandEvent(wxEVT_COMMAND_MENU_SELECTED, id);
192 commandEvent.SetEventObject(this);
193
7c0ea335 194 wxMenuItem *item = bar->FindItem(id);
84f7908b 195 if (item)
7c0ea335 196 {
84f7908b 197 if (!item->IsEnabled())
d1b20379 198 return true;
d4597e13 199
26c36d75
WS
200 if ((item->GetKind() == wxITEM_RADIO) && item->IsChecked() )
201 return true;
202
84f7908b
RR
203 if (item->IsCheckable())
204 {
205 item->Toggle();
0472ece7 206
84f7908b
RR
207 // use the new value
208 commandEvent.SetInt(item->IsChecked());
209 }
3ca6a5f0 210 }
7c0ea335 211
a01fe3d6 212 GetEventHandler()->ProcessEvent(commandEvent);
d1b20379 213 return true;
1e6feb95 214#else // !wxUSE_MENUS
d1b20379 215 return false;
1e6feb95 216#endif // wxUSE_MENUS/!wxUSE_MENUS
7c0ea335
VZ
217}
218
e39af974
JS
219// Do the UI update processing for this window. This is
220// provided for the application to call if it wants to
221// force a UI update, particularly for the menus and toolbar.
222void wxFrameBase::UpdateWindowUI(long flags)
223{
224 wxWindowBase::UpdateWindowUI(flags);
a62848fd 225
e39af974
JS
226#if wxUSE_TOOLBAR
227 if (GetToolBar())
228 GetToolBar()->UpdateWindowUI(flags);
229#endif
230
231#if wxUSE_MENUS
232 if (GetMenuBar())
233 {
234 if ((flags & wxUPDATE_UI_FROMIDLE) && !wxUSE_IDLEMENUUPDATES)
235 {
236 // If coming from an idle event, we only
237 // want to update the menus if we're
238 // in the wxUSE_IDLEMENUUPDATES configuration:
239 // so if we're not, do nothing
240 }
241 else
242 DoMenuUpdates();
243 }
96ac065f 244#endif // wxUSE_MENUS
e39af974
JS
245}
246
7c0ea335 247// ----------------------------------------------------------------------------
96ac065f 248// event handlers for status bar updates from menus
7c0ea335
VZ
249// ----------------------------------------------------------------------------
250
96ac065f
VZ
251#if wxUSE_MENUS && wxUSE_STATUSBAR
252
7c0ea335
VZ
253void wxFrameBase::OnMenuHighlight(wxMenuEvent& event)
254{
255#if wxUSE_STATUSBAR
722ed5be 256 (void)ShowMenuHelp(event.GetMenuId());
7c0ea335
VZ
257#endif // wxUSE_STATUSBAR
258}
259
d1b20379 260#if !wxUSE_IDLEMENUUPDATES
96ac065f 261void wxFrameBase::OnMenuOpen(wxMenuEvent& event)
d1b20379
DS
262#else
263void wxFrameBase::OnMenuOpen(wxMenuEvent& WXUNUSED(event))
264#endif
96ac065f
VZ
265{
266#if !wxUSE_IDLEMENUUPDATES
267 DoMenuUpdates(event.GetMenu());
268#endif // !wxUSE_IDLEMENUUPDATES
269}
270
271void wxFrameBase::OnMenuClose(wxMenuEvent& WXUNUSED(event))
272{
6d99eb3e 273 DoGiveHelp(wxEmptyString, false);
96ac065f
VZ
274}
275
276#endif // wxUSE_MENUS && wxUSE_STATUSBAR
277
e39af974
JS
278// Implement internal behaviour (menu updating on some platforms)
279void wxFrameBase::OnInternalIdle()
6522713c 280{
e2b6d07d 281 wxTopLevelWindow::OnInternalIdle();
a62848fd 282
0b30bb0b 283#if wxUSE_MENUS && wxUSE_IDLEMENUUPDATES
e39af974 284 if (wxUpdateUIEvent::CanUpdate(this))
0b30bb0b
JS
285 DoMenuUpdates();
286#endif
287}
288
7c0ea335
VZ
289// ----------------------------------------------------------------------------
290// status bar stuff
291// ----------------------------------------------------------------------------
292
293#if wxUSE_STATUSBAR
294
295wxStatusBar* wxFrameBase::CreateStatusBar(int number,
296 long style,
297 wxWindowID id,
298 const wxString& name)
299{
300 // the main status bar can only be created once (or else it should be
301 // deleted before calling CreateStatusBar() again)
302 wxCHECK_MSG( !m_frameStatusBar, (wxStatusBar *)NULL,
303 wxT("recreating status bar in wxFrame") );
304
a4f01f05 305 SetStatusBar(OnCreateStatusBar(number, style, id, name));
7c0ea335
VZ
306
307 return m_frameStatusBar;
308}
309
310wxStatusBar *wxFrameBase::OnCreateStatusBar(int number,
311 long style,
312 wxWindowID id,
313 const wxString& name)
314{
ed791986 315 wxStatusBar *statusBar = new wxStatusBar(this, id, style, name);
7c0ea335 316
7c0ea335
VZ
317 statusBar->SetFieldsCount(number);
318
319 return statusBar;
320}
321
322void wxFrameBase::SetStatusText(const wxString& text, int number)
323{
7c0ea335
VZ
324 wxCHECK_RET( m_frameStatusBar != NULL, wxT("no statusbar to set text for") );
325
326 m_frameStatusBar->SetStatusText(text, number);
327}
328
329void wxFrameBase::SetStatusWidths(int n, const int widths_field[] )
330{
7c0ea335
VZ
331 wxCHECK_RET( m_frameStatusBar != NULL, wxT("no statusbar to set widths for") );
332
333 m_frameStatusBar->SetStatusWidths(n, widths_field);
334
335 PositionStatusBar();
336}
337
1f361cdd 338void wxFrameBase::PushStatusText(const wxString& text, int number)
f6bcfd97 339{
1f361cdd
MB
340 wxCHECK_RET( m_frameStatusBar != NULL, wxT("no statusbar to set text for") );
341
342 m_frameStatusBar->PushStatusText(text, number);
343}
f6bcfd97 344
1f361cdd
MB
345void wxFrameBase::PopStatusText(int number)
346{
347 wxCHECK_RET( m_frameStatusBar != NULL, wxT("no statusbar to set text for") );
348
349 m_frameStatusBar->PopStatusText(number);
350}
351
722ed5be 352bool wxFrameBase::ShowMenuHelp(int menuId)
1f361cdd
MB
353{
354#if wxUSE_MENUS
f6bcfd97
BP
355 // if no help string found, we will clear the status bar text
356 wxString helpString;
fa36fe36 357 if ( menuId != wxID_SEPARATOR && menuId != -3 /* wxID_TITLE */ )
f6bcfd97 358 {
10816efb
VZ
359 const wxMenuItem * const item = FindItemInMenuBar(menuId);
360 if ( item )
361 helpString = item->GetHelp();
362
363 // notice that it's ok if we don't find the item because it might
364 // belong to the popup menu, so don't assert here
f6bcfd97
BP
365 }
366
6d99eb3e 367 DoGiveHelp(helpString, true);
f6bcfd97 368
1729813a 369 return !helpString.empty();
3379ed37 370#else // !wxUSE_MENUS
d1b20379 371 return false;
3379ed37 372#endif // wxUSE_MENUS/!wxUSE_MENUS
f6bcfd97
BP
373}
374
a4f01f05
VZ
375void wxFrameBase::SetStatusBar(wxStatusBar *statBar)
376{
377 bool hadBar = m_frameStatusBar != NULL;
378 m_frameStatusBar = statBar;
379
380 if ( (m_frameStatusBar != NULL) != hadBar )
381 {
382 PositionStatusBar();
383
384 DoLayout();
385 }
386}
387
7c0ea335
VZ
388#endif // wxUSE_STATUSBAR
389
f257ac87 390#if wxUSE_MENUS || wxUSE_TOOLBAR
6d99eb3e 391void wxFrameBase::DoGiveHelp(const wxString& help, bool show)
c60a36d5
VZ
392{
393#if wxUSE_STATUSBAR
96ac065f
VZ
394 if ( m_statusBarPane < 0 )
395 {
396 // status bar messages disabled
397 return;
398 }
399
400 wxStatusBar *statbar = GetStatusBar();
401 if ( !statbar )
402 return;
403
6d99eb3e
VZ
404 wxString text;
405 if ( show )
4cbc928a 406 {
6d99eb3e
VZ
407 // remember the old status bar text if this is the first time we're
408 // called since the menu has been opened as we're going to overwrite it
409 // in our DoGiveHelp() and we want to restore it when the menu is
410 // closed
411 //
412 // note that it would be logical to do this in OnMenuOpen() but under
413 // MSW we get an EVT_MENU_HIGHLIGHT before EVT_MENU_OPEN, strangely
414 // enough, and so this doesn't work and instead we use the ugly trick
415 // with using special m_oldStatusText value as "menu opened" (but it is
416 // arguably better than adding yet another member variable to wxFrame
417 // on all platforms)
96ac065f
VZ
418 if ( m_oldStatusText.empty() )
419 {
6d99eb3e
VZ
420 m_oldStatusText = statbar->GetStatusText(m_statusBarPane);
421 if ( m_oldStatusText.empty() )
422 {
423 // use special value to prevent us from doing this the next time
424 m_oldStatusText += _T('\0');
425 }
96ac065f 426 }
6d99eb3e
VZ
427
428 text = help;
429 }
430 else // hide help, restore the original text
431 {
432 text = m_oldStatusText;
433 m_oldStatusText.clear();
96ac065f 434 }
c60a36d5 435
6d99eb3e 436 statbar->SetStatusText(text, m_statusBarPane);
f428e6c5
WS
437#else
438 wxUnusedVar(text);
439 wxUnusedVar(show);
c60a36d5
VZ
440#endif // wxUSE_STATUSBAR
441}
f257ac87 442#endif // wxUSE_MENUS || wxUSE_TOOLBAR
c60a36d5
VZ
443
444
7c0ea335
VZ
445// ----------------------------------------------------------------------------
446// toolbar stuff
447// ----------------------------------------------------------------------------
448
449#if wxUSE_TOOLBAR
450
451wxToolBar* wxFrameBase::CreateToolBar(long style,
452 wxWindowID id,
453 const wxString& name)
454{
6a17b868 455 // the main toolbar can't be recreated (unless it was explicitly deleted
7c0ea335
VZ
456 // before)
457 wxCHECK_MSG( !m_frameToolBar, (wxToolBar *)NULL,
458 wxT("recreating toolbar in wxFrame") );
459
f9dae779
VZ
460 if ( style == -1 )
461 {
462 // use default style
463 //
464 // NB: we don't specify the default value in the method declaration
465 // because
466 // a) this allows us to have different defaults for different
467 // platforms (even if we don't have them right now)
468 // b) we don't need to include wx/toolbar.h in the header then
469 style = wxBORDER_NONE | wxTB_HORIZONTAL | wxTB_FLAT;
470 }
471
a4f01f05 472 SetToolBar(OnCreateToolBar(style, id, name));
7c0ea335
VZ
473
474 return m_frameToolBar;
475}
476
477wxToolBar* wxFrameBase::OnCreateToolBar(long style,
478 wxWindowID id,
479 const wxString& name)
480{
a9102b36
JS
481#if defined(__WXWINCE__) && defined(__POCKETPC__)
482 return new wxToolMenuBar(this, id,
483 wxDefaultPosition, wxDefaultSize,
484 style, name);
485#else
7c0ea335
VZ
486 return new wxToolBar(this, id,
487 wxDefaultPosition, wxDefaultSize,
488 style, name);
a9102b36 489#endif
7c0ea335
VZ
490}
491
a4f01f05
VZ
492void wxFrameBase::SetToolBar(wxToolBar *toolbar)
493{
494 bool hadBar = m_frameToolBar != NULL;
495 m_frameToolBar = toolbar;
496
497 if ( (m_frameToolBar != NULL) != hadBar )
498 {
499 PositionToolBar();
500
501 DoLayout();
502 }
503}
504
7c0ea335
VZ
505#endif // wxUSE_TOOLBAR
506
507// ----------------------------------------------------------------------------
6522713c 508// menus
7c0ea335
VZ
509// ----------------------------------------------------------------------------
510
1e6feb95
VZ
511#if wxUSE_MENUS
512
63fec618 513// update all menus
92f1a59c 514void wxFrameBase::DoMenuUpdates(wxMenu* menu)
63fec618 515{
92f1a59c 516 if (menu)
4d538595
DS
517 {
518 wxEvtHandler* source = GetEventHandler();
92f1a59c 519 menu->UpdateUI(source);
4d538595
DS
520 }
521 else
54517652 522 {
4d538595
DS
523 wxMenuBar* bar = GetMenuBar();
524 if (bar != NULL)
525 bar->UpdateMenus();
63fec618 526 }
63fec618 527}
1e6feb95 528
6522713c
VZ
529void wxFrameBase::DetachMenuBar()
530{
531 if ( m_frameMenuBar )
532 {
533 m_frameMenuBar->Detach();
534 m_frameMenuBar = NULL;
535 }
536}
537
538void wxFrameBase::AttachMenuBar(wxMenuBar *menubar)
539{
540 if ( menubar )
541 {
6522713c 542 menubar->Attach((wxFrame *)this);
3dbe38c3 543 m_frameMenuBar = menubar;
6522713c
VZ
544 }
545}
546
547void wxFrameBase::SetMenuBar(wxMenuBar *menubar)
548{
549 if ( menubar == GetMenuBar() )
550 {
551 // nothing to do
552 return;
553 }
554
555 DetachMenuBar();
556
a96b4743 557 this->AttachMenuBar(menubar);
6522713c
VZ
558}
559
10816efb
VZ
560const wxMenuItem *wxFrameBase::FindItemInMenuBar(int menuId) const
561{
562 const wxMenuBar * const menuBar = GetMenuBar();
563
564 return menuBar ? menuBar->FindItem(menuId) : NULL;
565}
566
1e6feb95 567#endif // wxUSE_MENUS