1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/carbon/menu.cpp
3 // Purpose: wxMenu, wxMenuBar, wxMenuItem
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
13 // headers & declarations
14 // ============================================================================
19 #include "wx/wxprec.h"
28 #include "wx/menuitem.h"
31 #include "wx/osx/private.h"
32 #include "wx/stockitem.h"
34 // other standard headers
35 // ----------------------
38 // under carbon there's no such thing as a MenuItemRef, everything is done
39 // on the 'parent' menu via index APIs (first line having index 1 !)
40 // so to make things still work, we store the wxMenuItemImpl instance as a
41 // RefCon at the respective menu line
43 class wxMenuItemCarbonImpl
: public wxMenuItemImpl
46 wxMenuItemCarbonImpl( wxMenuItem
* peer
) : wxMenuItemImpl(peer
)
48 // the parent menu ref is only set, once the item has been attached
49 m_parentMenuRef
= NULL
;
52 ~wxMenuItemCarbonImpl();
54 void SetBitmap( const wxBitmap
& bitmap
)
56 MenuItemIndex i
= FindMenuItemIndex() ;
62 ControlButtonContentInfo info
;
63 wxMacCreateBitmapButton( &info
, bitmap
) ;
64 if ( info
.contentType
!= kControlNoContent
)
66 if ( info
.contentType
== kControlContentIconRef
)
67 SetMenuItemIconHandle( m_parentMenuRef
, i
,
68 kMenuIconRefType
, (Handle
) info
.u
.iconRef
) ;
69 else if ( info
.contentType
== kControlContentCGImageRef
)
70 SetMenuItemIconHandle( m_parentMenuRef
, i
,
71 kMenuCGImageRefType
, (Handle
) info
.u
.imageRef
) ;
73 wxMacReleaseBitmapButton( &info
) ;
79 void Enable( bool enable
)
81 MenuItemIndex i
= FindMenuItemIndex() ;
85 if ( GetWXPeer()->GetId() == wxApp::s_macPreferencesMenuItemId
)
88 EnableMenuCommand( NULL
, kHICommandPreferences
) ;
90 DisableMenuCommand( NULL
, kHICommandPreferences
) ;
92 else if ( GetWXPeer()->GetId() == wxApp::s_macExitMenuItemId
)
95 EnableMenuCommand( NULL
, kHICommandQuit
) ;
97 DisableMenuCommand( NULL
, kHICommandQuit
) ;
101 EnableMenuItem(m_parentMenuRef
, i
);
103 DisableMenuItem(m_parentMenuRef
, i
);
105 if ( GetWXPeer()->IsSubMenu() )
107 UMAEnableMenuItem( GetWXPeer()->GetSubMenu()->GetHMenu() , 0 , enable
) ;
112 void Check( bool check
)
114 MenuItemIndex i
= FindMenuItemIndex() ;
118 ::SetItemMark( m_parentMenuRef
, i
, 0x12 ) ; // checkmark
120 ::SetItemMark( m_parentMenuRef
, i
, 0 ) ; // no mark
124 void Hide( bool hide
)
126 MenuItemIndex i
= FindMenuItemIndex() ;
130 ChangeMenuItemAttributes( m_parentMenuRef
, i
, kMenuItemAttrHidden
, 0 );
132 ChangeMenuItemAttributes( m_parentMenuRef
, i
, 0 , kMenuItemAttrHidden
);
136 void SetLabel( const wxString
& text
, wxAcceleratorEntry
*entry
)
138 MenuItemIndex i
= FindMenuItemIndex() ;
141 SetMenuItemTextWithCFString( m_parentMenuRef
, i
, wxCFStringRef(text
));
142 UMASetMenuItemShortcut( m_parentMenuRef
, i
, entry
) ;
146 void * GetHMenuItem() { return NULL
; }
150 void AttachToParent( MenuRef parentMenuRef
, MenuItemIndex index
)
152 m_parentMenuRef
= parentMenuRef
;
153 if ( m_parentMenuRef
&& index
> 0 )
154 SetMenuItemRefCon( m_parentMenuRef
, index
, (URefCon
) this );
157 MenuItemIndex
FindMenuItemIndex()
159 MenuItemIndex hit
= 0 ;
160 if ( m_parentMenuRef
)
162 for ( MenuItemIndex i
= 1 ; i
<= CountMenuItems(m_parentMenuRef
) ; ++i
)
164 URefCon storedRef
= 0;
165 GetMenuItemRefCon(m_parentMenuRef
, i
, &storedRef
);
166 if ( storedRef
== (URefCon
) this )
176 MenuRef m_parentMenuRef
;
183 class wxMenuCarbonImpl
: public wxMenuImpl
186 wxMenuCarbonImpl( wxMenu
* peer
, MenuRef menu
) : wxMenuImpl(peer
), m_osxMenu(menu
)
190 virtual ~wxMenuCarbonImpl();
193 virtual void InsertOrAppend(wxMenuItem
*pItem
, size_t pos
)
195 // MacOS counts menu items from 1 and inserts after, therefore having the
196 // same effect as wx 0 based and inserting before, we must correct pos
197 // after however for updates to be correct
199 MenuItemIndex index
= pos
;
200 if ( pos
== (size_t) -1 )
201 index
= CountMenuItems(m_osxMenu
);
203 if ( pItem
->IsSeparator() )
205 InsertMenuItemTextWithCFString( m_osxMenu
, CFSTR(""), index
, kMenuItemAttrSeparator
, 0);
206 // now switch to the Carbon 1 based counting
211 InsertMenuItemTextWithCFString( m_osxMenu
, CFSTR("placeholder"), index
, 0, 0 );
213 // now switch to the Carbon 1 based counting
215 if ( pItem
->IsSubMenu() )
217 MenuRef submenu
= pItem
->GetSubMenu()->GetHMenu();
218 SetMenuItemHierarchicalMenu(m_osxMenu
, index
, submenu
);
219 // carbon is using the title of the submenu, eg in the menubar
220 SetMenuTitleWithCFString(submenu
, wxCFStringRef(wxStripMenuCodes(pItem
->GetLabel())));
224 SetMenuItemCommandID( m_osxMenu
, index
, wxIdToMacCommand(pItem
->GetId()) ) ;
228 wxMenuItemCarbonImpl
* impl
= (wxMenuItemCarbonImpl
*) pItem
->GetPeer();
229 impl
->AttachToParent( m_osxMenu
, index
);
230 // only now can all settings be updated correctly
231 pItem
->UpdateItemText();
232 pItem
->UpdateItemStatus();
233 pItem
->UpdateItemBitmap();
236 virtual void Remove( wxMenuItem
*pItem
)
238 wxMenuItemCarbonImpl
* impl
= (wxMenuItemCarbonImpl
*) pItem
->GetPeer();
241 MenuItemIndex i
= impl
->FindMenuItemIndex();
244 DeleteMenuItem(m_osxMenu
, i
);
245 impl
->AttachToParent( NULL
, 0 );
250 virtual void MakeRoot()
252 SetRootMenu( m_osxMenu
);
255 virtual void SetTitle( const wxString
& text
)
257 SetMenuTitleWithCFString(m_osxMenu
, wxCFStringRef(text
));
260 WXHMENU
GetHMenu() { return m_osxMenu
; }
262 static wxMenuImpl
* Create( wxMenu
* peer
, const wxString
& title
);
263 static wxMenuImpl
* CreateRootMenu( wxMenu
* peer
);
265 wxCFRef
<MenuRef
> m_osxMenu
;
268 // static const short kwxMacAppleMenuId = 1 ;
270 // Find an item given the Macintosh Menu Reference
272 WX_DECLARE_HASH_MAP(WXHMENU
, wxMenu
*, wxPointerHash
, wxPointerEqual
, MacMenuMap
);
274 static MacMenuMap wxWinMacMenuList
;
276 wxMenu
*wxFindMenuFromMacMenu(WXHMENU inMenuRef
)
278 MacMenuMap::iterator node
= wxWinMacMenuList
.find(inMenuRef
);
280 return (node
== wxWinMacMenuList
.end()) ? NULL
: node
->second
;
283 void wxAssociateMenuWithMacMenu(WXHMENU inMenuRef
, wxMenu
*menu
) ;
284 void wxAssociateMenuWithMacMenu(WXHMENU inMenuRef
, wxMenu
*menu
)
286 // adding NULL MenuRef is (first) surely a result of an error and
287 // (secondly) breaks menu command processing
288 wxCHECK_RET( inMenuRef
!= (WXHMENU
) NULL
, wxT("attempt to add a NULL MenuRef to menu list") );
290 wxWinMacMenuList
[inMenuRef
] = menu
;
293 void wxRemoveMacMenuAssociation(wxMenu
*menu
) ;
294 void wxRemoveMacMenuAssociation(wxMenu
*menu
)
296 // iterate over all the elements in the class
297 MacMenuMap::iterator it
;
298 for ( it
= wxWinMacMenuList
.begin(); it
!= wxWinMacMenuList
.end(); ++it
)
300 if ( it
->second
== menu
)
302 wxWinMacMenuList
.erase(it
);
308 wxMenuCarbonImpl::~wxMenuCarbonImpl()
310 wxRemoveMacMenuAssociation( GetWXPeer() );
313 wxMenuImpl
* wxMenuImpl::Create( wxMenu
* peer
, const wxString
& title
)
316 static SInt16 s_macNextMenuId
= 3;
318 CreateNewMenu( s_macNextMenuId
++ , 0 , &menu
) ;
321 wxLogLastError(wxT("CreateNewMenu failed"));
325 wxMenuImpl
* c
= new wxMenuCarbonImpl( peer
, menu
);
327 wxAssociateMenuWithMacMenu( menu
, peer
) ;
335 wxMenuItemCarbonImpl::~wxMenuItemCarbonImpl()
340 wxMenuItemImpl
* wxMenuItemImpl::Create( wxMenuItem
* peer
,
341 wxMenu
* WXUNUSED(pParentMenu
),
343 const wxString
& WXUNUSED(text
),
344 wxAcceleratorEntry
*WXUNUSED(entry
),
345 const wxString
& WXUNUSED(strHelp
),
346 wxItemKind
WXUNUSED(kind
),
347 wxMenu
*WXUNUSED(pSubMenu
) )
349 wxMenuItemImpl
* c
= NULL
;
351 c
= new wxMenuItemCarbonImpl( peer
);
355 void wxInsertMenuItemsInMenu(wxMenu
* menu
, MenuRef wm
, MenuItemIndex insertAfter
)
357 wxMenuItemList::compatibility_iterator node
;
359 wxMenu
*subMenu
= NULL
;
360 bool newItems
= false;
362 for (node
= menu
->GetMenuItems().GetFirst(); node
; node
= node
->GetNext())
364 item
= (wxMenuItem
*)node
->GetData();
365 subMenu
= item
->GetSubMenu() ;
368 wxInsertMenuItemsInMenu(subMenu
, (MenuRef
)subMenu
->GetHMenu(), 0);
370 if ( item
->IsSeparator() )
373 InsertMenuItemTextWithCFString( wm
,
374 CFSTR(""), insertAfter
, kMenuItemAttrSeparator
, 0);
381 entry
= wxAcceleratorEntry::Create( item
->GetItemLabel() ) ;
383 MenuItemIndex winListPos
= (MenuItemIndex
)-1;
384 OSStatus err
= GetIndMenuItemWithCommandID(wm
,
385 wxIdToMacCommand ( item
->GetId() ), 1, NULL
, &winListPos
);
387 if ( wm
&& err
== menuItemNotFoundErr
)
389 // NB: the only way to determine whether or not we should add
390 // a separator is to know if we've added menu items to the menu
391 // before the separator.
393 UMAInsertMenuItem(wm
, wxStripMenuCodes(item
->GetItemLabel()) , wxFont::GetDefaultEncoding(), insertAfter
, entry
);
394 SetMenuItemCommandID( wm
, insertAfter
+1 , wxIdToMacCommand ( item
->GetId() ) ) ;
395 SetMenuItemRefCon( wm
, insertAfter
+1 , (URefCon
) item
) ;