1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/frame.cpp
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
30 #include "wx/msw/wrapcctl.h" // include <commctrl.h> "properly"
34 #include "wx/dialog.h"
35 #include "wx/settings.h"
36 #include "wx/dcclient.h"
40 #include "wx/toolbar.h"
41 #include "wx/statusbr.h"
42 #include "wx/menuitem.h"
45 #include "wx/msw/private.h"
47 #if defined(__POCKETPC__) || defined(__SMARTPHONE__)
50 #include "wx/msw/winundef.h"
53 #include "wx/generic/statusbr.h"
55 #ifdef __WXUNIVERSAL__
56 #include "wx/univ/theme.h"
57 #include "wx/univ/colschem.h"
58 #endif // __WXUNIVERSAL__
60 // ----------------------------------------------------------------------------
62 // ----------------------------------------------------------------------------
64 #if wxUSE_MENUS_NATIVE
65 extern wxMenu
*wxCurrentPopupMenu
;
66 #endif // wxUSE_MENUS_NATIVE
68 // ----------------------------------------------------------------------------
70 // ----------------------------------------------------------------------------
72 BEGIN_EVENT_TABLE(wxFrame
, wxFrameBase
)
73 EVT_SYS_COLOUR_CHANGED(wxFrame::OnSysColourChanged
)
76 #if wxUSE_EXTENDED_RTTI
77 WX_DEFINE_FLAGS( wxFrameStyle
)
79 wxBEGIN_FLAGS( wxFrameStyle
)
80 // new style border flags, we put them first to
81 // use them for streaming out
82 wxFLAGS_MEMBER(wxBORDER_SIMPLE
)
83 wxFLAGS_MEMBER(wxBORDER_SUNKEN
)
84 wxFLAGS_MEMBER(wxBORDER_DOUBLE
)
85 wxFLAGS_MEMBER(wxBORDER_RAISED
)
86 wxFLAGS_MEMBER(wxBORDER_STATIC
)
87 wxFLAGS_MEMBER(wxBORDER_NONE
)
89 // old style border flags
90 wxFLAGS_MEMBER(wxSIMPLE_BORDER
)
91 wxFLAGS_MEMBER(wxSUNKEN_BORDER
)
92 wxFLAGS_MEMBER(wxDOUBLE_BORDER
)
93 wxFLAGS_MEMBER(wxRAISED_BORDER
)
94 wxFLAGS_MEMBER(wxSTATIC_BORDER
)
95 wxFLAGS_MEMBER(wxBORDER
)
97 // standard window styles
98 wxFLAGS_MEMBER(wxTAB_TRAVERSAL
)
99 wxFLAGS_MEMBER(wxCLIP_CHILDREN
)
100 wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW
)
101 wxFLAGS_MEMBER(wxWANTS_CHARS
)
102 wxFLAGS_MEMBER(wxFULL_REPAINT_ON_RESIZE
)
103 wxFLAGS_MEMBER(wxALWAYS_SHOW_SB
)
104 wxFLAGS_MEMBER(wxVSCROLL
)
105 wxFLAGS_MEMBER(wxHSCROLL
)
108 wxFLAGS_MEMBER(wxSTAY_ON_TOP
)
109 wxFLAGS_MEMBER(wxCAPTION
)
110 #if WXWIN_COMPATIBILITY_2_6
111 wxFLAGS_MEMBER(wxTHICK_FRAME
)
112 #endif // WXWIN_COMPATIBILITY_2_6
113 wxFLAGS_MEMBER(wxSYSTEM_MENU
)
114 wxFLAGS_MEMBER(wxRESIZE_BORDER
)
115 #if WXWIN_COMPATIBILITY_2_6
116 wxFLAGS_MEMBER(wxRESIZE_BOX
)
117 #endif // WXWIN_COMPATIBILITY_2_6
118 wxFLAGS_MEMBER(wxCLOSE_BOX
)
119 wxFLAGS_MEMBER(wxMAXIMIZE_BOX
)
120 wxFLAGS_MEMBER(wxMINIMIZE_BOX
)
122 wxFLAGS_MEMBER(wxFRAME_TOOL_WINDOW
)
123 wxFLAGS_MEMBER(wxFRAME_FLOAT_ON_PARENT
)
125 wxFLAGS_MEMBER(wxFRAME_SHAPED
)
127 wxEND_FLAGS( wxFrameStyle
)
129 IMPLEMENT_DYNAMIC_CLASS_XTI(wxFrame
, wxTopLevelWindow
,"wx/frame.h")
131 wxBEGIN_PROPERTIES_TABLE(wxFrame
)
132 wxEVENT_PROPERTY( Menu
, wxEVT_COMMAND_MENU_SELECTED
, wxCommandEvent
)
134 wxPROPERTY( Title
,wxString
, SetTitle
, GetTitle
, wxString() , 0 /*flags*/ , wxT("Helpstring") , wxT("group"))
135 wxPROPERTY_FLAGS( WindowStyle
, wxFrameStyle
, long , SetWindowStyleFlag
, GetWindowStyleFlag
, EMPTY_MACROVALUE
, 0 /*flags*/ , wxT("Helpstring") , wxT("group")) // style
136 wxPROPERTY( MenuBar
, wxMenuBar
* , SetMenuBar
, GetMenuBar
, EMPTY_MACROVALUE
, 0 /*flags*/ , wxT("Helpstring") , wxT("group"))
137 wxEND_PROPERTIES_TABLE()
139 wxBEGIN_HANDLERS_TABLE(wxFrame
)
140 wxEND_HANDLERS_TABLE()
142 wxCONSTRUCTOR_6( wxFrame
, wxWindow
* , Parent
, wxWindowID
, Id
, wxString
, Title
, wxPoint
, Position
, wxSize
, Size
, long , WindowStyle
)
145 IMPLEMENT_DYNAMIC_CLASS(wxFrame
, wxTopLevelWindow
)
148 // ============================================================================
150 // ============================================================================
152 // ----------------------------------------------------------------------------
153 // static class members
154 // ----------------------------------------------------------------------------
157 #if wxUSE_NATIVE_STATUSBAR
158 bool wxFrame::m_useNativeStatusBar
= true;
160 bool wxFrame::m_useNativeStatusBar
= false;
162 #endif // wxUSE_NATIVE_STATUSBAR
164 // ----------------------------------------------------------------------------
165 // creation/destruction
166 // ----------------------------------------------------------------------------
172 #endif // wxUSE_MENUS
178 m_wasMinimized
= false;
181 bool wxFrame::Create(wxWindow
*parent
,
183 const wxString
& title
,
187 const wxString
& name
)
189 if ( !wxTopLevelWindow::Create(parent
, id
, title
, pos
, size
, style
, name
) )
192 SetOwnBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE
));
194 #if defined(__SMARTPHONE__)
195 SetLeftMenu(wxID_EXIT
, _("Done"));
198 #if wxUSE_ACCEL && defined(__POCKETPC__)
199 // The guidelines state that Ctrl+Q should quit the app.
200 // Let's define an accelerator table to send wxID_EXIT.
201 wxAcceleratorEntry entries
[1];
202 entries
[0].Set(wxACCEL_CTRL
, 'Q', wxID_EXIT
);
203 wxAcceleratorTable
accel(1, entries
);
204 SetAcceleratorTable(accel
);
205 #endif // wxUSE_ACCEL && __POCKETPC__
212 m_isBeingDeleted
= true;
216 // ----------------------------------------------------------------------------
217 // wxFrame client size calculations
218 // ----------------------------------------------------------------------------
220 void wxFrame::DoSetClientSize(int width
, int height
)
222 // leave enough space for the status bar if we have (and show) it
224 wxStatusBar
*statbar
= GetStatusBar();
225 if ( statbar
&& statbar
->IsShown() )
227 height
+= statbar
->GetSize().y
;
229 #endif // wxUSE_STATUSBAR
231 // call GetClientAreaOrigin() to take the toolbar into account
232 wxPoint pt
= GetClientAreaOrigin();
237 wxToolBar
* const toolbar
= GetToolBar();
240 if ( toolbar
->HasFlag(wxTB_RIGHT
| wxTB_BOTTOM
) )
242 const wxSize sizeTB
= toolbar
->GetSize();
243 if ( toolbar
->HasFlag(wxTB_RIGHT
) )
248 //else: toolbar already taken into account by GetClientAreaOrigin()
250 #endif // wxUSE_TOOLBAR
252 wxTopLevelWindow::DoSetClientSize(width
, height
);
255 // Get size *available for subwindows* i.e. excluding menu bar, toolbar etc.
256 void wxFrame::DoGetClientSize(int *x
, int *y
) const
258 wxTopLevelWindow::DoGetClientSize(x
, y
);
260 // account for the possible toolbar
261 wxPoint pt
= GetClientAreaOrigin();
269 wxToolBar
* const toolbar
= GetToolBar();
272 if ( toolbar
->HasFlag(wxTB_RIGHT
| wxTB_BOTTOM
) )
274 const wxSize sizeTB
= toolbar
->GetSize();
275 if ( toolbar
->HasFlag(wxTB_RIGHT
) )
286 //else: toolbar already taken into account by GetClientAreaOrigin()
288 #endif // wxUSE_TOOLBAR
291 // adjust client area height to take the status bar into account
294 wxStatusBar
*statbar
= GetStatusBar();
295 if ( statbar
&& statbar
->IsShown() )
297 *y
-= statbar
->GetClientSize().y
;
300 #endif // wxUSE_STATUSBAR
303 // ----------------------------------------------------------------------------
304 // wxFrame: various geometry-related functions
305 // ----------------------------------------------------------------------------
307 void wxFrame::Raise()
309 ::SetForegroundWindow(GetHwnd());
312 // generate an artificial resize event
313 void wxFrame::SendSizeEvent()
317 RECT r
= wxGetWindowRect(GetHwnd());
319 (void)::PostMessage(GetHwnd(), WM_SIZE
,
320 IsMaximized() ? SIZE_MAXIMIZED
: SIZE_RESTORED
,
321 MAKELPARAM(r
.right
- r
.left
, r
.bottom
- r
.top
));
326 wxStatusBar
*wxFrame::OnCreateStatusBar(int number
,
329 const wxString
& name
)
331 wxStatusBar
*statusBar
wxDUMMY_INITIALIZE(NULL
);
333 #if wxUSE_NATIVE_STATUSBAR
334 if ( !UsesNativeStatusBar() )
336 statusBar
= (wxStatusBar
*)new wxStatusBarGeneric(this, id
, style
);
341 statusBar
= new wxStatusBar(this, id
, style
, name
);
344 statusBar
->SetFieldsCount(number
);
349 void wxFrame::PositionStatusBar()
351 if ( !m_frameStatusBar
|| !m_frameStatusBar
->IsShown() )
355 GetClientSize(&w
, &h
);
358 m_frameStatusBar
->GetSize(&sw
, &sh
);
362 wxToolBar
* const toolbar
= GetToolBar();
363 if ( toolbar
&& !toolbar
->HasFlag(wxTB_TOP
) )
365 const wxSize sizeTB
= toolbar
->GetSize();
367 if ( toolbar
->HasFlag(wxTB_LEFT
| wxTB_RIGHT
) )
369 if ( toolbar
->HasFlag(wxTB_LEFT
) )
376 // we need to position the status bar below the toolbar
380 //else: no adjustments necessary for the toolbar on top
381 #endif // wxUSE_TOOLBAR
383 // Since we wish the status bar to be directly under the client area,
384 // we use the adjusted sizes without using wxSIZE_NO_ADJUSTMENTS.
385 m_frameStatusBar
->SetSize(x
, h
, w
, sh
);
388 #endif // wxUSE_STATUSBAR
390 #if wxUSE_MENUS_NATIVE
392 void wxFrame::AttachMenuBar(wxMenuBar
*menubar
)
394 #if defined(__SMARTPHONE__) && defined(__WXWINCE__)
396 wxMenu
*autoMenu
= NULL
;
398 if( menubar
->GetMenuCount() == 1 )
400 autoMenu
= wxTopLevelWindowMSW::ButtonMenu::DuplicateMenu(menubar
->GetMenu(0));
401 SetRightMenu(wxID_ANY
, menubar
->GetMenuLabel(0), autoMenu
);
405 autoMenu
= new wxMenu
;
407 for( size_t n
= 0; n
< menubar
->GetMenuCount(); n
++ )
409 wxMenu
*item
= menubar
->GetMenu(n
);
410 wxString label
= menubar
->GetMenuLabel(n
);
411 wxMenu
*new_item
= wxTopLevelWindowMSW::ButtonMenu::DuplicateMenu(item
);
412 autoMenu
->Append(wxID_ANY
, label
, new_item
);
415 SetRightMenu(wxID_ANY
, _("Menu"), autoMenu
);
418 #elif defined(WINCE_WITHOUT_COMMANDBAR)
421 wxToolMenuBar
* toolBar
= new wxToolMenuBar(this, wxID_ANY
,
422 wxDefaultPosition
, wxDefaultSize
,
423 wxBORDER_NONE
| wxTB_HORIZONTAL
,
424 wxToolBarNameStr
, menubar
);
426 menubar
->SetToolBar(toolBar
);
428 // Now adjust size for menu bar
431 //When the main window is created using CW_USEDEFAULT the height of the
432 // is created is not taken into account). So we resize the window after
433 // if a menubar is present
436 ::GetWindowRect((HWND
) GetHWND(), &rc
);
437 // adjust for menu / titlebar height
438 rc
.bottom
-= (2*menuHeight
-1);
440 ::MoveWindow((HWND
) GetHWND(), rc
.left
, rc
.top
, rc
.right
, rc
.bottom
, FALSE
);
444 wxFrameBase::AttachMenuBar(menubar
);
448 // actually remove the menu from the frame
449 m_hMenu
= (WXHMENU
)0;
450 InternalSetMenuBar();
452 else // set new non NULL menu bar
454 #if !defined(__WXWINCE__) || defined(WINCE_WITH_COMMANDBAR)
455 // Can set a menubar several times.
456 if ( menubar
->GetHMenu() )
458 m_hMenu
= menubar
->GetHMenu();
462 m_hMenu
= menubar
->Create();
466 wxFAIL_MSG( _T("failed to create menu bar") );
471 InternalSetMenuBar();
475 void wxFrame::InternalSetMenuBar()
477 #if defined(__WXMICROWIN__) || defined(__WXWINCE__)
480 if ( !::SetMenu(GetHwnd(), (HMENU
)m_hMenu
) )
482 wxLogLastError(wxT("SetMenu"));
487 #endif // wxUSE_MENUS_NATIVE
489 // Responds to colour changes, and passes event on to children.
490 void wxFrame::OnSysColourChanged(wxSysColourChangedEvent
& event
)
492 SetOwnBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE
));
496 if ( m_frameStatusBar
)
498 wxSysColourChangedEvent event2
;
499 event2
.SetEventObject( m_frameStatusBar
);
500 m_frameStatusBar
->HandleWindowEvent(event2
);
502 #endif // wxUSE_STATUSBAR
504 // Propagate the event to the non-top-level children
505 wxWindow::OnSysColourChanged(event
);
508 // Pass true to show full screen, false to restore.
509 bool wxFrame::ShowFullScreen(bool show
, long style
)
511 // TODO-CE: add support for CE
512 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
513 if ( IsFullScreen() == show
)
518 // zap the toolbar, menubar, and statusbar if needed
520 // TODO: hide commandbar for WINCE_WITH_COMMANDBAR
522 wxToolBar
*theToolBar
= GetToolBar();
524 if ((style
& wxFULLSCREEN_NOTOOLBAR
) && theToolBar
)
526 if ( theToolBar
->IsShown() )
528 theToolBar
->SetSize(wxDefaultCoord
,0);
529 theToolBar
->Show(false);
531 else // prevent it from being restored later
533 style
&= ~wxFULLSCREEN_NOTOOLBAR
;
536 #endif // wxUSE_TOOLBAR
538 if (style
& wxFULLSCREEN_NOMENUBAR
)
539 SetMenu((HWND
)GetHWND(), (HMENU
) NULL
);
542 wxStatusBar
*theStatusBar
= GetStatusBar();
544 // Save the number of fields in the statusbar
545 if ((style
& wxFULLSCREEN_NOSTATUSBAR
) && theStatusBar
)
547 if ( theStatusBar
->IsShown() )
548 theStatusBar
->Show(false);
550 style
&= ~wxFULLSCREEN_NOSTATUSBAR
;
552 #endif // wxUSE_STATUSBAR
554 else // restore to normal
556 // restore the toolbar, menubar, and statusbar if we had hid them
558 wxToolBar
*theToolBar
= GetToolBar();
560 if ((m_fsStyle
& wxFULLSCREEN_NOTOOLBAR
) && theToolBar
)
562 theToolBar
->Show(true);
564 #endif // wxUSE_TOOLBAR
567 if (m_fsStyle
& wxFULLSCREEN_NOMENUBAR
)
569 WXHMENU menu
= m_hMenu
;
571 #if wxUSE_MDI_ARCHITECTURE
572 wxMDIParentFrame
*frame
= wxDynamicCast(this, wxMDIParentFrame
);
575 wxMDIChildFrame
*child
= frame
->GetActiveChild();
578 menu
= child
->GetWinMenu();
581 #endif // wxUSE_MDI_ARCHITECTURE
585 ::SetMenu(GetHwnd(), (HMENU
)menu
);
588 #endif // wxUSE_MENUS
591 wxStatusBar
*theStatusBar
= GetStatusBar();
593 if ((m_fsStyle
& wxFULLSCREEN_NOSTATUSBAR
) && theStatusBar
)
595 theStatusBar
->Show(true);
598 #endif // wxUSE_STATUSBAR
600 #endif // !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
602 return wxFrameBase::ShowFullScreen(show
, style
);
605 // ----------------------------------------------------------------------------
606 // tool/status bar stuff
607 // ----------------------------------------------------------------------------
611 wxToolBar
* wxFrame::CreateToolBar(long style
, wxWindowID id
, const wxString
& name
)
613 #if defined(WINCE_WITHOUT_COMMANDBAR)
614 // We may already have a toolbar from calling SetMenuBar.
618 if ( wxFrameBase::CreateToolBar(style
, id
, name
) )
623 return m_frameToolBar
;
626 void wxFrame::PositionToolBar()
628 // TODO: we want to do something different in WinCE, because the toolbar
629 // should be associated with the commandbar, instead of being
630 // independent window.
631 #if !defined(WINCE_WITHOUT_COMMANDBAR)
632 wxToolBar
*toolbar
= GetToolBar();
633 if ( toolbar
&& toolbar
->IsShown() )
635 // don't call our (or even wxTopLevelWindow) version because we want
636 // the real (full) client area size, not excluding the tool/status bar
638 wxWindow::DoGetClientSize(&width
, &height
);
641 wxStatusBar
*statbar
= GetStatusBar();
642 if ( statbar
&& statbar
->IsShown() )
644 height
-= statbar
->GetClientSize().y
;
646 #endif // wxUSE_STATUSBAR
649 toolbar
->GetPosition( &tx
, &ty
);
650 toolbar
->GetSize( &tw
, &th
);
653 if ( toolbar
->HasFlag(wxTB_BOTTOM
) )
658 else if ( toolbar
->HasFlag(wxTB_RIGHT
) )
669 #if defined(WINCE_WITH_COMMANDBAR)
670 // We're using a commandbar - so we have to allow for it.
671 if (GetMenuBar() && GetMenuBar()->GetCommandBar())
674 ::GetWindowRect((HWND
) GetMenuBar()->GetCommandBar(), &rect
);
675 y
= rect
.bottom
- rect
.top
;
677 #endif // WINCE_WITH_COMMANDBAR
679 if ( toolbar
->HasFlag(wxTB_BOTTOM
) )
681 if ( ty
< 0 && ( -ty
== th
) )
683 if ( tx
< 0 && (-tx
== tw
) )
686 else if ( toolbar
->HasFlag(wxTB_RIGHT
) )
688 if( ty
< 0 && ( -ty
== th
) )
690 if( tx
< 0 && ( -tx
== tw
) )
695 if (ty
< 0 && (-ty
== th
))
697 if (tx
< 0 && (-tx
== tw
))
704 if ( toolbar
->IsVertical() )
713 // use the 'real' MSW position here, don't offset relativly to the
714 // client area origin
716 // Optimise such that we don't have to always resize the toolbar
717 // when the frame changes, otherwise we'll get a lot of flicker.
718 bool heightChanging
wxDUMMY_INITIALIZE(true);
719 bool widthChanging
wxDUMMY_INITIALIZE(true);
721 if ( toolbar
->IsVertical() )
723 // It's OK if the current height is greater than what can be shown.
724 heightChanging
= (desiredH
> th
) ;
725 widthChanging
= (desiredW
!= tw
) ;
727 // The next time around, we may not have to set the size
729 desiredH
= desiredH
+ 200;
733 // It's OK if the current width is greater than what can be shown.
734 widthChanging
= (desiredW
> tw
) ;
735 heightChanging
= (desiredH
!= th
) ;
737 // The next time around, we may not have to set the size
739 desiredW
= desiredW
+ 200;
742 if (tx
!= 0 || ty
!= 0 || widthChanging
|| heightChanging
)
743 toolbar
->SetSize(x
, y
, desiredW
, desiredH
, wxSIZE_NO_ADJUSTMENTS
);
746 #endif // !WINCE_WITH_COMMANDBAR
749 #endif // wxUSE_TOOLBAR
751 // ----------------------------------------------------------------------------
752 // frame state (iconized/maximized/...)
753 // ----------------------------------------------------------------------------
755 // propagate our state change to all child frames: this allows us to emulate X
756 // Windows behaviour where child frames float independently of the parent one
757 // on the desktop, but are iconized/restored with it
758 void wxFrame::IconizeChildFrames(bool bIconize
)
760 m_iconized
= bIconize
;
762 for ( wxWindowList::compatibility_iterator node
= GetChildren().GetFirst();
764 node
= node
->GetNext() )
766 wxWindow
*win
= node
->GetData();
768 // iconizing the frames with this style under Win95 shell puts them at
769 // the bottom of the screen (as the MDI children) instead of making
770 // them appear in the taskbar because they are, by virtue of this
771 // style, not managed by the taskbar - instead leave Windows take care
773 if ( win
->GetWindowStyle() & wxFRAME_TOOL_WINDOW
)
776 // the child MDI frames are a special case and should not be touched by
777 // the parent frame - instead, they are managed by the user
778 wxFrame
*frame
= wxDynamicCast(win
, wxFrame
);
780 #if wxUSE_MDI_ARCHITECTURE
781 && !frame
->IsMDIChild()
782 #endif // wxUSE_MDI_ARCHITECTURE
785 // we don't want to restore the child frames which had been
786 // iconized even before we were iconized, so save the child frame
787 // status when iconizing the parent frame and check it when
791 frame
->m_wasMinimized
= frame
->IsIconized();
794 // note that we shouldn't touch the hidden frames neither because
795 // iconizing/restoring them would show them as a side effect
796 if ( !frame
->m_wasMinimized
&& frame
->IsShown() )
797 frame
->Iconize(bIconize
);
802 WXHICON
wxFrame::GetDefaultIcon() const
804 // we don't have any standard icons (any more)
808 // ===========================================================================
809 // message processing
810 // ===========================================================================
812 // ---------------------------------------------------------------------------
814 // ---------------------------------------------------------------------------
816 bool wxFrame::MSWDoTranslateMessage(wxFrame
*frame
, WXMSG
*pMsg
)
818 if ( wxWindow::MSWTranslateMessage(pMsg
) )
821 #if wxUSE_MENUS && wxUSE_ACCEL && !defined(__WXUNIVERSAL__)
822 // try the menu bar accels
823 wxMenuBar
*menuBar
= GetMenuBar();
826 const wxAcceleratorTable
& acceleratorTable
= menuBar
->GetAccelTable();
827 return acceleratorTable
.Translate(frame
, pMsg
);
829 #endif // wxUSE_MENUS && wxUSE_ACCEL
834 // ---------------------------------------------------------------------------
835 // our private (non virtual) message handlers
836 // ---------------------------------------------------------------------------
838 bool wxFrame::HandlePaint()
841 if ( ::GetUpdateRect(GetHwnd(), &rect
, FALSE
) )
843 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
846 const wxIcon
& icon
= GetIcon();
847 HICON hIcon
= icon
.Ok() ? GetHiconOf(icon
)
848 : (HICON
)GetDefaultIcon();
850 // Hold a pointer to the dc so long as the OnPaint() message
851 // is being processed
853 HDC hdc
= ::BeginPaint(GetHwnd(), &ps
);
855 // Erase background before painting or we get white background
856 MSWDefWindowProc(WM_ICONERASEBKGND
, (WORD
)(LONG
)ps
.hdc
, 0L);
861 ::GetClientRect(GetHwnd(), &rect
);
863 // FIXME: why hardcoded?
864 static const int icon_width
= 32;
865 static const int icon_height
= 32;
867 int icon_x
= (int)((rect
.right
- icon_width
)/2);
868 int icon_y
= (int)((rect
.bottom
- icon_height
)/2);
870 ::DrawIcon(hdc
, icon_x
, icon_y
, hIcon
);
873 ::EndPaint(GetHwnd(), &ps
);
880 return wxWindow::HandlePaint();
885 // nothing to paint - processed
890 bool wxFrame::HandleSize(int WXUNUSED(x
), int WXUNUSED(y
), WXUINT id
)
892 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
897 // only do it it if we were iconized before, otherwise resizing the
898 // parent frame has a curious side effect of bringing it under it's
903 // restore all child frames too
904 IconizeChildFrames(false);
906 (void)SendIconizeEvent(false);
910 // iconize all child frames too
911 IconizeChildFrames(true);
916 #endif // !__WXWINCE__
922 #endif // wxUSE_STATUSBAR
926 #endif // wxUSE_TOOLBAR
928 #if defined(WINCE_WITH_COMMANDBAR)
929 // Position the menu command bar
930 if (GetMenuBar() && GetMenuBar()->GetCommandBar())
933 ::GetWindowRect((HWND
) GetMenuBar()->GetCommandBar(), &rect
);
934 wxSize clientSz
= GetClientSize();
936 if ( !::MoveWindow((HWND
) GetMenuBar()->GetCommandBar(), 0, 0, clientSz
.x
, rect
.bottom
- rect
.top
, true ) )
938 wxLogLastError(wxT("MoveWindow"));
942 #endif // WINCE_WITH_COMMANDBAR
945 // call the base class version to generate the appropriate events
949 bool wxFrame::HandleCommand(WXWORD id_
, WXWORD cmd
, WXHWND control
)
951 // sign extend to int from short before comparing with the other int ids
952 int id
= (signed short)id_
;
956 // In case it's e.g. a toolbar.
957 wxWindow
*win
= wxFindWinFromHandle(control
);
959 return win
->MSWCommand(cmd
, id
);
962 // handle here commands from menus and accelerators
963 if ( cmd
== 0 || cmd
== 1 )
965 #if wxUSE_MENUS_NATIVE
966 if ( wxCurrentPopupMenu
)
968 wxMenu
*popupMenu
= wxCurrentPopupMenu
;
969 wxCurrentPopupMenu
= NULL
;
971 return popupMenu
->MSWCommand(cmd
, id
);
973 #endif // wxUSE_MENUS_NATIVE
975 #if defined(__SMARTPHONE__) && defined(__WXWINCE__)
976 // handle here commands from Smartphone menu bar
977 if ( wxTopLevelWindow::HandleCommand(id
, cmd
, control
) )
981 #endif // __SMARTPHONE__ && __WXWINCE__
983 if ( ProcessCommand(id
) )
994 bool wxFrame::HandleMenuSelect(WXWORD nItem
, WXWORD flags
, WXHMENU hMenu
)
997 if ( flags
== 0xFFFF && hMenu
== 0 )
999 // menu was removed from screen
1002 #ifndef __WXMICROWIN__
1003 else if ( !(flags
& MF_POPUP
) && !(flags
& MF_SEPARATOR
) )
1010 // don't give hints for separators (doesn't make sense) nor for the
1011 // items opening popup menus (they don't have them anyhow) but do clear
1012 // the status line - otherwise, we would be left with the help message
1013 // for the previous item which doesn't apply any more
1014 DoGiveHelp(wxEmptyString
, true);
1019 wxMenuEvent
event(wxEVT_MENU_HIGHLIGHT
, item
);
1020 event
.SetEventObject(this);
1022 return HandleWindowEvent(event
);
1025 bool wxFrame::HandleMenuLoop(const wxEventType
& evtType
, WXWORD isPopup
)
1027 // we don't have the menu id here, so we use the id to specify if the event
1028 // was from a popup menu or a normal one
1029 wxMenuEvent
event(evtType
, isPopup
? -1 : 0);
1030 event
.SetEventObject(this);
1032 return HandleWindowEvent(event
);
1035 bool wxFrame::HandleInitMenuPopup(WXHMENU hMenu
)
1037 wxMenu
* menu
= NULL
;
1040 int nCount
= GetMenuBar()->GetMenuCount();
1041 for (int n
= 0; n
< nCount
; n
++)
1043 if (GetMenuBar()->GetMenu(n
)->GetHMenu() == hMenu
)
1045 menu
= GetMenuBar()->GetMenu(n
);
1051 wxMenuEvent
event(wxEVT_MENU_OPEN
, 0, menu
);
1052 event
.SetEventObject(this);
1054 return HandleWindowEvent(event
);
1057 #endif // wxUSE_MENUS
1059 // ---------------------------------------------------------------------------
1060 // the window proc for wxFrame
1061 // ---------------------------------------------------------------------------
1063 WXLRESULT
wxFrame::MSWWindowProc(WXUINT message
, WXWPARAM wParam
, WXLPARAM lParam
)
1066 bool processed
= false;
1071 // if we can't close, tell the system that we processed the
1072 // message - otherwise it would close us
1073 processed
= !Close();
1077 processed
= HandleSize(LOWORD(lParam
), HIWORD(lParam
), wParam
);
1084 UnpackCommand((WXWPARAM
)wParam
, (WXLPARAM
)lParam
,
1087 processed
= HandleCommand(id
, cmd
, (WXHWND
)hwnd
);
1092 processed
= HandlePaint();
1095 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
1097 case WM_INITMENUPOPUP
:
1098 processed
= HandleInitMenuPopup((WXHMENU
) wParam
);
1105 UnpackMenuSelect(wParam
, lParam
, &item
, &flags
, &hmenu
);
1107 processed
= HandleMenuSelect(item
, flags
, hmenu
);
1111 case WM_EXITMENULOOP
:
1112 processed
= HandleMenuLoop(wxEVT_MENU_CLOSE
, (WXWORD
)wParam
);
1114 #endif // wxUSE_MENUS
1116 case WM_QUERYDRAGICON
:
1118 const wxIcon
& icon
= GetIcon();
1119 HICON hIcon
= icon
.Ok() ? GetHiconOf(icon
)
1120 : (HICON
)GetDefaultIcon();
1122 processed
= rc
!= 0;
1125 #endif // !__WXMICROWIN__
1129 rc
= wxFrameBase::MSWWindowProc(message
, wParam
, lParam
);
1134 // ----------------------------------------------------------------------------
1135 // wxFrame size management: we exclude the areas taken by menu/status/toolbars
1136 // from the client area, so the client area is what's really available for the
1138 // ----------------------------------------------------------------------------
1140 // get the origin of the client area in the client coordinates
1141 wxPoint
wxFrame::GetClientAreaOrigin() const
1143 wxPoint pt
= wxTopLevelWindow::GetClientAreaOrigin();
1145 #if wxUSE_TOOLBAR && !defined(__WXUNIVERSAL__) && \
1146 (!defined(__WXWINCE__) || (_WIN32_WCE >= 400 && !defined(__POCKETPC__) && !defined(__SMARTPHONE__)))
1147 wxToolBar
* const toolbar
= GetToolBar();
1148 if ( toolbar
&& toolbar
->IsShown() )
1150 const wxSize sizeTB
= toolbar
->GetSize();
1152 if ( toolbar
->HasFlag(wxTB_TOP
) )
1156 else if ( toolbar
->HasFlag(wxTB_LEFT
) )
1161 #endif // wxUSE_TOOLBAR
1163 #if defined(WINCE_WITH_COMMANDBAR)
1164 if (GetMenuBar() && GetMenuBar()->GetCommandBar())
1167 ::GetWindowRect((HWND
) GetMenuBar()->GetCommandBar(), &rect
);
1168 pt
.y
+= (rect
.bottom
- rect
.top
);