1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk1/menu.cpp 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  14 #include "wx/stockitem.h" 
  20     #include "wx/bitmap.h" 
  27 #include "wx/gtk1/private.h" 
  29 #include <gdk/gdkkeysyms.h> 
  31 #define ACCEL_OBJECT        GtkObject 
  32 #define ACCEL_OBJECTS(a)    (a)->attach_objects 
  33 #define ACCEL_OBJ_CAST(obj) GTK_OBJECT(obj) 
  35 // we use normal item but with a special id for the menu title 
  36 static const int wxGTK_TITLE_ID 
= -3; 
  38 // defined in window.cpp 
  39 extern guint32 wxGtkTimeLastClick
; 
  41 //----------------------------------------------------------------------------- 
  43 //----------------------------------------------------------------------------- 
  45 extern void wxapp_install_idle_handler(); 
  49 static wxString 
GetGtkHotKey( const wxMenuItem
& item 
); 
  52 //----------------------------------------------------------------------------- 
  54 //----------------------------------------------------------------------------- 
  56 static wxString 
wxReplaceUnderscore( const wxString
& title 
) 
  60     // GTK 1.2 wants to have "_" instead of "&" for accelerators 
  63     while (*pc 
!= wxT('\0')) 
  65         if ((*pc 
== wxT('&')) && (*(pc
+1) == wxT('&'))) 
  67             // "&" is doubled to indicate "&" instead of accelerator 
  71         else if (*pc 
== wxT('&')) 
  77             if ( *pc 
== wxT('_') ) 
  79                 // underscores must be doubled to prevent them from being 
  80                 // interpreted as accelerator character prefix by GTK 
  89     // wxPrintf( wxT("before %s after %s\n"), title.c_str(), str.c_str() ); 
  94 static wxString 
wxConvertFromGTKToWXLabel(const wxString
& gtkLabel
) 
  97     for ( const wxChar 
*pc 
= gtkLabel
.c_str(); *pc
; pc
++ ) 
  99         // '_' is the escape character for GTK+. 
 101         if ( *pc 
== wxT('_') && *(pc
+1) == wxT('_')) 
 103             // An underscore was escaped. 
 107         else if ( *pc 
== wxT('_') ) 
 109             // Convert GTK+ hotkey symbol to wxWidgets/Windows standard 
 112         else if ( *pc 
== wxT('&') ) 
 114             // Double the ampersand to escape it as far as wxWidgets is concerned 
 119             // don't remove ampersands '&' since if we have them in the menu title 
 120             // it means that they were doubled to indicate "&" instead of accelerator 
 129 //----------------------------------------------------------------------------- 
 130 // activate message from GTK 
 131 //----------------------------------------------------------------------------- 
 133 static void DoCommonMenuCallbackCode(wxMenu 
*menu
, wxMenuEvent
& event
) 
 136         wxapp_install_idle_handler(); 
 138     event
.SetEventObject( menu 
); 
 140     wxEvtHandler
* handler 
= menu
->GetEventHandler(); 
 141     if (handler 
&& handler
->ProcessEvent(event
)) 
 144     wxWindow 
*win 
= menu
->GetInvokingWindow(); 
 146         win
->HandleWindowEvent( event 
); 
 151 static void gtk_menu_open_callback( GtkWidget 
*WXUNUSED(widget
), wxMenu 
*menu 
) 
 153     wxMenuEvent 
event(wxEVT_MENU_OPEN
, -1, menu
); 
 155     DoCommonMenuCallbackCode(menu
, event
); 
 158 static void gtk_menu_close_callback( GtkWidget 
*WXUNUSED(widget
), wxMenuBar 
*menubar 
) 
 160     if ( !menubar
->GetMenuCount() ) 
 162         // if menubar is empty we can't call GetMenu(0) below 
 166     wxMenuEvent 
event( wxEVT_MENU_CLOSE
, -1, NULL 
); 
 168     DoCommonMenuCallbackCode(menubar
->GetMenu(0), event
); 
 173 //----------------------------------------------------------------------------- 
 175 //----------------------------------------------------------------------------- 
 177 IMPLEMENT_DYNAMIC_CLASS(wxMenuBar
,wxWindow
) 
 179 void wxMenuBar::Init(size_t n
, wxMenu 
*menus
[], const wxString titles
[], long style
) 
 181     // the parent window is known after wxFrame::SetMenu() 
 182     m_needParent 
= false; 
 184     m_invokingWindow 
= NULL
; 
 186     if (!PreCreation( NULL
, wxDefaultPosition
, wxDefaultSize 
) || 
 187         !CreateBase( NULL
, -1, wxDefaultPosition
, wxDefaultSize
, style
, wxDefaultValidator
, wxT("menubar") )) 
 189         wxFAIL_MSG( wxT("wxMenuBar creation failed") ); 
 193     m_menubar 
= gtk_menu_bar_new(); 
 194     m_accel 
= gtk_accel_group_new(); 
 196     if (style 
& wxMB_DOCKABLE
) 
 198         m_widget 
= gtk_handle_box_new(); 
 199         gtk_container_add( GTK_CONTAINER(m_widget
), GTK_WIDGET(m_menubar
) ); 
 200         gtk_widget_show( GTK_WIDGET(m_menubar
) ); 
 204         m_widget 
= GTK_WIDGET(m_menubar
); 
 211     for (size_t i 
= 0; i 
< n
; ++i 
) 
 212         Append(menus
[i
], titles
[i
]); 
 214     // VZ: for some reason connecting to menus "deactivate" doesn't work (we 
 215     //     don't get it when the menu is dismissed by clicking outside the 
 216     //     toolbar) so we connect to the global one, even if it means that we 
 217     //     can't pass the menu which was closed in wxMenuEvent object 
 218     gtk_signal_connect( GTK_OBJECT(GTK_MENU_SHELL(m_menubar
)), 
 220                         GTK_SIGNAL_FUNC(gtk_menu_close_callback
), 
 225 wxMenuBar::wxMenuBar(size_t n
, wxMenu 
*menus
[], const wxString titles
[], long style
) 
 227     Init(n
, menus
, titles
, style
); 
 230 wxMenuBar::wxMenuBar(long style
) 
 232     Init(0, NULL
, NULL
, style
); 
 235 wxMenuBar::wxMenuBar() 
 237     Init(0, NULL
, NULL
, 0); 
 240 wxMenuBar::~wxMenuBar() 
 244 static void wxMenubarUnsetInvokingWindow( wxMenu 
*menu
, wxWindow 
*win 
) 
 246     menu
->SetInvokingWindow( NULL 
); 
 248     wxWindow 
*top_frame 
= win
; 
 249     while (top_frame
->GetParent() && !(top_frame
->IsTopLevel())) 
 250         top_frame 
= top_frame
->GetParent(); 
 252     // support for native hot keys 
 253     gtk_accel_group_detach( menu
->m_accel
, ACCEL_OBJ_CAST(top_frame
->m_widget
) ); 
 255     wxMenuItemList::compatibility_iterator node 
= menu
->GetMenuItems().GetFirst(); 
 258         wxMenuItem 
*menuitem 
= node
->GetData(); 
 259         if (menuitem
->IsSubMenu()) 
 260             wxMenubarUnsetInvokingWindow( menuitem
->GetSubMenu(), win 
); 
 261         node 
= node
->GetNext(); 
 265 static void wxMenubarSetInvokingWindow( wxMenu 
*menu
, wxWindow 
*win 
) 
 267     menu
->SetInvokingWindow( win 
); 
 269     wxWindow 
*top_frame 
= win
; 
 270     while (top_frame
->GetParent() && !(top_frame
->IsTopLevel())) 
 271         top_frame 
= top_frame
->GetParent(); 
 273     // support for native hot keys 
 274     ACCEL_OBJECT 
*obj 
= ACCEL_OBJ_CAST(top_frame
->m_widget
); 
 275     if ( !g_slist_find( ACCEL_OBJECTS(menu
->m_accel
), obj 
) ) 
 276         gtk_accel_group_attach( menu
->m_accel
, obj 
); 
 278     wxMenuItemList::compatibility_iterator node 
= menu
->GetMenuItems().GetFirst(); 
 281         wxMenuItem 
*menuitem 
= node
->GetData(); 
 282         if (menuitem
->IsSubMenu()) 
 283             wxMenubarSetInvokingWindow( menuitem
->GetSubMenu(), win 
); 
 284         node 
= node
->GetNext(); 
 288 void wxMenuBar::SetInvokingWindow( wxWindow 
*win 
) 
 290     m_invokingWindow 
= win
; 
 291     wxWindow 
*top_frame 
= win
; 
 292     while (top_frame
->GetParent() && !(top_frame
->IsTopLevel())) 
 293         top_frame 
= top_frame
->GetParent(); 
 295     // support for native key accelerators indicated by underscroes 
 296     ACCEL_OBJECT 
*obj 
= ACCEL_OBJ_CAST(top_frame
->m_widget
); 
 297     if ( !g_slist_find( ACCEL_OBJECTS(m_accel
), obj 
) ) 
 298         gtk_accel_group_attach( m_accel
, obj 
); 
 300     wxMenuList::compatibility_iterator node 
= m_menus
.GetFirst(); 
 303         wxMenu 
*menu 
= node
->GetData(); 
 304         wxMenubarSetInvokingWindow( menu
, win 
); 
 305         node 
= node
->GetNext(); 
 309 void wxMenuBar::UnsetInvokingWindow( wxWindow 
*win 
) 
 311     m_invokingWindow 
= NULL
; 
 312     wxWindow 
*top_frame 
= win
; 
 313     while (top_frame
->GetParent() && !(top_frame
->IsTopLevel())) 
 314         top_frame 
= top_frame
->GetParent(); 
 316     // support for native key accelerators indicated by underscroes 
 317     gtk_accel_group_detach( m_accel
, ACCEL_OBJ_CAST(top_frame
->m_widget
) ); 
 319     wxMenuList::compatibility_iterator node 
= m_menus
.GetFirst(); 
 322         wxMenu 
*menu 
= node
->GetData(); 
 323         wxMenubarUnsetInvokingWindow( menu
, win 
); 
 324         node 
= node
->GetNext(); 
 328 bool wxMenuBar::Append( wxMenu 
*menu
, const wxString 
&title 
) 
 330     if ( !wxMenuBarBase::Append( menu
, title 
) ) 
 333     return GtkAppend(menu
, title
); 
 336 bool wxMenuBar::GtkAppend(wxMenu 
*menu
, const wxString
& title
, int pos
) 
 338     wxString 
str( wxReplaceUnderscore( title 
) ); 
 340     // This doesn't have much effect right now. 
 341     menu
->SetTitle( str 
); 
 343     // The "m_owner" is the "menu item" 
 344     menu
->m_owner 
= gtk_menu_item_new_with_label( wxGTK_CONV( str 
) ); 
 345     GtkLabel 
*label 
= GTK_LABEL( GTK_BIN(menu
->m_owner
)->child 
); 
 347     gtk_label_set_text( label
, wxGTK_CONV( str 
) ); 
 349     guint accel_key 
= gtk_label_parse_uline (GTK_LABEL(label
), wxGTK_CONV( str 
) ); 
 350     if (accel_key 
!= GDK_VoidSymbol
) 
 352         gtk_widget_add_accelerator (menu
->m_owner
, 
 354                                     m_accel
, //gtk_menu_ensure_uline_accel_group(GTK_MENU(m_menubar)), 
 360     gtk_widget_show( menu
->m_owner 
); 
 362     gtk_menu_item_set_submenu( GTK_MENU_ITEM(menu
->m_owner
), menu
->m_menu 
); 
 365         gtk_menu_shell_append( GTK_MENU_SHELL(m_menubar
), menu
->m_owner 
); 
 367         gtk_menu_shell_insert( GTK_MENU_SHELL(m_menubar
), menu
->m_owner
, pos 
); 
 369     gtk_signal_connect( GTK_OBJECT(menu
->m_owner
), "activate", 
 370                         GTK_SIGNAL_FUNC(gtk_menu_open_callback
), 
 373     // m_invokingWindow is set after wxFrame::SetMenuBar(). This call enables 
 374     // addings menu later on. 
 375     if (m_invokingWindow
) 
 377         wxMenubarSetInvokingWindow( menu
, m_invokingWindow 
); 
 379             // OPTIMISE ME:  we should probably cache this, or pass it 
 380             //               directly, but for now this is a minimal 
 381             //               change to validate the new dynamic sizing. 
 382             //               see (and refactor :) similar code in Remove 
 385         wxFrame 
*frame 
= wxDynamicCast( m_invokingWindow
, wxFrame 
); 
 388             frame
->UpdateMenuBarSize(); 
 394 bool wxMenuBar::Insert(size_t pos
, wxMenu 
*menu
, const wxString
& title
) 
 396     if ( !wxMenuBarBase::Insert(pos
, menu
, title
) ) 
 401     if ( !GtkAppend(menu
, title
, (int)pos
) ) 
 407 wxMenu 
*wxMenuBar::Replace(size_t pos
, wxMenu 
*menu
, const wxString
& title
) 
 409     // remove the old item and insert a new one 
 410     wxMenu 
*menuOld 
= Remove(pos
); 
 411     if ( menuOld 
&& !Insert(pos
, menu
, title
) ) 
 416     // either Insert() succeeded or Remove() failed and menuOld is NULL 
 420 wxMenu 
*wxMenuBar::Remove(size_t pos
) 
 422     wxMenu 
*menu 
= wxMenuBarBase::Remove(pos
); 
 426     gtk_menu_item_remove_submenu( GTK_MENU_ITEM(menu
->m_owner
) ); 
 427     gtk_container_remove(GTK_CONTAINER(m_menubar
), menu
->m_owner
); 
 429     gtk_widget_destroy( menu
->m_owner 
); 
 430     menu
->m_owner 
= NULL
; 
 432     if (m_invokingWindow
) 
 434         // OPTIMISE ME:  see comment in GtkAppend 
 435         wxFrame 
*frame 
= wxDynamicCast( m_invokingWindow
, wxFrame 
); 
 438             frame
->UpdateMenuBarSize(); 
 444 static int FindMenuItemRecursive( const wxMenu 
*menu
, const wxString 
&menuString
, const wxString 
&itemString 
) 
 446     if (wxMenuItem::GetLabelText(wxConvertFromGTKToWXLabel(menu
->GetTitle())) == wxMenuItem::GetLabelText(menuString
)) 
 448         int res 
= menu
->FindItem( itemString 
); 
 449         if (res 
!= wxNOT_FOUND
) 
 453     wxMenuItemList::compatibility_iterator node 
= menu
->GetMenuItems().GetFirst(); 
 456         wxMenuItem 
*item 
= node
->GetData(); 
 457         if (item
->IsSubMenu()) 
 458             return FindMenuItemRecursive(item
->GetSubMenu(), menuString
, itemString
); 
 460         node 
= node
->GetNext(); 
 466 int wxMenuBar::FindMenuItem( const wxString 
&menuString
, const wxString 
&itemString 
) const 
 468     wxMenuList::compatibility_iterator node 
= m_menus
.GetFirst(); 
 471         wxMenu 
*menu 
= node
->GetData(); 
 472         int res 
= FindMenuItemRecursive( menu
, menuString
, itemString
); 
 475         node 
= node
->GetNext(); 
 481 // Find a wxMenuItem using its id. Recurses down into sub-menus 
 482 static wxMenuItem
* FindMenuItemByIdRecursive(const wxMenu
* menu
, int id
) 
 484     wxMenuItem
* result 
= menu
->FindChildItem(id
); 
 486     wxMenuItemList::compatibility_iterator node 
= menu
->GetMenuItems().GetFirst(); 
 487     while ( node 
&& result 
== NULL 
) 
 489         wxMenuItem 
*item 
= node
->GetData(); 
 490         if (item
->IsSubMenu()) 
 492             result 
= FindMenuItemByIdRecursive( item
->GetSubMenu(), id 
); 
 494         node 
= node
->GetNext(); 
 500 wxMenuItem
* wxMenuBar::FindItem( int id
, wxMenu 
**menuForItem 
) const 
 502     wxMenuItem
* result 
= 0; 
 503     wxMenuList::compatibility_iterator node 
= m_menus
.GetFirst(); 
 504     while (node 
&& result 
== 0) 
 506         wxMenu 
*menu 
= node
->GetData(); 
 507         result 
= FindMenuItemByIdRecursive( menu
, id 
); 
 508         node 
= node
->GetNext(); 
 513         *menuForItem 
= result 
? result
->GetMenu() : NULL
; 
 519 void wxMenuBar::EnableTop( size_t pos
, bool flag 
) 
 521     wxMenuList::compatibility_iterator node 
= m_menus
.Item( pos 
); 
 523     wxCHECK_RET( node
, wxT("menu not found") ); 
 525     wxMenu
* menu 
= node
->GetData(); 
 528         gtk_widget_set_sensitive( menu
->m_owner
, flag 
); 
 531 wxString 
wxMenuBar::GetMenuLabel( size_t pos 
) const 
 533     wxMenuList::compatibility_iterator node 
= m_menus
.Item( pos 
); 
 535     wxCHECK_MSG( node
, wxT("invalid"), wxT("menu not found") ); 
 537     wxMenu
* menu 
= node
->GetData(); 
 539     return wxConvertFromGTKToWXLabel(menu
->GetTitle()); 
 542 void wxMenuBar::SetMenuLabel( size_t pos
, const wxString
& label 
) 
 544     wxMenuList::compatibility_iterator node 
= m_menus
.Item( pos 
); 
 546     wxCHECK_RET( node
, wxT("menu not found") ); 
 548     wxMenu
* menu 
= node
->GetData(); 
 550     const wxString 
str( wxReplaceUnderscore( label 
) ); 
 552     menu
->SetTitle( str 
); 
 556         GtkLabel 
*glabel 
= GTK_LABEL( GTK_BIN(menu
->m_owner
)->child 
); 
 559         gtk_label_set( glabel
, wxGTK_CONV( str 
) ); 
 561         /* reparse key accel */ 
 562         (void)gtk_label_parse_uline (GTK_LABEL(glabel
), wxGTK_CONV( str 
) ); 
 563         gtk_accel_label_refetch( GTK_ACCEL_LABEL(glabel
) ); 
 568 //----------------------------------------------------------------------------- 
 570 //----------------------------------------------------------------------------- 
 573 static void gtk_menu_clicked_callback( GtkWidget 
*widget
, wxMenu 
*menu 
) 
 576         wxapp_install_idle_handler(); 
 578     int id 
= menu
->FindMenuIdByMenuItem(widget
); 
 580     /* should find it for normal (not popup) menu */ 
 581     wxASSERT_MSG( (id 
!= -1) || (menu
->GetInvokingWindow() != NULL
), 
 582                   _T("menu item not found in gtk_menu_clicked_callback") ); 
 584     if (!menu
->IsEnabled(id
)) 
 587     wxMenuItem
* item 
= menu
->FindChildItem( id 
); 
 588     wxCHECK_RET( item
, wxT("error in menu item callback") ); 
 590     if ( item
->GetId() == wxGTK_TITLE_ID 
) 
 592         // ignore events from the menu title 
 596     if (item
->IsCheckable()) 
 598         bool isReallyChecked 
= item
->IsChecked(), 
 599             isInternallyChecked 
= item
->wxMenuItemBase::IsChecked(); 
 601         // ensure that the internal state is always consistent with what is 
 602         // shown on the screen 
 603         item
->wxMenuItemBase::Check(isReallyChecked
); 
 605         // we must not report the events for the radio button going up nor the 
 606         // events resulting from the calls to wxMenuItem::Check() 
 607         if ( (item
->GetKind() == wxITEM_RADIO 
&& !isReallyChecked
) || 
 608              (isInternallyChecked 
== isReallyChecked
) ) 
 615     // Is this menu on a menubar?  (possibly nested) 
 616     wxFrame
* frame 
= NULL
; 
 617     if(menu
->IsAttached()) 
 618         frame 
= menu
->GetMenuBar()->GetFrame(); 
 620     // FIXME: why do we have to call wxFrame::GetEventHandler() directly here? 
 621     //        normally wxMenu::SendEvent() should be enough, if it doesn't work 
 622     //        in wxGTK then we have a bug in wxMenu::GetInvokingWindow() which 
 623     //        should be fixed instead of working around it here... 
 626         // If it is attached then let the frame send the event. 
 627         // Don't call frame->ProcessCommand(id) because it toggles 
 628         // checkable items and we've already done that above. 
 629         wxCommandEvent 
commandEvent(wxEVT_COMMAND_MENU_SELECTED
, id
); 
 630         commandEvent
.SetEventObject(frame
); 
 631         if (item
->IsCheckable()) 
 632             commandEvent
.SetInt(item
->IsChecked()); 
 633         commandEvent
.SetEventObject(menu
); 
 635         frame
->HandleWindowEvent(commandEvent
); 
 639         // otherwise let the menu have it 
 640         menu
->SendEvent(id
, item
->IsCheckable() ? item
->IsChecked() : -1); 
 645 //----------------------------------------------------------------------------- 
 647 //----------------------------------------------------------------------------- 
 650 static void gtk_menu_hilight_callback( GtkWidget 
*widget
, wxMenu 
*menu 
) 
 652     if (g_isIdle
) wxapp_install_idle_handler(); 
 654     int id 
= menu
->FindMenuIdByMenuItem(widget
); 
 656     wxASSERT( id 
!= -1 ); // should find it! 
 658     if (!menu
->IsEnabled(id
)) 
 661     wxMenuEvent 
event( wxEVT_MENU_HIGHLIGHT
, id 
); 
 662     event
.SetEventObject( menu 
); 
 664     wxEvtHandler
* handler 
= menu
->GetEventHandler(); 
 665     if (handler 
&& handler
->ProcessEvent(event
)) 
 668     wxWindow 
*win 
= menu
->GetInvokingWindow(); 
 669     if (win
) win
->HandleWindowEvent( event 
); 
 673 //----------------------------------------------------------------------------- 
 675 //----------------------------------------------------------------------------- 
 678 static void gtk_menu_nolight_callback( GtkWidget 
*widget
, wxMenu 
*menu 
) 
 680     if (g_isIdle
) wxapp_install_idle_handler(); 
 682     int id 
= menu
->FindMenuIdByMenuItem(widget
); 
 684     wxASSERT( id 
!= -1 ); // should find it! 
 686     if (!menu
->IsEnabled(id
)) 
 689     wxMenuEvent 
event( wxEVT_MENU_HIGHLIGHT
, -1 ); 
 690     event
.SetEventObject( menu 
); 
 692     wxEvtHandler
* handler 
= menu
->GetEventHandler(); 
 693     if (handler 
&& handler
->ProcessEvent(event
)) 
 696     wxWindow 
*win 
= menu
->GetInvokingWindow(); 
 698         win
->HandleWindowEvent( event 
); 
 702 //----------------------------------------------------------------------------- 
 704 //----------------------------------------------------------------------------- 
 706 IMPLEMENT_DYNAMIC_CLASS(wxMenuItem
, wxObject
) 
 708 wxMenuItem 
*wxMenuItemBase::New(wxMenu 
*parentMenu
, 
 710                                 const wxString
& name
, 
 711                                 const wxString
& help
, 
 715     return new wxMenuItem(parentMenu
, id
, name
, help
, kind
, subMenu
); 
 718 wxMenuItem::wxMenuItem(wxMenu 
*parentMenu
, 
 720                        const wxString
& text
, 
 721                        const wxString
& help
, 
 724           : wxMenuItemBase(parentMenu
, id
, text
, help
, kind
, subMenu
) 
 729 wxMenuItem::wxMenuItem(wxMenu 
*parentMenu
, 
 731                        const wxString
& text
, 
 732                        const wxString
& help
, 
 735           : wxMenuItemBase(parentMenu
, id
, text
, help
, 
 736                            isCheckable 
? wxITEM_CHECK 
: wxITEM_NORMAL
, subMenu
) 
 741 void wxMenuItem::Init() 
 743     m_labelWidget 
= NULL
; 
 749 wxMenuItem::~wxMenuItem() 
 751    // don't delete menu items, the menus take care of that 
 754 wxString 
wxMenuItem::GetItemLabel() const 
 756     wxString label 
= wxConvertFromGTKToWXLabel(m_text
); 
 757     if (!m_hotKey
.IsEmpty()) 
 758         label 
= label 
+ wxT("\t") + m_hotKey
; 
 762 void wxMenuItem::SetItemLabel( const wxString
& string 
) 
 764     wxString str 
= string
; 
 765     if ( str
.empty() && !IsSeparator() ) 
 767         wxASSERT_MSG(wxIsStockID(GetId()), wxT("A non-stock menu item with an empty label?")); 
 768         str 
= wxGetStockLabel(GetId(), wxSTOCK_WITH_ACCELERATOR 
| 
 769                                        wxSTOCK_WITH_MNEMONIC
); 
 772     // Some optimization to avoid flicker 
 773     wxString oldLabel 
= m_text
; 
 774     oldLabel 
= wxStripMenuCodes(oldLabel
); 
 775     oldLabel
.Replace(wxT("_"), wxEmptyString
); 
 776     wxString label1 
= wxStripMenuCodes(str
); 
 777     wxString oldhotkey 
= GetHotKey();    // Store the old hotkey in Ctrl-foo format 
 778     wxCharBuffer oldbuf 
= wxGTK_CONV( GetGtkHotKey(*this) );  // and as <control>foo 
 782     if (oldLabel 
== label1 
&& 
 783              oldhotkey 
== GetHotKey())    // Make sure we can change a hotkey even if the label is unaltered 
 790             label 
= (GtkLabel
*) m_labelWidget
; 
 792             label 
= GTK_LABEL( GTK_BIN(m_menuItem
)->child 
); 
 795         gtk_label_set( label
, wxGTK_CONV( m_text 
) ); 
 798         (void)gtk_label_parse_uline (GTK_LABEL(label
), wxGTK_CONV(m_text
) ); 
 799         gtk_accel_label_refetch( GTK_ACCEL_LABEL(label
) ); 
 803     GdkModifierType accel_mods
; 
 804     gtk_accelerator_parse( (const char*) oldbuf
, &accel_key
, &accel_mods
); 
 807         gtk_widget_remove_accelerator( GTK_WIDGET(m_menuItem
), 
 808                                        m_parentMenu
->m_accel
, 
 813     wxCharBuffer buf 
= wxGTK_CONV( GetGtkHotKey(*this) ); 
 814     gtk_accelerator_parse( (const char*) buf
, &accel_key
, &accel_mods
); 
 817         gtk_widget_add_accelerator( GTK_WIDGET(m_menuItem
), 
 819                                     m_parentMenu
->m_accel
, 
 826 // it's valid for this function to be called even if m_menuItem == NULL 
 827 void wxMenuItem::DoSetText( const wxString
& str 
) 
 829     // '\t' is the deliminator indicating a hot key 
 831     text
.reserve(str
.length()); 
 833     const wxChar 
*pc 
= str
; 
 834     while ( (*pc 
!= wxT('\0')) && (*pc 
!= wxT('\t')) ) 
 836         if ((*pc 
== wxT('&')) && (*(pc
+1) == wxT('&'))) 
 838             // "&" is doubled to indicate "&" instead of accelerator 
 842         else if (*pc 
== wxT('&')) 
 846         else if ( *pc 
== wxT('_') )    // escape underscores 
 857     m_hotKey 
= wxEmptyString
; 
 859     if ( *pc 
== wxT('\t') ) 
 870 wxAcceleratorEntry 
*wxMenuItem::GetAccel() const 
 878     // accelerator parsing code looks for them after a TAB, so insert a dummy 
 881     label 
<< wxT('\t') << GetHotKey(); 
 883     return wxAcceleratorEntry::Create(label
); 
 886 #endif // wxUSE_ACCEL 
 888 void wxMenuItem::Check( bool check 
) 
 890     wxCHECK_RET( m_menuItem
, wxT("invalid menu item") ); 
 892     if (check 
== m_isChecked
) 
 895     wxMenuItemBase::Check( check 
); 
 901             gtk_check_menu_item_set_state( (GtkCheckMenuItem
*)m_menuItem
, (gint
)check 
); 
 905             wxFAIL_MSG( _T("can't check this item") ); 
 909 void wxMenuItem::Enable( bool enable 
) 
 911     wxCHECK_RET( m_menuItem
, wxT("invalid menu item") ); 
 913     gtk_widget_set_sensitive( m_menuItem
, enable 
); 
 914     wxMenuItemBase::Enable( enable 
); 
 917 bool wxMenuItem::IsChecked() const 
 919     wxCHECK_MSG( m_menuItem
, false, wxT("invalid menu item") ); 
 921     wxCHECK_MSG( IsCheckable(), false, 
 922                  wxT("can't get state of uncheckable item!") ); 
 924     return ((GtkCheckMenuItem
*)m_menuItem
)->active 
!= 0; 
 927 //----------------------------------------------------------------------------- 
 929 //----------------------------------------------------------------------------- 
 931 IMPLEMENT_DYNAMIC_CLASS(wxMenu
,wxEvtHandler
) 
 935     m_accel 
= gtk_accel_group_new(); 
 936     m_menu 
= gtk_menu_new(); 
 937     // NB: keep reference to the menu so that it is not destroyed behind 
 938     //     our back by GTK+ e.g. when it is removed from menubar: 
 939     gtk_widget_ref(m_menu
); 
 943     // Tearoffs are entries, just like separators. So if we want this 
 944     // menu to be a tear-off one, we just append a tearoff entry 
 946     if ( m_style 
& wxMENU_TEAROFF 
) 
 948         GtkWidget 
*tearoff 
= gtk_tearoff_menu_item_new(); 
 950         gtk_menu_append(GTK_MENU(m_menu
), tearoff
); 
 955     // append the title as the very first entry if we have it 
 956     if ( !m_title
.empty() ) 
 958         Append(wxGTK_TITLE_ID
, m_title
); 
 965    WX_CLEAR_LIST(wxMenuItemList
, m_items
); 
 967    if ( GTK_IS_WIDGET( m_menu 
)) 
 970        gtk_widget_unref( m_menu 
); 
 971        // if the menu is inserted in another menu at this time, there was 
 972        // one more reference to it: 
 974            gtk_widget_destroy( m_menu 
); 
 978 bool wxMenu::GtkAppend(wxMenuItem 
*mitem
, int pos
) 
 983     GtkLabel
* label 
= NULL
; 
 985     if ( mitem
->IsSeparator() ) 
 988         menuItem 
= gtk_menu_item_new(); 
 990     else if (mitem
->GetBitmap().Ok()) 
 992         text 
= mitem
->wxMenuItemBase::GetItemLabel(); 
 993         const wxBitmap 
*bitmap 
= &mitem
->GetBitmap(); 
 997         menuItem 
= gtk_menu_item_new_with_label( wxGTK_CONV( text 
) ); 
 998         label 
= GTK_LABEL( GTK_BIN(menuItem
)->child 
); 
1002     else // a normal item 
1004         // text has "_" instead of "&" after mitem->SetItemLabel() so don't use it 
1005         text 
=  mitem
->wxMenuItemBase::GetItemLabel() ; 
1007         switch ( mitem
->GetKind() ) 
1011                 menuItem 
= gtk_check_menu_item_new_with_label( wxGTK_CONV( text 
) ); 
1012                 label 
= GTK_LABEL( GTK_BIN(menuItem
)->child 
); 
1014                 gtk_label_set_text( label
, wxGTK_CONV( text 
) ); 
1021                 GSList 
*group 
= NULL
; 
1022                 if ( m_prevRadio 
== NULL 
) 
1024                     // start of a new radio group 
1025                     m_prevRadio 
= menuItem 
= gtk_radio_menu_item_new_with_label( group
, wxGTK_CONV( text 
) ); 
1026                     label 
= GTK_LABEL( GTK_BIN(menuItem
)->child 
); 
1028                     gtk_label_set_text( label
, wxGTK_CONV( text 
) ); 
1030                 else // continue the radio group 
1032                     group 
= gtk_radio_menu_item_group (GTK_RADIO_MENU_ITEM (m_prevRadio
)); 
1033                     m_prevRadio 
= menuItem 
= gtk_radio_menu_item_new_with_label( group
, wxGTK_CONV( text 
) ); 
1034                     label 
= GTK_LABEL( GTK_BIN(menuItem
)->child 
); 
1040                 wxFAIL_MSG( _T("unexpected menu item kind") ); 
1045                 menuItem 
= gtk_menu_item_new_with_label( wxGTK_CONV( text 
) ); 
1046                 label 
= GTK_LABEL( GTK_BIN(menuItem
)->child 
); 
1055     GdkModifierType accel_mods
; 
1056     wxCharBuffer buf 
= wxGTK_CONV( GetGtkHotKey(*mitem
) ); 
1058     // wxPrintf( wxT("item: %s hotkey %s\n"), mitem->GetItemLabel().c_str(), GetGtkHotKey(*mitem).c_str() ); 
1059     gtk_accelerator_parse( (const char*) buf
, &accel_key
, &accel_mods
); 
1062         gtk_widget_add_accelerator (GTK_WIDGET(menuItem
), 
1071         gtk_menu_shell_append(GTK_MENU_SHELL(m_menu
), menuItem
); 
1073         gtk_menu_shell_insert(GTK_MENU_SHELL(m_menu
), menuItem
, pos
); 
1075     gtk_widget_show( menuItem 
); 
1077     if ( !mitem
->IsSeparator() ) 
1079         wxASSERT_MSG( menuItem
, wxT("invalid menuitem") ); 
1081         gtk_signal_connect( GTK_OBJECT(menuItem
), "select", 
1082                             GTK_SIGNAL_FUNC(gtk_menu_hilight_callback
), 
1085         gtk_signal_connect( GTK_OBJECT(menuItem
), "deselect", 
1086                             GTK_SIGNAL_FUNC(gtk_menu_nolight_callback
), 
1089         if ( mitem
->IsSubMenu() && mitem
->GetKind() != wxITEM_RADIO 
&& mitem
->GetKind() != wxITEM_CHECK 
) 
1091             gtk_menu_item_set_submenu( GTK_MENU_ITEM(menuItem
), mitem
->GetSubMenu()->m_menu 
); 
1093             gtk_widget_show( mitem
->GetSubMenu()->m_menu 
); 
1095             // if adding a submenu to a menu already existing in the menu bar, we 
1096             // must set invoking window to allow processing events from this 
1098             if ( m_invokingWindow 
) 
1099                 wxMenubarSetInvokingWindow(mitem
->GetSubMenu(), m_invokingWindow
); 
1103             gtk_signal_connect( GTK_OBJECT(menuItem
), "activate", 
1104                                 GTK_SIGNAL_FUNC(gtk_menu_clicked_callback
), 
1108         guint accel_key 
= gtk_label_parse_uline (GTK_LABEL(label
), wxGTK_CONV( text 
) ); 
1109         if (accel_key 
!= GDK_VoidSymbol
) 
1111             gtk_widget_add_accelerator (menuItem
, 
1113                                         gtk_menu_ensure_uline_accel_group(GTK_MENU(m_menu
)), 
1120     mitem
->SetMenuItem(menuItem
); 
1124         // This doesn't even exist! 
1125         // gtk_widget_lock_accelerators(mitem->GetMenuItem()); 
1131 wxMenuItem
* wxMenu::DoAppend(wxMenuItem 
*mitem
) 
1133     if (!GtkAppend(mitem
)) 
1136     return wxMenuBase::DoAppend(mitem
); 
1139 wxMenuItem
* wxMenu::DoInsert(size_t pos
, wxMenuItem 
*item
) 
1141     if ( !wxMenuBase::DoInsert(pos
, item
) ) 
1145     if ( !GtkAppend(item
, (int)pos
) ) 
1151 wxMenuItem 
*wxMenu::DoRemove(wxMenuItem 
*item
) 
1153     if ( !wxMenuBase::DoRemove(item
) ) 
1156     // TODO: this code doesn't delete the item factory item and this seems 
1157     //       impossible as of GTK 1.2.6. 
1158     gtk_widget_destroy( item
->GetMenuItem() ); 
1163 int wxMenu::FindMenuIdByMenuItem( GtkWidget 
*menuItem 
) const 
1165     wxMenuItemList::compatibility_iterator node 
= m_items
.GetFirst(); 
1168         wxMenuItem 
*item 
= node
->GetData(); 
1169         if (item
->GetMenuItem() == menuItem
) 
1170             return item
->GetId(); 
1171         node 
= node
->GetNext(); 
1177 // ---------------------------------------------------------------------------- 
1179 // ---------------------------------------------------------------------------- 
1183 static wxString 
GetGtkHotKey( const wxMenuItem
& item 
) 
1187     wxAcceleratorEntry 
*accel 
= item
.GetAccel(); 
1190         int flags 
= accel
->GetFlags(); 
1191         if ( flags 
& wxACCEL_ALT 
) 
1192             hotkey 
+= wxT("<alt>"); 
1193         if ( flags 
& wxACCEL_CTRL 
) 
1194             hotkey 
+= wxT("<control>"); 
1195         if ( flags 
& wxACCEL_SHIFT 
) 
1196             hotkey 
+= wxT("<shift>"); 
1198         int code 
= accel
->GetKeyCode(); 
1225                 hotkey 
+= wxString::Format(wxT("F%d"), code 
- WXK_F1 
+ 1); 
1228                 // TODO: we should use gdk_keyval_name() (a.k.a. 
1229                 //       XKeysymToString) here as well as hardcoding the keysym 
1230                 //       names this might be not portable 
1232                 hotkey 
<< wxT("Insert" ); 
1235                 hotkey 
<< wxT("Delete" ); 
1238                 hotkey 
<< wxT("Up" ); 
1241                 hotkey 
<< wxT("Down" ); 
1244                 hotkey 
<< wxT("Page_Up" ); 
1247                 hotkey 
<< wxT("Page_Down" ); 
1250                 hotkey 
<< wxT("Left" ); 
1253                 hotkey 
<< wxT("Right" ); 
1256                 hotkey 
<< wxT("Home" ); 
1259                 hotkey 
<< wxT("End" ); 
1262                 hotkey 
<< wxT("Return" ); 
1265                 hotkey 
<< wxT("BackSpace" ); 
1268                 hotkey 
<< wxT("Tab" ); 
1271                 hotkey 
<< wxT("Esc" ); 
1274                 hotkey 
<< wxT("space" ); 
1277                 hotkey 
<< wxT("Multiply" ); 
1280                 hotkey 
<< wxT("Add" ); 
1283                 hotkey 
<< wxT("Separator" ); 
1286                 hotkey 
<< wxT("Subtract" ); 
1289                 hotkey 
<< wxT("Decimal" ); 
1292                 hotkey 
<< wxT("Divide" ); 
1295                 hotkey 
<< wxT("Cancel" ); 
1298                 hotkey 
<< wxT("Clear" ); 
1301                 hotkey 
<< wxT("Menu" ); 
1304                 hotkey 
<< wxT("Pause" ); 
1307                 hotkey 
<< wxT("Capital" ); 
1310                 hotkey 
<< wxT("Select" ); 
1313                 hotkey 
<< wxT("Print" ); 
1316                 hotkey 
<< wxT("Execute" ); 
1319                 hotkey 
<< wxT("Snapshot" ); 
1322                 hotkey 
<< wxT("Help" ); 
1325                 hotkey 
<< wxT("Num_Lock" ); 
1328                 hotkey 
<< wxT("Scroll_Lock" ); 
1330             case WXK_NUMPAD_INSERT
: 
1331                 hotkey 
<< wxT("KP_Insert" ); 
1333             case WXK_NUMPAD_DELETE
: 
1334                 hotkey 
<< wxT("KP_Delete" ); 
1336              case WXK_NUMPAD_SPACE
: 
1337                 hotkey 
<< wxT("KP_Space" ); 
1339             case WXK_NUMPAD_TAB
: 
1340                 hotkey 
<< wxT("KP_Tab" ); 
1342             case WXK_NUMPAD_ENTER
: 
1343                 hotkey 
<< wxT("KP_Enter" ); 
1345             case WXK_NUMPAD_F1
: case WXK_NUMPAD_F2
: case WXK_NUMPAD_F3
: 
1347                 hotkey 
+= wxString::Format(wxT("KP_F%d"), code 
- WXK_NUMPAD_F1 
+ 1); 
1349             case WXK_NUMPAD_HOME
: 
1350                 hotkey 
<< wxT("KP_Home" ); 
1352             case WXK_NUMPAD_LEFT
: 
1353                 hotkey 
<< wxT("KP_Left" ); 
1356                 hotkey 
<< wxT("KP_Up" ); 
1358             case WXK_NUMPAD_RIGHT
: 
1359                 hotkey 
<< wxT("KP_Right" ); 
1361             case WXK_NUMPAD_DOWN
: 
1362                 hotkey 
<< wxT("KP_Down" ); 
1364             case WXK_NUMPAD_PAGEUP
: 
1365                 hotkey 
<< wxT("KP_Page_Up" ); 
1367             case WXK_NUMPAD_PAGEDOWN
: 
1368                 hotkey 
<< wxT("KP_Page_Down" ); 
1370             case WXK_NUMPAD_END
: 
1371                 hotkey 
<< wxT("KP_End" ); 
1373             case WXK_NUMPAD_BEGIN
: 
1374                 hotkey 
<< wxT("KP_Begin" ); 
1376             case WXK_NUMPAD_EQUAL
: 
1377                 hotkey 
<< wxT("KP_Equal" ); 
1379             case WXK_NUMPAD_MULTIPLY
: 
1380                 hotkey 
<< wxT("KP_Multiply" ); 
1382             case WXK_NUMPAD_ADD
: 
1383                 hotkey 
<< wxT("KP_Add" ); 
1385             case WXK_NUMPAD_SEPARATOR
: 
1386                 hotkey 
<< wxT("KP_Separator" ); 
1388             case WXK_NUMPAD_SUBTRACT
: 
1389                 hotkey 
<< wxT("KP_Subtract" ); 
1391             case WXK_NUMPAD_DECIMAL
: 
1392                 hotkey 
<< wxT("KP_Decimal" ); 
1394             case WXK_NUMPAD_DIVIDE
: 
1395                 hotkey 
<< wxT("KP_Divide" ); 
1397            case WXK_NUMPAD0
: case WXK_NUMPAD1
: case WXK_NUMPAD2
: 
1398            case WXK_NUMPAD3
: case WXK_NUMPAD4
: case WXK_NUMPAD5
: 
1399            case WXK_NUMPAD6
: case WXK_NUMPAD7
: case WXK_NUMPAD8
: case WXK_NUMPAD9
: 
1400                 hotkey 
+= wxString::Format(wxT("KP_%d"), code 
- WXK_NUMPAD0
); 
1402             case WXK_WINDOWS_LEFT
: 
1403                 hotkey 
<< wxT("Super_L" ); 
1405             case WXK_WINDOWS_RIGHT
: 
1406                 hotkey 
<< wxT("Super_R" ); 
1408             case WXK_WINDOWS_MENU
: 
1409                 hotkey 
<< wxT("Menu" ); 
1412                 hotkey 
<< wxT("Command" ); 
1414           /* These probably wouldn't work as there is no SpecialX in gdk/keynames.txt 
1415             case WXK_SPECIAL1: case WXK_SPECIAL2: case WXK_SPECIAL3: case WXK_SPECIAL4: 
1416             case WXK_SPECIAL5: case WXK_SPECIAL6: case WXK_SPECIAL7: case WXK_SPECIAL8: 
1417             case WXK_SPECIAL9:  case WXK_SPECIAL10:  case WXK_SPECIAL11: case WXK_SPECIAL12: 
1418             case WXK_SPECIAL13: case WXK_SPECIAL14: case WXK_SPECIAL15: case WXK_SPECIAL16: 
1419             case WXK_SPECIAL17: case WXK_SPECIAL18: case WXK_SPECIAL19:  case WXK_SPECIAL20: 
1420                 hotkey += wxString::Format(wxT("Special%d"), code - WXK_SPECIAL1 + 1); 
1423                 // if there are any other keys wxAcceleratorEntry::Create() may 
1424                 // return, we should process them here 
1429                     wxString name 
= wxGTK_CONV_BACK( gdk_keyval_name((guint
)code
) ); 
1430                     if ( !name
.empty() ) 
1437                 wxFAIL_MSG( wxT("unknown keyboard accel") ); 
1446 #endif // wxUSE_ACCEL 
1448 // ---------------------------------------------------------------------------- 
1449 // Pop-up menu stuff 
1450 // ---------------------------------------------------------------------------- 
1452 #if wxUSE_MENUS_NATIVE 
1454 extern "C" WXDLLIMPEXP_CORE
 
1455 void gtk_pop_hide_callback( GtkWidget 
*WXUNUSED(widget
), bool* is_waiting  
) 
1457     *is_waiting 
= false; 
1460 WXDLLIMPEXP_CORE 
void SetInvokingWindow( wxMenu 
*menu
, wxWindow
* win 
) 
1462     menu
->SetInvokingWindow( win 
); 
1464     wxMenuItemList::compatibility_iterator node 
= menu
->GetMenuItems().GetFirst(); 
1467         wxMenuItem 
*menuitem 
= node
->GetData(); 
1468         if (menuitem
->IsSubMenu()) 
1470             SetInvokingWindow( menuitem
->GetSubMenu(), win 
); 
1473         node 
= node
->GetNext(); 
1477 extern "C" WXDLLIMPEXP_CORE
 
1478 void wxPopupMenuPositionCallback( GtkMenu 
*menu
, 
1480                                   gpointer user_data 
) 
1482     // ensure that the menu appears entirely on screen 
1484     gtk_widget_get_child_requisition(GTK_WIDGET(menu
), &req
); 
1486     wxSize sizeScreen 
= wxGetDisplaySize(); 
1487     wxPoint 
*pos 
= (wxPoint
*)user_data
; 
1489     gint xmax 
= sizeScreen
.x 
- req
.width
, 
1490          ymax 
= sizeScreen
.y 
- req
.height
; 
1492     *x 
= pos
->x 
< xmax 
? pos
->x 
: xmax
; 
1493     *y 
= pos
->y 
< ymax 
? pos
->y 
: ymax
; 
1496 bool wxWindowGTK::DoPopupMenu( wxMenu 
*menu
, int x
, int y 
) 
1498     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid window") ); 
1500     wxCHECK_MSG( menu 
!= NULL
, false, wxT("invalid popup-menu") ); 
1502     // NOTE: if you change this code, you need to update 
1503     //       the same code in taskbar.cpp as well. This 
1504     //       is ugly code duplication, I know. 
1506     SetInvokingWindow( menu
, this ); 
1510     bool is_waiting 
= true; 
1512     gulong handler 
= gtk_signal_connect( GTK_OBJECT(menu
->m_menu
), 
1514                                          GTK_SIGNAL_FUNC(gtk_pop_hide_callback
), 
1515                                          (gpointer
)&is_waiting 
); 
1519     GtkMenuPositionFunc posfunc
; 
1520     if ( x 
== -1 && y 
== -1 ) 
1522         // use GTK's default positioning algorithm 
1528         pos 
= ClientToScreen(wxPoint(x
, y
)); 
1530         posfunc 
= wxPopupMenuPositionCallback
; 
1533     wxMenuEvent 
eventOpen(wxEVT_MENU_OPEN
, -1, menu
); 
1534     DoCommonMenuCallbackCode(menu
, eventOpen
); 
1537                   GTK_MENU(menu
->m_menu
), 
1538                   NULL
,           // parent menu shell 
1539                   NULL
,           // parent menu item 
1540                   posfunc
,                      // function to position it 
1541                   userdata
,                     // client data 
1542                   0,                            // button used to activate it 
1543                   wxGtkTimeLastClick            
// the time of activation 
1548         gtk_main_iteration(); 
1551     gtk_signal_disconnect(GTK_OBJECT(menu
->m_menu
), handler
); 
1553     wxMenuEvent 
eventClose(wxEVT_MENU_CLOSE
, -1, menu
); 
1554     DoCommonMenuCallbackCode(menu
, eventClose
); 
1559 #endif // wxUSE_MENUS_NATIVE