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 // FIXME-VC6: Only VC6 doesn't have this in its standard headers so this
61 // could be removed once support for it is dropped.
62 #ifndef WM_UNINITMENUPOPUP
63 #define WM_UNINITMENUPOPUP 0x0125
66 // ----------------------------------------------------------------------------
68 // ----------------------------------------------------------------------------
70 #if wxUSE_MENUS || wxUSE_MENUS_NATIVE
71 extern wxMenu
*wxCurrentPopupMenu
;
72 #endif // wxUSE_MENUS || wxUSE_MENUS_NATIVE
74 // ----------------------------------------------------------------------------
76 // ----------------------------------------------------------------------------
78 BEGIN_EVENT_TABLE(wxFrame
, wxFrameBase
)
79 EVT_SYS_COLOUR_CHANGED(wxFrame::OnSysColourChanged
)
82 // ============================================================================
84 // ============================================================================
86 // ----------------------------------------------------------------------------
87 // static class members
88 // ----------------------------------------------------------------------------
91 #if wxUSE_NATIVE_STATUSBAR
92 bool wxFrame::m_useNativeStatusBar
= true;
94 bool wxFrame::m_useNativeStatusBar
= false;
96 #endif // wxUSE_NATIVE_STATUSBAR
98 // ----------------------------------------------------------------------------
99 // creation/destruction
100 // ----------------------------------------------------------------------------
106 #endif // wxUSE_MENUS
112 m_wasMinimized
= false;
115 bool wxFrame::Create(wxWindow
*parent
,
117 const wxString
& title
,
121 const wxString
& name
)
123 if ( !wxTopLevelWindow::Create(parent
, id
, title
, pos
, size
, style
, name
) )
126 SetOwnBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE
));
128 #if defined(__SMARTPHONE__)
129 SetLeftMenu(wxID_EXIT
, _("Done"));
132 #if wxUSE_ACCEL && defined(__POCKETPC__)
133 // The guidelines state that Ctrl+Q should quit the app.
134 // Let's define an accelerator table to send wxID_EXIT.
135 wxAcceleratorEntry entries
[1];
136 entries
[0].Set(wxACCEL_CTRL
, 'Q', wxID_EXIT
);
137 wxAcceleratorTable
accel(1, entries
);
138 SetAcceleratorTable(accel
);
139 #endif // wxUSE_ACCEL && __POCKETPC__
151 // ----------------------------------------------------------------------------
152 // wxFrame client size calculations
153 // ----------------------------------------------------------------------------
155 void wxFrame::DoSetClientSize(int width
, int height
)
157 // leave enough space for the status bar if we have (and show) it
159 wxStatusBar
*statbar
= GetStatusBar();
160 if ( statbar
&& statbar
->IsShown() )
162 height
+= statbar
->GetSize().y
;
164 #endif // wxUSE_STATUSBAR
166 // call GetClientAreaOrigin() to take the toolbar into account
167 wxPoint pt
= GetClientAreaOrigin();
172 wxToolBar
* const toolbar
= GetToolBar();
175 if ( toolbar
->HasFlag(wxTB_RIGHT
| wxTB_BOTTOM
) )
177 const wxSize sizeTB
= toolbar
->GetSize();
178 if ( toolbar
->HasFlag(wxTB_RIGHT
) )
183 //else: toolbar already taken into account by GetClientAreaOrigin()
185 #endif // wxUSE_TOOLBAR
187 wxTopLevelWindow::DoSetClientSize(width
, height
);
190 // Get size *available for subwindows* i.e. excluding menu bar, toolbar etc.
191 void wxFrame::DoGetClientSize(int *x
, int *y
) const
193 wxTopLevelWindow::DoGetClientSize(x
, y
);
195 // account for the possible toolbar
196 wxPoint pt
= GetClientAreaOrigin();
204 wxToolBar
* const toolbar
= GetToolBar();
207 if ( toolbar
->HasFlag(wxTB_RIGHT
| wxTB_BOTTOM
) )
209 const wxSize sizeTB
= toolbar
->GetSize();
210 if ( toolbar
->HasFlag(wxTB_RIGHT
) )
221 //else: toolbar already taken into account by GetClientAreaOrigin()
223 #endif // wxUSE_TOOLBAR
226 // adjust client area height to take the status bar into account
229 wxStatusBar
*statbar
= GetStatusBar();
230 if ( statbar
&& statbar
->IsShown() )
232 *y
-= statbar
->GetSize().y
;
235 #endif // wxUSE_STATUSBAR
238 // ----------------------------------------------------------------------------
239 // wxFrame: various geometry-related functions
240 // ----------------------------------------------------------------------------
242 // generate an artificial resize event
243 void wxFrame::SendSizeEvent(int flags
)
247 RECT r
= wxGetWindowRect(GetHwnd());
249 if ( flags
& wxSEND_EVENT_POST
)
251 ::PostMessage(GetHwnd(), WM_SIZE
,
252 IsMaximized() ? SIZE_MAXIMIZED
: SIZE_RESTORED
,
253 MAKELPARAM(r
.right
- r
.left
, r
.bottom
- r
.top
));
257 ::SendMessage(GetHwnd(), WM_SIZE
,
258 IsMaximized() ? SIZE_MAXIMIZED
: SIZE_RESTORED
,
259 MAKELPARAM(r
.right
- r
.left
, r
.bottom
- r
.top
));
265 wxStatusBar
*wxFrame::OnCreateStatusBar(int number
,
268 const wxString
& name
)
270 wxStatusBar
*statusBar
wxDUMMY_INITIALIZE(NULL
);
272 #if wxUSE_NATIVE_STATUSBAR
273 if ( !UsesNativeStatusBar() )
275 statusBar
= (wxStatusBar
*)new wxStatusBarGeneric(this, id
, style
);
280 statusBar
= new wxStatusBar(this, id
, style
, name
);
283 statusBar
->SetFieldsCount(number
);
288 void wxFrame::PositionStatusBar()
290 if ( !m_frameStatusBar
|| !m_frameStatusBar
->IsShown() )
294 GetClientSize(&w
, &h
);
297 m_frameStatusBar
->GetSize(&sw
, &sh
);
301 wxToolBar
* const toolbar
= GetToolBar();
302 if ( toolbar
&& !toolbar
->HasFlag(wxTB_TOP
) )
304 const wxSize sizeTB
= toolbar
->GetSize();
306 if ( toolbar
->HasFlag(wxTB_LEFT
| wxTB_RIGHT
) )
308 if ( toolbar
->HasFlag(wxTB_LEFT
) )
315 // we need to position the status bar below the toolbar
319 //else: no adjustments necessary for the toolbar on top
320 #endif // wxUSE_TOOLBAR
322 // Since we wish the status bar to be directly under the client area,
323 // we use the adjusted sizes without using wxSIZE_NO_ADJUSTMENTS.
324 m_frameStatusBar
->SetSize(x
, h
, w
, sh
);
327 #endif // wxUSE_STATUSBAR
329 #if wxUSE_MENUS_NATIVE
331 void wxFrame::AttachMenuBar(wxMenuBar
*menubar
)
333 #if defined(__SMARTPHONE__) && defined(__WXWINCE__)
335 wxMenu
*autoMenu
= NULL
;
337 if( menubar
->GetMenuCount() == 1 )
339 autoMenu
= wxTopLevelWindowMSW::ButtonMenu::DuplicateMenu(menubar
->GetMenu(0));
340 SetRightMenu(wxID_ANY
, menubar
->GetMenuLabel(0), autoMenu
);
344 autoMenu
= new wxMenu
;
346 for( size_t n
= 0; n
< menubar
->GetMenuCount(); n
++ )
348 wxMenu
*item
= menubar
->GetMenu(n
);
349 wxString label
= menubar
->GetMenuLabel(n
);
350 wxMenu
*new_item
= wxTopLevelWindowMSW::ButtonMenu::DuplicateMenu(item
);
351 autoMenu
->Append(wxID_ANY
, label
, new_item
);
354 SetRightMenu(wxID_ANY
, _("Menu"), autoMenu
);
357 #elif defined(WINCE_WITHOUT_COMMANDBAR)
360 wxToolMenuBar
* toolBar
= new wxToolMenuBar(this, wxID_ANY
,
361 wxDefaultPosition
, wxDefaultSize
,
362 wxBORDER_NONE
| wxTB_HORIZONTAL
,
363 wxToolBarNameStr
, menubar
);
365 menubar
->SetToolBar(toolBar
);
368 // When the main window is created using CW_USEDEFAULT the height of the
369 // menubar is not taken into account, so we resize it afterwards if a
370 // menubar is present
371 HWND hwndMenuBar
= SHFindMenuBar(GetHwnd());
375 ::GetWindowRect(hwndMenuBar
, &mbRect
);
376 const int menuHeight
= mbRect
.bottom
- mbRect
.top
;
379 ::GetWindowRect(GetHwnd(), &rc
);
380 // adjust for menu / titlebar height
381 rc
.bottom
-= (2*menuHeight
-1);
383 ::MoveWindow(GetHwnd(), rc
.left
, rc
.top
, rc
.right
, rc
.bottom
, FALSE
);
387 wxFrameBase::AttachMenuBar(menubar
);
391 // actually remove the menu from the frame
392 m_hMenu
= (WXHMENU
)0;
393 InternalSetMenuBar();
395 else // set new non NULL menu bar
397 #if !defined(__WXWINCE__) || defined(WINCE_WITH_COMMANDBAR)
398 // Can set a menubar several times.
399 if ( menubar
->GetHMenu() )
401 m_hMenu
= menubar
->GetHMenu();
405 m_hMenu
= menubar
->Create();
409 wxFAIL_MSG( wxT("failed to create menu bar") );
414 InternalSetMenuBar();
418 void wxFrame::InternalSetMenuBar()
420 #if defined(__WXMICROWIN__) || defined(__WXWINCE__)
423 if ( !::SetMenu(GetHwnd(), (HMENU
)m_hMenu
) )
425 wxLogLastError(wxT("SetMenu"));
430 #endif // wxUSE_MENUS_NATIVE
433 wxMenu
* wxFrame::MSWFindMenuFromHMENU(WXHMENU hMenu
)
435 return GetMenuBar() ? GetMenuBar()->MSWGetMenu(hMenu
) : NULL
;
437 #endif // wxUSE_MENUS
439 // Responds to colour changes, and passes event on to children.
440 void wxFrame::OnSysColourChanged(wxSysColourChangedEvent
& event
)
442 // Don't override the colour explicitly set by the user, if any.
445 SetOwnBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE
));
450 if ( m_frameStatusBar
)
452 wxSysColourChangedEvent event2
;
453 event2
.SetEventObject( m_frameStatusBar
);
454 m_frameStatusBar
->HandleWindowEvent(event2
);
456 #endif // wxUSE_STATUSBAR
458 // Propagate the event to the non-top-level children
459 wxWindow::OnSysColourChanged(event
);
462 // Pass true to show full screen, false to restore.
463 bool wxFrame::ShowFullScreen(bool show
, long style
)
465 // TODO-CE: add support for CE
466 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
467 if ( IsFullScreen() == show
)
472 // zap the toolbar, menubar, and statusbar if needed
474 // TODO: hide commandbar for WINCE_WITH_COMMANDBAR
476 wxToolBar
*theToolBar
= GetToolBar();
478 if ((style
& wxFULLSCREEN_NOTOOLBAR
) && theToolBar
)
480 if ( theToolBar
->IsShown() )
482 theToolBar
->SetSize(wxDefaultCoord
,0);
483 theToolBar
->Show(false);
485 else // prevent it from being restored later
487 style
&= ~wxFULLSCREEN_NOTOOLBAR
;
490 #endif // wxUSE_TOOLBAR
492 if (style
& wxFULLSCREEN_NOMENUBAR
)
493 SetMenu((HWND
)GetHWND(), (HMENU
) NULL
);
496 wxStatusBar
*theStatusBar
= GetStatusBar();
498 // Save the number of fields in the statusbar
499 if ((style
& wxFULLSCREEN_NOSTATUSBAR
) && theStatusBar
)
501 if ( theStatusBar
->IsShown() )
502 theStatusBar
->Show(false);
504 style
&= ~wxFULLSCREEN_NOSTATUSBAR
;
506 #endif // wxUSE_STATUSBAR
508 else // restore to normal
510 // restore the toolbar, menubar, and statusbar if we had hid them
512 wxToolBar
*theToolBar
= GetToolBar();
514 if ((m_fsStyle
& wxFULLSCREEN_NOTOOLBAR
) && theToolBar
)
516 theToolBar
->Show(true);
518 #endif // wxUSE_TOOLBAR
521 if (m_fsStyle
& wxFULLSCREEN_NOMENUBAR
)
523 const WXHMENU hmenu
= MSWGetActiveMenu();
525 ::SetMenu(GetHwnd(), (HMENU
)hmenu
);
527 #endif // wxUSE_MENUS
530 wxStatusBar
*theStatusBar
= GetStatusBar();
532 if ((m_fsStyle
& wxFULLSCREEN_NOSTATUSBAR
) && theStatusBar
)
534 theStatusBar
->Show(true);
537 #endif // wxUSE_STATUSBAR
539 #endif // !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
541 return wxFrameBase::ShowFullScreen(show
, style
);
544 // ----------------------------------------------------------------------------
545 // tool/status bar stuff
546 // ----------------------------------------------------------------------------
550 wxToolBar
* wxFrame::CreateToolBar(long style
, wxWindowID id
, const wxString
& name
)
552 #if defined(WINCE_WITHOUT_COMMANDBAR)
553 // We may already have a toolbar from calling SetMenuBar.
557 if ( wxFrameBase::CreateToolBar(style
, id
, name
) )
562 return m_frameToolBar
;
565 void wxFrame::PositionToolBar()
567 // TODO: we want to do something different in WinCE, because the toolbar
568 // should be associated with the commandbar, instead of being
569 // independent window.
570 #if !defined(WINCE_WITHOUT_COMMANDBAR)
571 wxToolBar
*toolbar
= GetToolBar();
572 if ( toolbar
&& toolbar
->IsShown() )
574 // don't call our (or even wxTopLevelWindow) version because we want
575 // the real (full) client area size, not excluding the tool/status bar
577 wxWindow::DoGetClientSize(&width
, &height
);
580 wxStatusBar
*statbar
= GetStatusBar();
581 if ( statbar
&& statbar
->IsShown() )
583 height
-= statbar
->GetClientSize().y
;
585 #endif // wxUSE_STATUSBAR
588 toolbar
->GetPosition( &tx
, &ty
);
589 toolbar
->GetSize( &tw
, &th
);
592 if ( toolbar
->HasFlag(wxTB_BOTTOM
) )
597 else if ( toolbar
->HasFlag(wxTB_RIGHT
) )
608 #if defined(WINCE_WITH_COMMANDBAR)
609 // We're using a commandbar - so we have to allow for it.
610 if (GetMenuBar() && GetMenuBar()->GetCommandBar())
613 ::GetWindowRect((HWND
) GetMenuBar()->GetCommandBar(), &rect
);
614 y
= rect
.bottom
- rect
.top
;
616 #endif // WINCE_WITH_COMMANDBAR
618 if ( toolbar
->HasFlag(wxTB_BOTTOM
) )
620 if ( ty
< 0 && ( -ty
== th
) )
622 if ( tx
< 0 && (-tx
== tw
) )
625 else if ( toolbar
->HasFlag(wxTB_RIGHT
) )
627 if( ty
< 0 && ( -ty
== th
) )
629 if( tx
< 0 && ( -tx
== tw
) )
634 if (ty
< 0 && (-ty
== th
))
636 if (tx
< 0 && (-tx
== tw
))
643 if ( toolbar
->IsVertical() )
654 // use the 'real' MSW position here, don't offset relatively to the
655 // client area origin
656 toolbar
->SetSize(x
, y
, desiredW
, desiredH
, wxSIZE_NO_ADJUSTMENTS
);
659 #endif // !WINCE_WITH_COMMANDBAR
662 #endif // wxUSE_TOOLBAR
664 // ----------------------------------------------------------------------------
665 // frame state (iconized/maximized/...)
666 // ----------------------------------------------------------------------------
668 // propagate our state change to all child frames: this allows us to emulate X
669 // Windows behaviour where child frames float independently of the parent one
670 // on the desktop, but are iconized/restored with it
671 void wxFrame::IconizeChildFrames(bool bIconize
)
673 m_iconized
= bIconize
;
675 for ( wxWindowList::compatibility_iterator node
= GetChildren().GetFirst();
677 node
= node
->GetNext() )
679 wxWindow
*win
= node
->GetData();
681 // iconizing the frames with this style under Win95 shell puts them at
682 // the bottom of the screen (as the MDI children) instead of making
683 // them appear in the taskbar because they are, by virtue of this
684 // style, not managed by the taskbar - instead leave Windows take care
686 if ( win
->GetWindowStyle() & wxFRAME_TOOL_WINDOW
)
689 // the child MDI frames are a special case and should not be touched by
690 // the parent frame - instead, they are managed by the user
691 wxFrame
*frame
= wxDynamicCast(win
, wxFrame
);
693 #if wxUSE_MDI_ARCHITECTURE
694 && !frame
->IsMDIChild()
695 #endif // wxUSE_MDI_ARCHITECTURE
698 // we don't want to restore the child frames which had been
699 // iconized even before we were iconized, so save the child frame
700 // status when iconizing the parent frame and check it when
704 frame
->m_wasMinimized
= frame
->IsIconized();
707 // note that we shouldn't touch the hidden frames neither because
708 // iconizing/restoring them would show them as a side effect
709 if ( !frame
->m_wasMinimized
&& frame
->IsShown() )
710 frame
->Iconize(bIconize
);
715 WXHICON
wxFrame::GetDefaultIcon() const
717 // we don't have any standard icons (any more)
721 // ===========================================================================
722 // message processing
723 // ===========================================================================
725 // ---------------------------------------------------------------------------
727 // ---------------------------------------------------------------------------
729 bool wxFrame::MSWDoTranslateMessage(wxFrame
*frame
, WXMSG
*pMsg
)
731 if ( wxWindow::MSWTranslateMessage(pMsg
) )
734 #if wxUSE_MENUS && wxUSE_ACCEL && !defined(__WXUNIVERSAL__)
735 // try the menu bar accelerators
736 wxMenuBar
*menuBar
= GetMenuBar();
737 if ( menuBar
&& menuBar
->GetAcceleratorTable()->Translate(frame
, pMsg
) )
739 #endif // wxUSE_MENUS && wxUSE_ACCEL
744 // ---------------------------------------------------------------------------
745 // our private (non virtual) message handlers
746 // ---------------------------------------------------------------------------
748 bool wxFrame::HandleSize(int WXUNUSED(x
), int WXUNUSED(y
), WXUINT id
)
750 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
755 // only do it it if we were iconized before, otherwise resizing the
756 // parent frame has a curious side effect of bringing it under it's
761 // restore all child frames too
762 IconizeChildFrames(false);
764 (void)SendIconizeEvent(false);
768 // iconize all child frames too
769 IconizeChildFrames(true);
774 #endif // !__WXWINCE__
780 #endif // wxUSE_STATUSBAR
784 #endif // wxUSE_TOOLBAR
786 #if defined(WINCE_WITH_COMMANDBAR)
787 // Position the menu command bar
788 if (GetMenuBar() && GetMenuBar()->GetCommandBar())
791 ::GetWindowRect((HWND
) GetMenuBar()->GetCommandBar(), &rect
);
792 wxSize clientSz
= GetClientSize();
794 if ( !::MoveWindow((HWND
) GetMenuBar()->GetCommandBar(), 0, 0, clientSz
.x
, rect
.bottom
- rect
.top
, true ) )
796 wxLogLastError(wxT("MoveWindow"));
800 #endif // WINCE_WITH_COMMANDBAR
803 // call the base class version to generate the appropriate events
807 bool wxFrame::HandleCommand(WXWORD id
, WXWORD cmd
, WXHWND control
)
811 #if defined(WINCE_WITHOUT_COMMANDBAR)
812 if (GetToolBar() && GetToolBar()->FindById(id
))
813 return GetToolBar()->MSWCommand(cmd
, id
);
816 // we only need to handle the menu and accelerator commands from the items
817 // of our menu bar, base wxWindow class already handles the rest
818 if ( !control
&& (cmd
== 0 /* menu */ || cmd
== 1 /* accel */) )
820 #if wxUSE_MENUS_NATIVE
821 if ( !wxCurrentPopupMenu
)
822 #endif // wxUSE_MENUS_NATIVE
824 wxMenuItem
* const mitem
= FindItemInMenuBar((signed short)id
);
826 return ProcessCommand(mitem
);
829 #endif // wxUSE_MENUS
831 return wxFrameBase::HandleCommand(id
, cmd
, control
);;
834 // ---------------------------------------------------------------------------
835 // the window proc for wxFrame
836 // ---------------------------------------------------------------------------
838 WXLRESULT
wxFrame::MSWWindowProc(WXUINT message
, WXWPARAM wParam
, WXLPARAM lParam
)
841 bool processed
= false;
846 // if we can't close, tell the system that we processed the
847 // message - otherwise it would close us
848 processed
= !Close();
852 processed
= HandleSize(LOWORD(lParam
), HIWORD(lParam
), wParam
);
859 UnpackCommand((WXWPARAM
)wParam
, (WXLPARAM
)lParam
,
862 HandleCommand(id
, cmd
, (WXHWND
)hwnd
);
864 // don't pass WM_COMMAND to the base class whether we processed
865 // it or not because we did generate an event for it (our
866 // HandleCommand() calls the base class version) and we must
867 // not do it again or the handlers which skip the event would
873 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__)
874 case WM_QUERYDRAGICON
:
876 const wxIcon
& icon
= GetIcon();
877 HICON hIcon
= icon
.IsOk() ? GetHiconOf(icon
)
878 : (HICON
)GetDefaultIcon();
879 rc
= (WXLRESULT
)hIcon
;
883 #endif // !__WXMICROWIN__
887 rc
= wxFrameBase::MSWWindowProc(message
, wParam
, lParam
);
892 // ----------------------------------------------------------------------------
893 // wxFrame size management: we exclude the areas taken by menu/status/toolbars
894 // from the client area, so the client area is what's really available for the
896 // ----------------------------------------------------------------------------
898 // get the origin of the client area in the client coordinates
899 wxPoint
wxFrame::GetClientAreaOrigin() const
901 wxPoint pt
= wxTopLevelWindow::GetClientAreaOrigin();
903 #if wxUSE_TOOLBAR && !defined(__WXUNIVERSAL__) && \
904 (!defined(__WXWINCE__) || (_WIN32_WCE >= 400 && !defined(__POCKETPC__) && !defined(__SMARTPHONE__)))
905 wxToolBar
* const toolbar
= GetToolBar();
906 if ( toolbar
&& toolbar
->IsShown() )
908 const wxSize sizeTB
= toolbar
->GetSize();
910 if ( toolbar
->HasFlag(wxTB_TOP
) )
914 else if ( toolbar
->HasFlag(wxTB_LEFT
) )
919 #endif // wxUSE_TOOLBAR
921 #if defined(WINCE_WITH_COMMANDBAR)
922 if (GetMenuBar() && GetMenuBar()->GetCommandBar())
925 ::GetWindowRect((HWND
) GetMenuBar()->GetCommandBar(), &rect
);
926 pt
.y
+= (rect
.bottom
- rect
.top
);