1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxMenu, wxMenuBar, wxMenuItem
4 // Author: David Webster
8 // Copyright: (c) David Webster
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "menu.h"
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
29 #include "wx/ownerdrw.h"
32 #include "wx/os2/private.h"
34 // other standard headers
37 // ----------------------------------------------------------------------------
39 // ----------------------------------------------------------------------------
41 extern wxMenu
* wxCurrentPopupMenu
;
43 // ----------------------------------------------------------------------------
45 // ----------------------------------------------------------------------------
48 // The (popup) menu title has this special id
50 static const int idMenuTitle
= -2;
52 // ----------------------------------------------------------------------------
54 // ----------------------------------------------------------------------------
56 IMPLEMENT_DYNAMIC_CLASS(wxMenu
, wxEvtHandler
)
57 IMPLEMENT_DYNAMIC_CLASS(wxMenuBar
, wxEvtHandler
)
59 // ----------------------------------------------------------------------------
60 // static function for translating menu labels
61 // ----------------------------------------------------------------------------
63 static wxString
TextToLabel(const wxString
& rTitle
)
67 for (pc
= rTitle
; *pc
!= wxT('\0'); pc
++ )
71 if (*(pc
+1) == wxT('&'))
79 // else if (*pc == wxT('/'))
81 // Title << wxT('\\');
85 if ( *pc
== wxT('~') )
87 // tildes must be doubled to prevent them from being
88 // interpreted as accelerator character prefix by PM ???
97 // ============================================================================
99 // ============================================================================
101 // ---------------------------------------------------------------------------
102 // wxMenu construction, adding and removing menu items
103 // ---------------------------------------------------------------------------
106 // Construct a menu with optional title (then use append)
113 // Create the menu (to be used as a submenu or a popup)
115 if ((m_hMenu
= ::WinCreateWindow( HWND_DESKTOP
116 ,(const wxChar
*)WC_MENU
130 wxLogLastError("WinLoadMenu");
132 m_vMenuData
.iPosition
= 0;
133 m_vMenuData
.afStyle
= MIS_SUBMENU
| MIS_TEXT
;
134 m_vMenuData
.afAttribute
= (USHORT
)0;
135 m_vMenuData
.id
= (USHORT
)0;
136 m_vMenuData
.hwndSubMenu
= m_hMenu
;
137 m_vMenuData
.hItem
= NULLHANDLE
;
140 // If we have a title, insert it in the beginning of the menu
142 if (!m_title
.IsEmpty())
149 } // end of wxMenu::Init
152 // The wxWindow destructor will take care of deleting the submenus.
157 // We should free PM resources only if PM doesn't do it for us
158 // which happens if we're attached to a menubar or a submenu of another
160 if (!IsAttached() && !GetParent())
162 if (!::WinDestroyWindow((HWND
)GetHmenu()) )
164 wxLogLastError("WinDestroyWindow");
172 WX_CLEAR_ARRAY(m_vAccels
);
173 #endif // wxUSE_ACCEL
174 } // end of wxMenu::~wxMenu
178 // this will take effect during the next call to Append()
180 } // end of wxMenu::Break
184 int wxMenu::FindAccel(
189 size_t nCount
= m_vAccels
.GetCount();
191 for (n
= 0; n
< nCount
; n
++)
193 if (m_vAccels
[n
]->m_command
== nId
)
197 } // end of wxMenu::FindAccel
199 void wxMenu::UpdateAccel(
204 // Find the (new) accel for this item
206 wxAcceleratorEntry
* pAccel
= wxGetAccelFromString(pItem
->GetText());
209 pAccel
->m_command
= pItem
->GetId();
214 int n
= FindAccel(pItem
->GetId());
216 if (n
== wxNOT_FOUND
)
219 // No old, add new if any
222 m_vAccels
.Add(pAccel
);
224 return; // skipping RebuildAccelTable() below
229 // Replace old with new or just remove the old one if no new
234 m_vAccels
[n
] = pAccel
;
241 m_menuBar
->RebuildAccelTable();
243 } // wxMenu::UpdateAccel
245 #endif // wxUSE_ACCEL
248 // Append a new item or submenu to the menu
250 bool wxMenu::DoInsertOrAppend(
261 #endif // wxUSE_ACCEL
263 memset(&vItem
, '\0', sizeof(vItem
));
266 // If "Break" has just been called, insert a menu break before this item
267 // (and don't forget to reset the flag)
271 vItem
.afStyle
|= MIS_BREAK
;
276 // Menu items that are being inserted into a submenu MUST have a
277 // MENUITEM structure separate from the parent menu so we must use
278 // a local vItem not the object's m_vMenuItem as that is the MENUITEM
279 // associated with the parent submenu.
281 if (pItem
->IsSeparator())
283 vItem
.afStyle
|= MIS_SEPARATOR
;
287 // Id is the numeric id for normal menu items and HMENU for submenus as
288 // required by ::MM_INSERTITEM message API
291 wxMenu
* pSubmenu
= pItem
->GetSubMenu();
293 if (pSubmenu
!= NULL
)
295 wxASSERT_MSG(pSubmenu
->GetHMenu(), wxT("invalid submenu"));
296 pSubmenu
->SetParent(this);
298 vItem
.iPosition
= 0; // submenus have a 0 position
299 vItem
.id
= (USHORT
)pSubmenu
->GetHMenu();
300 vItem
.afStyle
|= MIS_SUBMENU
| MIS_TEXT
;
304 vItem
.id
= pItem
->GetId();
309 #if wxUSE_OWNER_DRAWN
310 if (pItem
->IsOwnerDrawn())
313 // Want to get {Measure|Draw}Item messages?
314 // item draws itself, pass pointer to it in data parameter
315 // Will eventually need to set the image handle somewhere into vItem.hItem
317 vItem
.afStyle
|= MIS_OWNERDRAW
;
318 pData
= (BYTE
*)pItem
;
319 // vItem.hItem = ????
325 // Menu is just a normal string (passed in data parameter)
327 wxSetShortCutKey((wxChar
*)pItem
->GetText().c_str());
328 vItem
.afStyle
|= MIS_TEXT
;
329 pData
= (char*)pItem
->GetText().c_str();
334 if (pSubmenu
== NULL
)
337 // -1 means append at end
339 if (nPos
== (size_t)-1)
341 vItem
.iPosition
= MIT_END
;
345 vItem
.iPosition
= nPos
;
349 rc
= (APIRET
)::WinSendMsg(GetHmenu(), MM_INSERTITEM
, (MPARAM
)&vItem
, (MPARAM
)pData
);
350 if (rc
== MIT_MEMERROR
|| rc
== MIT_ERROR
)
352 vError
= ::WinGetLastError(vHabmain
);
353 sError
= wxPMErrorToStr(vError
);
354 wxLogError("Error inserting or appending a menuitem. Error: %s\n", sError
);
355 wxLogLastError("Insert or AppendMenu");
361 // If we're already attached to the menubar, we must update it
365 m_menuBar
->Refresh();
370 } // end of wxMenu::DoInsertOrAppend
372 bool wxMenu::DoAppend(
376 return wxMenuBase::DoAppend(pItem
) && DoInsertOrAppend(pItem
);
379 bool wxMenu::DoInsert(
384 return ( wxMenuBase::DoInsert( nPos
386 DoInsertOrAppend( pItem
389 } // end of wxMenu::DoInsert
391 wxMenuItem
* wxMenu::DoRemove(
396 // We need to find the items position in the child list
399 wxMenuItemList::Node
* pNode
= GetMenuItems().GetFirst();
401 for (nPos
= 0; pNode
; nPos
++)
403 if (pNode
->GetData() == pItem
)
405 pNode
= pNode
->GetNext();
409 // DoRemove() (unlike Remove) can only be called for existing item!
411 wxCHECK_MSG(pNode
, NULL
, wxT("bug in wxMenu::Remove logic"));
415 // Remove the corresponding accel from the accel table
417 int n
= FindAccel(pItem
->GetId());
419 if (n
!= wxNOT_FOUND
)
425 #endif // wxUSE_ACCEL
427 // Remove the item from the menu
429 ::WinSendMsg( GetHmenu()
431 ,MPFROM2SHORT(pItem
->GetId(), TRUE
)
437 // Otherwise, the chane won't be visible
439 m_menuBar
->Refresh();
443 // And from internal data structures
445 return wxMenuBase::DoRemove(pItem
);
446 } // end of wxMenu::DoRemove
448 // ---------------------------------------------------------------------------
449 // accelerator helpers
450 // ---------------------------------------------------------------------------
455 // Create the wxAcceleratorEntries for our accels and put them into provided
456 // array - return the number of accels we have
458 size_t wxMenu::CopyAccels(
459 wxAcceleratorEntry
* pAccels
462 size_t nCount
= GetAccelCount();
464 for (size_t n
= 0; n
< nCount
; n
++)
466 *pAccels
++ = *m_vAccels
[n
];
469 } // end of wxMenu::CopyAccels
471 #endif // wxUSE_ACCEL
473 // ---------------------------------------------------------------------------
475 // ---------------------------------------------------------------------------
477 void wxMenu::SetTitle(
478 const wxString
& rLabel
481 bool bHasNoTitle
= m_title
.IsEmpty();
482 HWND hMenu
= GetHmenu();
487 if (!rLabel
.IsEmpty())
489 if (!::WinSetWindowText(hMenu
, rLabel
.c_str()))
491 wxLogLastError("SetMenuTitle");
497 if (rLabel
.IsEmpty() )
499 ::WinSendMsg( GetHmenu()
501 ,MPFROM2SHORT(hMenu
, TRUE
)
510 if (!::WinSetWindowText(hMenu
, rLabel
.c_str()))
512 wxLogLastError("SetMenuTitle");
516 } // end of wxMenu::SetTitle
518 // ---------------------------------------------------------------------------
520 // ---------------------------------------------------------------------------
522 bool wxMenu::OS2Command(
523 WXUINT
WXUNUSED(uParam
)
528 // Ignore commands from the menu title
531 if (vId
!= (WXWORD
)idMenuTitle
)
533 wxCommandEvent
vEvent(wxEVT_COMMAND_MENU_SELECTED
);
535 vEvent
.SetEventObject(this);
538 ProcessCommand(vEvent
);
541 } // end of wxMenu::OS2Command
543 bool wxMenu::ProcessCommand(
544 wxCommandEvent
& rEvent
547 bool bProcessed
= FALSE
;
549 #if wxUSE_MENU_CALLBACK
555 (void)(*(m_callback
))(*this, rEvent
);
558 #endif // wxUSE_MENU_CALLBACK
561 // Try the menu's event handler
563 if (!bProcessed
&& GetEventHandler())
565 bProcessed
= GetEventHandler()->ProcessEvent(rEvent
);
569 // Try the window the menu was popped up from (and up through the
571 wxWindow
* pWin
= GetInvokingWindow();
573 if (!bProcessed
&& pWin
)
574 bProcessed
= pWin
->GetEventHandler()->ProcessEvent(rEvent
);
576 } // end of wxMenu::ProcessCommand
578 // ---------------------------------------------------------------------------
580 // ---------------------------------------------------------------------------
587 // Menu can be in at most one menubar because otherwise they would both
588 // delete the menu pointer
590 wxASSERT_MSG(!m_menuBar
, wxT("menu belongs to 2 menubars, expect a crash"));
591 m_menuBar
= pMenubar
;
594 void wxMenu::Detach()
596 wxASSERT_MSG( m_menuBar
, wxT("can't detach menu if it's not attached") );
598 } // end of wxMenu::Detach
600 wxWindow
* wxMenu::GetWindow() const
602 if (m_invokingWindow
!= NULL
)
603 return m_invokingWindow
;
604 else if ( m_menuBar
!= NULL
)
605 return m_menuBar
->GetFrame();
608 } // end of wxMenu::GetWindow
610 // ---------------------------------------------------------------------------
612 // ---------------------------------------------------------------------------
614 void wxMenuBar::Init()
616 m_eventHandler
= this;
617 m_pMenuBarFrame
= NULL
;
619 } // end of wxMenuBar::Init
621 wxMenuBar::wxMenuBar()
624 } // end of wxMenuBar::wxMenuBar
626 wxMenuBar::wxMenuBar(
627 long WXUNUSED(lStyle
)
631 } // end of wxMenuBar::wxMenuBar
633 wxMenuBar::wxMenuBar(
636 , const wxString sTitles
[]
641 m_titles
.Alloc(nCount
);
642 for ( int i
= 0; i
< nCount
; i
++ )
644 m_menus
.Append(vMenus
[i
]);
645 m_titles
.Add(sTitles
[i
]);
646 vMenus
[i
]->Attach(this);
648 } // end of wxMenuBar::wxMenuBar
650 wxMenuBar::~wxMenuBar()
652 } // end of wxMenuBar::~wxMenuBar
654 // ---------------------------------------------------------------------------
656 // ---------------------------------------------------------------------------
658 void wxMenuBar::Refresh()
660 wxCHECK_RET( IsAttached(), wxT("can't refresh unatteched menubar") );
662 WinSendMsg(GetWinHwnd(m_pMenuBarFrame
), WM_UPDATEFRAME
, (MPARAM
)FCF_MENU
, (MPARAM
)0);
663 } // end of wxMenuBar::Refresh
665 WXHMENU
wxMenuBar::Create()
669 HWND hMenuBar
= NULLHANDLE
;
674 wxCHECK_MSG(!m_hMenu
, TRUE
, wxT("menubar already created"));
677 // Menubars should be associated with a frame otherwise they are popups
679 if (m_pMenuBarFrame
!= NULL
)
680 hFrame
= GetWinHwnd(m_pMenuBarFrame
);
682 hFrame
= HWND_DESKTOP
;
684 // Create an empty menu and then fill it with insertions
686 if (!wxWindow::OS2Create( hFrame
689 ,MS_ACTIONBAR
| WS_SYNCPAINT
| WS_VISIBLE
701 wxLogLastError("CreateMenu");
705 size_t nCount
= GetMenuCount();
707 hMenuBar
= GetHwnd();
708 for (size_t i
= 0; i
< nCount
; i
++)
716 // Set the parent and owner of the submenues to be the menubar, not the desktop
718 if (!::WinSetParent(m_menus
[i
]->m_vMenuData
.hwndSubMenu
, hMenuBar
, FALSE
))
720 vError
= ::WinGetLastError(vHabmain
);
721 sError
= wxPMErrorToStr(vError
);
722 wxLogError("Error setting parent for submenu. Error: %s\n", sError
);
726 if (!::WinSetOwner(m_menus
[i
]->m_vMenuData
.hwndSubMenu
, hMenuBar
))
728 vError
= ::WinGetLastError(vHabmain
);
729 sError
= wxPMErrorToStr(vError
);
730 wxLogError("Error setting parent for submenu. Error: %s\n", sError
);
734 m_menus
[i
]->m_vMenuData
.iPosition
= i
;
736 rc
= (APIRET
)::WinSendMsg(hMenuBar
, MM_INSERTITEM
, (MPARAM
)&m_menus
[i
]->m_vMenuData
, (MPARAM
)m_titles
[i
].c_str());
737 if (rc
== MIT_MEMERROR
|| rc
== MIT_ERROR
)
739 vError
= ::WinGetLastError(vHabmain
);
740 sError
= wxPMErrorToStr(vError
);
741 wxLogError("Error inserting or appending a menuitem. Error: %s\n", sError
);
747 } // end of wxMenuBar::Create
749 // ---------------------------------------------------------------------------
750 // wxMenuBar functions to work with the top level submenus
751 // ---------------------------------------------------------------------------
754 // NB: we don't support owner drawn top level items for now, if we do these
755 // functions would have to be changed to use wxMenuItem as well
757 void wxMenuBar::EnableTop(
762 wxCHECK_RET(IsAttached(), wxT("doesn't work with unattached menubars"));
767 uFlag
= MIA_DISABLED
;
769 nId
= SHORT1FROMMR(::WinSendMsg((HWND
)m_hMenu
, MM_ITEMIDFROMPOSITION
, MPFROMSHORT(nPos
), (MPARAM
)0));
770 if (nId
== MIT_ERROR
)
772 wxLogLastError("LogLastError");
775 ::WinSendMsg((HWND
)m_hMenu
, MM_SETITEMATTR
, MPFROM2SHORT(nId
, TRUE
), MPFROM2SHORT(uFlag
, uFlag
));
777 } // end of wxMenuBar::EnableTop
779 void wxMenuBar::SetLabelTop(
781 , const wxString
& rLabel
787 wxCHECK_RET(nPos
< GetMenuCount(), wxT("invalid menu index"));
788 m_titles
[nPos
] = rLabel
;
795 nId
= SHORT1FROMMR(::WinSendMsg((HWND
)m_hMenu
, MM_ITEMIDFROMPOSITION
, MPFROMSHORT(nPos
), (MPARAM
)0));
796 if (nId
== MIT_ERROR
)
798 wxLogLastError("LogLastError");
801 if(!::WinSendMsg( (HWND
)m_hMenu
803 ,MPFROM2SHORT(nId
, TRUE
)
807 wxLogLastError("QueryItem");
811 if (::WinSendMsg(GetHmenu(), MM_SETITEMTEXT
, MPFROMSHORT(nId
), (MPARAM
)rLabel
.c_str()));
813 wxLogLastError("ModifyMenu");
816 } // end of wxMenuBar::SetLabelTop
818 wxString
wxMenuBar::GetLabelTop(
822 wxCHECK_MSG( nPos
< GetMenuCount(), wxEmptyString
,
823 wxT("invalid menu index in wxMenuBar::GetLabelTop") );
824 return m_titles
[nPos
];
825 } // end of wxMenuBar::GetLabelTop
827 // ---------------------------------------------------------------------------
828 // wxMenuBar construction
829 // ---------------------------------------------------------------------------
831 wxMenu
* wxMenuBar::Replace(
834 , const wxString
& rTitle
838 wxString Title
= TextToLabel(rTitle
);
839 wxMenu
* pMenuOld
= wxMenuBarBase::Replace( nPos
845 nId
= SHORT1FROMMR(::WinSendMsg((HWND
)m_hMenu
, MM_ITEMIDFROMPOSITION
, MPFROMSHORT(nPos
), (MPARAM
)0));
846 if (nId
== MIT_ERROR
)
848 wxLogLastError("LogLastError");
853 m_titles
[nPos
] = Title
;
856 ::WinSendMsg((HWND
)m_hMenu
, MM_DELETEITEM
, MPFROM2SHORT(nId
, TRUE
), (MPARAM
)0);
857 ::WinSendMsg((HWND
)m_hMenu
, MM_INSERTITEM
, (MPARAM
)&pMenu
->m_vMenuData
, (MPARAM
)Title
.c_str());
860 if (pMenuOld
->HasAccels() || pMenu
->HasAccels())
863 // Need to rebuild accell table
867 #endif // wxUSE_ACCEL
871 } // end of wxMenuBar::Replace
873 bool wxMenuBar::Insert(
876 , const wxString
& rTitle
879 wxString Title
= TextToLabel(rTitle
);
880 if (!wxMenuBarBase::Insert( nPos
886 m_titles
.Insert( Title
894 ::WinSendMsg((HWND
)m_hMenu
, MM_INSERTITEM
, (MPARAM
)&pMenu
->m_vMenuData
, (MPARAM
)Title
.c_str());
896 if (pMenu
->HasAccels())
898 // need to rebuild accell table
901 #endif // wxUSE_ACCEL
905 } // end of wxMenuBar::Insert
907 bool wxMenuBar::Append(
909 , const wxString
& rTitle
912 WXHMENU hSubmenu
= pMenu
? pMenu
->GetHMenu() : 0;
914 wxCHECK_MSG(hSubmenu
, FALSE
, wxT("can't append invalid menu to menubar"));
916 wxSetShortCutKey((wxChar
*)rTitle
.c_str());
918 wxString Title
= TextToLabel(rTitle
);
919 if (!wxMenuBarBase::Append(pMenu
, Title
))
927 pMenu
->m_vMenuData
.iPosition
= MIT_END
;
928 ::WinSendMsg((HWND
)m_hMenu
, MM_INSERTITEM
, (MPARAM
)&pMenu
->m_vMenuData
, (MPARAM
)Title
.c_str());
930 if (pMenu
->HasAccels())
933 // Need to rebuild accell table
937 #endif // wxUSE_ACCEL
941 } // end of wxMenuBar::Append
943 wxMenu
* wxMenuBar::Remove(
947 wxMenu
* pMenu
= wxMenuBarBase::Remove(nPos
);
953 nId
= SHORT1FROMMR(::WinSendMsg((HWND
)GetHmenu(), MM_ITEMIDFROMPOSITION
, MPFROMSHORT(nPos
), (MPARAM
)0));
954 if (nId
== MIT_ERROR
)
956 wxLogLastError("LogLastError");
961 ::WinSendMsg((HWND
)GetHmenu(), MM_DELETEITEM
, MPFROM2SHORT(nId
, TRUE
), (MPARAM
)0);
965 if (pMenu
->HasAccels())
968 // Need to rebuild accell table
972 #endif // wxUSE_ACCEL
975 m_titles
.Remove(nPos
);
977 } // end of wxMenuBar::Remove
981 void wxMenuBar::RebuildAccelTable()
984 // Merge the accelerators of all menus into one accel table
986 size_t nAccelCount
= 0;
988 size_t nCount
= GetMenuCount();
990 for (i
= 0; i
< nCount
; i
++)
992 nAccelCount
+= m_menus
[i
]->GetAccelCount();
997 wxAcceleratorEntry
* pAccelEntries
= new wxAcceleratorEntry
[nAccelCount
];
1000 for (i
= 0; i
< nCount
; i
++)
1002 nAccelCount
+= m_menus
[i
]->CopyAccels(&pAccelEntries
[nAccelCount
]);
1004 m_vAccelTable
= wxAcceleratorTable( nAccelCount
1007 delete [] pAccelEntries
;
1009 } // end of wxMenuBar::RebuildAccelTable
1011 #endif // wxUSE_ACCEL
1013 void wxMenuBar::Attach(
1017 wxASSERT_MSG( !IsAttached(), wxT("menubar already attached!") );
1018 m_pMenuBarFrame
= pFrame
;
1021 RebuildAccelTable();
1022 #endif // wxUSE_ACCEL
1023 } // end of wxMenuBar::Attach
1025 void wxMenuBar::Detach()
1027 ::WinDestroyWindow((HWND
)m_hMenu
);
1028 m_hMenu
= (WXHMENU
)NULL
;
1029 m_pMenuBarFrame
= NULL
;
1030 } // end of wxMenuBar::Detach
1032 // ---------------------------------------------------------------------------
1033 // wxMenuBar searching for menu items
1034 // ---------------------------------------------------------------------------
1037 // Find the itemString in menuString, and return the item id or wxNOT_FOUND
1039 int wxMenuBar::FindMenuItem(
1040 const wxString
& rMenuString
1041 , const wxString
& rItemString
1044 wxString sMenuLabel
= wxStripMenuCodes(rMenuString
);
1045 size_t nCount
= GetMenuCount();
1047 for (size_t i
= 0; i
< nCount
; i
++)
1049 wxString sTitle
= wxStripMenuCodes(m_titles
[i
]);
1051 if (rMenuString
== sTitle
)
1052 return m_menus
[i
]->FindItem(rItemString
);
1055 } // end of wxMenuBar::FindMenuItem
1057 wxMenuItem
* wxMenuBar::FindItem(
1059 , wxMenu
** ppItemMenu
1065 wxMenuItem
* pItem
= NULL
;
1066 size_t nCount
= GetMenuCount();
1068 for (size_t i
= 0; !pItem
&& (i
< nCount
); i
++)
1070 pItem
= m_menus
[i
]->FindItem( nId
1075 } // end of wxMenuBar::FindItem
1078 // wxWindows' default shortcut identifier is WIN32's "&" but
1079 // OS2's is "~" so we have to change this and must watch for the
1080 // sequence, "&&" converting only the first one
1082 void wxSetShortCutKey(
1086 for (int i
= 0; zText
[i
] != '\0'; i
++)
1088 if (zText
[i
] == '&')
1091 if (zText
[i
+1] == '&')