1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxMenu, wxMenuBar, wxMenuItem 
   4 // Author:      Stefan Csomor 
   8 // Copyright:   (c) Stefan Csomor 
   9 // Licence:       wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13 #pragma implementation "menu.h" 
  14 #pragma implementation "menuitem.h" 
  17 // ============================================================================ 
  18 // headers & declarations 
  19 // ============================================================================ 
  26 #include "wx/menuitem.h" 
  27 #include "wx/window.h" 
  32 #include "wx/mac/uma.h" 
  34 // other standard headers 
  35 // ---------------------- 
  38 #if !USE_SHARED_LIBRARY 
  39 IMPLEMENT_DYNAMIC_CLASS(wxMenu
, wxEvtHandler
) 
  40 IMPLEMENT_DYNAMIC_CLASS(wxMenuBar
, wxEvtHandler
) 
  43 // the (popup) menu title has this special id 
  44 static const int idMenuTitle 
= -3; 
  45 static MenuItemIndex firstUserHelpMenuItem 
= 0 ; 
  47 const short kwxMacMenuBarResource 
= 1 ; 
  48 const short kwxMacAppleMenuId 
= 1 ; 
  51 // Find an item given the Macintosh Menu Reference 
  53 wxList 
wxWinMacMenuList(wxKEY_INTEGER
); 
  54 wxMenu 
*wxFindMenuFromMacMenu(MenuRef inMenuRef
) 
  56     wxNode 
*node 
= wxWinMacMenuList
.Find((long)inMenuRef
); 
  59     return (wxMenu 
*)node
->GetData(); 
  62 void wxAssociateMenuWithMacMenu(MenuRef inMenuRef
, wxMenu 
*menu
) ; 
  63 void wxAssociateMenuWithMacMenu(MenuRef inMenuRef
, wxMenu 
*menu
) 
  65     // adding NULL MenuRef is (first) surely a result of an error and 
  66     // (secondly) breaks menu command processing 
  67     wxCHECK_RET( inMenuRef 
!= (MenuRef
) NULL
, wxT("attempt to add a NULL MenuRef to menu list") ); 
  69     if ( !wxWinMacMenuList
.Find((long)inMenuRef
) ) 
  70         wxWinMacMenuList
.Append((long)inMenuRef
, menu
); 
  73 void wxRemoveMacMenuAssociation(wxMenu 
*menu
) ; 
  74 void wxRemoveMacMenuAssociation(wxMenu 
*menu
) 
  76     wxWinMacMenuList
.DeleteObject(menu
); 
  80 // ============================================================================ 
  82 // ============================================================================ 
  83 static void wxMenubarUnsetInvokingWindow( wxMenu 
*menu 
) ; 
  84 static void wxMenubarSetInvokingWindow( wxMenu 
*menu
, wxWindow 
*win 
); 
  88 // Construct a menu with optional title (then use append) 
  91 short wxMenu::s_macNextMenuId 
= 3 ; 
  93 short wxMenu::s_macNextMenuId 
= 2 ; 
  99     m_startRadioGroup 
= -1; 
 102     m_macMenuId 
= s_macNextMenuId
++; 
 103     m_hMenu 
= UMANewMenu(m_macMenuId
, m_title
, wxFont::GetDefaultEncoding() ); 
 107         wxLogLastError(wxT("UMANewMenu failed")); 
 110     wxAssociateMenuWithMacMenu( (MenuRef
)m_hMenu 
, this ) ; 
 112     // if we have a title, insert it in the beginning of the menu 
 115         Append(idMenuTitle
, m_title
) ; 
 122     wxRemoveMacMenuAssociation( this ) ; 
 123     if (MAC_WXHMENU(m_hMenu
)) 
 124         ::DisposeMenu(MAC_WXHMENU(m_hMenu
)); 
 129     // not available on the mac platform 
 132 void wxMenu::Attach(wxMenuBarBase 
*menubar
) 
 134     wxMenuBase::Attach(menubar
); 
 139 // function appends a new item or submenu to the menu 
 140 // append a new item or submenu to the menu 
 141 bool wxMenu::DoInsertOrAppend(wxMenuItem 
*pItem
, size_t pos
) 
 143     wxASSERT_MSG( pItem 
!= NULL
, wxT("can't append NULL item to the menu") ); 
 145     if ( pItem
->IsSeparator() ) 
 147         if ( pos 
== (size_t)-1 ) 
 148             MacAppendMenu(MAC_WXHMENU(m_hMenu
), "\p-"); 
 150             MacInsertMenuItem(MAC_WXHMENU(m_hMenu
), "\p-" , pos
); 
 154         wxMenu 
*pSubMenu 
= pItem
->GetSubMenu() ; 
 155         if ( pSubMenu 
!= NULL 
) 
 157                wxASSERT_MSG( pSubMenu
->m_hMenu 
!= NULL 
, wxT("invalid submenu added")); 
 158             pSubMenu
->m_menuParent 
= this ; 
 160             if (wxMenuBar::MacGetInstalledMenuBar() == GetMenuBar()) 
 162                 pSubMenu
->MacBeforeDisplay( true ) ; 
 165             if ( pos 
== (size_t)-1 ) 
 166                 UMAAppendSubMenuItem(MAC_WXHMENU(m_hMenu
), pItem
->GetText(), wxFont::GetDefaultEncoding() , pSubMenu
->m_macMenuId
); 
 168                 UMAInsertSubMenuItem(MAC_WXHMENU(m_hMenu
), pItem
->GetText(), wxFont::GetDefaultEncoding()  , pos
, pSubMenu
->m_macMenuId
); 
 169             pItem
->UpdateItemBitmap() ; 
 170             pItem
->UpdateItemStatus() ; 
 174             if ( pos 
== (size_t)-1 ) 
 176                 UMAAppendMenuItem(MAC_WXHMENU(m_hMenu
), wxT("a") , wxFont::GetDefaultEncoding() ); 
 177                 pos 
= CountMenuItems(MAC_WXHMENU(m_hMenu
)) ; 
 181                 // MacOS counts menu items from 1 and inserts after, therefore having the 
 182                 // same effect as wx 0 based and inserting before, we must correct pos 
 183                 // after however for updates to be correct 
 184                 UMAInsertMenuItem(MAC_WXHMENU(m_hMenu
), wxT("a"), wxFont::GetDefaultEncoding(), pos
); 
 188             SetMenuItemCommandID( MAC_WXHMENU(m_hMenu
) , pos 
, pItem
->GetId() ) ; 
 189             SetMenuItemRefCon( MAC_WXHMENU(m_hMenu
) , pos 
, (UInt32
) pItem 
) ; 
 190             pItem
->UpdateItemText() ; 
 191             pItem
->UpdateItemBitmap() ; 
 192             pItem
->UpdateItemStatus() ; 
 194               if ( pItem
->GetId() == idMenuTitle 
) 
 196                 UMAEnableMenuItem(MAC_WXHMENU(m_hMenu
) , pos 
, false ) ; 
 200     // if we're already attached to the menubar, we must update it 
 203         GetMenuBar()->Refresh(); 
 208 void wxMenu::EndRadioGroup() 
 210     // we're not inside a radio group any longer 
 211     m_startRadioGroup 
= -1; 
 214 wxMenuItem
* wxMenu::DoAppend(wxMenuItem 
*item
) 
 216     wxCHECK_MSG( item
, NULL
, _T("NULL item in wxMenu::DoAppend") ); 
 220     if ( item
->GetKind() == wxITEM_RADIO 
) 
 222         int count 
= GetMenuItemCount(); 
 224         if ( m_startRadioGroup 
== -1 ) 
 226             // start a new radio group 
 227             m_startRadioGroup 
= count
; 
 229             // for now it has just one element 
 230             item
->SetAsRadioGroupStart(); 
 231             item
->SetRadioGroupEnd(m_startRadioGroup
); 
 233             // ensure that we have a checked item in the radio group 
 236         else // extend the current radio group 
 238             // we need to update its end item 
 239             item
->SetRadioGroupStart(m_startRadioGroup
); 
 240             wxMenuItemList::Node 
*node 
= GetMenuItems().Item(m_startRadioGroup
); 
 244                 node
->GetData()->SetRadioGroupEnd(count
); 
 248                 wxFAIL_MSG( _T("where is the radio group start item?") ); 
 252     else // not a radio item 
 257     if ( !wxMenuBase::DoAppend(item
) || !DoInsertOrAppend(item
) ) 
 264         // check the item initially 
 271 wxMenuItem
* wxMenu::DoInsert(size_t pos
, wxMenuItem 
*item
) 
 273     if (wxMenuBase::DoInsert(pos
, item
) && DoInsertOrAppend(item
, pos
)) 
 279 wxMenuItem 
*wxMenu::DoRemove(wxMenuItem 
*item
) 
 281     // we need to find the items position in the child list 
 283     wxMenuItemList::Node 
*node 
= GetMenuItems().GetFirst(); 
 284     for ( pos 
= 0; node
; pos
++ ) 
 286         if ( node
->GetData() == item 
) 
 289         node 
= node
->GetNext(); 
 292     // DoRemove() (unlike Remove) can only be called for existing item! 
 293     wxCHECK_MSG( node
, NULL
, wxT("bug in wxMenu::Remove logic") ); 
 295     ::DeleteMenuItem(MAC_WXHMENU(m_hMenu
) , pos 
+ 1); 
 299         // otherwise, the change won't be visible 
 300         GetMenuBar()->Refresh(); 
 303     // and from internal data structures 
 304     return wxMenuBase::DoRemove(item
); 
 307 void wxMenu::SetTitle(const wxString
& label
) 
 310     UMASetMenuTitle(MAC_WXHMENU(m_hMenu
) , label 
, wxFont::GetDefaultEncoding() ) ; 
 312 bool wxMenu::ProcessCommand(wxCommandEvent 
& event
) 
 314     bool processed 
= FALSE
; 
 316     // Try the menu's event handler 
 317     if ( !processed 
&& GetEventHandler()) 
 319         processed 
= GetEventHandler()->ProcessEvent(event
); 
 322     // Try the window the menu was popped up from (and up through the 
 324     wxWindow 
*win 
= GetInvokingWindow(); 
 325     if ( !processed 
&& win 
) 
 326         processed 
= win
->GetEventHandler()->ProcessEvent(event
); 
 332 // --------------------------------------------------------------------------- 
 334 // --------------------------------------------------------------------------- 
 336 wxWindow 
*wxMenu::GetWindow() const 
 338     if ( m_invokingWindow 
!= NULL 
) 
 339         return m_invokingWindow
; 
 340     else if ( GetMenuBar() != NULL
) 
 341         return (wxWindow 
*) GetMenuBar()->GetFrame(); 
 346 // helper functions returning the mac menu position for a certain item, note that this is 
 347 // mac-wise 1 - based, i.e. the first item has index 1 whereas on MSWin it has pos 0 
 349 int wxMenu::MacGetIndexFromId( int id 
) 
 352     wxMenuItemList::Node 
*node 
= GetMenuItems().GetFirst(); 
 353     for ( pos 
= 0; node
; pos
++ ) 
 355         if ( node
->GetData()->GetId() == id 
) 
 358         node 
= node
->GetNext(); 
 367 int wxMenu::MacGetIndexFromItem( wxMenuItem 
*pItem 
) 
 370     wxMenuItemList::Node 
*node 
= GetMenuItems().GetFirst(); 
 371     for ( pos 
= 0; node
; pos
++ ) 
 373         if ( node
->GetData() == pItem 
) 
 376         node 
= node
->GetNext(); 
 385 void wxMenu::MacEnableMenu( bool bDoEnable 
) 
 387     UMAEnableMenuItem(MAC_WXHMENU(m_hMenu
) , 0 , bDoEnable 
) ; 
 392 // MacOS needs to know about submenus somewhere within this menu 
 393 // before it can be displayed , also hide special menu items like preferences 
 394 // that are handled by the OS 
 395 void wxMenu::MacBeforeDisplay( bool isSubMenu 
) 
 397     wxMenuItem
* previousItem 
= NULL 
; 
 399     wxMenuItemList::Node 
*node
; 
 401     for (pos 
= 0, node 
= GetMenuItems().GetFirst(); node
; node 
= node
->GetNext(), pos
++) 
 403         item 
= (wxMenuItem 
*)node
->GetData(); 
 404         wxMenu
* subMenu 
= item
->GetSubMenu() ; 
 407             subMenu
->MacBeforeDisplay( true ) ; 
 412             if ( UMAGetSystemVersion() >= 0x1000 ) 
 414                 if ( item
->GetId() == wxApp::s_macPreferencesMenuItemId 
|| item
->GetId() == wxApp::s_macExitMenuItemId
) 
 416                     ChangeMenuItemAttributes( MAC_WXHMENU( GetHMenu() ) , pos 
+ 1, kMenuItemAttrHidden
, 0 ); 
 417                     if ( GetMenuItems().GetCount() == pos 
+ 1 && 
 418                             previousItem 
!= NULL 
&& 
 419                                 previousItem
->IsSeparator() ) 
 421                         ChangeMenuItemAttributes( MAC_WXHMENU( GetHMenu() ) , pos 
, kMenuItemAttrHidden
, 0 ); 
 427         previousItem 
= item 
; 
 431         ::InsertMenu(MAC_WXHMENU( GetHMenu()), -1); 
 434 // undo all changes from the MacBeforeDisplay call 
 435 void wxMenu::MacAfterDisplay( bool isSubMenu 
) 
 438         ::DeleteMenu(MacGetMenuId()); 
 440     wxMenuItem
* previousItem 
= NULL 
; 
 442     wxMenuItemList::Node 
*node
; 
 444     for (pos 
= 0, node 
= GetMenuItems().GetFirst(); node
; node 
= node
->GetNext(), pos
++) 
 446         item 
= (wxMenuItem 
*)node
->GetData(); 
 447         wxMenu
* subMenu 
= item
->GetSubMenu() ; 
 450             subMenu
->MacAfterDisplay( true ) ; 
 454             // no need to undo hidings 
 456         previousItem 
= item 
; 
 464 Mac Implementation note : 
 466 The Mac has only one global menubar, so we attempt to install the currently 
 467 active menubar from a frame, we currently don't take into account mdi-frames 
 468 which would ask for menu-merging 
 470 Secondly there is no mac api for changing a menubar that is not the current 
 471 menubar, so we have to wait for preparing the actual menubar until the 
 472 wxMenubar is to be used 
 474 We can in subsequent versions use MacInstallMenuBar to provide some sort of 
 475 auto-merge for MDI in case this will be necessary 
 479 wxMenuBar
* wxMenuBar::s_macInstalledMenuBar 
= NULL 
; 
 480 wxMenuBar
* wxMenuBar::s_macCommonMenuBar 
= NULL 
; 
 482 void wxMenuBar::Init() 
 484     m_eventHandler 
= this; 
 485     m_menuBarFrame 
= NULL
; 
 486     m_invokingWindow 
= (wxWindow
*) NULL
; 
 489 wxMenuBar::wxMenuBar() 
 494 wxMenuBar::wxMenuBar( long WXUNUSED(style
) ) 
 500 wxMenuBar::wxMenuBar(int count
, wxMenu 
*menus
[], const wxString titles
[]) 
 504     m_titles
.Alloc(count
); 
 506     for ( int i 
= 0; i 
< count
; i
++ ) 
 508         m_menus
.Append(menus
[i
]); 
 509         m_titles
.Add(titles
[i
]); 
 511         menus
[i
]->Attach(this); 
 515 wxMenuBar::~wxMenuBar() 
 517     if (s_macCommonMenuBar 
== this) 
 518         s_macCommonMenuBar 
= NULL
; 
 519     if (s_macInstalledMenuBar 
== this) 
 522         s_macInstalledMenuBar 
= NULL
; 
 527 void wxMenuBar::Refresh(bool WXUNUSED(eraseBackground
), const wxRect 
*WXUNUSED(rect
)) 
 529     wxCHECK_RET( IsAttached(), wxT("can't refresh unatteched menubar") ); 
 534 void wxMenuBar::MacInstallMenuBar() 
 536     if ( s_macInstalledMenuBar 
== this ) 
 539     MenuBarHandle menubar 
= NULL 
; 
 540 #if TARGET_API_MAC_OSX 
 541     menubar 
= NewHandleClear( 6 /* sizeof( MenuBarHeader ) */ ) ; 
 543     menubar 
= NewHandleClear( 12 ) ; 
 544     (*menubar
)[3] = 0x0a ; 
 546     ::SetMenuBar( menubar 
) ; 
 547     DisposeMenuBar( menubar 
) ; 
 548     MenuHandle appleMenu 
= NULL 
; 
 549     char appleMenuTitle
[3] = { 01 , kMenuAppleLogoFilledGlyph 
, 0 } ; 
 551     verify_noerr( CreateNewMenu( kwxMacAppleMenuId 
, 0 , &appleMenu 
) ) ; 
 552     verify_noerr( SetMenuTitle( appleMenu 
, (ConstStr255Param
) appleMenuTitle 
) ); 
 553     MacInsertMenuItem( appleMenu 
, "\pAbout..." , 0 ) ; 
 554     MacInsertMenu( appleMenu 
, 0 ) ; 
 556     // clean-up the help menu before adding new items 
 557     MenuHandle mh 
= NULL 
; 
 558     if ( UMAGetHelpMenu( &mh 
, &firstUserHelpMenuItem
) == noErr 
) 
 560         for ( int i 
= CountMenuItems( mh 
) ; i 
>= firstUserHelpMenuItem 
; --i 
) 
 562             DeleteMenuItem( mh 
, i 
) ; 
 570     if ( UMAGetSystemVersion() >= 0x1000 && wxApp::s_macPreferencesMenuItemId
) 
 572         wxMenuItem 
*item 
= FindItem( wxApp::s_macPreferencesMenuItemId 
, NULL 
) ; 
 573         if ( item 
== NULL 
|| !(item
->IsEnabled()) ) 
 574             DisableMenuCommand( NULL 
, kHICommandPreferences 
) ; 
 576             EnableMenuCommand( NULL 
, kHICommandPreferences 
) ; 
 579        for (size_t i 
= 0; i 
< m_menus
.GetCount(); i
++) 
 581         wxMenuItemList::Node 
*node
; 
 584         wxMenu
* menu 
= m_menus
[i
] , *subMenu 
= NULL 
; 
 586         if( m_titles
[i
] == wxT("?") || m_titles
[i
] == wxT("&?")  || m_titles
[i
] == wxApp::s_macHelpMenuTitleName 
) 
 593               for (pos 
= 0 , node 
= menu
->GetMenuItems().GetFirst(); node
; node 
= node
->GetNext(), pos
++) 
 595                  item 
= (wxMenuItem 
*)node
->GetData(); 
 596                  subMenu 
= item
->GetSubMenu() ; 
 599                     // we don't support hierarchical menus in the help menu yet 
 603                     if ( item
->IsSeparator() ) 
 606                             MacAppendMenu(mh
, "\p-" ); 
 610                         wxAcceleratorEntry
* entry 
= wxGetAccelFromString( item
->GetText() ) ; 
 612                         if ( item
->GetId() == wxApp::s_macAboutMenuItemId 
) 
 614                                 UMASetMenuItemText( GetMenuHandle( kwxMacAppleMenuId 
) , 1 , item
->GetText() , wxFont::GetDefaultEncoding() ); 
 615                                 UMAEnableMenuItem( GetMenuHandle( kwxMacAppleMenuId 
) , 1 , true ); 
 616                                 SetMenuItemCommandID( GetMenuHandle( kwxMacAppleMenuId 
) , 1 , item
->GetId() ) ; 
 617                                 SetMenuItemRefCon(GetMenuHandle( kwxMacAppleMenuId 
) , 1 , (UInt32
)item 
) ; 
 618                                 UMASetMenuItemShortcut( GetMenuHandle( kwxMacAppleMenuId 
) , 1 , entry 
) ; 
 624                                 UMAAppendMenuItem(mh
, item
->GetText()  , wxFont::GetDefaultEncoding(), entry
); 
 625                                 SetMenuItemCommandID( mh 
, CountMenuItems(mh
) , item
->GetId() ) ; 
 626                                 SetMenuItemRefCon( mh 
, CountMenuItems(mh
) , (UInt32
)item 
) ; 
 637             UMASetMenuTitle( MAC_WXHMENU(menu
->GetHMenu()) , m_titles
[i
], m_font
.GetEncoding()  ) ; 
 638             m_menus
[i
]->MacBeforeDisplay(false) ; 
 639             ::InsertMenu(MAC_WXHMENU(m_menus
[i
]->GetHMenu()), 0); 
 643     s_macInstalledMenuBar 
= this; 
 646 void wxMenuBar::EnableTop(size_t pos
, bool enable
) 
 648     wxCHECK_RET( IsAttached(), wxT("doesn't work with unattached menubars") ); 
 649     m_menus
[pos
]->MacEnableMenu( enable 
) ; 
 653 bool wxMenuBar::Enable( bool enable
) 
 655     wxCHECK_MSG( IsAttached(), false, wxT("doesn't work with unattached menubars") ); 
 657     for (i 
= 0; i 
< GetMenuCount(); i
++) 
 659         EnableTop(i
, enable
); 
 664 void wxMenuBar::SetLabelTop(size_t pos
, const wxString
& label
) 
 666     wxCHECK_RET( pos 
< GetMenuCount(), wxT("invalid menu index") ); 
 668     m_titles
[pos
] = label
; 
 675     m_menus
[pos
]->SetTitle( label 
) ; 
 676     if (wxMenuBar::s_macInstalledMenuBar 
== this) // are we currently installed ? 
 678         ::SetMenuBar( GetMenuBar() ) ; 
 683 wxString 
wxMenuBar::GetLabelTop(size_t pos
) const 
 685     wxCHECK_MSG( pos 
< GetMenuCount(), wxEmptyString
, 
 686                  wxT("invalid menu index in wxMenuBar::GetLabelTop") ); 
 688     return m_titles
[pos
]; 
 691 int wxMenuBar::FindMenu(const wxString
& title
) 
 693     wxString menuTitle 
= wxStripMenuCodes(title
); 
 695     size_t count 
= GetMenuCount(); 
 696     for ( size_t i 
= 0; i 
< count
; i
++ ) 
 698         wxString title 
= wxStripMenuCodes(m_titles
[i
]); 
 699         if ( menuTitle 
== title 
) 
 708 // --------------------------------------------------------------------------- 
 709 // wxMenuBar construction 
 710 // --------------------------------------------------------------------------- 
 712 // --------------------------------------------------------------------------- 
 713 // wxMenuBar construction 
 714 // --------------------------------------------------------------------------- 
 716 wxMenu 
*wxMenuBar::Replace(size_t pos
, wxMenu 
*menu
, const wxString
& title
) 
 718     wxMenu 
*menuOld 
= wxMenuBarBase::Replace(pos
, menu
, title
); 
 721     m_titles
[pos
] = title
; 
 725         if (s_macInstalledMenuBar 
== this) 
 727             menuOld
->MacAfterDisplay( false ) ; 
 728             ::DeleteMenu( menuOld
->MacGetMenuId() /* m_menus[pos]->MacGetMenuId() */ ) ; 
 730                 menu
->MacBeforeDisplay( false ) ; 
 731                 UMASetMenuTitle( MAC_WXHMENU(menu
->GetHMenu()) , title 
, m_font
.GetEncoding() ) ; 
 732                 if ( pos 
== m_menus
.GetCount() - 1) 
 734                     ::InsertMenu( MAC_WXHMENU(menu
->GetHMenu()) , 0 ) ; 
 738                     ::InsertMenu( MAC_WXHMENU(menu
->GetHMenu()) , m_menus
[pos
+1]->MacGetMenuId() ) ; 
 749 bool wxMenuBar::Insert(size_t pos
, wxMenu 
*menu
, const wxString
& title
) 
 751     if ( !wxMenuBarBase::Insert(pos
, menu
, title
) ) 
 754     m_titles
.Insert(title
, pos
); 
 756     UMASetMenuTitle( MAC_WXHMENU(menu
->GetHMenu()) , title 
, m_font
.GetEncoding() ) ; 
 758     if ( IsAttached() && s_macInstalledMenuBar 
== this ) 
 760         if (s_macInstalledMenuBar 
== this) 
 762             menu
->MacBeforeDisplay( false ) ; 
 763             if ( pos 
== (size_t) -1  || pos 
+ 1 == m_menus
.GetCount() ) 
 765                 ::InsertMenu( MAC_WXHMENU(menu
->GetHMenu()) , 0 ) ; 
 769                 ::InsertMenu( MAC_WXHMENU(menu
->GetHMenu()) , m_menus
[pos
+1]->MacGetMenuId() ) ; 
 778 wxMenu 
*wxMenuBar::Remove(size_t pos
) 
 780     wxMenu 
*menu 
= wxMenuBarBase::Remove(pos
); 
 786         if (s_macInstalledMenuBar 
== this) 
 788             ::DeleteMenu( menu
->MacGetMenuId() /* m_menus[pos]->MacGetMenuId() */ ) ; 
 794     m_titles
.RemoveAt(pos
); 
 799 bool wxMenuBar::Append(wxMenu 
*menu
, const wxString
& title
) 
 801     WXHMENU submenu 
= menu 
? menu
->GetHMenu() : 0; 
 802     wxCHECK_MSG( submenu
, FALSE
, wxT("can't append invalid menu to menubar") ); 
 804     if ( !wxMenuBarBase::Append(menu
, title
) ) 
 809     UMASetMenuTitle( MAC_WXHMENU(menu
->GetHMenu()) , title 
, m_font
.GetEncoding() ) ; 
 813         if (s_macInstalledMenuBar 
== this) 
 815             ::InsertMenu( MAC_WXHMENU(menu
->GetHMenu()) , 0 ) ; 
 821    // m_invokingWindow is set after wxFrame::SetMenuBar(). This call enables 
 822     // adding menu later on. 
 823     if (m_invokingWindow
) 
 824         wxMenubarSetInvokingWindow( menu
, m_invokingWindow 
); 
 829 static void wxMenubarUnsetInvokingWindow( wxMenu 
*menu 
) 
 831     menu
->SetInvokingWindow( (wxWindow
*) NULL 
); 
 833     wxMenuItemList::Node 
*node 
= menu
->GetMenuItems().GetFirst(); 
 836         wxMenuItem 
*menuitem 
= node
->GetData(); 
 837         if (menuitem
->IsSubMenu()) 
 838             wxMenubarUnsetInvokingWindow( menuitem
->GetSubMenu() ); 
 839         node 
= node
->GetNext(); 
 843 static void wxMenubarSetInvokingWindow( wxMenu 
*menu
, wxWindow 
*win 
) 
 845     menu
->SetInvokingWindow( win 
); 
 847     wxMenuItemList::Node 
*node 
= menu
->GetMenuItems().GetFirst(); 
 850         wxMenuItem 
*menuitem 
= node
->GetData(); 
 851         if (menuitem
->IsSubMenu()) 
 852             wxMenubarSetInvokingWindow( menuitem
->GetSubMenu() , win 
); 
 853         node 
= node
->GetNext(); 
 857 void wxMenuBar::UnsetInvokingWindow() 
 859     m_invokingWindow 
= (wxWindow
*) NULL
; 
 860     wxMenuList::Node 
*node 
= m_menus
.GetFirst(); 
 863         wxMenu 
*menu 
= node
->GetData(); 
 864         wxMenubarUnsetInvokingWindow( menu 
); 
 865         node 
= node
->GetNext(); 
 869 void wxMenuBar::SetInvokingWindow(wxFrame 
*frame
) 
 871     m_invokingWindow 
= frame
; 
 872     wxMenuList::Node 
*node 
= m_menus
.GetFirst(); 
 875         wxMenu 
*menu 
= node
->GetData(); 
 876         wxMenubarSetInvokingWindow( menu
, frame 
); 
 877         node 
= node
->GetNext(); 
 881 void wxMenuBar::Detach() 
 883     wxMenuBarBase::Detach() ; 
 886 void wxMenuBar::Attach(wxFrame 
*frame
) 
 888     wxMenuBarBase::Attach( frame 
) ; 
 890 // --------------------------------------------------------------------------- 
 891 // wxMenuBar searching for menu items 
 892 // --------------------------------------------------------------------------- 
 894 // Find the itemString in menuString, and return the item id or wxNOT_FOUND 
 895 int wxMenuBar::FindMenuItem(const wxString
& menuString
, 
 896                             const wxString
& itemString
) const 
 898     wxString menuLabel 
= wxStripMenuCodes(menuString
); 
 899     size_t count 
= GetMenuCount(); 
 900     for ( size_t i 
= 0; i 
< count
; i
++ ) 
 902         wxString title 
= wxStripMenuCodes(m_titles
[i
]); 
 903         if ( menuLabel 
== title 
) 
 904             return m_menus
[i
]->FindItem(itemString
); 
 910 wxMenuItem 
*wxMenuBar::FindItem(int id
, wxMenu 
**itemMenu
) const 
 915     wxMenuItem 
*item 
= NULL
; 
 916     size_t count 
= GetMenuCount(); 
 917     for ( size_t i 
= 0; !item 
&& (i 
< count
); i
++ ) 
 919         item 
= m_menus
[i
]->FindItem(id
, itemMenu
);