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/bitmap.h" 
  35     #include "wx/settings.h" 
  36     #include "wx/window.h" 
  38     #include "wx/string.h" 
  47 #include "wx/msw/private.h" 
  50 // Implemented in menu.cpp 
  51 UINT 
GetMenuState(HMENU hMenu
, UINT id
, UINT flags
) ; 
  54 // --------------------------------------------------------------------------- 
  56 // --------------------------------------------------------------------------- 
  59 #define GetHMenuOf(menu)    ((HMENU)menu->GetHMenu()) 
  61 // conditional compilation 
  63     #define OWNER_DRAWN_ONLY( code ) if ( IsOwnerDrawn() ) code 
  64 #else // !wxUSE_OWNER_DRAWN 
  65     #define OWNER_DRAWN_ONLY( code ) 
  66 #endif // wxUSE_OWNER_DRAWN/!wxUSE_OWNER_DRAWN 
  68 // ============================================================================ 
  70 // ============================================================================ 
  72 // ---------------------------------------------------------------------------- 
  73 // dynamic classes implementation 
  74 // ---------------------------------------------------------------------------- 
  76 #if wxUSE_EXTENDED_RTTI 
  78 bool wxMenuItemStreamingCallback( const wxObject 
*object
, wxWriter 
* , wxPersister 
* , wxxVariantArray 
& ) 
  80     const wxMenuItem 
* mitem 
= dynamic_cast<const wxMenuItem
*>(object
) ; 
  81     if ( mitem
->GetMenu() && !mitem
->GetMenu()->GetTitle().empty() ) 
  83         // we don't stream out the first two items for menus with a title, they will be reconstructed 
  84         if ( mitem
->GetMenu()->FindItemByPosition(0) == mitem 
|| mitem
->GetMenu()->FindItemByPosition(1) == mitem 
) 
  90 wxBEGIN_ENUM( wxItemKind 
) 
  91     wxENUM_MEMBER( wxITEM_SEPARATOR 
) 
  92     wxENUM_MEMBER( wxITEM_NORMAL 
) 
  93     wxENUM_MEMBER( wxITEM_CHECK 
) 
  94     wxENUM_MEMBER( wxITEM_RADIO 
) 
  95 wxEND_ENUM( wxItemKind 
) 
  97 IMPLEMENT_DYNAMIC_CLASS_XTI_CALLBACK(wxMenuItem
, wxObject
,"wx/menuitem.h",wxMenuItemStreamingCallback
) 
  99 wxBEGIN_PROPERTIES_TABLE(wxMenuItem
) 
 100     wxPROPERTY( Parent
,wxMenu
*, SetMenu
, GetMenu
, EMPTY_MACROVALUE 
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") ) 
 101     wxPROPERTY( Id
,int, SetId
, GetId
, EMPTY_MACROVALUE 
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") ) 
 102     wxPROPERTY( Text
, wxString 
, SetText
, GetText
, wxString(), 0 /*flags*/ , wxT("Helpstring") , wxT("group") ) 
 103     wxPROPERTY( Help
, wxString 
, SetHelp
, GetHelp
, wxString(), 0 /*flags*/ , wxT("Helpstring") , wxT("group") ) 
 104     wxREADONLY_PROPERTY( Kind
, wxItemKind 
, GetKind 
, EMPTY_MACROVALUE 
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") ) 
 105     wxPROPERTY( SubMenu
,wxMenu
*, SetSubMenu
, GetSubMenu
, EMPTY_MACROVALUE 
, 0 /*flags*/ , wxT("Helpstring") , wxT("group") ) 
 106     wxPROPERTY( Enabled 
, bool , Enable 
, IsEnabled 
, wxxVariant((bool)true) , 0 /*flags*/ , wxT("Helpstring") , wxT("group")) 
 107     wxPROPERTY( Checked 
, bool , Check 
, IsChecked 
, wxxVariant((bool)false) , 0 /*flags*/ , wxT("Helpstring") , wxT("group")) 
 108     wxPROPERTY( Checkable 
, bool , SetCheckable 
, IsCheckable 
, wxxVariant((bool)false) , 0 /*flags*/ , wxT("Helpstring") , wxT("group")) 
 109 wxEND_PROPERTIES_TABLE() 
 111 wxBEGIN_HANDLERS_TABLE(wxMenuItem
) 
 112 wxEND_HANDLERS_TABLE() 
 114 wxDIRECT_CONSTRUCTOR_6( wxMenuItem 
, wxMenu
* , Parent 
, int , Id 
, wxString 
, Text 
, wxString 
, Help 
, wxItemKind 
, Kind 
, wxMenu
* , SubMenu  
) 
 116 IMPLEMENT_DYNAMIC_CLASS(wxMenuItem
, wxObject
) 
 119 // ---------------------------------------------------------------------------- 
 121 // ---------------------------------------------------------------------------- 
 126 wxMenuItem::wxMenuItem(wxMenu 
*pParentMenu
, 
 128                        const wxString
& text
, 
 129                        const wxString
& strHelp
, 
 132           : wxMenuItemBase(pParentMenu
, id
, text
, strHelp
, kind
, pSubMenu
) 
 133 #if wxUSE_OWNER_DRAWN 
 134             , wxOwnerDrawn(text
, kind 
== wxITEM_CHECK
, true) 
 135 #endif // owner drawn 
 140 wxMenuItem::wxMenuItem(wxMenu 
*parentMenu
, 
 142                        const wxString
& text
, 
 143                        const wxString
& help
, 
 146           : wxMenuItemBase(parentMenu
, id
, text
, help
, 
 147                            isCheckable 
? wxITEM_CHECK 
: wxITEM_NORMAL
, subMenu
) 
 148 #if wxUSE_OWNER_DRAWN 
 149            , wxOwnerDrawn(text
, isCheckable
, true) 
 150 #endif // owner drawn 
 155 void wxMenuItem::Init() 
 157     m_radioGroup
.start 
= -1; 
 158     m_isRadioGroupStart 
= false; 
 160 #if  wxUSE_OWNER_DRAWN 
 162     // when the color is not valid, wxOwnerDraw takes the default ones. 
 163     // If we set the colors here and they are changed by the user during 
 164     // the execution, then the colors are not updated until the application 
 165     // is restarted and our menus look bad 
 166     SetTextColour(wxNullColour
); 
 167     SetBackgroundColour(wxNullColour
); 
 169     // setting default colors switched ownerdraw on: switch it off again 
 172     //  switch ownerdraw back on if using a non default margin 
 173     if ( GetId() != wxID_SEPARATOR 
) 
 174         SetMarginWidth(GetMarginWidth()); 
 176     // tell the owner drawing code to show the accel string as well 
 177     SetAccelString(m_text
.AfterFirst(_T('\t'))); 
 178 #endif // wxUSE_OWNER_DRAWN 
 181 wxMenuItem::~wxMenuItem() 
 188 // return the id for calling Win32 API functions 
 189 int wxMenuItem::GetRealId() const 
 191     return m_subMenu 
? (int)m_subMenu
->GetHMenu() : GetId(); 
 197 bool wxMenuItem::IsChecked() const 
 199     // fix that RTTI is always getting the correct state (separators cannot be checked, but the call below 
 201     if ( GetId() == wxID_SEPARATOR 
) 
 204     int flag 
= ::GetMenuState(GetHMenuOf(m_parentMenu
), GetId(), MF_BYCOMMAND
); 
 206     return (flag 
& MF_CHECKED
) != 0; 
 210 wxString 
wxMenuItemBase::GetLabelFromText(const wxString
& text
) 
 212     return wxStripMenuCodes(text
); 
 218 void wxMenuItem::SetAsRadioGroupStart() 
 220     m_isRadioGroupStart 
= true; 
 223 void wxMenuItem::SetRadioGroupStart(int start
) 
 225     wxASSERT_MSG( !m_isRadioGroupStart
, 
 226                   _T("should only be called for the next radio items") ); 
 228     m_radioGroup
.start 
= start
; 
 231 void wxMenuItem::SetRadioGroupEnd(int end
) 
 233     wxASSERT_MSG( m_isRadioGroupStart
, 
 234                   _T("should only be called for the first radio item") ); 
 236     m_radioGroup
.end 
= end
; 
 242 void wxMenuItem::Enable(bool enable
) 
 244     if ( m_isEnabled 
== enable 
) 
 247     long rc 
= EnableMenuItem(GetHMenuOf(m_parentMenu
), 
 250                              (enable 
? MF_ENABLED 
: MF_GRAYED
)); 
 253         wxLogLastError(wxT("EnableMenuItem")); 
 256     wxMenuItemBase::Enable(enable
); 
 259 void wxMenuItem::Check(bool check
) 
 261     wxCHECK_RET( IsCheckable(), wxT("only checkable items may be checked") ); 
 263     if ( m_isChecked 
== check 
) 
 266     int flags 
= check 
? MF_CHECKED 
: MF_UNCHECKED
; 
 267     HMENU hmenu 
= GetHMenuOf(m_parentMenu
); 
 269     if ( GetKind() == wxITEM_RADIO 
) 
 271         // it doesn't make sense to uncheck a radio item - what would this do? 
 275         // get the index of this item in the menu 
 276         const wxMenuItemList
& items 
= m_parentMenu
->GetMenuItems(); 
 277         int pos 
= items
.IndexOf(this); 
 278         wxCHECK_RET( pos 
!= wxNOT_FOUND
, 
 279                      _T("menuitem not found in the menu items list?") ); 
 281         // get the radio group range 
 285         if ( m_isRadioGroupStart 
) 
 287             // we already have all information we need 
 289             end 
= m_radioGroup
.end
; 
 291         else // next radio group item 
 293             // get the radio group end from the start item 
 294             start 
= m_radioGroup
.start
; 
 295             end 
= items
.Item(start
)->GetData()->m_radioGroup
.end
; 
 299         // calling CheckMenuRadioItem() with such parameters hangs my system 
 300         // (NT4 SP6) and I suspect this could happen to the others as well - so 
 302         wxCHECK_RET( start 
!= -1 && end 
!= -1, 
 303                      _T("invalid ::CheckMenuRadioItem() parameter(s)") ); 
 305         if ( !::CheckMenuRadioItem(hmenu
, 
 306                                    start
,   // the first radio group item 
 308                                    pos
,     // the one to check 
 311             wxLogLastError(_T("CheckMenuRadioItem")); 
 315         // also uncheck all the other items in this radio group 
 316         wxMenuItemList::compatibility_iterator node 
= items
.Item(start
); 
 317         for ( int n 
= start
; n 
<= end 
&& node
; n
++ ) 
 321                 node
->GetData()->m_isChecked 
= false; 
 324             node 
= node
->GetNext(); 
 329         if ( ::CheckMenuItem(hmenu
, 
 331                              MF_BYCOMMAND 
| flags
) == (DWORD
)-1 ) 
 333             wxFAIL_MSG( _T("CheckMenuItem() failed, item not in the menu?") ); 
 337     wxMenuItemBase::Check(check
); 
 340 void wxMenuItem::SetText(const wxString
& txt
) 
 344     // don't do anything if label didn't change 
 348     // wxMenuItemBase will do stock ID checks 
 349     wxMenuItemBase::SetText(text
); 
 351     // m_text could now be different from 'text' if we are a stock menu item, 
 352     // so use only m_text below 
 354     OWNER_DRAWN_ONLY( wxOwnerDrawn::SetName(m_text
) ); 
 355 #if wxUSE_OWNER_DRAWN 
 356     // tell the owner drawing code to to show the accel string as well 
 357     SetAccelString(m_text
.AfterFirst(_T('\t'))); 
 360     HMENU hMenu 
= GetHMenuOf(m_parentMenu
); 
 361     wxCHECK_RET( hMenu
, wxT("menuitem without menu") ); 
 364     m_parentMenu
->UpdateAccel(this); 
 365 #endif // wxUSE_ACCEL 
 367     UINT id 
= GetRealId(); 
 368     UINT flagsOld 
= ::GetMenuState(hMenu
, id
, MF_BYCOMMAND
); 
 369     if ( flagsOld 
== 0xFFFFFFFF ) 
 371         // It's not an error, it means that the menu item doesn't exist 
 372         //wxLogLastError(wxT("GetMenuState")); 
 378             // high byte contains the number of items in a submenu for submenus 
 380             flagsOld 
|= MF_POPUP
; 
 385 #if wxUSE_OWNER_DRAWN 
 386         if ( IsOwnerDrawn() ) 
 388             flagsOld 
|= MF_OWNERDRAW
; 
 389             data 
= (LPCTSTR
)this; 
 394             flagsOld 
|= MF_STRING
; 
 395             data 
= (wxChar
*) m_text
.c_str(); 
 399         // FIXME: complete this, applying the old 
 401         // However, the WinCE doc for SetMenuItemInfo 
 402         // says that you can't use it to set the menu 
 403         // item state; only data, id and type. 
 406         info
.cbSize 
= sizeof(info
); 
 407         info
.fMask 
= MIIM_TYPE
; 
 408         info
.fType 
= MFT_STRING
; 
 409         info
.cch 
= m_text
.length(); 
 410         info
.dwTypeData 
= (LPTSTR
) data 
; 
 411         if ( !::SetMenuItemInfo(hMenu
, id
, FALSE
, & info
) ) 
 413             wxLogLastError(wxT("SetMenuItemInfo")); 
 416         if ( ::ModifyMenu(hMenu
, id
, 
 417                           MF_BYCOMMAND 
| flagsOld
, 
 418                           id
, data
) == (int)0xFFFFFFFF ) 
 420             wxLogLastError(wxT("ModifyMenu")); 
 426 void wxMenuItem::SetCheckable(bool checkable
) 
 428     wxMenuItemBase::SetCheckable(checkable
); 
 429     OWNER_DRAWN_ONLY( wxOwnerDrawn::SetCheckable(checkable
) ); 
 432 // ---------------------------------------------------------------------------- 
 434 // ---------------------------------------------------------------------------- 
 436 wxMenuItem 
*wxMenuItemBase::New(wxMenu 
*parentMenu
, 
 438                                 const wxString
& name
, 
 439                                 const wxString
& help
, 
 443     return new wxMenuItem(parentMenu
, id
, name
, help
, kind
, subMenu
); 
 446 #endif // wxUSE_MENUS