]>
Commit | Line | Data |
---|---|---|
421a8431 | 1 | ///////////////////////////////////////////////////////////////////////////// |
11e62fe6 | 2 | // Name: src/cocoa/combobox.mm |
421a8431 | 3 | // Purpose: wxComboBox |
11e62fe6 | 4 | // Author: Ryan Norton |
421a8431 | 5 | // Modified by: |
8f248607 | 6 | // Created: 2005/02/16 |
421a8431 | 7 | // Copyright: (c) 2003 David Elliott |
526954c5 | 8 | // Licence: wxWindows licence |
421a8431 DE |
9 | ///////////////////////////////////////////////////////////////////////////// |
10 | ||
8f248607 RN |
11 | // |
12 | // Impl notes: | |
3103e8a9 | 13 | // There is no custom data source because doing so unnecessarily sacrifices |
4c51a665 | 14 | // some native autocompletion behaviour (we would have to make our own - |
8f248607 RN |
15 | // the SimpleComboBox sample does so in the developer folder that |
16 | // comes with OSX). One reason you might want this would be to have | |
3103e8a9 | 17 | // only one array or be able to display numbers returned by an NSNumber |
8f248607 RN |
18 | // from the methods. |
19 | // | |
20 | // One problem though is that wxCB_SORT isn't implemented... | |
21 | // | |
22 | // Also, not sure if it is correctly getting text field events since | |
23 | // I'm using SetNSComboBox instead of SetNSTextField | |
24 | // | |
25 | // doWxEvent is really hackish... but since there's only one event... | |
26 | // | |
27 | // Ideas for future improvement - other notes: | |
11e62fe6 WS |
28 | // Combox w/o wxCB_DROPDOWN doesn't seem to be implementable |
29 | //wxCB_READONLY Same as wxCB_DROPDOWN but only the strings specified as the combobox choices can be selected, it is impossible to select (even from a program) a string which is not in the choices list. | |
30 | //wxCB_SORT is possible with data source | |
8f248607 RN |
31 | // |
32 | // setIntercellSpacing:/setItemHeight: to autoadjust to number of inserted items? | |
33 | // | |
34 | /* | |
35 | //example of autocompletion from SimpleComboBox Example | |
36 | // ========================================================== | |
37 | // Combo box data source methods | |
38 | // ========================================================== | |
39 | ||
40 | - (int)numberOfItemsInComboBox:(NSComboBox *)aComboBox { | |
41 | return [genres count]; | |
42 | } | |
43 | - (id)comboBox:(NSComboBox *)aComboBox objectValueForItemAtIndex:(int)index { | |
44 | return [genres objectAtIndex:index]; | |
45 | } | |
46 | - (unsigned int)comboBox:(NSComboBox *)aComboBox indexOfItemWithStringValue:(NSString *)string { | |
47 | return [genres indexOfObject: string]; | |
48 | } | |
49 | ||
50 | - (NSString *) firstGenreMatchingPrefix:(NSString *)prefix { | |
51 | NSString *string = nil; | |
52 | NSString *lowercasePrefix = [prefix lowercaseString]; | |
53 | NSEnumerator *stringEnum = [genres objectEnumerator]; | |
54 | while ((string = [stringEnum nextObject])) { | |
11e62fe6 | 55 | if ([[string lowercaseString] hasPrefix: lowercasePrefix]) return string; |
8f248607 RN |
56 | } |
57 | return nil; | |
58 | } | |
59 | ||
60 | - (NSString *)comboBox:(NSComboBox *)aComboBox completedString:(NSString *)inputString { | |
61 | // This method is received after each character typed by the user, because we have checked the "completes" flag for genreComboBox in IB. | |
62 | // Given the inputString the user has typed, see if we can find a genre with the prefix, and return it as the suggested complete string. | |
63 | NSString *candidate = [self firstGenreMatchingPrefix: inputString]; | |
64 | return (candidate ? candidate : inputString); | |
65 | } | |
66 | */ | |
2ee09b55 | 67 | |
8f248607 RN |
68 | // ============================================================================ |
69 | // declarations | |
70 | // ============================================================================ | |
71 | ||
72 | // ---------------------------------------------------------------------------- | |
73 | // headers | |
74 | // ---------------------------------------------------------------------------- | |
75 | ||
76 | #include "wx/wxprec.h" | |
8228b893 | 77 | |
061896d1 DE |
78 | #if wxUSE_COMBOBOX |
79 | ||
a5bbd1cc WS |
80 | #include "wx/combobox.h" |
81 | ||
e7e1ad7d DE |
82 | #include "wx/cocoa/objc/objc_uniquifying.h" |
83 | ||
8f248607 RN |
84 | #ifndef WX_PRECOMP |
85 | #include "wx/window.h" | |
e4db172a | 86 | #include "wx/log.h" |
670f9935 | 87 | #include "wx/app.h" |
8f248607 RN |
88 | #endif // WX_PRECOMP |
89 | ||
8f248607 RN |
90 | #import <AppKit/NSComboBox.h> |
91 | #import <Foundation/NSNotification.h> | |
92 | #import <Foundation/NSString.h> | |
93 | ||
94 | // ---------------------------------------------------------------------------- | |
95 | // globals | |
96 | // ---------------------------------------------------------------------------- | |
97 | WX_IMPLEMENT_OBJC_INTERFACE_HASHMAP(NSComboBox) | |
98 | ||
99 | void wxCocoaNSComboBox::AssociateNSComboBox(WX_NSComboBox cocoaNSComboBox) | |
100 | { | |
101 | if(cocoaNSComboBox) | |
102 | { | |
103 | sm_cocoaHash.insert(wxCocoaNSComboBoxHash::value_type(cocoaNSComboBox,this)); | |
11e62fe6 | 104 | |
8f248607 RN |
105 | [[NSNotificationCenter defaultCenter] addObserver:(id)cocoaNSComboBox selector:@selector(comboBoxSelectionDidChange:) name:@"NSComboBoxSelectionDidChangeNotification" object:cocoaNSComboBox]; |
106 | [[NSNotificationCenter defaultCenter] addObserver:(id)cocoaNSComboBox selector:@selector(comboBoxSelectionDidChange:) name:@"NSComboBoxSelectionIsChangingNotification" object:cocoaNSComboBox]; | |
107 | [[NSNotificationCenter defaultCenter] addObserver:(id)cocoaNSComboBox selector:@selector(comboBoxSelectionDidChange:) name:@"NSComboBoxWillDismissNotification" object:cocoaNSComboBox]; | |
108 | [[NSNotificationCenter defaultCenter] addObserver:(id)cocoaNSComboBox selector:@selector(comboBoxSelectionDidChange:) name:@"NSComboBoxWillPopUpNotification" object:cocoaNSComboBox]; | |
109 | } | |
110 | } | |
111 | ||
112 | void wxCocoaNSComboBox::DisassociateNSComboBox(WX_NSComboBox cocoaNSComboBox) | |
113 | { | |
114 | if(cocoaNSComboBox) | |
115 | { | |
116 | sm_cocoaHash.erase(cocoaNSComboBox); | |
117 | [[NSNotificationCenter defaultCenter] removeObserver:(id)cocoaNSComboBox name:@"NSComboBoxSelectionDidChangeNotification" object:cocoaNSComboBox]; | |
118 | [[NSNotificationCenter defaultCenter] removeObserver:(id)cocoaNSComboBox name:@"NSComboBoxSelectionIsChangingNotification" object:cocoaNSComboBox]; | |
119 | [[NSNotificationCenter defaultCenter] removeObserver:(id)cocoaNSComboBox name:@"NSComboBoxWillDismissNotification" object:cocoaNSComboBox]; | |
120 | [[NSNotificationCenter defaultCenter] removeObserver:(id)cocoaNSComboBox name:@"NSComboBoxWillPopUpNotification" object:cocoaNSComboBox]; | |
121 | } | |
122 | } | |
123 | ||
124 | // ============================================================================ | |
125 | // @class wxPoserNSComboBox | |
126 | // ============================================================================ | |
127 | @interface wxPoserNSComboBox : NSComboBox | |
128 | { | |
129 | } | |
130 | ||
131 | - (void)comboBoxSelectionDidChange:(NSNotification *)notification; | |
132 | - (void)comboBoxSelectionIsChanging:(NSNotification *)notification; | |
133 | - (void)comboBoxWillDismiss:(NSNotification *)notification; | |
134 | - (void)comboBoxWillPopUp:(NSNotification *)notification; | |
135 | @end // wxPoserNSComboBox | |
e7e1ad7d | 136 | WX_DECLARE_GET_OBJC_CLASS(wxPoserNSComboBox,NSComboBox) |
8f248607 RN |
137 | |
138 | //WX_IMPLEMENT_POSER(wxPoserNSComboBox); | |
139 | @implementation wxPoserNSComboBox : NSComboBox | |
140 | ||
141 | - (void)comboBoxSelectionDidChange:(NSNotification *)notification | |
142 | { | |
143 | wxCocoaNSComboBox *win = wxCocoaNSComboBox::GetFromCocoa(self); | |
ce7fe42e | 144 | win->doWxEvent(wxEVT_COMBOBOX); |
8f248607 RN |
145 | } |
146 | ||
147 | - (void)comboBoxSelectionIsChanging:(NSNotification *)notification | |
148 | { | |
149 | //... | |
150 | } | |
151 | ||
152 | - (void)comboBoxWillDismiss:(NSNotification *)notification | |
153 | { | |
154 | //... | |
155 | } | |
156 | ||
157 | - (void)comboBoxWillPopUp:(NSNotification *)notification | |
158 | { | |
159 | //... | |
160 | } | |
161 | ||
162 | @end // implementation wxPoserNSComboBox | |
e7e1ad7d | 163 | WX_IMPLEMENT_GET_OBJC_CLASS(wxPoserNSComboBox,NSComboBox) |
8f248607 | 164 | |
421a8431 DE |
165 | #include "wx/cocoa/autorelease.h" |
166 | #include "wx/cocoa/string.h" | |
167 | ||
168 | #import <AppKit/NSComboBox.h> | |
169 | ||
e960ba59 | 170 | BEGIN_EVENT_TABLE(wxComboBox, wxControl) |
421a8431 | 171 | END_EVENT_TABLE() |
8f248607 | 172 | WX_IMPLEMENT_COCOA_OWNER(wxComboBox,NSComboBox,NSTextField,NSView) |
e960ba59 | 173 | WX_IMPLEMENT_COCOA_OWNER(wxComboBox,NSTextField,NSControl,NSView) |
421a8431 | 174 | |
584ad2a3 MB |
175 | bool wxComboBox::Create(wxWindow *parent, wxWindowID winid, |
176 | const wxString& value, | |
177 | const wxPoint& pos, | |
178 | const wxSize& size, | |
179 | const wxArrayString& choices, | |
180 | long style, | |
181 | const wxValidator& validator, | |
182 | const wxString& name) | |
183 | { | |
184 | wxCArrayString chs(choices); | |
185 | ||
186 | return Create(parent, winid, value, pos, size, chs.GetCount(), | |
187 | chs.GetStrings(), style, validator, name); | |
188 | } | |
189 | ||
421a8431 DE |
190 | bool wxComboBox::Create(wxWindow *parent, wxWindowID winid, |
191 | const wxString& value, | |
192 | const wxPoint& pos, | |
193 | const wxSize& size, | |
194 | int n, const wxString choices[], | |
195 | long style, | |
196 | const wxValidator& validator, | |
197 | const wxString& name) | |
198 | { | |
199 | wxAutoNSAutoreleasePool pool; | |
200 | if(!CreateControl(parent,winid,pos,size,style,validator,name)) | |
201 | return false; | |
11e62fe6 | 202 | |
421a8431 | 203 | m_cocoaNSView = NULL; |
e7e1ad7d | 204 | SetNSComboBox([[WX_GET_OBJC_CLASS(wxPoserNSComboBox) alloc] initWithFrame:MakeDefaultNSRect(size)]); |
421a8431 DE |
205 | [m_cocoaNSView release]; |
206 | [GetNSTextField() setStringValue:wxNSStringWithWxString(value.c_str())]; | |
207 | [GetNSControl() sizeToFit]; | |
208 | if(m_parent) | |
209 | m_parent->CocoaAddChild(this); | |
8d656ea9 DE |
210 | SetInitialFrameRect(pos,size); |
211 | ||
a236aa20 | 212 | wxComboBox::Append(n, choices); |
11e62fe6 | 213 | |
8f248607 | 214 | [GetNSComboBox() setCompletes:true]; //autocomplete :) |
11e62fe6 | 215 | |
421a8431 DE |
216 | return true; |
217 | } | |
218 | ||
219 | wxComboBox::~wxComboBox() | |
220 | { | |
8f248607 | 221 | DisassociateNSComboBox(GetNSComboBox()); |
421a8431 DE |
222 | } |
223 | ||
8f248607 | 224 | void wxComboBox::doWxEvent(int nEvent) |
421a8431 | 225 | { |
ce7fe42e | 226 | wxCommandEvent event2(wxEVT_COMBOBOX, GetId() ); |
8f248607 RN |
227 | event2.SetInt(GetSelection()); |
228 | event2.SetEventObject(this); | |
229 | event2.SetString(GetStringSelection()); | |
937013e0 | 230 | HandleWindowEvent(event2); |
8f248607 RN |
231 | |
232 | // For consistency with MSW and GTK, also send a text updated event | |
233 | // After all, the text is updated when a selection is made | |
ce7fe42e | 234 | wxCommandEvent TextEvent( wxEVT_TEXT, GetId() ); |
8f248607 RN |
235 | TextEvent.SetString( GetStringSelection() ); |
236 | TextEvent.SetEventObject( this ); | |
937013e0 | 237 | HandleWindowEvent( TextEvent ); |
8f248607 RN |
238 | } |
239 | ||
240 | ||
241 | void wxComboBox::SetSelection(int nSelection) | |
242 | { | |
243 | [GetNSComboBox() selectItemAtIndex:nSelection]; | |
421a8431 DE |
244 | } |
245 | ||
246 | wxString wxComboBox::GetStringSelection() | |
247 | { | |
8f248607 | 248 | return wxStringWithNSString([GetNSComboBox() objectValueOfSelectedItem]); |
421a8431 DE |
249 | } |
250 | ||
a236aa20 | 251 | void wxComboBox::DoClear() |
421a8431 | 252 | { |
8f248607 RN |
253 | [GetNSComboBox() removeAllItems]; |
254 | m_Datas.Clear(); | |
421a8431 DE |
255 | } |
256 | ||
a236aa20 | 257 | void wxComboBox::DoDeleteOneItem(unsigned int n) |
421a8431 | 258 | { |
aa61d352 VZ |
259 | [GetNSComboBox() removeItemAtIndex:n]; |
260 | m_Datas.RemoveAt(n); | |
421a8431 DE |
261 | } |
262 | ||
aa61d352 | 263 | unsigned int wxComboBox::GetCount() const |
421a8431 | 264 | { |
aa61d352 | 265 | return (unsigned int)[GetNSComboBox() numberOfItems]; |
421a8431 DE |
266 | } |
267 | ||
aa61d352 | 268 | wxString wxComboBox::GetString(unsigned int nIndex) const |
11e62fe6 WS |
269 | { |
270 | return wxStringWithNSString([GetNSComboBox() itemObjectValueAtIndex:nIndex]); | |
271 | } | |
421a8431 | 272 | |
aa61d352 | 273 | void wxComboBox::SetString(unsigned int nIndex, const wxString& szString) |
11e62fe6 | 274 | { |
8f248607 RN |
275 | wxAutoNSAutoreleasePool pool; |
276 | //FIXME: There appears to be no "set item data" method - maybe | |
277 | //an assignment would work? | |
278 | [GetNSComboBox() removeItemAtIndex:nIndex]; | |
11e62fe6 | 279 | [GetNSComboBox() insertItemWithObjectValue:wxNSStringWithWxString(szString) atIndex:nIndex]; |
421a8431 DE |
280 | } |
281 | ||
11e62fe6 WS |
282 | int wxComboBox::FindString(const wxString& szItem, bool bCase) const |
283 | { | |
284 | // FIXME: use wxItemContainerImmutable::FindString for bCase parameter | |
285 | return [GetNSComboBox() indexOfItemWithObjectValue:wxNSStringWithWxString(szItem)]; | |
286 | } | |
421a8431 DE |
287 | |
288 | int wxComboBox::GetSelection() const | |
11e62fe6 WS |
289 | { |
290 | return [GetNSComboBox() indexOfSelectedItem]; | |
291 | } | |
421a8431 | 292 | |
a236aa20 VZ |
293 | int wxComboBox::DoInsertItems(const wxArrayStringsAdapter& items, |
294 | unsigned int pos, | |
295 | void **clientData, | |
296 | wxClientDataType type) | |
421a8431 | 297 | { |
8f248607 | 298 | wxAutoNSAutoreleasePool pool; |
a236aa20 VZ |
299 | const unsigned int numITems = items.GetCount(); |
300 | for ( unsigned int i = 0; i < numITems; ++i, ++pos ) | |
301 | { | |
302 | [GetNSComboBox() insertItemWithObjectValue:wxNSStringWithWxString(items[i]) atIndex:(pos)]; | |
303 | m_Datas.Insert(NULL, pos); | |
304 | AssignNewItemClientData(pos, clientData, i, type); | |
305 | } | |
306 | return pos - 1; | |
421a8431 DE |
307 | } |
308 | ||
aa61d352 | 309 | void wxComboBox::DoSetItemClientData(unsigned int nIndex, void* pData) |
421a8431 | 310 | { |
8f248607 | 311 | m_Datas[nIndex] = pData; |
421a8431 DE |
312 | } |
313 | ||
aa61d352 | 314 | void* wxComboBox::DoGetItemClientData(unsigned int nIndex) const |
421a8431 | 315 | { |
8f248607 | 316 | return m_Datas[nIndex]; |
421a8431 DE |
317 | } |
318 | ||
e960ba59 DE |
319 | ///////////////////////////////////////////////////////////////////////////// |
320 | // wxTextEntry virtual implementations: | |
321 | ||
322 | void wxComboBox::WriteText(wxString const&) | |
323 | { | |
324 | } | |
325 | ||
326 | wxString wxComboBox::GetValue() const | |
327 | { | |
328 | wxAutoNSAutoreleasePool pool; | |
329 | return wxStringWithNSString([GetNSTextField() stringValue]); | |
330 | } | |
331 | ||
332 | void wxComboBox::Remove(long, long) | |
333 | { | |
334 | } | |
335 | ||
336 | void wxComboBox::Cut() | |
337 | { | |
338 | } | |
339 | ||
340 | void wxComboBox::Copy() | |
341 | { | |
342 | } | |
343 | ||
344 | void wxComboBox::Paste() | |
345 | { | |
346 | } | |
347 | ||
348 | void wxComboBox::Undo() | |
349 | { | |
350 | } | |
351 | ||
352 | void wxComboBox::Redo() | |
353 | { | |
354 | } | |
355 | ||
356 | bool wxComboBox::CanUndo() const | |
357 | { | |
358 | return false; | |
359 | } | |
360 | ||
361 | bool wxComboBox::CanRedo() const | |
362 | { | |
363 | return false; | |
364 | } | |
365 | ||
366 | void wxComboBox::SetInsertionPoint(long) | |
367 | { | |
368 | } | |
369 | ||
370 | long wxComboBox::GetInsertionPoint() const | |
371 | { | |
372 | return 0; | |
373 | } | |
374 | ||
375 | wxTextPos wxComboBox::GetLastPosition() const | |
376 | { | |
377 | // working - returns the size of the wxString | |
378 | return (long)(GetValue().Len()); | |
379 | } | |
380 | ||
381 | void wxComboBox::SetSelection(long, long) | |
382 | { | |
383 | } | |
384 | ||
385 | void wxComboBox::GetSelection(long*, long*) const | |
386 | { | |
387 | } | |
388 | ||
389 | bool wxComboBox::IsEditable() const | |
390 | { | |
391 | return [GetNSTextField() isEditable]; | |
392 | } | |
393 | ||
394 | void wxComboBox::SetEditable(bool editable) | |
395 | { | |
396 | // first ensure that the current value is stored (in case the user had not finished editing | |
397 | // before SetEditable(FALSE) was called) | |
398 | DoSetValue(GetValue(),1); | |
399 | ||
400 | [GetNSTextField() setEditable: editable]; | |
401 | ||
402 | // forces the focus on the textctrl to be lost - while focus is still maintained | |
403 | // after SetEditable(FALSE) the user may still edit the control | |
404 | // (might not the best way to do this..) | |
405 | [GetNSTextField() abortEditing]; | |
406 | } | |
407 | ||
a236aa20 | 408 | #endif // wxUSE_COMBOBOX |