1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxMenu, wxMenuBar, wxMenuItem
4 // Author: Julian Smart
5 // Modified by: Vadim Zeitlin
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
12 // ===========================================================================
14 // ===========================================================================
16 // ---------------------------------------------------------------------------
18 // ---------------------------------------------------------------------------
21 #pragma implementation "menu.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
39 #include "wx/ownerdrw.h"
42 #include "wx/msw/private.h"
43 #include "wx/msw/menu.h"
44 #include "wx/menuitem.h"
47 // other standard headers
50 // ----------------------------------------------------------------------------
52 // ----------------------------------------------------------------------------
54 extern wxMenu
*wxCurrentPopupMenu
;
56 // ----------------------------------------------------------------------------
58 // ----------------------------------------------------------------------------
60 // the (popup) menu title has this special id
61 static const int idMenuTitle
= -2;
63 // ----------------------------------------------------------------------------
65 // ----------------------------------------------------------------------------
67 #if !USE_SHARED_LIBRARY
68 IMPLEMENT_DYNAMIC_CLASS(wxMenu
, wxEvtHandler
)
69 IMPLEMENT_DYNAMIC_CLASS(wxMenuBar
, wxEvtHandler
)
72 // ============================================================================
74 // ============================================================================
76 // ---------------------------------------------------------------------------
77 // wxMenu construction, adding and removing menu items
78 // ---------------------------------------------------------------------------
80 // Construct a menu with optional title (then use append)
81 void wxMenu::Init(const wxString
& title
82 #ifdef WXWIN_COMPATIBILITY
83 , const wxFunction func
89 m_eventHandler
= this;
90 m_pInvokingWindow
= NULL
;
94 m_hMenu
= (WXHMENU
) CreatePopupMenu();
96 m_topLevelMenu
= this;
97 m_clientData
= (void*) NULL
;
101 Append(idMenuTitle
, m_title
) ;
105 #if WXWIN_COMPATIBILITY
110 // The wxWindow destructor will take care of deleting the submenus.
113 // free Windows resources
116 ::DestroyMenu((HMENU
)m_hMenu
);
121 wxNode
*node
= m_menuItems
.First();
124 wxMenuItem
*item
= (wxMenuItem
*)node
->Data();
126 // Delete child menus.
127 // Beware: they must not be appended to children list!!!
128 // (because order of delete is significant)
129 if ( item
->IsSubMenu() )
130 item
->DeleteSubMenu();
132 wxNode
*next
= node
->Next();
144 // function appends a new item or submenu to the menu
145 void wxMenu::Append(wxMenuItem
*pItem
)
147 wxCHECK_RET( pItem
!= NULL
, _T("can't append NULL item to the menu") );
150 // check for accelerators: they are given after '\t'
151 wxString label
= pItem
->GetName();
152 int posTab
= label
.Find(_T('\t'));
153 if ( posTab
!= wxNOT_FOUND
) {
154 // parse the accelerator string
156 int accelFlags
= wxACCEL_NORMAL
;
158 for ( size_t n
= (size_t)posTab
+ 1; n
< label
.Len(); n
++ ) {
159 if ( (label
[n
] == '+') || (label
[n
] == '-') ) {
160 if ( current
== _("ctrl") )
161 accelFlags
|= wxACCEL_CTRL
;
162 else if ( current
== _("alt") )
163 accelFlags
|= wxACCEL_ALT
;
164 else if ( current
== _("shift") )
165 accelFlags
|= wxACCEL_SHIFT
;
167 wxLogDebug(_T("Unknown accel modifier: '%s'"),
174 current
+= wxTolower(label
[n
]);
178 if ( current
.IsEmpty() ) {
179 wxLogDebug(_T("No accel key found, accel string ignored."));
182 if ( current
.Len() == 1 ) {
184 keyCode
= wxToupper(current
[0U]);
187 // it should be a function key
188 if ( current
[0U] == 'f' && isdigit(current
[1U]) &&
189 (current
.Len() == 2 ||
190 (current
.Len() == 3 && isdigit(current
[2U]))) ) {
192 wxSscanf(current
.c_str() + 1, _T("%d"), &n
);
194 keyCode
= VK_F1
+ n
- 1;
197 wxLogDebug(_T("Unrecognized accel key '%s', accel "
198 "string ignored."), current
.c_str());
205 m_accelKeyCodes
.Add(keyCode
);
206 m_accelFlags
.Add(accelFlags
);
207 m_accelIds
.Add(pItem
->GetId());
210 #endif // wxUSE_ACCEL
214 // if "Break" has just been called, insert a menu break before this item
215 // (and don't forget to reset the flag)
217 flags
|= MF_MENUBREAK
;
221 if ( pItem
->IsSeparator() ) {
222 flags
|= MF_SEPARATOR
;
225 // id is the numeric id for normal menu items and HMENU for submenus as
226 // required by ::AppendMenu() API
228 wxMenu
*submenu
= pItem
->GetSubMenu();
229 if ( submenu
!= NULL
) {
230 wxASSERT( submenu
->GetHMenu() != (WXHMENU
) NULL
);
232 id
= (UINT
)submenu
->GetHMenu();
233 submenu
->m_topLevelMenu
= m_topLevelMenu
;
234 submenu
->m_parent
= this;
235 submenu
->m_savehMenu
= (WXHMENU
)id
;
236 submenu
->m_hMenu
= 0;
246 #if wxUSE_OWNER_DRAWN
247 if ( pItem
->IsOwnerDrawn() ) { // want to get {Measure|Draw}Item messages?
248 // item draws itself, pass pointer to it in data parameter
249 flags
|= MF_OWNERDRAW
;
250 pData
= (LPCTSTR
)pItem
;
255 // menu is just a normal string (passed in data parameter)
260 if ( !::AppendMenu(GetHmenu(), flags
, id
, pData
) )
262 wxLogLastError("AppendMenu");
267 if ( id
== idMenuTitle
)
269 // visually select the menu title
271 mii
.cbSize
= sizeof(mii
);
272 mii
.fMask
= MIIM_STATE
;
273 mii
.fState
= MFS_DEFAULT
;
275 if ( !SetMenuItemInfo(GetHmenu(), (unsigned)id
, FALSE
, &mii
) )
277 wxLogLastError(_T("SetMenuItemInfo"));
282 m_menuItems
.Append(pItem
);
287 void wxMenu::AppendSeparator()
289 Append(new wxMenuItem(this, ID_SEPARATOR
));
293 void wxMenu::Append(int id
,
294 const wxString
& label
,
296 const wxString
& helpString
)
298 Append(new wxMenuItem(this, id
, label
, helpString
, FALSE
, SubMenu
));
301 // Ordinary menu item
302 void wxMenu::Append(int id
,
303 const wxString
& label
,
304 const wxString
& helpString
,
307 // 'checkable' parameter is useless for Windows.
308 Append(new wxMenuItem(this, id
, label
, helpString
, checkable
));
312 void wxMenu::Delete(int id
)
314 wxMenuItem
*item
= NULL
;
317 for (pos
= 0, node
= m_menuItems
.First(); node
; node
= node
->Next(), pos
++)
319 item
= (wxMenuItem
*)node
->Data();
320 if ( item
->GetId() == id
)
324 wxCHECK_RET( node
, _T("wxMenu::Delete(): item doesn't exist") );
326 HMENU menu
= GetHmenu();
328 wxMenu
*pSubMenu
= item
->GetSubMenu();
329 if ( pSubMenu
!= NULL
) {
330 RemoveMenu(menu
, (UINT
)pos
, MF_BYPOSITION
);
331 pSubMenu
->m_hMenu
= pSubMenu
->m_savehMenu
;
332 pSubMenu
->m_savehMenu
= 0;
333 pSubMenu
->m_parent
= NULL
;
334 // RemoveChild(item->subMenu);
335 pSubMenu
->m_topLevelMenu
= NULL
;
336 // TODO: Why isn't subMenu deleted here???
337 // Will put this in for now. Assuming this is supposed
338 // to delete the menu, not just remove it.
339 item
->DeleteSubMenu();
342 DeleteMenu(menu
, (UINT
)pos
, MF_BYPOSITION
);
345 m_menuItems
.DeleteNode(node
);
351 // ---------------------------------------------------------------------------
352 // accelerator helpers
353 // ---------------------------------------------------------------------------
355 // create the wxAcceleratorEntries for our accels and put them into provided
356 // array - return the number of accels we have
357 size_t wxMenu::CopyAccels(wxAcceleratorEntry
*accels
) const
359 size_t count
= GetAccelCount();
360 for ( size_t n
= 0; n
< count
; n
++ )
362 (*accels
++).Set(m_accelFlags
[n
], m_accelKeyCodes
[n
], m_accelIds
[n
]);
368 #endif // wxUSE_ACCEL
370 // ---------------------------------------------------------------------------
371 // wxMenu functions implemented in wxMenuItem
372 // ---------------------------------------------------------------------------
374 void wxMenu::Enable(int id
, bool Flag
)
376 wxMenuItem
*item
= FindItemForId(id
);
377 wxCHECK_RET( item
!= NULL
, _T("can't enable non-existing menu item") );
382 bool wxMenu::IsEnabled(int id
) const
384 wxMenuItem
*item
= FindItemForId(id
);
385 wxCHECK_MSG( item
!= NULL
, FALSE
, _T("invalid item id") );
387 return item
->IsEnabled();
390 void wxMenu::Check(int id
, bool Flag
)
392 wxMenuItem
*item
= FindItemForId(id
);
393 wxCHECK_RET( item
!= NULL
, _T("can't get status of non-existing menu item") );
398 bool wxMenu::IsChecked(int id
) const
400 wxMenuItem
*item
= FindItemForId(id
);
401 wxCHECK_MSG( item
!= NULL
, FALSE
, _T("invalid item id") );
403 return item
->IsChecked();
406 void wxMenu::SetLabel(int id
, const wxString
& label
)
408 wxMenuItem
*item
= FindItemForId(id
) ;
409 wxCHECK_RET( item
, _T("wxMenu::SetLabel: no such item") );
411 item
->SetName(label
);
414 wxString
wxMenu::GetLabel(int id
) const
417 wxMenuItem
*pItem
= FindItemForId(id
) ;
419 label
= pItem
->GetName() ;
421 wxFAIL_MSG(_T("wxMenu::GetLabel: item doesn't exist"));
426 void wxMenu::SetHelpString(int itemId
, const wxString
& helpString
)
428 wxMenuItem
*item
= FindItemForId (itemId
);
430 item
->SetHelp(helpString
);
432 wxFAIL_MSG(_T("wxMenu::SetHelpString: item doesn't exist"));
435 wxString
wxMenu::GetHelpString (int itemId
) const
438 wxMenuItem
*item
= FindItemForId (itemId
);
440 help
= item
->GetHelp();
442 wxFAIL_MSG(_T("wxMenu::GetHelpString: item doesn't exist"));
447 // ---------------------------------------------------------------------------
449 // ---------------------------------------------------------------------------
451 void wxMenu::SetTitle(const wxString
& label
)
453 bool hasNoTitle
= m_title
.IsEmpty();
456 HMENU hMenu
= GetHmenu();
460 if ( !label
.IsEmpty() )
462 if ( !InsertMenu(hMenu
, 0u, MF_BYPOSITION
| MF_STRING
,
463 (unsigned)idMenuTitle
, m_title
) ||
464 !InsertMenu(hMenu
, 1u, MF_BYPOSITION
, (unsigned)-1, NULL
) )
466 wxLogLastError(_T("InsertMenu"));
472 if ( label
.IsEmpty() )
474 // remove the title and the separator after it
475 if ( !RemoveMenu(hMenu
, 0, MF_BYPOSITION
) ||
476 !RemoveMenu(hMenu
, 0, MF_BYPOSITION
) )
478 wxLogLastError("RemoveMenu");
484 if ( !ModifyMenu(hMenu
, 0u,
485 MF_BYPOSITION
| MF_STRING
,
486 (unsigned)idMenuTitle
, m_title
) )
488 wxLogLastError("ModifyMenu");
494 // put the title string in bold face
495 if ( !m_title
.IsEmpty() )
498 mii
.cbSize
= sizeof(mii
);
499 mii
.fMask
= MIIM_STATE
;
500 mii
.fState
= MFS_DEFAULT
;
502 if ( !SetMenuItemInfo(hMenu
, (unsigned)idMenuTitle
, FALSE
, &mii
) )
504 wxLogLastError("SetMenuItemInfo");
510 const wxString
wxMenu::GetTitle() const
515 // ---------------------------------------------------------------------------
517 // ---------------------------------------------------------------------------
519 bool wxMenu::MSWCommand(WXUINT
WXUNUSED(param
), WXWORD id
)
521 // ignore commands from the menu title
523 // NB: VC++ generates wrong assembler for `if ( id != idMenuTitle )'!!
524 if ( id
!= (WXWORD
)idMenuTitle
)
526 wxCommandEvent
event(wxEVT_COMMAND_MENU_SELECTED
);
527 event
.SetEventObject( this );
530 ProcessCommand(event
);
536 bool wxMenu::ProcessCommand(wxCommandEvent
& event
)
538 bool processed
= FALSE
;
540 #if WXWIN_COMPATIBILITY
544 (void)(*(m_callback
))(*this, event
);
547 #endif // WXWIN_COMPATIBILITY
549 // Try the menu's event handler
550 if ( !processed
&& GetEventHandler())
552 processed
= GetEventHandler()->ProcessEvent(event
);
555 // Try the window the menu was popped up from (and up through the
557 wxWindow
*win
= GetInvokingWindow();
558 if ( !processed
&& win
)
559 processed
= win
->GetEventHandler()->ProcessEvent(event
);
564 // ---------------------------------------------------------------------------
566 // ---------------------------------------------------------------------------
568 // Finds the item id matching the given string, -1 if not found.
569 int wxMenu::FindItem (const wxString
& itemString
) const
571 wxString itemLabel
= wxStripMenuCodes(itemString
);
572 for ( wxNode
*node
= m_menuItems
.First(); node
; node
= node
->Next() )
574 wxMenuItem
*item
= (wxMenuItem
*)node
->Data();
575 if ( item
->IsSubMenu() )
577 int ans
= item
->GetSubMenu()->FindItem(itemString
);
578 if ( ans
!= wxNOT_FOUND
)
581 else if ( !item
->IsSeparator() )
583 wxString label
= wxStripMenuCodes(item
->GetName());
584 if ( itemLabel
== label
)
585 return item
->GetId();
592 wxMenuItem
*wxMenu::FindItemForId(int itemId
, wxMenu
** itemMenu
) const
597 wxMenuItem
*item
= NULL
;
598 for ( wxNode
*node
= m_menuItems
.First(); node
&& !item
; node
= node
->Next() )
600 item
= (wxMenuItem
*)node
->Data();
602 if ( item
->GetId() == itemId
)
605 *itemMenu
= (wxMenu
*)this;
607 else if ( item
->IsSubMenu() )
609 item
= item
->GetSubMenu()->FindItemForId(itemId
, itemMenu
);
613 // don't exit the loop
621 // ---------------------------------------------------------------------------
623 // ---------------------------------------------------------------------------
625 void wxMenu::Attach(wxMenuBar
*menubar
)
627 // menu can be in at most one menubar because otherwise they would both
628 // delete the menu pointer
629 wxASSERT_MSG( !m_menuBar
, _T("menu belongs to 2 menubars, expect a crash") );
632 m_savehMenu
= m_hMenu
;
636 void wxMenu::Detach()
638 wxASSERT_MSG( m_menuBar
, _T("can't detach menu if it's not attached") );
640 m_hMenu
= m_savehMenu
;
644 // ---------------------------------------------------------------------------
646 // ---------------------------------------------------------------------------
648 void wxMenuBar::Init()
650 m_eventHandler
= this;
654 m_menuBarFrame
= NULL
;
658 wxMenuBar::wxMenuBar()
663 wxMenuBar::wxMenuBar( long WXUNUSED(style
) )
668 wxMenuBar::wxMenuBar(int count
, wxMenu
*menus
[], const wxString titles
[])
674 m_titles
= new wxString
[count
];
677 for ( i
= 0; i
< count
; i
++ )
678 m_titles
[i
] = titles
[i
];
680 for ( i
= 0; i
< count
; i
++ )
681 m_menus
[i
]->Attach(this);
684 wxMenuBar::~wxMenuBar()
686 for ( int i
= 0; i
< m_menuCount
; i
++ )
695 // ---------------------------------------------------------------------------
697 // ---------------------------------------------------------------------------
699 void wxMenuBar::Refresh()
701 wxCHECK_RET( m_menuBarFrame
, _T("can't refresh a menubar withotu a frame") );
703 DrawMenuBar((HWND
)m_menuBarFrame
->GetHWND()) ;
706 WXHMENU
wxMenuBar::Create()
708 wxCHECK_MSG( !m_hMenu
, TRUE
, _T("menubar already created") );
710 m_hMenu
= (WXHMENU
)::CreateMenu();
714 wxLogLastError("CreateMenu");
718 for ( int i
= 0; i
< m_menuCount
; i
++ )
720 if ( !::AppendMenu((HMENU
)m_hMenu
, MF_POPUP
| MF_STRING
,
721 (UINT
)m_menus
[i
]->GetHMenu(),
724 wxLogLastError("AppendMenu");
732 // ---------------------------------------------------------------------------
733 // wxMenuBar functions forwarded to wxMenuItem
734 // ---------------------------------------------------------------------------
736 // Must only be used AFTER menu has been attached to frame,
737 // otherwise use individual menus to enable/disable items
738 void wxMenuBar::Enable(int id
, bool enable
)
740 wxMenu
*itemMenu
= NULL
;
741 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
743 wxCHECK_RET( item
, _T("attempt to enable an item which doesn't exist") );
745 item
->Enable(enable
);
748 void wxMenuBar::EnableTop(int pos
, bool enable
)
750 int flag
= enable
? MF_ENABLED
: MF_GRAYED
;;
752 EnableMenuItem((HMENU
)m_hMenu
, pos
, MF_BYPOSITION
| flag
);
755 // Must only be used AFTER menu has been attached to frame,
756 // otherwise use individual menus
757 void wxMenuBar::Check(int id
, bool check
)
759 wxMenu
*itemMenu
= NULL
;
760 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
762 wxCHECK_RET( item
, _T("attempt to check an item which doesn't exist") );
763 wxCHECK_RET( item
->IsCheckable(), _T("attempt to check an uncheckable item") );
768 bool wxMenuBar::IsChecked(int id
) const
770 wxMenu
*itemMenu
= NULL
;
771 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
773 wxCHECK_MSG( item
, FALSE
, _T("wxMenuBar::IsChecked(): no such item") );
775 int flag
= ::GetMenuState(GetHmenuOf(itemMenu
), id
, MF_BYCOMMAND
);
777 return (flag
& MF_CHECKED
) != 0;
780 bool wxMenuBar::IsEnabled(int id
) const
782 wxMenu
*itemMenu
= NULL
;
783 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
785 wxCHECK_MSG( item
, FALSE
, _T("wxMenuBar::IsEnabled(): no such item") );
787 int flag
= ::GetMenuState(GetHmenuOf(itemMenu
), id
, MF_BYCOMMAND
) ;
789 return (flag
& MF_ENABLED
) != 0;
792 void wxMenuBar::SetLabel(int id
, const wxString
& label
)
794 wxMenu
*itemMenu
= NULL
;
795 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
797 wxCHECK_RET( item
, _T("wxMenuBar::SetLabel(): no such item") );
799 item
->SetName(label
);
802 wxString
wxMenuBar::GetLabel(int id
) const
804 wxMenu
*itemMenu
= NULL
;
805 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
807 wxCHECK_MSG( item
, _T(""), _T("wxMenuBar::GetLabel(): no such item") );
809 return item
->GetName();
812 void wxMenuBar::SetHelpString (int id
, const wxString
& helpString
)
814 wxMenu
*itemMenu
= NULL
;
815 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
817 wxCHECK_RET( item
, _T("wxMenuBar::SetHelpString(): no such item") );
819 item
->SetHelp(helpString
);
822 wxString
wxMenuBar::GetHelpString (int id
) const
824 wxMenu
*itemMenu
= NULL
;
825 wxMenuItem
*item
= FindItemForId(id
, &itemMenu
) ;
827 wxCHECK_MSG( item
, _T(""), _T("wxMenuBar::GetHelpString(): no such item") );
829 return item
->GetHelp();
832 // ---------------------------------------------------------------------------
833 // wxMenuBar functions to work with the top level submenus
834 // ---------------------------------------------------------------------------
836 // NB: we don't support owner drawn top level items for now, if we do these
837 // functions would have to be changed to use wxMenuItem as well
839 void wxMenuBar::SetLabelTop(int pos
, const wxString
& label
)
842 UINT flagsOld
= ::GetMenuState((HMENU
)m_hMenu
, pos
, MF_BYPOSITION
);
843 if ( flagsOld
== 0xFFFFFFFF )
845 wxLogLastError(_T("GetMenuState"));
850 if ( flagsOld
& MF_POPUP
)
852 // HIBYTE contains the number of items in the submenu in this case
854 id
= (UINT
)::GetSubMenu((HMENU
)m_hMenu
, pos
) ;
861 if ( ::ModifyMenu(GetHmenu(), pos
, MF_BYPOSITION
| MF_STRING
| flagsOld
,
862 id
, label
) == 0xFFFFFFFF )
864 wxLogLastError("ModifyMenu");
868 wxString
wxMenuBar::GetLabelTop(int pos
) const
870 int len
= ::GetMenuString((HMENU
)m_hMenu
, pos
, NULL
, 0, MF_BYCOMMAND
);
872 len
++; // for the NUL character
874 ::GetMenuString(GetHmenu(), pos
, label
.GetWriteBuf(len
), len
, MF_BYCOMMAND
);
875 label
.UngetWriteBuf();
880 // ---------------------------------------------------------------------------
881 // wxMenuBar notifications
882 // ---------------------------------------------------------------------------
884 bool wxMenuBar::OnDelete(wxMenu
*a_menu
, int pos
)
886 if ( !m_menuBarFrame
)
889 if ( ::RemoveMenu((HMENU
)m_hMenu
, (UINT
)pos
, MF_BYPOSITION
) )
891 // VZ: I'm not sure about what's going on here, so I leave an assert
892 wxASSERT_MSG( m_menus
[pos
] == a_menu
, _T("what is this parameter for??") );
896 if ( m_menuBarFrame
)
903 wxLogLastError("RemoveMenu");
909 bool wxMenuBar::OnAppend(wxMenu
*a_menu
, const wxChar
*title
)
911 WXHMENU submenu
= a_menu
->GetHMenu();
915 if ( !m_menuBarFrame
)
918 a_menu
->Attach(this);
920 if ( !::AppendMenu(GetHmenu(), MF_POPUP
| MF_STRING
,
921 (UINT
)submenu
, title
) )
923 wxLogLastError(_T("AppendMenu"));
931 // ---------------------------------------------------------------------------
932 // wxMenuBar construction
933 // ---------------------------------------------------------------------------
935 void wxMenuBar::Append (wxMenu
* menu
, const wxString
& title
)
937 if (!OnAppend(menu
, title
))
941 wxMenu
**new_menus
= new wxMenu
*[m_menuCount
];
942 wxString
*new_titles
= new wxString
[m_menuCount
];
945 for (i
= 0; i
< m_menuCount
- 1; i
++)
947 new_menus
[i
] = m_menus
[i
];
949 new_titles
[i
] = m_titles
[i
];
950 m_titles
[i
] = _T("");
958 m_titles
= new_titles
;
960 m_menus
[m_menuCount
- 1] = (wxMenu
*)menu
;
961 m_titles
[m_menuCount
- 1] = title
;
963 menu
->SetParent(this);
966 void wxMenuBar::Delete(wxMenu
* menu
, int i
)
972 for (ii
= 0; ii
< m_menuCount
; ii
++) {
973 if (m_menus
[ii
] == menu
)
976 if (ii
>= m_menuCount
)
979 if (ii
< 0 || ii
>= m_menuCount
)
984 if (!OnDelete(menu
, ii
))
987 menu
->SetParent(NULL
);
990 for (j
= ii
; j
< m_menuCount
; j
++) {
991 m_menus
[j
] = m_menus
[j
+ 1];
992 m_titles
[j
] = m_titles
[j
+ 1];
996 void wxMenuBar::Attach(wxFrame
*frame
)
998 wxASSERT_MSG( !m_menuBarFrame
, _T("menubar already attached!") );
1000 m_menuBarFrame
= frame
;
1003 // create the accel table - we consider that the menubar construction is
1005 size_t nAccelCount
= 0;
1007 for ( i
= 0; i
< m_menuCount
; i
++ )
1009 nAccelCount
+= m_menus
[i
]->GetAccelCount();
1014 wxAcceleratorEntry
*accelEntries
= new wxAcceleratorEntry
[nAccelCount
];
1017 for ( i
= 0; i
< m_menuCount
; i
++ )
1019 nAccelCount
+= m_menus
[i
]->CopyAccels(&accelEntries
[nAccelCount
]);
1022 m_accelTable
= wxAcceleratorTable(nAccelCount
, accelEntries
);
1024 delete [] accelEntries
;
1026 #endif // wxUSE_ACCEL
1029 // ---------------------------------------------------------------------------
1030 // wxMenuBar searching for menu items
1031 // ---------------------------------------------------------------------------
1033 // Find the itemString in menuString, and return the item id or wxNOT_FOUND
1034 int wxMenuBar::FindMenuItem(const wxString
& menuString
,
1035 const wxString
& itemString
) const
1037 wxString menuLabel
= wxStripMenuCodes(menuString
);
1038 for ( int i
= 0; i
< m_menuCount
; i
++ )
1040 wxString title
= wxStripMenuCodes(m_titles
[i
]);
1041 if ( menuString
== title
)
1042 return m_menus
[i
]->FindItem(itemString
);
1048 wxMenuItem
*wxMenuBar::FindItemForId (int id
, wxMenu
**itemMenu
) const
1053 wxMenuItem
*item
= NULL
;
1054 for ( int i
= 0; !item
&& (i
< m_menuCount
); i
++ )
1056 item
= m_menus
[i
]->FindItemForId(id
, itemMenu
);
1063 // ----------------------------------------------------------------------------
1065 // ----------------------------------------------------------------------------
1067 wxWindow
*wxMenu::GetWindow() const
1069 if ( m_pInvokingWindow
!= NULL
)
1070 return m_pInvokingWindow
;
1071 else if ( m_menuBar
!= NULL
)
1072 return m_menuBar
->GetFrame();
1077 WXHMENU
wxMenu::GetHMenu() const
1081 else if ( m_savehMenu
!= 0 )
1084 wxFAIL_MSG(_T("wxMenu without HMENU"));
1089 // Update a menu and all submenus recursively. source is the object that has
1090 // the update event handlers defined for it. If NULL, the menu or associated
1091 // window will be used.
1092 void wxMenu::UpdateUI(wxEvtHandler
* source
)
1094 if (!source
&& GetInvokingWindow())
1095 source
= GetInvokingWindow()->GetEventHandler();
1097 source
= GetEventHandler();
1101 wxNode
* node
= GetItems().First();
1104 wxMenuItem
* item
= (wxMenuItem
*) node
->Data();
1105 if ( !item
->IsSeparator() )
1107 wxWindowID id
= item
->GetId();
1108 wxUpdateUIEvent
event(id
);
1109 event
.SetEventObject( source
);
1111 if (source
->ProcessEvent(event
))
1113 if (event
.GetSetText())
1114 SetLabel(id
, event
.GetText());
1115 if (event
.GetSetChecked())
1116 Check(id
, event
.GetChecked());
1117 if (event
.GetSetEnabled())
1118 Enable(id
, event
.GetEnabled());
1121 if (item
->GetSubMenu())
1122 item
->GetSubMenu()->UpdateUI(source
);
1124 node
= node
->Next();