implemented wxMenuBar ctor taking array of menus/titles for all ports; added optional...
[wxWidgets.git] / src / cocoa / menu.mm
1 /////////////////////////////////////////////////////////////////////////////
2 // Name:        cocoa/menu.cpp
3 // Purpose:     wxMenu and wxMenuBar implementation
4 // Author:      David Elliott
5 // Modified by:
6 // Created:     2002/12/09
7 // RCS-ID:      $Id: 
8 // Copyright:   (c) 2002 David Elliott
9 // Licence:     wxWidgets licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 #include "wx/wxprec.h"
21 #ifndef WX_PRECOMP
22     #include "wx/menu.h"
23     #include "wx/log.h"
24 #endif // WX_PRECOMP
25
26 #include "wx/cocoa/autorelease.h"
27 #include "wx/cocoa/string.h"
28
29 #import <Foundation/NSString.h>
30 #import <AppKit/NSMenu.h>
31
32 #if wxUSE_MENUS
33
34 // ----------------------------------------------------------------------------
35 // globals
36 // ----------------------------------------------------------------------------
37
38 // ============================================================================
39 // wxMenu implementation
40 // ============================================================================
41
42 IMPLEMENT_DYNAMIC_CLASS(wxMenu,wxEvtHandler)
43
44 bool wxMenu::Create(const wxString& title, long style)
45 {
46     wxAutoNSAutoreleasePool pool;
47     m_cocoaNSMenu = [[NSMenu alloc] initWithTitle: wxNSStringWithWxString(title)];
48     AssociateNSMenu(m_cocoaNSMenu);
49     return true;
50 }
51
52 wxMenu::~wxMenu()
53 {
54     DisassociateNSMenu(m_cocoaNSMenu);
55     if(!m_cocoaDeletes)
56         [m_cocoaNSMenu release];
57 }
58
59 wxMenuItem* wxMenu::DoAppend(wxMenuItem *item)
60 {
61     wxAutoNSAutoreleasePool pool;
62     if(!wxMenuBase::DoAppend(item))
63         return NULL;
64     [m_cocoaNSMenu addItem: item->GetNSMenuItem()];
65     return item;
66 }
67
68 wxMenuItem* wxMenu::DoInsert(unsigned long pos, wxMenuItem *item)
69 {
70     wxAutoNSAutoreleasePool pool;
71     if(!wxMenuBase::DoInsert(pos,item))
72         return NULL;
73     [m_cocoaNSMenu insertItem:item->GetNSMenuItem() atIndex:pos];
74     return item;
75 }
76
77 wxMenuItem* wxMenu::DoRemove(wxMenuItem *item)
78 {
79     wxAutoNSAutoreleasePool pool;
80     wxMenuItem *retitem = wxMenuBase::DoRemove(item);
81     wxASSERT(retitem->GetNSMenuItem());
82     [m_cocoaNSMenu removeItem:retitem->GetNSMenuItem()];
83     return retitem;
84 }
85
86 // This autoreleases the menu on the assumption that something is
87 // going to retain it shortly (for instance, it is going to be returned from
88 // an overloaded [NSStatusItem menu] or from the applicationDockMenu:
89 // NSApplication delegate method.
90 //
91 // It then sets a bool flag m_cocoaDeletes.  When the NSMenu is dealloc'd
92 // (dealloc is the Cocoa destructor) we delete ourselves.  In this manner we
93 // can be available for Cocoa calls until Cocoa is finished with us.
94 //
95 // I can see very few reasons to undo this.  Nevertheless, it is implemented.
96 void wxMenu::SetCocoaDeletes(bool cocoaDeletes)
97 {
98     if(m_cocoaDeletes==cocoaDeletes)
99         return;
100     m_cocoaDeletes = cocoaDeletes;
101     if(m_cocoaDeletes)
102         [m_cocoaNSMenu autorelease];
103     else
104         [m_cocoaNSMenu retain];
105 }
106
107 void wxMenu::Cocoa_dealloc()
108 {
109     if(m_cocoaDeletes)
110         delete this;
111 }
112
113 // ============================================================================
114 // wxMenuBar implementation
115 // ============================================================================
116 IMPLEMENT_DYNAMIC_CLASS(wxMenuBar,wxWindow)
117
118 bool wxMenuBar::Create(long style)
119 {
120     wxAutoNSAutoreleasePool pool;
121     m_cocoaNSMenu = [[NSMenu alloc] initWithTitle: @"wxMenuBar"];
122
123     NSMenuItem *dummyItem = [[NSMenuItem alloc] initWithTitle:@"App menu"
124         /* Note: title gets clobbered by app name anyway */
125         action:nil keyEquivalent:@""];
126     [m_cocoaNSMenu addItem:dummyItem];
127     [dummyItem release];
128     return true;
129 }
130
131 wxMenuBar::wxMenuBar(size_t n, wxMenu *menus[], const wxString titles[], long WXUNUSED(style))
132 {
133     Create(style);
134
135     for(size_t i = 0; i < n; ++i )
136         Append(menus[i], titles[i]);
137 }
138
139 wxMenuBar::~wxMenuBar()
140 {
141     [m_cocoaNSMenu release];
142 }
143
144 bool wxMenuBar::Append( wxMenu *menu, const wxString &title )
145 {
146     wxAutoNSAutoreleasePool pool;
147     wxLogTrace(wxTRACE_COCOA,wxT("append menu=%p, title=%s"),menu,title.c_str());
148     if(!wxMenuBarBase::Append(menu,title))
149         return false;
150     wxASSERT(menu);
151     wxASSERT(menu->GetNSMenu());
152     NSString *menuTitle = wxInitNSStringWithWxString([NSString alloc], wxStripMenuCodes(title));
153     NSMenuItem *newItem = [[NSMenuItem alloc] initWithTitle:menuTitle action:NULL keyEquivalent:@""];
154     [menu->GetNSMenu() setTitle:menuTitle];
155     [newItem setSubmenu:menu->GetNSMenu()];
156
157     [m_cocoaNSMenu addItem:newItem];
158
159     [menuTitle release];
160     [newItem release];
161     return true;
162 }
163
164 bool wxMenuBar::Insert(size_t pos, wxMenu *menu, const wxString& title)
165 {
166     wxAutoNSAutoreleasePool pool;
167     wxLogTrace(wxTRACE_COCOA,wxT("insert pos=%lu, menu=%p, title=%s"),pos,menu,title.c_str());
168     // Get the current menu at this position
169     wxMenu *nextmenu = GetMenu(pos);
170     if(!wxMenuBarBase::Insert(pos,menu,title))
171         return false;
172     wxASSERT(menu);
173     wxASSERT(menu->GetNSMenu());
174     NSString *menuTitle = wxInitNSStringWithWxString([NSString alloc], title);
175     NSMenuItem *newItem = [[NSMenuItem alloc] initWithTitle:menuTitle action:NULL keyEquivalent:@""];
176     [menu->GetNSMenu() setTitle:menuTitle];
177     [newItem setSubmenu:menu->GetNSMenu()];
178
179     int itemindex = [m_cocoaNSMenu indexOfItemWithSubmenu:nextmenu->GetNSMenu()];
180     wxASSERT(itemindex>=0);
181     [m_cocoaNSMenu insertItem:newItem atIndex:itemindex];
182
183     [menuTitle release];
184     [newItem release];
185     return true;
186 }
187
188 wxMenu *wxMenuBar::Replace(size_t pos, wxMenu *menu, const wxString& title)
189 {
190     return NULL;
191 }
192
193 wxMenu *wxMenuBar::Remove(size_t pos)
194 {
195     wxMenu *menu = wxMenuBarBase::Remove(pos);
196     wxASSERT(menu);
197     int itemindex = [GetNSMenu() indexOfItemWithSubmenu:menu->GetNSMenu()];
198     wxASSERT(itemindex>=0);
199     [m_cocoaNSMenu removeItemAtIndex:itemindex];
200     return menu;
201 }
202
203
204 void wxMenuBar::EnableTop(size_t pos, bool enable)
205 {
206 }
207
208 bool wxMenuBar::IsEnabledTop(size_t pos) const
209 {
210     return false;
211 }
212
213 void wxMenuBar::SetLabelTop(size_t pos, const wxString& label)
214 {
215 }
216
217 wxString wxMenuBar::GetLabelTop(size_t pos) const
218 {
219     wxMenu *menu = GetMenu(pos);
220     int itemindex = [m_cocoaNSMenu indexOfItemWithSubmenu:menu->GetNSMenu()];
221     wxASSERT(itemindex>=0);
222     return wxStringWithNSString([[m_cocoaNSMenu itemAtIndex:itemindex] title]);
223 }
224
225 void wxMenuBar::Attach(wxFrame *frame)
226 {
227     wxMenuBarBase::Attach(frame);
228 }
229
230 void wxMenuBar::Detach()
231 {
232     wxMenuBarBase::Detach();
233 }
234
235 wxSize wxMenuBar::DoGetBestClientSize() const
236 {
237     return wxDefaultSize;
238 }
239
240 #endif // wxUSE_MENUS