1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/cocoa/combobox.mm
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 #include "wx/wxprec.h"
16 #include "wx/combobox.h"
17 #include "wx/evtloop.h"
21 #include "wx/dcclient.h"
24 #include "wx/osx/cocoa/private/textimpl.h"
28 @interface wxNSTableDataSource : NSObject wxOSX_10_6_AND_LATER(<NSComboBoxDataSource>)
30 wxNSComboBoxControl* impl;
33 - (NSInteger)numberOfItemsInComboBox:(NSComboBox *)aComboBox;
34 - (id)comboBox:(NSComboBox *)aComboBox objectValueForItemAtIndex:(NSInteger)index;
39 @implementation wxNSComboBox
43 static BOOL initialized = NO;
47 wxOSXCocoaClassAddWXMethods( self );
53 [fieldEditor release];
57 // Over-riding NSComboBox onKeyDown method doesn't work for key events.
58 // Ensure that we can use our own wxNSTextFieldEditor to catch key events.
59 // See windowWillReturnFieldEditor in nonownedwnd.mm.
60 // Key events will be caught and handled via wxNSTextFieldEditor onkey...
61 // methods in textctrl.mm.
63 - (void) setFieldEditor:(wxNSTextFieldEditor*) editor
65 if ( editor != fieldEditor )
68 [fieldEditor release];
73 - (wxNSTextFieldEditor*) fieldEditor
78 - (void)controlTextDidChange:(NSNotification *)aNotification
80 wxUnusedVar(aNotification);
81 wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self );
82 if ( impl && impl->ShouldSendEvents() )
84 wxWindow* wxpeer = (wxWindow*) impl->GetWXPeer();
86 wxCommandEvent event(wxEVT_TEXT, wxpeer->GetId());
87 event.SetEventObject( wxpeer );
88 event.SetString( static_cast<wxComboBox*>(wxpeer)->GetValue() );
89 wxpeer->HandleWindowEvent( event );
94 - (void)comboBoxSelectionDidChange:(NSNotification *)notification
96 wxUnusedVar(notification);
97 wxWidgetCocoaImpl* impl = (wxWidgetCocoaImpl* ) wxWidgetImpl::FindFromWXWidget( self );
98 if ( impl && impl->ShouldSendEvents())
100 wxComboBox* wxpeer = static_cast<wxComboBox*>(impl->GetWXPeer());
102 const int sel = wxpeer->GetSelection();
104 wxCommandEvent event(wxEVT_COMBOBOX, wxpeer->GetId());
105 event.SetEventObject( wxpeer );
107 event.SetString( wxpeer->GetString(sel) );
108 // For some reason, wxComboBox::GetValue will not return the newly selected item
109 // while we're inside this callback, so use AddPendingEvent to make sure
110 // GetValue() returns the right value.
112 wxpeer->GetEventHandler()->AddPendingEvent( event );
119 wxNSComboBoxControl::wxNSComboBoxControl( wxComboBox *wxPeer, WXWidget w )
120 : wxNSTextFieldControl(wxPeer, wxPeer, w)
122 m_comboBox = (NSComboBox*)w;
125 wxNSComboBoxControl::~wxNSComboBoxControl()
129 void wxNSComboBoxControl::mouseEvent(WX_NSEvent event, WXWidget slf, void *_cmd)
131 // NSComboBox has its own event loop, which reacts very badly to our synthetic
132 // events used to signal when a wxEvent is posted, so during that time we switch
133 // the wxEventLoop::WakeUp implementation to a lower-level version
136 wxEventLoop* const loop = (wxEventLoop*) wxEventLoopBase::GetActive();
138 if ( loop != NULL && [event type] == NSLeftMouseDown )
141 loop->OSXUseLowLevelWakeup(true);
144 wxOSX_EventHandlerPtr superimpl = (wxOSX_EventHandlerPtr) [[slf superclass] instanceMethodForSelector:(SEL)_cmd];
145 superimpl(slf, (SEL)_cmd, event);
149 loop->OSXUseLowLevelWakeup(false);
153 int wxNSComboBoxControl::GetSelectedItem() const
155 return [m_comboBox indexOfSelectedItem];
158 void wxNSComboBoxControl::SetSelectedItem(int item)
162 if ( item != wxNOT_FOUND )
164 wxASSERT_MSG( item >= 0 && item < [m_comboBox numberOfItems],
165 "Inavlid item index." );
166 [m_comboBox selectItemAtIndex: item];
168 else // remove current selection (if we have any)
170 const int sel = GetSelectedItem();
171 if ( sel != wxNOT_FOUND )
172 [m_comboBox deselectItemAtIndex:sel];
178 int wxNSComboBoxControl::GetNumberOfItems() const
180 return [m_comboBox numberOfItems];
183 void wxNSComboBoxControl::InsertItem(int pos, const wxString& item)
185 [m_comboBox insertItemWithObjectValue:wxCFStringRef( item , m_wxPeer->GetFont().GetEncoding() ).AsNSString() atIndex:pos];
188 void wxNSComboBoxControl::RemoveItem(int pos)
191 [m_comboBox removeItemAtIndex:pos];
195 void wxNSComboBoxControl::Clear()
198 [m_comboBox removeAllItems];
199 [m_comboBox setStringValue:@""];
203 wxString wxNSComboBoxControl::GetStringAtIndex(int pos) const
205 return wxCFStringRef::AsString([m_comboBox itemObjectValueAtIndex:pos], m_wxPeer->GetFont().GetEncoding());
208 int wxNSComboBoxControl::FindString(const wxString& text) const
210 NSInteger nsresult = [m_comboBox indexOfItemWithObjectValue:wxCFStringRef( text , m_wxPeer->GetFont().GetEncoding() ).AsNSString()];
213 if (nsresult == NSNotFound)
214 result = wxNOT_FOUND;
216 result = (int) nsresult;
220 void wxNSComboBoxControl::Popup()
222 id ax = NSAccessibilityUnignoredDescendant(m_comboBox);
223 [ax accessibilitySetValue: [NSNumber numberWithBool: YES] forAttribute: NSAccessibilityExpandedAttribute];
226 void wxNSComboBoxControl::Dismiss()
228 id ax = NSAccessibilityUnignoredDescendant(m_comboBox);
229 [ax accessibilitySetValue: [NSNumber numberWithBool: NO] forAttribute: NSAccessibilityExpandedAttribute];
232 void wxNSComboBoxControl::SetEditable(bool editable)
234 // TODO: unfortunately this does not work, setEditable just means the same as CB_READONLY
235 // I don't see a way to access the text field directly
237 // Behavior NONE <- SELECTECTABLE
238 [m_comboBox setEditable:editable];
241 wxWidgetImplType* wxWidgetImpl::CreateComboBox( wxComboBox* wxpeer,
242 wxWindowMac* WXUNUSED(parent),
243 wxWindowID WXUNUSED(id),
244 wxMenu* WXUNUSED(menu),
248 long WXUNUSED(extraStyle))
250 NSRect r = wxOSXGetFrameForControl( wxpeer, pos , size ) ;
251 wxNSComboBox* v = [[wxNSComboBox alloc] initWithFrame:r];
252 if (style & wxCB_READONLY)
254 wxNSComboBoxControl* c = new wxNSComboBoxControl( wxpeer, v );
258 wxSize wxComboBox::DoGetBestSize() const
260 int lbWidth = GetCount() > 0 ? 20 : 100; // some defaults
261 wxSize baseSize = wxWindow::DoGetBestSize();
262 int lbHeight = baseSize.y;
266 wxClientDC dc(const_cast<wxComboBox*>(this));
268 // Find the widest line
269 for(unsigned int i = 0; i < GetCount(); i++)
271 wxString str(GetString(i));
273 wxCoord width, height ;
274 dc.GetTextExtent( str , &width, &height);
277 lbWidth = wxMax( lbWidth, wLine ) ;
280 // Add room for the popup arrow
281 lbWidth += 2 * lbHeight ;
284 return wxSize( lbWidth, lbHeight );
287 #endif // wxUSE_COMBOBOX