1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/listbox_osx.cpp
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
12 #include "wx/wxprec.h"
16 #include "wx/listbox.h"
22 #include "wx/settings.h"
23 #include "wx/arrstr.h"
24 #include "wx/dcclient.h"
27 BEGIN_EVENT_TABLE(wxListBox
, wxControl
)
30 #include "wx/osx/private.h"
32 // ============================================================================
33 // list box control implementation
34 // ============================================================================
36 wxListBox::wxListBox()
40 bool wxListBox::Create(
45 const wxArrayString
& choices
,
47 const wxValidator
& validator
,
48 const wxString
& name
)
50 wxCArrayString
chs(choices
);
53 parent
, id
, pos
, size
, chs
.GetCount(), chs
.GetStrings(),
54 style
, validator
, name
);
57 wxListWidgetImpl
* wxListBox::GetListPeer() const
59 wxListWidgetImpl
* impl
= dynamic_cast<wxListWidgetImpl
*> ( GetPeer() );
63 bool wxListBox::Create(
69 const wxString choices
[],
71 const wxValidator
& validator
,
72 const wxString
& name
)
75 m_blockEvents
= false;
77 if ( ! (style
& wxNO_BORDER
) )
78 style
= (style
& ~wxBORDER_MASK
) | wxSUNKEN_BORDER
;
80 wxASSERT_MSG( !(style
& wxLB_MULTIPLE
) || !(style
& wxLB_EXTENDED
),
81 wxT("only a single listbox selection mode can be specified") );
83 if ( !wxListBoxBase::Create( parent
, id
, pos
, size
, style
& ~(wxHSCROLL
| wxVSCROLL
), validator
, name
) )
87 m_strings
.sorted
= new wxSortedArrayString
;
89 m_strings
.unsorted
= new wxArrayString
;
91 SetPeer(wxWidgetImpl::CreateListBox( this, parent
, id
, pos
, size
, style
, GetExtraStyle() ));
93 MacPostControlCreate( pos
, size
);
95 m_textColumn
= GetListPeer()->InsertTextColumn(0,wxEmptyString
);
99 // Needed because it is a wxControlWithItems
100 SetInitialSize( size
);
105 wxListBox::~wxListBox()
107 m_blockEvents
= true;
109 m_blockEvents
= false;
111 // make sure no native events get sent to a object in destruction
115 delete m_strings
.sorted
;
117 delete m_strings
.unsorted
;
119 m_strings
.sorted
= NULL
;
122 void wxListBox::FreeData()
125 m_strings
.sorted
->Clear();
127 m_strings
.unsorted
->Clear();
129 m_itemsClientData
.Clear();
131 GetListPeer()->ListClear();
134 void wxListBox::DoSetFirstItem(int n
)
136 GetListPeer()->ListScrollTo( n
);
139 void wxListBox::EnsureVisible(int n
)
141 GetListPeer()->ListScrollTo( n
);
144 void wxListBox::DoDeleteOneItem(unsigned int n
)
146 wxCHECK_RET( IsValid(n
), wxT("invalid index in wxListBox::Delete") );
148 m_blockEvents
= true;
150 m_strings
.sorted
->RemoveAt(n
);
152 m_strings
.unsorted
->RemoveAt(n
);
154 m_itemsClientData
.RemoveAt(n
);
156 GetListPeer()->ListDelete( n
);
157 m_blockEvents
= false;
159 UpdateOldSelections();
162 void wxListBox::DoClear()
164 m_blockEvents
= true;
166 m_blockEvents
= false;
168 UpdateOldSelections();
171 // ----------------------------------------------------------------------------
173 // ----------------------------------------------------------------------------
175 void wxListBox::DoSetSelection(int n
, bool select
)
177 wxCHECK_RET( n
== wxNOT_FOUND
|| IsValid(n
),
178 wxT("invalid index in wxListBox::SetSelection") );
180 m_blockEvents
= true;
182 if ( n
== wxNOT_FOUND
)
183 GetListPeer()->ListDeselectAll();
185 GetListPeer()->ListSetSelection( n
, select
, HasMultipleSelection() );
187 m_blockEvents
= false;
189 UpdateOldSelections();
192 bool wxListBox::IsSelected(int n
) const
194 wxCHECK_MSG( IsValid(n
), false, wxT("invalid index in wxListBox::Selected") );
196 return GetListPeer()->ListIsSelected( n
);
199 // Return number of selections and an array of selected integers
200 int wxListBox::GetSelections(wxArrayInt
& aSelections
) const
202 return GetListPeer()->ListGetSelections( aSelections
);
205 // Get single selection, for single choice list items
206 int wxListBox::GetSelection() const
208 return GetListPeer()->ListGetSelection();
211 int wxListBox::DoListHitTest(const wxPoint
& inpoint
) const
213 return GetListPeer()->DoListHitTest( inpoint
);
216 // ----------------------------------------------------------------------------
218 // ----------------------------------------------------------------------------
220 void wxListBox::GetValueCallback( unsigned int n
, wxListWidgetColumn
* col
, wxListWidgetCellValue
& value
)
222 if ( col
== m_textColumn
)
223 value
.Set( GetString( n
) );
226 void wxListBox::SetValueCallback( unsigned int WXUNUSED(n
), wxListWidgetColumn
* WXUNUSED(col
) , wxListWidgetCellValue
& WXUNUSED(value
) )
230 wxSize
wxListBox::DoGetBestSize() const
232 int lbWidth
= 100; // some defaults
237 wxClientDC
dc(const_cast<wxListBox
*>(this));
238 dc
.SetFont(GetFont());
240 // Find the widest line
241 for (unsigned int i
= 0; i
< GetCount(); i
++)
243 wxString
str( GetString( i
) );
245 wxCoord width
, height
;
246 dc
.GetTextExtent( str
, &width
, &height
);
248 lbWidth
= wxMax( lbWidth
, wLine
);
251 // Add room for the scrollbar
252 lbWidth
+= wxSystemSettings::GetMetric( wxSYS_VSCROLL_X
);
254 // And just a bit more
257 wxCoord width
, height
;
258 dc
.GetTextExtent( wxT("XX") , &width
, &height
);
262 // don't make the listbox too tall (limit height to around 10 items)
263 // but don't make it too small neither
264 lbHeight
= wxMax( (cy
+ 4) * wxMin( wxMax( GetCount(), 3 ), 10 ), 70 );
267 return wxSize( lbWidth
, lbHeight
);
270 void wxListBox::Refresh(bool eraseBack
, const wxRect
*rect
)
272 wxControl::Refresh( eraseBack
, rect
);
275 // Some custom controls depend on this
276 /* static */ wxVisualAttributes
277 wxListBox::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
279 wxVisualAttributes attr
;
281 attr
.colFg
= wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOWTEXT
);
282 attr
.colBg
= wxSystemSettings::GetColour( wxSYS_COLOUR_LISTBOX
);
283 static wxFont font
= wxFont(wxOSX_SYSTEM_FONT_VIEWS
);
289 // below is all code copied from univ
291 // ----------------------------------------------------------------------------
292 // client data handling
293 // ----------------------------------------------------------------------------
295 void wxListBox::DoSetItemClientData(unsigned int n
, void* clientData
)
297 m_itemsClientData
[n
] = clientData
;
300 void *wxListBox::DoGetItemClientData(unsigned int n
) const
302 return m_itemsClientData
[n
];
305 // ----------------------------------------------------------------------------
307 // ----------------------------------------------------------------------------
309 unsigned int wxListBox::GetCount() const
311 return IsSorted() ? m_strings
.sorted
->size()
312 : m_strings
.unsorted
->size();
315 wxString
wxListBox::GetString(unsigned int n
) const
317 return IsSorted() ? m_strings
.sorted
->Item(n
)
318 : m_strings
.unsorted
->Item(n
);
321 int wxListBox::FindString(const wxString
& s
, bool bCase
) const
323 return IsSorted() ? m_strings
.sorted
->Index(s
, bCase
)
324 : m_strings
.unsorted
->Index(s
, bCase
);
327 // ----------------------------------------------------------------------------
328 // adding/inserting strings
329 // ----------------------------------------------------------------------------
331 void wxListBox::OnItemInserted(unsigned int WXUNUSED(pos
))
335 int wxListBox::DoInsertItems(const wxArrayStringsAdapter
& items
,
338 wxClientDataType type
)
340 int idx
= wxNOT_FOUND
;
341 unsigned int startpos
= pos
;
343 const unsigned int numItems
= items
.GetCount();
344 for ( unsigned int i
= 0; i
< numItems
; ++i
)
346 const wxString
& item
= items
[i
];
347 idx
= IsSorted() ? m_strings
.sorted
->Add(item
)
348 : (m_strings
.unsorted
->Insert(item
, pos
), pos
++);
350 m_itemsClientData
.Insert(NULL
, idx
);
351 AssignNewItemClientData(idx
, clientData
, i
, type
);
353 GetListPeer()->ListInsert(startpos
+i
);
358 GetListPeer()->UpdateLineToEnd(startpos
);
360 // Inserting the items may scroll the listbox down to show the last
361 // selected one but we don't want to do it as it could result in e.g. the
362 // first items of a listbox be hidden immediately after its creation so
363 // show the first selected item instead. Ideal would probably be to
364 // preserve the old selection unchanged, in fact, but I don't know how to
365 // get the first visible item so for now do at least this.
366 SetFirstItem(startpos
);
368 UpdateOldSelections();
373 void wxListBox::SetString(unsigned int n
, const wxString
& s
)
375 wxCHECK_RET( !IsSorted(), wxT("can't set string in sorted listbox") );
378 (*m_strings
.sorted
)[n
] = s
;
380 (*m_strings
.unsorted
)[n
] = s
;
382 GetListPeer()->UpdateLine(n
);
386 // common event handling
389 void wxListBox::HandleLineEvent( unsigned int n
, bool doubleClick
)
391 wxCommandEvent
event( doubleClick
? wxEVT_COMMAND_LISTBOX_DOUBLECLICKED
:
392 wxEVT_COMMAND_LISTBOX_SELECTED
, GetId() );
393 event
.SetEventObject( this );
394 if ( HasClientObjectData() )
395 event
.SetClientObject( GetClientObject(n
) );
396 else if ( HasClientUntypedData() )
397 event
.SetClientData( GetClientData(n
) );
398 event
.SetString( GetString(n
) );
400 event
.SetExtraLong( 1 );
401 HandleWindowEvent(event
);
405 // common list cell value operations
408 void wxListWidgetCellValue::Check( bool check
)
410 Set( check
? 1 : 0 );
413 bool wxListWidgetCellValue::IsChecked() const
415 return GetIntValue() != 0;
420 #endif // wxUSE_LISTBOX