1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
13 #pragma implementation "choice.h"
16 #include "wx/wxprec.h"
20 #include "wx/choice.h"
22 #include "wx/mac/uma.h"
24 #if !USE_SHARED_LIBRARY
25 IMPLEMENT_DYNAMIC_CLASS(wxChoice
, wxControl
)
28 extern MenuHandle
NewUniqueMenu() ;
32 if ( HasClientObjectData() )
34 size_t i
, max
= GetCount();
36 for ( i
= 0; i
< max
; ++i
)
37 delete GetClientObject(i
);
40 // DeleteMenu( m_macPopUpMenuId ) ;
41 // DisposeMenu( m_macPopUpMenuHandle ) ;
44 bool wxChoice::Create(wxWindow
*parent
, wxWindowID id
,
47 const wxArrayString
& choices
,
49 const wxValidator
& validator
,
52 wxCArrayString
chs(choices
);
54 return Create(parent
, id
, pos
, size
, chs
.GetCount(), chs
.GetStrings(),
55 style
, validator
, name
);
58 bool wxChoice::Create(wxWindow
*parent
, wxWindowID id
,
61 int n
, const wxString choices
[],
63 const wxValidator
& validator
,
66 m_macIsUserPane
= false ;
68 if ( !wxChoiceBase::Create(parent
, id
, pos
, size
, style
, validator
, name
) )
71 Rect bounds
= wxMacGetBoundsForControl( this , pos
, size
) ;
73 m_peer
= new wxMacControl(this) ;
74 verify_noerr ( CreatePopupButtonControl( MAC_WXHWND(parent
->MacGetTopLevelWindowRef()) , &bounds
, CFSTR("") ,
75 -12345 , false /* no variable width */ , 0 , 0 , 0 , m_peer
->GetControlRefAddr() ) );
78 m_macPopUpMenuHandle
= NewUniqueMenu() ;
79 m_peer
->SetData
<MenuHandle
>( kControlNoPart
, kControlPopupButtonMenuHandleTag
, (MenuHandle
) m_macPopUpMenuHandle
) ;
80 m_peer
->SetValueAndRange( n
> 0 ? 1 : 0 , 0 , 0 ) ;
81 MacPostControlCreate(pos
,size
) ;
84 if ( style
& wxCB_SORT
)
86 m_strings
= wxArrayString(1) ; // autosort
90 for ( int i
= 0; i
< n
; i
++ )
94 SetBestSize(size
); // Needed because it is a wxControlWithItems
98 // ----------------------------------------------------------------------------
99 // adding/deleting items to/from the list
100 // ----------------------------------------------------------------------------
101 int wxChoice::DoAppend(const wxString
& item
)
104 wxArrayString::iterator insertPoint
;
107 if (GetWindowStyle() & wxCB_SORT
)
109 insertPoint
= std::lower_bound( m_strings
.begin(), m_strings
.end(), item
);
110 index
= insertPoint
- m_strings
.begin();
114 insertPoint
= m_strings
.end();
115 index
= m_strings
.size();
118 m_strings
.insert( insertPoint
, item
);
120 size_t index
= m_strings
.Add( item
) ;
122 m_datas
.Insert( NULL
, index
) ;
123 UMAInsertMenuItem(MAC_WXHMENU( m_macPopUpMenuHandle
) , item
, m_font
.GetEncoding() , index
);
124 DoSetItemClientData( index
, NULL
) ;
125 m_peer
->SetMaximum( GetCount() ) ;
129 int wxChoice::DoInsert(const wxString
& item
, int pos
)
131 wxCHECK_MSG(!(GetWindowStyle() & wxCB_SORT
), -1, wxT("can't insert into sorted list"));
132 wxCHECK_MSG((pos
>=0) && (pos
<=GetCount()), -1, wxT("invalid index"));
134 if (pos
== GetCount())
135 return DoAppend(item
);
137 UMAInsertMenuItem(MAC_WXHMENU( m_macPopUpMenuHandle
) , item
, m_font
.GetEncoding() , pos
);
138 m_strings
.Insert( item
, pos
) ;
139 m_datas
.Insert( NULL
, pos
) ;
140 DoSetItemClientData( pos
, NULL
) ;
141 m_peer
->SetMaximum( GetCount() ) ;
145 void wxChoice::Delete(int n
)
147 wxCHECK_RET( n
< GetCount(), wxT("invalid item index in wxChoice::Delete") );
148 if ( HasClientObjectData() )
150 delete GetClientObject(n
);
152 ::DeleteMenuItem( MAC_WXHMENU(m_macPopUpMenuHandle
) , n
+ 1) ;
153 m_strings
.RemoveAt( n
) ;
154 m_datas
.RemoveAt( n
) ;
155 m_peer
->SetMaximum( GetCount() ) ;
158 void wxChoice::Clear()
161 for ( int i
= 0 ; i
< GetCount() ; i
++ )
163 ::DeleteMenuItem( MAC_WXHMENU(m_macPopUpMenuHandle
) , 1 ) ;
167 m_peer
->SetMaximum( 0 ) ;
170 void wxChoice::FreeData()
172 if ( HasClientObjectData() )
174 size_t count
= GetCount();
175 for ( size_t n
= 0; n
< count
; n
++ )
177 delete GetClientObject(n
);
182 // ----------------------------------------------------------------------------
184 // ----------------------------------------------------------------------------
185 int wxChoice::GetSelection() const
187 return m_peer
->GetValue() -1 ;
190 void wxChoice::SetSelection(int n
)
192 m_peer
->SetValue( n
+ 1 ) ;
195 // ----------------------------------------------------------------------------
196 // string list functions
197 // ----------------------------------------------------------------------------
199 int wxChoice::GetCount() const
201 return m_strings
.GetCount() ;
204 int wxChoice::FindString(const wxString
& s
) const
206 return m_strings
.Index( s
, true , false) ;
209 void wxChoice::SetString(int n
, const wxString
& s
)
212 // apple menu pos is 1-based
213 UMASetMenuItemText( MAC_WXHMENU(m_macPopUpMenuHandle
) , n
+ 1 , s
, wxFont::GetDefaultEncoding() ) ;
216 wxString
wxChoice::GetString(int n
) const
218 wxCHECK_MSG( n
>= 0 && (size_t)n
< m_strings
.GetCount(), _T(""),
219 _T("wxChoice::GetString(): invalid index") );
221 return m_strings
[n
] ;
224 // ----------------------------------------------------------------------------
226 // ----------------------------------------------------------------------------
227 void wxChoice::DoSetItemClientData( int n
, void* clientData
)
229 wxCHECK_RET( n
>= 0 && (size_t)n
< m_datas
.GetCount(),
230 wxT("invalid index in wxChoice::SetClientData") );
232 m_datas
[n
] = (char*) clientData
;
235 void *wxChoice::DoGetItemClientData(int n
) const
237 wxCHECK_MSG( n
>= 0 && (size_t)n
< m_datas
.GetCount(), NULL
,
238 wxT("invalid index in wxChoice::GetClientData") );
239 return (void *)m_datas
[n
];
242 void wxChoice::DoSetItemClientObject( int n
, wxClientData
* clientData
)
244 DoSetItemClientData(n
, clientData
);
247 wxClientData
* wxChoice::DoGetItemClientObject( int n
) const
249 return (wxClientData
*)DoGetItemClientData(n
);
252 wxInt32
wxChoice::MacControlHit(WXEVENTHANDLERREF
WXUNUSED(handler
) , WXEVENTREF
WXUNUSED(event
) )
254 wxCommandEvent
event(wxEVT_COMMAND_CHOICE_SELECTED
, m_windowId
);
255 int n
= GetSelection();
256 // actually n should be made sure by the os to be a valid selection, but ...
260 event
.SetString(GetStringSelection());
261 event
.SetEventObject(this);
262 if ( HasClientObjectData() )
263 event
.SetClientObject( GetClientObject(n
) );
264 else if ( HasClientUntypedData() )
265 event
.SetClientData( GetClientData(n
) );
266 ProcessCommand(event
);
271 wxSize
wxChoice::DoGetBestSize() const
273 int lbWidth
= GetCount() > 0 ? 20 : 100; // some defaults
278 GetThemeMetric(kThemeMetricPopupButtonHeight
, &metric
);
282 wxMacPortStateHelper
st( UMAGetWindowPort( (WindowRef
) MacGetTopLevelWindowRef() ) ) ;
285 ::TextFont( m_font
.MacGetFontNum() ) ;
286 ::TextSize( m_font
.MacGetFontSize() ) ;
287 ::TextFace( m_font
.MacGetFontStyle() ) ;
291 ::TextFont( kFontIDMonaco
) ;
295 // Find the widest line
296 for(int i
= 0; i
< GetCount(); i
++) {
297 wxString
str(GetString(i
));
301 ::GetThemeTextDimensions( wxMacCFStringHolder( str
, m_font
.GetEncoding() ) ,
302 kThemeCurrentPortFont
,
309 wLine
= ::TextWidth( str
.c_str() , 0 , str
.Length() ) ;
311 lbWidth
= wxMax(lbWidth
, wLine
);
313 // Add room for the popup arrow
314 lbWidth
+= 2 * lbHeight
;
315 // And just a bit more
316 int cx
= ::TextWidth( "X" , 0 , 1 ) ;
320 return wxSize(lbWidth
, lbHeight
);
323 #endif // wxUSE_CHOICE