1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/menuitem.cpp
3 // Purpose: wxMenuItem implementation
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
12 // ===========================================================================
14 // ===========================================================================
16 // ---------------------------------------------------------------------------
18 // ---------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
29 #include "wx/menuitem.h"
30 #include "wx/stockitem.h"
34 #include "wx/dcmemory.h"
36 #include "wx/bitmap.h"
37 #include "wx/settings.h"
38 #include "wx/window.h"
40 #include "wx/string.h"
49 #include "wx/msw/private.h"
50 #include "wx/msw/dc.h"
53 // Implemented in menu.cpp
54 UINT
GetMenuState(HMENU hMenu
, UINT id
, UINT flags
) ;
58 #include "wx/msw/uxtheme.h"
61 // ---------------------------------------------------------------------------
63 // ---------------------------------------------------------------------------
66 #define GetHMenuOf(menu) ((HMENU)menu->GetHMenu())
68 // ============================================================================
70 // ============================================================================
74 #include "wx/fontutil.h"
75 #include "wx/msw/private/metrics.h"
77 #ifndef SPI_GETKEYBOARDCUES
78 #define SPI_GETKEYBOARDCUES 0x100A
81 #ifndef DSS_HIDEPREFIX
82 #define DSS_HIDEPREFIX 0x0200
89 MENU_MENUITEM_TMSCHEMA
= 1,
90 MENU_SEPARATOR_TMSCHEMA
= 6,
91 MENU_POPUPBACKGROUND
= 9,
92 MENU_POPUPBORDERS
= 10,
94 MENU_POPUPCHECKBACKGROUND
= 12,
95 MENU_POPUPGUTTER
= 13,
97 MENU_POPUPSEPARATOR
= 15,
98 MENU_POPUPSUBMENU
= 16,
110 enum POPUPCHECKBACKGROUNDSTATES
117 enum POPUPCHECKSTATES
119 MC_CHECKMARKNORMAL
= 1,
120 MC_CHECKMARKDISABLED
= 2,
122 MC_BULLETDISABLED
= 4,
125 const int TMT_MENUFONT
= 803;
126 const int TMT_BORDERSIZE
= 2403;
127 const int TMT_CONTENTMARGINS
= 3602;
128 const int TMT_SIZINGMARGINS
= 3601;
130 #endif // wxUSE_UXTHEME
132 #endif // wxUSE_OWNER_DRAWN
134 // ----------------------------------------------------------------------------
135 // dynamic classes implementation
136 // ----------------------------------------------------------------------------
138 #if wxUSE_EXTENDED_RTTI
140 bool wxMenuItemStreamingCallback( const wxObject
*object
, wxWriter
* , wxPersister
* , wxxVariantArray
& )
142 const wxMenuItem
* mitem
= dynamic_cast<const wxMenuItem
*>(object
) ;
143 if ( mitem
->GetMenu() && !mitem
->GetMenu()->GetTitle().empty() )
145 // we don't stream out the first two items for menus with a title, they will be reconstructed
146 if ( mitem
->GetMenu()->FindItemByPosition(0) == mitem
|| mitem
->GetMenu()->FindItemByPosition(1) == mitem
)
152 wxBEGIN_ENUM( wxItemKind
)
153 wxENUM_MEMBER( wxITEM_SEPARATOR
)
154 wxENUM_MEMBER( wxITEM_NORMAL
)
155 wxENUM_MEMBER( wxITEM_CHECK
)
156 wxENUM_MEMBER( wxITEM_RADIO
)
157 wxEND_ENUM( wxItemKind
)
159 IMPLEMENT_DYNAMIC_CLASS_XTI_CALLBACK(wxMenuItem
, wxObject
,"wx/menuitem.h",wxMenuItemStreamingCallback
)
161 wxBEGIN_PROPERTIES_TABLE(wxMenuItem
)
162 wxPROPERTY( Parent
,wxMenu
*, SetMenu
, GetMenu
, EMPTY_MACROVALUE
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") )
163 wxPROPERTY( Id
,int, SetId
, GetId
, EMPTY_MACROVALUE
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") )
164 wxPROPERTY( Text
, wxString
, SetText
, GetText
, wxString(), 0 /*flags*/ , wxT("Helpstring") , wxT("group") )
165 wxPROPERTY( Help
, wxString
, SetHelp
, GetHelp
, wxString(), 0 /*flags*/ , wxT("Helpstring") , wxT("group") )
166 wxREADONLY_PROPERTY( Kind
, wxItemKind
, GetKind
, EMPTY_MACROVALUE
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") )
167 wxPROPERTY( SubMenu
,wxMenu
*, SetSubMenu
, GetSubMenu
, EMPTY_MACROVALUE
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") )
168 wxPROPERTY( Enabled
, bool , Enable
, IsEnabled
, wxxVariant((bool)true) , 0 /*flags*/ , wxT("Helpstring") , wxT("group"))
169 wxPROPERTY( Checked
, bool , Check
, IsChecked
, wxxVariant((bool)false) , 0 /*flags*/ , wxT("Helpstring") , wxT("group"))
170 wxPROPERTY( Checkable
, bool , SetCheckable
, IsCheckable
, wxxVariant((bool)false) , 0 /*flags*/ , wxT("Helpstring") , wxT("group"))
171 wxEND_PROPERTIES_TABLE()
173 wxBEGIN_HANDLERS_TABLE(wxMenuItem
)
174 wxEND_HANDLERS_TABLE()
176 wxDIRECT_CONSTRUCTOR_6( wxMenuItem
, wxMenu
* , Parent
, int , Id
, wxString
, Text
, wxString
, Help
, wxItemKind
, Kind
, wxMenu
* , SubMenu
)
178 IMPLEMENT_DYNAMIC_CLASS(wxMenuItem
, wxObject
)
181 // ----------------------------------------------------------------------------
183 // ----------------------------------------------------------------------------
185 #if wxUSE_OWNER_DRAWN
190 // helper class to keep information about metrics and other stuff
191 // needed for measuring and drawing menu item
211 Margins ItemMargin
; // popup item margins
213 Margins CheckMargin
; // popup check margins
214 Margins CheckBgMargin
; // popup check background margins
216 Margins ArrowMargin
; // popup submenu arrow margins
218 Margins SeparatorMargin
; // popup separator margins
220 SIZE CheckSize
; // popup check size metric
221 SIZE ArrowSize
; // popup submenu arrow size metric
222 SIZE SeparatorSize
; // popup separator size metric
224 int TextBorder
; // popup border space between
225 // item text and gutter
227 int AccelBorder
; // popup border space between
228 // item text and accelerator
230 int ArrowBorder
; // popup border space between
231 // item accelerator and submenu arrow
233 int Offset
; // system added space at the end of the menu,
234 // add this offset for remove the extra space
236 wxFont Font
; // default menu font
238 bool AlwaysShowCues
; // must keyboard cues always be shown?
240 bool Theme
; // is data initialized for FullTheme?
242 static const MenuDrawData
* Get()
244 // notice that s_menuData can't be created as a global variable because
245 // it needs a window to initialize and no windows exist at the time of
246 // globals initialization yet
249 static MenuDrawData s_menuData
;
250 ms_instance
= &s_menuData
;
254 bool theme
= MenuLayout() == FullTheme
;
255 if ( ms_instance
->Theme
!= theme
)
257 #endif // wxUSE_UXTHEME
267 // get the theme engine or NULL if themes
268 // are not available or not supported on menu
269 static wxUxThemeEngine
*GetUxThemeEngine()
272 if ( MenuLayout() == FullTheme
)
273 return wxUxThemeEngine::GetIfActive();
274 #endif // wxUSE_UXTHEME
281 FullTheme
, // full menu themes (Vista or new)
282 PseudoTheme
, // pseudo menu themes (on XP)
286 static MenuLayoutType
MenuLayout()
288 MenuLayoutType menu
= Classic
;
290 if ( wxUxThemeEngine::GetIfActive() != NULL
)
292 static wxWinVersion ver
= wxGetWinVersion();
293 if ( ver
>= wxWinVersion_Vista
)
295 else if ( ver
== wxWinVersion_XP
)
298 #endif // wxUSE_UXTHEME
305 static MenuDrawData
* ms_instance
;
308 MenuDrawData
* MenuDrawData::ms_instance
= NULL
;
310 void MenuDrawData::Init()
313 wxUxThemeEngine
* theme
= GetUxThemeEngine();
316 wxWindow
* window
= static_cast<wxApp
*>(wxApp::GetInstance())->GetTopWindow();
317 wxUxThemeHandle
hTheme(window
, L
"MENU");
319 theme
->GetThemeMargins(hTheme
, NULL
, MENU_POPUPITEM
, 0,
320 TMT_CONTENTMARGINS
, NULL
,
321 reinterpret_cast<MARGINS
*>(&ItemMargin
));
323 theme
->GetThemeMargins(hTheme
, NULL
, MENU_POPUPCHECK
, 0,
324 TMT_CONTENTMARGINS
, NULL
,
325 reinterpret_cast<MARGINS
*>(&CheckMargin
));
326 theme
->GetThemeMargins(hTheme
, NULL
, MENU_POPUPCHECKBACKGROUND
, 0,
327 TMT_CONTENTMARGINS
, NULL
,
328 reinterpret_cast<MARGINS
*>(&CheckBgMargin
));
330 theme
->GetThemeMargins(hTheme
, NULL
, MENU_POPUPSUBMENU
, 0,
331 TMT_CONTENTMARGINS
, NULL
,
332 reinterpret_cast<MARGINS
*>(&ArrowMargin
));
334 theme
->GetThemeMargins(hTheme
, NULL
, MENU_POPUPSEPARATOR
, 0,
335 TMT_SIZINGMARGINS
, NULL
,
336 reinterpret_cast<MARGINS
*>(&SeparatorMargin
));
338 theme
->GetThemePartSize(hTheme
, NULL
, MENU_POPUPCHECK
, 0,
339 NULL
, TS_TRUE
, &CheckSize
);
341 theme
->GetThemePartSize(hTheme
, NULL
, MENU_POPUPSUBMENU
, 0,
342 NULL
, TS_TRUE
, &ArrowSize
);
344 theme
->GetThemePartSize(hTheme
, NULL
, MENU_POPUPSEPARATOR
, 0,
345 NULL
, TS_TRUE
, &SeparatorSize
);
347 theme
->GetThemeInt(hTheme
, MENU_POPUPBACKGROUND
, 0, TMT_BORDERSIZE
, &TextBorder
);
354 wxNativeFontInfo fontInfo
;
355 theme
->GetThemeSysFont(hTheme
, TMT_MENUFONT
, &fontInfo
.lf
);
356 Font
= wxFont(fontInfo
);
360 // native menu doesn't uses the vertical margins
361 ItemMargin
.top
= ItemMargin
.bottom
= 0;
363 // native menu uses small top margin for separator
364 if ( SeparatorMargin
.top
>= 2 )
365 SeparatorMargin
.top
-= 2;
368 #endif // wxUSE_UXTHEME
370 const NONCLIENTMETRICS
& metrics
= wxMSWImpl::GetNonClientMetrics();
372 ItemMargin
= Margins();
375 CheckMargin
.right
= ::GetSystemMetrics(SM_CXEDGE
);
377 CheckMargin
.bottom
= ::GetSystemMetrics(SM_CYEDGE
);
379 CheckBgMargin
= Margins();
381 CheckSize
.cx
= ::GetSystemMetrics(SM_CXMENUCHECK
);
382 CheckSize
.cy
= ::GetSystemMetrics(SM_CYMENUCHECK
);
384 ArrowMargin
= Margins();
386 ArrowSize
= CheckSize
;
388 // separator height with margins
389 int sepFullSize
= metrics
.iMenuHeight
/ 2;
391 SeparatorMargin
.left
=
392 SeparatorMargin
.right
= 1;
393 SeparatorMargin
.top
=
394 SeparatorMargin
.bottom
= sepFullSize
/ 2 - 1;
396 SeparatorSize
.cx
= 1;
397 SeparatorSize
.cy
= sepFullSize
- SeparatorMargin
.top
- SeparatorMargin
.bottom
;
405 Font
= wxFont(wxNativeFontInfo(metrics
.lfMenuFont
));
411 if ( ::SystemParametersInfo(SPI_GETKEYBOARDCUES
, 0, &value
, 0) == 0 )
413 // if it's not supported, we must be on an old Windows version
414 // which always shows them
418 AlwaysShowCues
= value
== 1;
422 } // anonymous namespace
424 #endif // wxUSE_OWNER_DRAWN
430 wxMenuItem::wxMenuItem(wxMenu
*pParentMenu
,
432 const wxString
& text
,
433 const wxString
& strHelp
,
436 : wxMenuItemBase(pParentMenu
, id
, text
, strHelp
, kind
, pSubMenu
)
441 #if WXWIN_COMPATIBILITY_2_8
442 wxMenuItem::wxMenuItem(wxMenu
*parentMenu
,
444 const wxString
& text
,
445 const wxString
& help
,
448 : wxMenuItemBase(parentMenu
, id
, text
, help
,
449 isCheckable
? wxITEM_CHECK
: wxITEM_NORMAL
, subMenu
)
455 void wxMenuItem::Init()
457 m_radioGroup
.start
= -1;
458 m_isRadioGroupStart
= false;
460 #if wxUSE_OWNER_DRAWN
462 // when the color is not valid, wxOwnerDraw takes the default ones.
463 // If we set the colors here and they are changed by the user during
464 // the execution, then the colors are not updated until the application
465 // is restarted and our menus look bad
466 SetTextColour(wxNullColour
);
467 SetBackgroundColour(wxNullColour
);
469 // setting default colors switched ownerdraw on: switch it off again
470 SetOwnerDrawn(false);
472 // switch ownerdraw back on if using a non default margin
473 if ( !IsSeparator() )
474 SetMarginWidth(GetMarginWidth());
476 #endif // wxUSE_OWNER_DRAWN
479 wxMenuItem::~wxMenuItem()
486 // return the id for calling Win32 API functions
487 WXWPARAM
wxMenuItem::GetMSWId() const
489 // we must use ids in unsigned short range with Windows functions, if we
490 // pass ids > USHRT_MAX to them they get very confused (e.g. start
491 // generating WM_COMMAND messages with negative high word of wParam), so
492 // use the cast to ensure the id is in range
493 return m_subMenu
? wxPtrToUInt(m_subMenu
->GetHMenu())
494 : static_cast<unsigned short>(GetId());
500 bool wxMenuItem::IsChecked() const
502 // fix that RTTI is always getting the correct state (separators cannot be
503 // checked, but the Windows call below returns true
507 // the item might not be attached to a menu yet
509 // TODO: shouldn't we just always call the base class version? It seems
510 // like it ought to always be in sync
512 return wxMenuItemBase::IsChecked();
514 HMENU hmenu
= GetHMenuOf(m_parentMenu
);
515 int flag
= ::GetMenuState(hmenu
, GetMSWId(), MF_BYCOMMAND
);
517 return (flag
& MF_CHECKED
) != 0;
523 void wxMenuItem::SetAsRadioGroupStart()
525 m_isRadioGroupStart
= true;
528 void wxMenuItem::SetRadioGroupStart(int start
)
530 wxASSERT_MSG( !m_isRadioGroupStart
,
531 wxT("should only be called for the next radio items") );
533 m_radioGroup
.start
= start
;
536 void wxMenuItem::SetRadioGroupEnd(int end
)
538 wxASSERT_MSG( m_isRadioGroupStart
,
539 wxT("should only be called for the first radio item") );
541 m_radioGroup
.end
= end
;
547 void wxMenuItem::Enable(bool enable
)
549 if ( m_isEnabled
== enable
)
554 long rc
= EnableMenuItem(GetHMenuOf(m_parentMenu
),
557 (enable
? MF_ENABLED
: MF_GRAYED
));
561 wxLogLastError(wxT("EnableMenuItem"));
565 wxMenuItemBase::Enable(enable
);
568 void wxMenuItem::Check(bool check
)
570 wxCHECK_RET( IsCheckable(), wxT("only checkable items may be checked") );
572 if ( m_isChecked
== check
)
577 int flags
= check
? MF_CHECKED
: MF_UNCHECKED
;
578 HMENU hmenu
= GetHMenuOf(m_parentMenu
);
580 if ( GetKind() == wxITEM_RADIO
)
582 // it doesn't make sense to uncheck a radio item -- what would this
587 // get the index of this item in the menu
588 const wxMenuItemList
& items
= m_parentMenu
->GetMenuItems();
589 int pos
= items
.IndexOf(this);
590 wxCHECK_RET( pos
!= wxNOT_FOUND
,
591 wxT("menuitem not found in the menu items list?") );
593 // get the radio group range
597 if ( m_isRadioGroupStart
)
599 // we already have all information we need
601 end
= m_radioGroup
.end
;
603 else // next radio group item
605 // get the radio group end from the start item
606 start
= m_radioGroup
.start
;
607 end
= items
.Item(start
)->GetData()->m_radioGroup
.end
;
611 // calling CheckMenuRadioItem() with such parameters hangs my system
612 // (NT4 SP6) and I suspect this could happen to the others as well,
614 wxCHECK_RET( start
!= -1 && end
!= -1,
615 wxT("invalid ::CheckMenuRadioItem() parameter(s)") );
617 if ( !::CheckMenuRadioItem(hmenu
,
618 start
, // the first radio group item
620 pos
, // the one to check
623 wxLogLastError(wxT("CheckMenuRadioItem"));
627 // also uncheck all the other items in this radio group
628 wxMenuItemList::compatibility_iterator node
= items
.Item(start
);
629 for ( int n
= start
; n
<= end
&& node
; n
++ )
633 node
->GetData()->m_isChecked
= false;
636 node
= node
->GetNext();
641 if ( ::CheckMenuItem(hmenu
,
643 MF_BYCOMMAND
| flags
) == (DWORD
)-1 )
645 wxFAIL_MSG(wxT("CheckMenuItem() failed, item not in the menu?"));
650 wxMenuItemBase::Check(check
);
653 void wxMenuItem::SetItemLabel(const wxString
& txt
)
657 // don't do anything if label didn't change
661 // wxMenuItemBase will do stock ID checks
662 wxMenuItemBase::SetItemLabel(text
);
664 // the item can be not attached to any menu yet and SetItemLabel() is still
665 // valid to call in this case and should do nothing else
670 m_parentMenu
->UpdateAccel(this);
671 #endif // wxUSE_ACCEL
673 const UINT id
= GetMSWId();
674 HMENU hMenu
= GetHMenuOf(m_parentMenu
);
675 if ( !hMenu
|| ::GetMenuState(hMenu
, id
, MF_BYCOMMAND
) == (UINT
)-1 )
678 #if wxUSE_OWNER_DRAWN
679 if ( IsOwnerDrawn() )
681 // we don't need to do anything for owner drawn items, they will redraw
682 // themselves using the new text the next time they're displayed
685 #endif // owner drawn
687 // update the text of the native menu item
688 WinStruct
<MENUITEMINFO
> info
;
690 // surprisingly, calling SetMenuItemInfo() with just MIIM_STRING doesn't
691 // work as it resets the menu bitmap, so we need to first get the old item
692 // state and then modify it
693 const bool isLaterThanWin95
= wxGetWinVersion() > wxWinVersion_95
;
694 info
.fMask
= MIIM_STATE
|
699 if ( isLaterThanWin95
)
700 info
.fMask
|= MIIM_BITMAP
| MIIM_FTYPE
;
702 info
.fMask
|= MIIM_TYPE
;
703 if ( !::GetMenuItemInfo(hMenu
, id
, FALSE
, &info
) )
705 wxLogLastError(wxT("GetMenuItemInfo"));
709 if ( isLaterThanWin95
)
710 info
.fMask
|= MIIM_STRING
;
711 //else: MIIM_TYPE already specified
712 info
.dwTypeData
= (LPTSTR
)m_text
.wx_str();
713 info
.cch
= m_text
.length();
714 if ( !::SetMenuItemInfo(hMenu
, id
, FALSE
, &info
) )
716 wxLogLastError(wxT("SetMenuItemInfo"));
720 #if wxUSE_OWNER_DRAWN
722 int wxMenuItem::MeasureAccelWidth() const
724 wxString accel
= GetItemLabel().AfterFirst(wxT('\t'));
732 dc
.GetTextExtent(accel
, &w
, NULL
);
737 wxString
wxMenuItem::GetName() const
739 return GetItemLabelText();
742 bool wxMenuItem::OnMeasureItem(size_t *width
, size_t *height
)
744 const MenuDrawData
* data
= MenuDrawData::Get();
746 if ( IsOwnerDrawn() )
748 *width
= data
->ItemMargin
.left
+ data
->ItemMargin
.right
;
749 *height
= data
->ItemMargin
.top
+ data
->ItemMargin
.bottom
;
753 *width
+= data
->SeparatorSize
.cx
754 + data
->SeparatorMargin
.left
+ data
->SeparatorMargin
.right
;
755 *height
+= data
->SeparatorSize
.cy
756 + data
->SeparatorMargin
.top
+ data
->SeparatorMargin
.bottom
;
760 wxString str
= GetName();
768 dc
.GetTextExtent(str
, &w
, &h
);
770 *width
= data
->TextBorder
+ w
+ data
->AccelBorder
;
773 w
= m_parentMenu
->GetMaxAccelWidth();
775 *width
+= w
+ data
->ArrowBorder
;
777 *width
+= data
->Offset
;
778 *width
+= data
->ArrowMargin
.left
+ data
->ArrowSize
.cx
+ data
->ArrowMargin
.right
;
780 else // don't draw the text, just the bitmap (if any)
788 if ( IsOwnerDrawn() )
790 // width of menu icon with margins in ownerdrawn menu
791 // if any bitmap is not set, the width of space reserved for icon
792 // image is equal to the width of std check mark,
793 // if bitmap is set, then the width is set to the width of the widest
794 // bitmap in menu (GetMarginWidth()) unless std check mark is wider,
795 // then it's is set to std mark's width
796 int imgWidth
= wxMax(GetMarginWidth(), data
->CheckSize
.cx
)
797 + data
->CheckMargin
.left
+ data
->CheckMargin
.right
;
799 *width
+= imgWidth
+ data
->CheckBgMargin
.left
+ data
->CheckBgMargin
.right
;
802 if ( m_bmpChecked
.IsOk() || m_bmpChecked
.IsOk() )
804 // get size of bitmap always return valid value (0 for invalid bitmap),
805 // so we don't needed check if bitmap is valid ;)
806 size_t heightBmp
= wxMax(m_bmpChecked
.GetHeight(), m_bmpUnchecked
.GetHeight());
807 size_t widthtBmp
= wxMax(m_bmpChecked
.GetWidth(), m_bmpUnchecked
.GetWidth());
809 if ( IsOwnerDrawn() )
811 heightBmp
+= data
->CheckMargin
.top
+ data
->CheckMargin
.bottom
;
815 // we must allocate enough space for the bitmap
819 // Is BMP height larger than text height?
820 if ( *height
< heightBmp
)
824 // make sure that this item is at least as tall as the system menu height
825 const size_t menuHeight
= data
->CheckMargin
.top
+ data
->CheckMargin
.bottom
826 + data
->CheckSize
.cy
;
827 if (*height
< menuHeight
)
828 *height
= menuHeight
;
833 bool wxMenuItem::OnDrawItem(wxDC
& dc
, const wxRect
& rc
,
834 wxODAction
WXUNUSED(act
), wxODStatus stat
)
836 const MenuDrawData
* data
= MenuDrawData::Get();
838 wxMSWDCImpl
*impl
= (wxMSWDCImpl
*) dc
.GetImpl();
839 HDC hdc
= GetHdcOf(*impl
);
842 wxCopyRectToRECT(rc
, rect
);
844 int imgWidth
= wxMax(GetMarginWidth(), data
->CheckSize
.cx
);
846 if ( IsOwnerDrawn() )
848 // font and colors to use
852 wxColour colText1
, colBack1
;
853 GetColourToUse(stat
, colText1
, colBack1
);
855 DWORD colText
= wxColourToPalRGB(colText1
);
856 DWORD colBack
= wxColourToPalRGB(colBack1
);
858 // calculate metrics of item parts
864 SetRect(&rcSelection
,
865 rect
.left
+ data
->ItemMargin
.left
,
866 rect
.top
+ data
->ItemMargin
.top
,
867 rect
.right
- data
->ItemMargin
.right
,
868 rect
.bottom
- data
->ItemMargin
.bottom
);
870 SetRect(&rcSeparator
,
871 rcSelection
.left
+ data
->SeparatorMargin
.left
,
872 rcSelection
.top
+ data
->SeparatorMargin
.top
,
873 rcSelection
.right
- data
->SeparatorMargin
.right
,
874 rcSelection
.bottom
- data
->SeparatorMargin
.bottom
);
876 CopyRect(&rcGutter
, &rcSelection
);
877 rcGutter
.right
= data
->ItemMargin
.left
878 + data
->CheckBgMargin
.left
879 + data
->CheckMargin
.left
881 + data
->CheckMargin
.right
882 + data
->CheckBgMargin
.right
;
884 CopyRect(&rcText
, &rcSelection
);
885 rcText
.left
= rcGutter
.right
+ data
->TextBorder
;
887 // we draw the text label vertically centered, but this results in it
888 // being 1px too low compared to native menus for some reason, fix it
889 if ( data
->MenuLayout() != MenuDrawData::FullTheme
)
893 wxUxThemeEngine
* theme
= MenuDrawData::GetUxThemeEngine();
896 POPUPITEMSTATES state
;
897 if ( stat
& wxODDisabled
)
899 state
= (stat
& wxODSelected
) ? MPI_DISABLEDHOT
902 else if ( stat
& wxODSelected
)
911 wxUxThemeHandle
hTheme(GetMenu()->GetWindow(), L
"MENU");
913 if ( theme
->IsThemeBackgroundPartiallyTransparent(hTheme
,
914 MENU_POPUPITEM
, state
) )
916 theme
->DrawThemeBackground(hTheme
, hdc
,
917 MENU_POPUPBACKGROUND
,
921 theme
->DrawThemeBackground(hTheme
, hdc
, MENU_POPUPGUTTER
,
926 rcSeparator
.left
= rcGutter
.right
;
927 theme
->DrawThemeBackground(hTheme
, hdc
, MENU_POPUPSEPARATOR
,
928 0, &rcSeparator
, NULL
);
932 theme
->DrawThemeBackground(hTheme
, hdc
, MENU_POPUPITEM
,
933 state
, &rcSelection
, NULL
);
937 #endif // wxUSE_UXTHEME
941 DrawEdge(hdc
, &rcSeparator
, EDGE_ETCHED
, BF_TOP
);
945 AutoHBRUSH
hbr(colBack
);
946 SelectInHDC
selBrush(hdc
, hbr
);
947 ::FillRect(hdc
, &rcSelection
, hbr
);
952 // using native API because it recognizes '&'
954 COLORREF colOldText
= ::SetTextColor(hdc
, colText
);
955 COLORREF colOldBack
= ::SetBkColor(hdc
, colBack
);
957 int prevMode
= SetBkMode(hdc
, TRANSPARENT
);
959 SelectInHDC
selFont(hdc
, GetHfontOf(font
));
962 // item text name without menemonic for calculating size
963 wxString text
= GetName();
966 ::GetTextExtentPoint32(hdc
, text
.c_str(), text
.length(), &textSize
);
968 // item text name with menemonic
969 text
= GetItemLabel().BeforeFirst('\t');
971 int flags
= DST_PREFIXTEXT
;
972 // themes menu is using specified color for disabled labels
973 if ( data
->MenuLayout() == MenuDrawData::Classic
&&
974 (stat
& wxODDisabled
) && !(stat
& wxODSelected
) )
975 flags
|= DSS_DISABLED
;
977 if ( (stat
& wxODHidePrefix
) && !data
->AlwaysShowCues
)
978 flags
|= DSS_HIDEPREFIX
;
981 int y
= rcText
.top
+ (rcText
.bottom
- rcText
.top
- textSize
.cy
) / 2;
983 ::DrawState(hdc
, NULL
, NULL
, (LPARAM
)text
.wx_str(),
984 text
.length(), x
, y
, 0, 0, flags
);
986 // ::SetTextAlign(hdc, TA_RIGHT) doesn't work with DSS_DISABLED or DSS_MONO
987 // as the last parameter in DrawState() (at least with Windows98). So we have
988 // to take care of right alignment ourselves.
989 wxString accel
= GetItemLabel().AfterFirst(wxT('\t'));
990 if ( !accel
.empty() )
993 ::GetTextExtentPoint32(hdc
, accel
.c_str(), accel
.length(), &accelSize
);
995 int flags
= DST_TEXT
;
996 // themes menu is using specified color for disabled labels
997 if ( data
->MenuLayout() == MenuDrawData::Classic
&&
998 (stat
& wxODDisabled
) && !(stat
& wxODSelected
) )
999 flags
|= DSS_DISABLED
;
1001 int x
= rcText
.right
- data
->ArrowMargin
.left
1002 - data
->ArrowSize
.cx
1003 - data
->ArrowMargin
.right
1004 - data
->ArrowBorder
;
1006 // right align accel on FullTheme menu, left otherwise
1007 if ( data
->MenuLayout() == MenuDrawData::FullTheme
)
1010 x
-= m_parentMenu
->GetMaxAccelWidth();
1012 int y
= rcText
.top
+ (rcText
.bottom
- rcText
.top
- accelSize
.cy
) / 2;
1014 ::DrawState(hdc
, NULL
, NULL
, (LPARAM
)accel
.wx_str(),
1015 accel
.length(), x
, y
, 0, 0, flags
);
1018 ::SetBkMode(hdc
, prevMode
);
1019 ::SetBkColor(hdc
, colOldBack
);
1020 ::SetTextColor(hdc
, colOldText
);
1028 rect
.left
+ data
->ItemMargin
.left
1029 + data
->CheckBgMargin
.left
1030 + data
->CheckMargin
.left
,
1031 rect
.top
+ data
->ItemMargin
.top
1032 + data
->CheckBgMargin
.top
1033 + data
->CheckMargin
.top
,
1034 rect
.left
+ data
->ItemMargin
.left
1035 + data
->CheckBgMargin
.left
1036 + data
->CheckMargin
.left
1038 rect
.bottom
- data
->ItemMargin
.bottom
1039 - data
->CheckBgMargin
.bottom
1040 - data
->CheckMargin
.bottom
);
1042 if ( IsCheckable() && !m_bmpChecked
.Ok() )
1044 if ( stat
& wxODChecked
)
1046 DrawStdCheckMark((WXHDC
)hdc
, &rcImg
, stat
);
1053 if ( stat
& wxODDisabled
)
1055 bmp
= GetDisabledBitmap();
1060 // for not checkable bitmaps we should always use unchecked one
1061 // because their checked bitmap is not set
1062 bmp
= GetBitmap(!IsCheckable() || (stat
& wxODChecked
));
1065 if ( bmp
.Ok() && stat
& wxODDisabled
)
1067 // we need to grey out the bitmap as we don't have any specific
1069 wxImage imgGrey
= bmp
.ConvertToImage().ConvertToGreyscale();
1071 bmp
= wxBitmap(imgGrey
);
1073 #endif // wxUSE_IMAGE
1078 wxMemoryDC
dcMem(&dc
);
1079 dcMem
.SelectObjectAsSource(bmp
);
1082 int nBmpWidth
= bmp
.GetWidth(),
1083 nBmpHeight
= bmp
.GetHeight();
1085 // there should be enough space!
1086 wxASSERT( nBmpWidth
<= imgWidth
&& nBmpHeight
<= (rcImg
.bottom
- rcImg
.top
) );
1088 int x
= rcImg
.left
+ (imgWidth
- nBmpWidth
) / 2;
1089 int y
= rcImg
.top
+ (rcImg
.bottom
- rcImg
.top
- nBmpHeight
) / 2;
1090 dc
.Blit(x
, y
, nBmpWidth
, nBmpHeight
, &dcMem
, 0, 0, wxCOPY
, true);
1101 // helper function for draw coloured check mark
1102 void DrawColorCheckMark(HDC hdc
, int x
, int y
, int cx
, int cy
, HDC hdcCheckMask
, int idxColor
)
1104 const COLORREF colBlack
= RGB(0, 0, 0);
1105 const COLORREF colWhite
= RGB(255, 255, 255);
1107 COLORREF colOldText
= ::SetTextColor(hdc
, colBlack
);
1108 COLORREF colOldBack
= ::SetBkColor(hdc
, colWhite
);
1109 int prevMode
= SetBkMode(hdc
, TRANSPARENT
);
1111 // memory DC for color bitmap
1112 MemoryHDC
hdcMem(hdc
);
1113 CompatibleBitmap
hbmpMem(hdc
, cx
, cy
);
1114 SelectInHDC
selMem(hdcMem
, hbmpMem
);
1116 RECT rect
= { 0, 0, cx
, cy
};
1117 ::FillRect(hdcMem
, &rect
, ::GetSysColorBrush(idxColor
));
1119 const COLORREF colCheck
= ::GetSysColor(idxColor
);
1120 if ( colCheck
== colWhite
)
1122 ::BitBlt(hdc
, x
, y
, cx
, cy
, hdcCheckMask
, 0, 0, MERGEPAINT
);
1123 ::BitBlt(hdc
, x
, y
, cx
, cy
, hdcMem
, 0, 0, SRCAND
);
1127 if ( colCheck
!= colBlack
)
1129 const DWORD ROP_DSna
= 0x00220326; // dest = (NOT src) AND dest
1130 ::BitBlt(hdcMem
, 0, 0, cx
, cy
, hdcCheckMask
, 0, 0, ROP_DSna
);
1133 ::BitBlt(hdc
, x
, y
, cx
, cy
, hdcCheckMask
, 0, 0, SRCAND
);
1134 ::BitBlt(hdc
, x
, y
, cx
, cy
, hdcMem
, 0, 0, SRCPAINT
);
1137 ::SetBkMode(hdc
, prevMode
);
1138 ::SetBkColor(hdc
, colOldBack
);
1139 ::SetTextColor(hdc
, colOldText
);
1142 } // anonymous namespace
1144 void wxMenuItem::DrawStdCheckMark(WXHDC hdc_
, const RECT
* rc
, wxODStatus stat
)
1146 HDC hdc
= (HDC
)hdc_
;
1149 wxUxThemeEngine
* theme
= MenuDrawData::GetUxThemeEngine();
1152 wxUxThemeHandle
hTheme(GetMenu()->GetWindow(), L
"MENU");
1154 const MenuDrawData
* data
= MenuDrawData::Get();
1156 // rect for background must be without check margins
1159 rc
->left
- data
->CheckMargin
.left
,
1160 rc
->top
- data
->CheckMargin
.top
,
1161 rc
->right
+ data
->CheckMargin
.right
,
1162 rc
->bottom
+ data
->CheckMargin
.bottom
);
1164 POPUPCHECKBACKGROUNDSTATES stateCheckBg
= (stat
& wxODDisabled
)
1168 theme
->DrawThemeBackground(hTheme
, hdc
, MENU_POPUPCHECKBACKGROUND
,
1169 stateCheckBg
, &rcBg
, NULL
);
1171 POPUPCHECKSTATES stateCheck
;
1172 if ( GetKind() == wxITEM_CHECK
)
1174 stateCheck
= (stat
& wxODDisabled
) ? MC_CHECKMARKDISABLED
1175 : MC_CHECKMARKNORMAL
;
1179 stateCheck
= (stat
& wxODDisabled
) ? MC_BULLETDISABLED
1183 theme
->DrawThemeBackground(hTheme
, hdc
, MENU_POPUPCHECK
,
1184 stateCheck
, rc
, NULL
);
1187 #endif // wxUSE_UXTHEME
1189 int cx
= rc
->right
- rc
->left
;
1190 int cy
= rc
->bottom
- rc
->top
;
1192 // first create mask of check mark
1193 MemoryHDC
hdcMask(hdc
);
1194 MonoBitmap
hbmpMask(cx
, cy
);
1195 SelectInHDC
selMask(hdcMask
,hbmpMask
);
1197 // then draw a check mark into it
1198 UINT stateCheck
= (GetKind() == wxITEM_CHECK
) ? DFCS_MENUCHECK
1200 RECT rect
= { 0, 0, cx
, cy
};
1201 ::DrawFrameControl(hdcMask
, &rect
, DFC_MENU
, stateCheck
);
1203 // first draw shadow if disabled
1204 if ( (stat
& wxODDisabled
) && !(stat
& wxODSelected
) )
1206 DrawColorCheckMark(hdc
, rc
->left
+ 1, rc
->top
+ 1,
1207 cx
, cy
, hdcMask
, COLOR_3DHILIGHT
);
1210 // then draw a check mark
1211 int color
= COLOR_MENUTEXT
;
1212 if ( stat
& wxODDisabled
)
1213 color
= COLOR_BTNSHADOW
;
1214 else if ( stat
& wxODSelected
)
1215 color
= COLOR_HIGHLIGHTTEXT
;
1217 DrawColorCheckMark(hdc
, rc
->left
, rc
->top
, cx
, cy
, hdcMask
, color
);
1221 void wxMenuItem::GetFontToUse(wxFont
& font
) const
1225 font
= MenuDrawData::Get()->Font
;
1228 void wxMenuItem::GetColourToUse(wxODStatus stat
, wxColour
& colText
, wxColour
& colBack
) const
1231 wxUxThemeEngine
* theme
= MenuDrawData::GetUxThemeEngine();
1234 wxUxThemeHandle
hTheme(GetMenu()->GetWindow(), L
"MENU");
1236 if ( stat
& wxODDisabled
)
1238 wxRGBToColour(colText
, theme
->GetThemeSysColor(hTheme
, COLOR_GRAYTEXT
));
1242 colText
= GetTextColour();
1243 if ( !colText
.IsOk() )
1244 wxRGBToColour(colText
, theme
->GetThemeSysColor(hTheme
, COLOR_MENUTEXT
));
1247 if ( stat
& wxODSelected
)
1249 wxRGBToColour(colBack
, theme
->GetThemeSysColor(hTheme
, COLOR_HIGHLIGHT
));
1253 colBack
= GetBackgroundColour();
1254 if ( !colBack
.IsOk() )
1255 wxRGBToColour(colBack
, theme
->GetThemeSysColor(hTheme
, COLOR_MENU
));
1259 #endif // wxUSE_UXTHEME
1261 wxOwnerDrawn::GetColourToUse(stat
, colText
, colBack
);
1264 #endif // wxUSE_OWNER_DRAWN
1266 // ----------------------------------------------------------------------------
1268 // ----------------------------------------------------------------------------
1270 wxMenuItem
*wxMenuItemBase::New(wxMenu
*parentMenu
,
1272 const wxString
& name
,
1273 const wxString
& help
,
1277 return new wxMenuItem(parentMenu
, id
, name
, help
, kind
, subMenu
);
1280 #endif // wxUSE_MENUS