Commit | Line | Data |
---|---|---|
6c8a980f VZ |
1 | ///////////////////////////////////////////////////////////////////////////// |
2 | // Name: wx/ctrlsub.h (read: "wxConTRoL with SUBitems") | |
3 | // Purpose: wxControlWithItems interface | |
4 | // Author: Vadim Zeitlin | |
5 | // Modified by: | |
6 | // Created: 22.10.99 | |
7 | // RCS-ID: $Id$ | |
77ffb593 | 8 | // Copyright: (c) wxWidgets team |
65571936 | 9 | // Licence: wxWindows licence |
6c8a980f VZ |
10 | ///////////////////////////////////////////////////////////////////////////// |
11 | ||
12 | #ifndef _WX_CTRLSUB_H_BASE_ | |
13 | #define _WX_CTRLSUB_H_BASE_ | |
14 | ||
2ecf902b WS |
15 | #include "wx/defs.h" |
16 | ||
1e6feb95 VZ |
17 | #if wxUSE_CONTROLS |
18 | ||
a236aa20 | 19 | #include "wx/arrstr.h" |
6c8a980f VZ |
20 | #include "wx/control.h" // base class |
21 | ||
22 | // ---------------------------------------------------------------------------- | |
1e6feb95 | 23 | // wxItemContainer defines an interface which is implemented by all controls |
6c8a980f VZ |
24 | // which have string subitems each of which may be selected. |
25 | // | |
8ba7c771 VZ |
26 | // It is decomposed in wxItemContainerImmutable which omits all methods |
27 | // adding/removing items and is used by wxRadioBox and wxItemContainer itself. | |
28 | // | |
1e6feb95 VZ |
29 | // Examples: wxListBox, wxCheckListBox, wxChoice and wxComboBox (which |
30 | // implements an extended interface deriving from this one) | |
6c8a980f VZ |
31 | // ---------------------------------------------------------------------------- |
32 | ||
53a2db12 | 33 | class WXDLLIMPEXP_CORE wxItemContainerImmutable |
8ba7c771 VZ |
34 | { |
35 | public: | |
36 | wxItemContainerImmutable() { } | |
37 | virtual ~wxItemContainerImmutable(); | |
38 | ||
39 | // accessing strings | |
40 | // ----------------- | |
41 | ||
aa61d352 | 42 | virtual unsigned int GetCount() const = 0; |
8ba7c771 VZ |
43 | bool IsEmpty() const { return GetCount() == 0; } |
44 | ||
aa61d352 | 45 | virtual wxString GetString(unsigned int n) const = 0; |
8ba7c771 | 46 | wxArrayString GetStrings() const; |
aa61d352 | 47 | virtual void SetString(unsigned int n, const wxString& s) = 0; |
853dcc57 WS |
48 | |
49 | // finding string natively is either case sensitive or insensitive | |
50 | // but never both so fall back to this base version for not | |
51 | // supported search type | |
52 | virtual int FindString(const wxString& s, bool bCase = false) const | |
53 | { | |
aa61d352 | 54 | unsigned int count = GetCount(); |
853dcc57 | 55 | |
aa61d352 | 56 | for ( unsigned int i = 0; i < count ; ++i ) |
853dcc57 WS |
57 | { |
58 | if (GetString(i).IsSameAs( s , bCase )) | |
aa61d352 | 59 | return (int)i; |
853dcc57 WS |
60 | } |
61 | ||
62 | return wxNOT_FOUND; | |
63 | } | |
8ba7c771 VZ |
64 | |
65 | ||
66 | // selection | |
67 | // --------- | |
68 | ||
69 | virtual void SetSelection(int n) = 0; | |
70 | virtual int GetSelection() const = 0; | |
71 | ||
72 | // set selection to the specified string, return false if not found | |
73 | bool SetStringSelection(const wxString& s); | |
74 | ||
75 | // return the selected string or empty string if none | |
0ec1179b | 76 | virtual wxString GetStringSelection() const; |
8ba7c771 VZ |
77 | |
78 | // this is the same as SetSelection( for single-selection controls but | |
79 | // reads better for multi-selection ones | |
80 | void Select(int n) { SetSelection(n); } | |
81 | ||
789f6795 WS |
82 | |
83 | protected: | |
789f6795 | 84 | // check that the index is valid |
a236aa20 VZ |
85 | bool IsValid(unsigned int n) const { return n < GetCount(); } |
86 | bool IsValidInsert(unsigned int n) const { return n <= GetCount(); } | |
8ba7c771 VZ |
87 | }; |
88 | ||
a236aa20 VZ |
89 | // ---------------------------------------------------------------------------- |
90 | // wxItemContainer extends wxItemContainerImmutable interface with methods | |
91 | // for adding/removing items. | |
92 | // | |
93 | // Classes deriving from this one must override DoInsertItems() to implement | |
94 | // adding items to the control. This can often be implemented more efficiently | |
95 | // than simply looping over the elements and inserting them but if this is not | |
96 | // the case, the generic DoInsertItemsInLoop can be used in implementation, but | |
97 | // in this case DoInsertItem() needs to be overridden. | |
98 | // ---------------------------------------------------------------------------- | |
99 | ||
53a2db12 | 100 | class WXDLLIMPEXP_CORE wxItemContainer : public wxItemContainerImmutable |
6c8a980f | 101 | { |
a236aa20 VZ |
102 | private: |
103 | // AppendItems() and InsertItems() helpers just call DoAppend/InsertItems() | |
104 | // after doing some checks | |
105 | // | |
106 | // NB: they're defined here so that they're inlined when used in public part | |
107 | int AppendItems(const wxArrayStringsAdapter& items, | |
108 | void **clientData, | |
109 | wxClientDataType type) | |
110 | { | |
111 | if ( items.IsEmpty() ) | |
112 | return wxNOT_FOUND; | |
113 | ||
114 | return DoAppendItems(items, clientData, type); | |
115 | } | |
116 | ||
117 | int AppendItems(const wxArrayStringsAdapter& items) | |
118 | { | |
119 | return AppendItems(items, NULL, wxClientData_None); | |
120 | } | |
121 | ||
122 | int AppendItems(const wxArrayStringsAdapter& items, void **clientData) | |
123 | { | |
131b1fba | 124 | wxASSERT_MSG( GetClientDataType() != wxClientData_Object, |
9a83f860 | 125 | wxT("can't mix different types of client data") ); |
a236aa20 VZ |
126 | |
127 | return AppendItems(items, clientData, wxClientData_Void); | |
128 | } | |
129 | ||
130 | int AppendItems(const wxArrayStringsAdapter& items, | |
131 | wxClientData **clientData) | |
132 | { | |
131b1fba | 133 | wxASSERT_MSG( GetClientDataType() != wxClientData_Void, |
9a83f860 | 134 | wxT("can't mix different types of client data") ); |
a236aa20 | 135 | |
5c33522f | 136 | return AppendItems(items, reinterpret_cast<void **>(clientData), |
a236aa20 VZ |
137 | wxClientData_Object); |
138 | } | |
139 | ||
140 | int InsertItems(const wxArrayStringsAdapter& items, | |
141 | unsigned int pos, | |
142 | void **clientData, | |
143 | wxClientDataType type) | |
144 | { | |
9a83f860 | 145 | wxASSERT_MSG( !IsSorted(), wxT("can't insert items in sorted control") ); |
a236aa20 VZ |
146 | |
147 | wxCHECK_MSG( pos <= GetCount(), wxNOT_FOUND, | |
9a83f860 | 148 | wxT("position out of range") ); |
a236aa20 VZ |
149 | |
150 | // not all derived classes handle empty arrays correctly in | |
151 | // DoInsertItems() and besides it really doesn't make much sense to do | |
152 | // this (for append it could correspond to creating an initially empty | |
153 | // control but why would anybody need to insert 0 items?) | |
154 | wxCHECK_MSG( !items.IsEmpty(), wxNOT_FOUND, | |
9a83f860 | 155 | wxT("need something to insert") ); |
a236aa20 VZ |
156 | |
157 | return DoInsertItems(items, pos, clientData, type); | |
158 | } | |
159 | ||
160 | int InsertItems(const wxArrayStringsAdapter& items, unsigned int pos) | |
161 | { | |
162 | return InsertItems(items, pos, NULL, wxClientData_None); | |
163 | } | |
164 | ||
165 | int InsertItems(const wxArrayStringsAdapter& items, | |
166 | unsigned int pos, | |
167 | void **clientData) | |
168 | { | |
131b1fba | 169 | wxASSERT_MSG( GetClientDataType() != wxClientData_Object, |
9a83f860 | 170 | wxT("can't mix different types of client data") ); |
a236aa20 VZ |
171 | |
172 | return InsertItems(items, pos, clientData, wxClientData_Void); | |
173 | } | |
174 | ||
175 | int InsertItems(const wxArrayStringsAdapter& items, | |
176 | unsigned int pos, | |
177 | wxClientData **clientData) | |
178 | { | |
131b1fba | 179 | wxASSERT_MSG( GetClientDataType() != wxClientData_Void, |
9a83f860 | 180 | wxT("can't mix different types of client data") ); |
a236aa20 VZ |
181 | |
182 | return InsertItems(items, pos, | |
5c33522f | 183 | reinterpret_cast<void **>(clientData), |
a236aa20 VZ |
184 | wxClientData_Object); |
185 | } | |
186 | ||
6c8a980f | 187 | public: |
6463b9f5 | 188 | wxItemContainer() { m_clientDataItemsType = wxClientData_None; } |
799ea011 | 189 | virtual ~wxItemContainer(); |
6c8a980f VZ |
190 | |
191 | // adding items | |
192 | // ------------ | |
193 | ||
a236aa20 VZ |
194 | // append single item, return its position in the control (which can be |
195 | // different from the last one if the control is sorted) | |
1e6feb95 | 196 | int Append(const wxString& item) |
a236aa20 | 197 | { return AppendItems(item); } |
1e6feb95 | 198 | int Append(const wxString& item, void *clientData) |
a236aa20 | 199 | { return AppendItems(item, &clientData); } |
1e6feb95 | 200 | int Append(const wxString& item, wxClientData *clientData) |
a236aa20 VZ |
201 | { return AppendItems(item, &clientData); } |
202 | ||
203 | // append several items at once to the control, return the position of the | |
204 | // last item appended | |
205 | int Append(const wxArrayString& items) | |
206 | { return AppendItems(items); } | |
207 | int Append(const wxArrayString& items, void **clientData) | |
208 | { return AppendItems(items, clientData); } | |
209 | int Append(const wxArrayString& items, wxClientData **clientData) | |
210 | { return AppendItems(items, clientData); } | |
211 | int Append(unsigned int n, const wxString *items) | |
212 | { return AppendItems(wxArrayStringsAdapter(n, items)); } | |
213 | int Append(unsigned int n, const wxString *items, void **clientData) | |
214 | { return AppendItems(wxArrayStringsAdapter(n, items), clientData); } | |
215 | int Append(unsigned int n, | |
216 | const wxString *items, | |
217 | wxClientData **clientData) | |
218 | { return AppendItems(wxArrayStringsAdapter(n, items), clientData); } | |
219 | ||
220 | // only for RTTI needs (separate name) | |
221 | void AppendString(const wxString& item) | |
222 | { Append(item); } | |
223 | ||
224 | ||
225 | // inserting items: not for sorted controls! | |
226 | // ----------------------------------------- | |
227 | ||
228 | // insert single item at the given position, return its effective position | |
aa61d352 | 229 | int Insert(const wxString& item, unsigned int pos) |
a236aa20 VZ |
230 | { return InsertItems(item, pos); } |
231 | int Insert(const wxString& item, unsigned int pos, void *clientData) | |
232 | { return InsertItems(item, pos, &clientData); } | |
233 | int Insert(const wxString& item, unsigned int pos, wxClientData *clientData) | |
234 | { return InsertItems(item, pos, &clientData); } | |
235 | ||
236 | // insert several items at once into the control, return the index of the | |
237 | // last item inserted | |
238 | int Insert(const wxArrayString& items, unsigned int pos) | |
239 | { return InsertItems(items, pos); } | |
240 | int Insert(const wxArrayString& items, unsigned int pos, void **clientData) | |
241 | { return InsertItems(items, pos, clientData); } | |
242 | int Insert(const wxArrayString& items, | |
243 | unsigned int pos, | |
244 | wxClientData **clientData) | |
245 | { return InsertItems(items, pos, clientData); } | |
246 | int Insert(unsigned int n, const wxString *items, unsigned int pos) | |
247 | { return InsertItems(wxArrayStringsAdapter(n, items), pos); } | |
248 | int Insert(unsigned int n, | |
249 | const wxString *items, | |
250 | unsigned int pos, | |
251 | void **clientData) | |
252 | { return InsertItems(wxArrayStringsAdapter(n, items), pos, clientData); } | |
253 | int Insert(unsigned int n, | |
254 | const wxString *items, | |
255 | unsigned int pos, | |
256 | wxClientData **clientData) | |
257 | { return InsertItems(wxArrayStringsAdapter(n, items), pos, clientData); } | |
258 | ||
259 | ||
260 | // replacing items | |
261 | // --------------- | |
262 | ||
263 | void Set(const wxArrayString& items) | |
264 | { Clear(); Append(items); } | |
265 | void Set(const wxArrayString& items, void **clientData) | |
266 | { Clear(); Append(items, clientData); } | |
267 | void Set(const wxArrayString& items, wxClientData **clientData) | |
268 | { Clear(); Append(items, clientData); } | |
269 | void Set(unsigned int n, const wxString *items) | |
270 | { Clear(); Append(n, items); } | |
271 | void Set(unsigned int n, const wxString *items, void **clientData) | |
272 | { Clear(); Append(n, items, clientData); } | |
273 | void Set(unsigned int n, const wxString *items, wxClientData **clientData) | |
274 | { Clear(); Append(n, items, clientData); } | |
243dbf1a | 275 | |
6c8a980f VZ |
276 | // deleting items |
277 | // -------------- | |
278 | ||
a236aa20 VZ |
279 | void Clear(); |
280 | void Delete(unsigned int pos); | |
6c8a980f | 281 | |
6c8a980f | 282 | |
131b1fba VZ |
283 | // various accessors |
284 | // ----------------- | |
285 | ||
286 | // The control may maintain its items in a sorted order in which case | |
287 | // items are automatically inserted at the right position when they are | |
288 | // inserted or appended. Derived classes have to override this method if | |
289 | // they implement sorting, typically by returning HasFlag(wxXX_SORT) | |
290 | virtual bool IsSorted() const { return false; } | |
291 | ||
292 | ||
6c8a980f | 293 | // client data stuff |
a236aa20 VZ |
294 | // ----------------- |
295 | ||
aa61d352 VZ |
296 | void SetClientData(unsigned int n, void* clientData); |
297 | void* GetClientData(unsigned int n) const; | |
6c8a980f | 298 | |
ab989357 VZ |
299 | // SetClientObject() takes ownership of the pointer, GetClientObject() |
300 | // returns it but keeps the ownership while DetachClientObject() expects | |
301 | // the caller to delete the pointer and also resets the internally stored | |
302 | // one to NULL for this item | |
aa61d352 VZ |
303 | void SetClientObject(unsigned int n, wxClientData* clientData); |
304 | wxClientData* GetClientObject(unsigned int n) const; | |
ab989357 | 305 | wxClientData* DetachClientObject(unsigned int n); |
6c8a980f | 306 | |
131b1fba VZ |
307 | // return the type of client data stored in this control: usually it just |
308 | // returns m_clientDataItemsType but must be overridden in the controls | |
309 | // which delegate their client data storage to another one (e.g. wxChoice | |
310 | // in wxUniv which stores data in wxListBox which it uses anyhow); don't | |
311 | // forget to override SetClientDataType() if you override this one | |
312 | // | |
313 | // NB: for this to work no code should ever access m_clientDataItemsType | |
314 | // directly but only via this function! | |
315 | virtual wxClientDataType GetClientDataType() const | |
316 | { return m_clientDataItemsType; } | |
317 | ||
a236aa20 | 318 | bool HasClientData() const |
131b1fba | 319 | { return GetClientDataType() != wxClientData_None; } |
6c8a980f | 320 | bool HasClientObjectData() const |
131b1fba | 321 | { return GetClientDataType() == wxClientData_Object; } |
6c8a980f | 322 | bool HasClientUntypedData() const |
131b1fba | 323 | { return GetClientDataType() == wxClientData_Void; } |
a236aa20 | 324 | |
6c8a980f | 325 | protected: |
a236aa20 VZ |
326 | // there is usually no need to override this method but you can do it if it |
327 | // is more convenient to only do "real" insertions in DoInsertItems() and | |
328 | // to implement items appending here (in which case DoInsertItems() should | |
329 | // call this method if pos == GetCount() as it can still be called in this | |
330 | // case if public Insert() is called with such position) | |
331 | virtual int DoAppendItems(const wxArrayStringsAdapter& items, | |
332 | void **clientData, | |
333 | wxClientDataType type) | |
334 | { | |
335 | return DoInsertItems(items, GetCount(), clientData, type); | |
336 | } | |
6c8a980f | 337 | |
a236aa20 VZ |
338 | // this method must be implemented to insert the items into the control at |
339 | // position pos which can be GetCount() meaning that the items should be | |
340 | // appended; for the sorted controls the position can be ignored | |
341 | // | |
342 | // the derived classes typically use AssignNewItemClientData() to | |
343 | // associate the data with the items as they're being inserted | |
344 | // | |
345 | // the method should return the index of the position the last item was | |
346 | // inserted into or wxNOT_FOUND if an error occurred | |
347 | virtual int DoInsertItems(const wxArrayStringsAdapter & items, | |
348 | unsigned int pos, | |
349 | void **clientData, | |
350 | wxClientDataType type) = 0; | |
351 | ||
352 | // before the client data is set for the first time for the control which | |
353 | // hadn't had it before, DoInitItemClientData() is called which gives the | |
354 | // derived class the possibility to initialize its client data storage only | |
355 | // when client data is really used | |
356 | virtual void DoInitItemClientData() { } | |
357 | virtual void DoSetItemClientData(unsigned int n, void *clientData) = 0; | |
358 | virtual void *DoGetItemClientData(unsigned int n) const = 0; | |
359 | ||
360 | virtual void DoClear() = 0; | |
361 | virtual void DoDeleteOneItem(unsigned int pos) = 0; | |
362 | ||
363 | ||
364 | // methods useful for the derived classes which don't have any better way | |
365 | // of adding multiple items to the control than doing it one by one: such | |
366 | // classes should call DoInsertItemsInLoop() from their DoInsert() and | |
367 | // override DoInsertOneItem() to perform the real insertion | |
368 | virtual int DoInsertOneItem(const wxString& item, unsigned int pos); | |
369 | int DoInsertItemsInLoop(const wxArrayStringsAdapter& items, | |
370 | unsigned int pos, | |
371 | void **clientData, | |
372 | wxClientDataType type); | |
373 | ||
374 | ||
375 | // helper for DoInsertItems(): n is the index into clientData, pos is the | |
376 | // position of the item in the control | |
377 | void AssignNewItemClientData(unsigned int pos, | |
378 | void **clientData, | |
379 | unsigned int n, | |
380 | wxClientDataType type); | |
381 | ||
382 | // free the client object associated with the item at given position and | |
383 | // set it to NULL (must only be called if HasClientObjectData()) | |
384 | void ResetItemClientObject(unsigned int n); | |
6c8a980f | 385 | |
131b1fba VZ |
386 | // set the type of the client data stored in this control: override this if |
387 | // you override GetClientDataType() | |
388 | virtual void SetClientDataType(wxClientDataType clientDataItemsType) | |
389 | { | |
390 | m_clientDataItemsType = clientDataItemsType; | |
391 | } | |
593ac33e | 392 | |
131b1fba | 393 | private: |
6c8a980f VZ |
394 | // the type of the client data for the items |
395 | wxClientDataType m_clientDataItemsType; | |
396 | }; | |
397 | ||
0ae0bb79 VZ |
398 | // this macro must (unfortunately) be used in any class deriving from both |
399 | // wxItemContainer and wxControl because otherwise there is ambiguity when | |
400 | // calling GetClientXXX() functions -- the compiler can't choose between the | |
401 | // two versions | |
402 | #define wxCONTROL_ITEMCONTAINER_CLIENTDATAOBJECT_RECAST \ | |
403 | void SetClientData(void *data) \ | |
9ebb7cad | 404 | { wxEvtHandler::SetClientData(data); } \ |
0ae0bb79 | 405 | void *GetClientData() const \ |
9ebb7cad | 406 | { return wxEvtHandler::GetClientData(); } \ |
0ae0bb79 | 407 | void SetClientObject(wxClientData *data) \ |
9ebb7cad | 408 | { wxEvtHandler::SetClientObject(data); } \ |
0ae0bb79 | 409 | wxClientData *GetClientObject() const \ |
9ebb7cad | 410 | { return wxEvtHandler::GetClientObject(); } \ |
aa61d352 | 411 | void SetClientData(unsigned int n, void* clientData) \ |
0ae0bb79 | 412 | { wxItemContainer::SetClientData(n, clientData); } \ |
aa61d352 | 413 | void* GetClientData(unsigned int n) const \ |
0ae0bb79 | 414 | { return wxItemContainer::GetClientData(n); } \ |
aa61d352 | 415 | void SetClientObject(unsigned int n, wxClientData* clientData) \ |
0ae0bb79 | 416 | { wxItemContainer::SetClientObject(n, clientData); } \ |
aa61d352 | 417 | wxClientData* GetClientObject(unsigned int n) const \ |
0ae0bb79 VZ |
418 | { return wxItemContainer::GetClientObject(n); } |
419 | ||
53a2db12 | 420 | class WXDLLIMPEXP_CORE wxControlWithItemsBase : public wxControl, |
a236aa20 | 421 | public wxItemContainer |
1e6feb95 VZ |
422 | { |
423 | public: | |
a236aa20 | 424 | wxControlWithItemsBase() { } |
0ae0bb79 | 425 | |
1e6feb95 VZ |
426 | // we have to redefine these functions here to avoid ambiguities in classes |
427 | // deriving from us which would arise otherwise because both base classses | |
428 | // have the methods with the same names - hopefully, a smart compiler can | |
429 | // optimize away these simple inline wrappers so we don't suffer much from | |
430 | // this | |
0ae0bb79 | 431 | wxCONTROL_ITEMCONTAINER_CLIENTDATAOBJECT_RECAST |
fc7a2a60 | 432 | |
d4864e97 VZ |
433 | // usually the controls like list/combo boxes have their own background |
434 | // colour | |
435 | virtual bool ShouldInheritColours() const { return false; } | |
436 | ||
929bd5fd | 437 | protected: |
593ac33e VZ |
438 | // fill in the client object or data field of the event as appropriate |
439 | // | |
440 | // calls InitCommandEvent() and, if n != wxNOT_FOUND, also sets the per | |
441 | // item client data | |
442 | void InitCommandEventWithItems(wxCommandEvent& event, int n); | |
443 | ||
fc7a2a60 | 444 | private: |
c0c133e1 | 445 | wxDECLARE_NO_COPY_CLASS(wxControlWithItemsBase); |
1e6feb95 VZ |
446 | }; |
447 | ||
a236aa20 VZ |
448 | // define the platform-specific wxControlWithItems class |
449 | #if defined(__WXMSW__) | |
450 | #include "wx/msw/ctrlsub.h" | |
451 | #elif defined(__WXMOTIF__) | |
452 | #include "wx/motif/ctrlsub.h" | |
453 | #else | |
53a2db12 | 454 | class WXDLLIMPEXP_CORE wxControlWithItems : public wxControlWithItemsBase |
a236aa20 VZ |
455 | { |
456 | public: | |
457 | wxControlWithItems() { } | |
458 | ||
459 | private: | |
460 | DECLARE_ABSTRACT_CLASS(wxControlWithItems) | |
c0c133e1 | 461 | wxDECLARE_NO_COPY_CLASS(wxControlWithItems); |
a236aa20 VZ |
462 | }; |
463 | #endif | |
c6179a84 | 464 | |
1e6feb95 | 465 | #endif // wxUSE_CONTROLS |
6c8a980f | 466 | |
1e6feb95 | 467 | #endif // _WX_CTRLSUB_H_BASE_ |