1 ///////////////////////////////////////////////////////////////////////////////
 
   3 // Purpose:     wxMenuItem implementation
 
   4 // Author:      David Elliott
 
   8 // Copyright:   2002-2004 David Elliott
 
   9 // Licence:     wxWindows licence
 
  10 ///////////////////////////////////////////////////////////////////////////////
 
  12 // ============================================================================
 
  14 // ============================================================================
 
  16 // ----------------------------------------------------------------------------
 
  18 // ----------------------------------------------------------------------------
 
  20 #include "wx/wxprec.h"
 
  23     #include "wx/menuitem.h"
 
  29 #include "wx/cocoa/ObjcPose.h"
 
  30 #include "wx/cocoa/autorelease.h"
 
  31 #include "wx/cocoa/string.h"
 
  33 #import <AppKit/NSMenuItem.h>
 
  34 #import <AppKit/NSMenu.h>
 
  35 #import <Foundation/NSString.h>
 
  36 #import <AppKit/NSCell.h> // NSOnState, NSOffState
 
  40 // ----------------------------------------------------------------------------
 
  41 // functions prototypes
 
  42 // ----------------------------------------------------------------------------
 
  44 // ============================================================================
 
  45 // @class wxNSMenuItemTarget
 
  46 // ============================================================================
 
  47 @interface wxNSMenuItemTarget : NSObject
 
  51 - (void)wxMenuItemAction: (id)sender;
 
  52 - (BOOL)validateMenuItem: (id)menuItem;
 
  53 @end //interface wxNSMenuItemTarget
 
  55 @implementation wxNSMenuItemTarget : NSObject
 
  57 - (void)wxMenuItemAction: (id)sender
 
  59     wxLogTrace(wxTRACE_COCOA,wxT("wxMenuItemAction"));
 
  60     wxMenuItem *item = wxMenuItem::GetFromCocoa(sender);
 
  61     wxCHECK_RET(item,wxT("wxMenuItemAction received but no wxMenuItem exists!"));
 
  63     wxMenu *menu = item->GetMenu();
 
  64     wxCHECK_RET(menu,wxT("wxMenuItemAction received but wxMenuItem is not in a wxMenu"));
 
  65     wxMenuBar *menubar = menu->GetMenuBar();
 
  68         wxFrame *frame = menubar->GetFrame();
 
  69         wxCHECK_RET(frame, wxT("wxMenuBar MUST be attached to a wxFrame!"));
 
  70         frame->ProcessCommand(item->GetId());
 
  74 - (BOOL)validateMenuItem: (id)menuItem
 
  76     // TODO: Do wxWidgets validation here and avoid sending during idle time
 
  77     wxLogTrace(wxTRACE_COCOA,wxT("wxMenuItemAction"));
 
  78     wxMenuItem *item = wxMenuItem::GetFromCocoa(menuItem);
 
  79     wxCHECK_MSG(item,NO,wxT("validateMenuItem received but no wxMenuItem exists!"));
 
  80     return item->IsEnabled();
 
  83 @end //implementation wxNSMenuItemTarget
 
  85 // ============================================================================
 
  86 // wxMenuItemCocoa implementation
 
  87 // ============================================================================
 
  88 IMPLEMENT_DYNAMIC_CLASS(wxMenuItem, wxObject)
 
  89 wxMenuItemCocoaHash wxMenuItemCocoa::sm_cocoaHash;
 
  91 struct objc_object *wxMenuItemCocoa::sm_cocoaTarget = [[wxNSMenuItemTarget alloc] init];
 
  93 // ----------------------------------------------------------------------------
 
  95 // ----------------------------------------------------------------------------
 
  97 wxMenuItem *wxMenuItemBase::New(wxMenu *parentMenu,
 
 100                                 const wxString& help,
 
 104     return new wxMenuItem(parentMenu, itemid, name, help, kind, subMenu);
 
 108 wxString wxMenuItemBase::GetLabelFromText(const wxString& text)
 
 110     return wxStripMenuCodes(text);
 
 113 // ----------------------------------------------------------------------------
 
 115 // ----------------------------------------------------------------------------
 
 116 wxMenuItemCocoa::wxMenuItemCocoa(wxMenu *pParentMenu,
 
 118                        const wxString& strName,
 
 119                        const wxString& strHelp,
 
 122           : wxMenuItemBase(pParentMenu, itemid, strName, strHelp, kind, pSubMenu)
 
 124     wxAutoNSAutoreleasePool pool;
 
 125     if(m_kind == wxITEM_SEPARATOR)
 
 126         m_cocoaNSMenuItem = [[NSMenuItem separatorItem] retain];
 
 129         NSString *menuTitle = wxInitNSStringWithWxString([NSString alloc],wxStripMenuCodes(strName));
 
 130         m_cocoaNSMenuItem = [[NSMenuItem alloc] initWithTitle:menuTitle action:@selector(wxMenuItemAction:) keyEquivalent:@""];
 
 131         sm_cocoaHash.insert(wxMenuItemCocoaHash::value_type(m_cocoaNSMenuItem,this));
 
 132         [m_cocoaNSMenuItem setTarget:sm_cocoaTarget];
 
 135             wxASSERT(pSubMenu->GetNSMenu());
 
 136             [pSubMenu->GetNSMenu() setTitle:menuTitle];
 
 137             [m_cocoaNSMenuItem setSubmenu:pSubMenu->GetNSMenu()];
 
 143 wxMenuItem::~wxMenuItem()
 
 145     sm_cocoaHash.erase(m_cocoaNSMenuItem);
 
 146     [m_cocoaNSMenuItem release];
 
 149 // ----------------------------------------------------------------------------
 
 151 // ----------------------------------------------------------------------------
 
 153 void wxMenuItem::SetBitmaps(const wxBitmap& bmpChecked,
 
 154         const wxBitmap& bmpUnchecked)
 
 156     wxCHECK_RET(m_kind != wxITEM_SEPARATOR, wxT("Separator items do not have bitmaps."));
 
 157     wxAutoNSAutoreleasePool pool;
 
 158     m_bmpChecked = bmpChecked;
 
 159     m_bmpUnchecked = bmpUnchecked;
 
 162         [m_cocoaNSMenuItem setOnStateImage: bmpChecked.GetNSImage(true)];
 
 163         [m_cocoaNSMenuItem setOffStateImage: bmpUnchecked.GetNSImage(true)];
 
 167         wxASSERT_MSG(!bmpUnchecked.Ok(),wxT("Normal menu items should only have one bitmap"));
 
 168         [m_cocoaNSMenuItem setImage: bmpChecked.GetNSImage(true)];
 
 175 void wxMenuItem::Enable(bool bDoEnable)
 
 177     wxMenuItemBase::Enable(bDoEnable);
 
 178     // NOTE: Nothing to do, we respond to validateMenuItem instead
 
 181 void wxMenuItem::Check(bool check)
 
 183     wxCHECK_RET( IsCheckable(), wxT("only checkable items may be checked") );
 
 184     if(m_isChecked == check)
 
 186     wxAutoNSAutoreleasePool pool;
 
 187     if(GetKind() == wxITEM_RADIO)
 
 189         // it doesn't make sense to uncheck a radio item - what would this do?
 
 192         const wxMenuItemList& items = m_parentMenu->GetMenuItems();
 
 193         // First search backwards for other radio items
 
 194         wxMenuItemList::compatibility_iterator radioStart = items.Find(this);
 
 195         for(wxMenuItemList::compatibility_iterator prevNode = radioStart;
 
 196             prevNode && (prevNode->GetData()->GetKind() == wxITEM_RADIO);
 
 197             prevNode = prevNode->GetPrevious())
 
 199             radioStart = prevNode;
 
 201         // Now starting there set the state of every item until we're
 
 202         // out of radio items to set.
 
 203         for(wxMenuItemList::compatibility_iterator node = radioStart;
 
 204             node && (node->GetData()->GetKind() == wxITEM_RADIO);
 
 205             node = node->GetNext())
 
 207             wxMenuItem *item = node->GetData();
 
 208             bool checkItem = (item == this);
 
 209             item->wxMenuItemBase::Check(checkItem);
 
 210             [item->m_cocoaNSMenuItem setState: checkItem?NSOnState:NSOffState];
 
 213     else // normal check (non-radio) item
 
 215         wxMenuItemBase::Check(check);
 
 216         [m_cocoaNSMenuItem setState: check?NSOnState:NSOffState];
 
 220 void wxMenuItem::SetText(const wxString& label)
 
 222     wxMenuItemBase::SetText(label);
 
 223     wxCHECK_RET(m_kind != wxITEM_SEPARATOR, wxT("Separator items do not have titles."));
 
 224     [m_cocoaNSMenuItem setTitle: wxNSStringWithWxString(wxStripMenuCodes(label))];
 
 227 void wxMenuItem::SetCheckable(bool checkable)
 
 229     wxCHECK_RET(m_kind != wxITEM_SEPARATOR, wxT("Separator items cannot be turned into normal menu items."));
 
 230     wxMenuItemBase::SetCheckable(checkable);
 
 231     // NOTE: Cocoa does not discern between unchecked and normal items
 
 234 #endif // wxUSE_MENUS