put the write end of the pipe in non blocking mode to avoid deadlocks and return...
[wxWidgets.git] / src / motif / listbox.cpp
CommitLineData
4bb6408c
JS
1///////////////////////////////////////////////////////////////////////////////
2// Name: listbox.cpp
3// Purpose: wxListBox
4// Author: Julian Smart
5// Modified by:
6// Created: 17/09/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
65571936 9// Licence: wxWindows licence
4bb6408c
JS
10///////////////////////////////////////////////////////////////////////////////
11
14f355c2 12#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
29006414 13 #pragma implementation "listbox.h"
4bb6408c
JS
14#endif
15
1248b41f
MB
16// For compilers that support precompilation, includes "wx.h".
17#include "wx/wxprec.h"
18
4dff3400
JJ
19#ifdef __VMS
20#define XtParent XTPARENT
21#define XtDisplay XTDISPLAY
22#endif
23
24# include "wx/listbox.h"
4bb6408c
JS
25#include "wx/settings.h"
26#include "wx/dynarray.h"
27#include "wx/log.h"
f97c9854 28#include "wx/utils.h"
584ad2a3 29#include "wx/arrstr.h"
f97c9854 30
338dd992
JJ
31#ifdef __VMS__
32#pragma message disable nosimpint
33#endif
f97c9854 34#include <Xm/List.h>
338dd992
JJ
35#ifdef __VMS__
36#pragma message enable nosimpint
37#endif
f97c9854 38#include "wx/motif/private.h"
4bb6408c 39
29006414 40 IMPLEMENT_DYNAMIC_CLASS(wxListBox, wxControl)
4bb6408c 41
29006414
VZ
42static void wxListBoxCallback(Widget w,
43 XtPointer clientData,
44 XmListCallbackStruct * cbs);
f97c9854 45
99ab3e3f
MB
46// ----------------------------------------------------------------------------
47// wxSizeKeeper
48// ----------------------------------------------------------------------------
49
50// helper class to reduce code duplication
51class wxSizeKeeper
52{
53 int m_x, m_y;
54 wxWindow* m_w;
55public:
56 wxSizeKeeper( wxWindow* w )
57 : m_w( w )
58 {
59 m_w->GetSize( &m_x, &m_y );
60 }
61
62 void Restore()
63 {
64 int x, y;
65
66 m_w->GetSize( &x, &y );
67 if( x != m_x || y != m_y )
68 m_w->SetSize( -1, -1, m_x, m_y );
69 }
70};
71
4bb6408c
JS
72// ============================================================================
73// list box control implementation
74// ============================================================================
75
76// Listbox item
99ab3e3f 77wxListBox::wxListBox()
4bb6408c 78{
f97c9854 79 m_noItems = 0;
4bb6408c
JS
80}
81
82bool wxListBox::Create(wxWindow *parent, wxWindowID id,
83 const wxPoint& pos,
84 const wxSize& size,
85 int n, const wxString choices[],
86 long style,
87 const wxValidator& validator,
88 const wxString& name)
89{
99ab3e3f
MB
90 if( !wxControl::CreateControl( parent, id, pos, size, style,
91 validator, name ) )
92 return FALSE;
93
f97c9854 94 m_noItems = n;
94b49b93 95 m_backgroundColour = * wxWHITE;
29006414 96
f97c9854 97 Widget parentWidget = (Widget) parent->GetClientWidget();
29006414 98
da494b40 99 WXFontType fontType = (WXFontType)NULL;
e1aae528
MB
100
101 if( m_font.Ok() )
102 {
da494b40 103 fontType = m_font.GetFontType(XtDisplay(parentWidget));
e1aae528
MB
104 }
105
106 Arg args[4];
99ab3e3f 107 int count = 0;
e1aae528
MB
108 XtSetArg( args[count], XmNlistSizePolicy, XmCONSTANT ); ++count;
109 XtSetArg( args[count], XmNselectionPolicy,
99ab3e3f
MB
110 ( m_windowStyle & wxLB_MULTIPLE ) ? XmMULTIPLE_SELECT :
111 ( m_windowStyle & wxLB_EXTENDED ) ? XmEXTENDED_SELECT :
112 XmBROWSE_SELECT );
113 ++count;
da494b40 114 if( fontType )
e1aae528 115 {
da494b40 116 XtSetArg( args[count], (String)wxFont::GetFontTag(), fontType );
e1aae528
MB
117 ++count;
118 }
99ab3e3f 119 if( m_windowStyle & wxLB_ALWAYS_SB )
f97c9854 120 {
e1aae528 121 XtSetArg( args[count], XmNscrollBarDisplayPolicy, XmSTATIC );
99ab3e3f 122 ++count;
f97c9854 123 }
29006414 124
d3a80c92
MB
125 Widget listWidget =
126 XmCreateScrolledList(parentWidget,
127 wxConstCast(name.c_str(), char), args, count);
29006414 128
f97c9854 129 m_mainWidget = (WXWidget) listWidget;
29006414 130
a4294b78 131 Set(n, choices);
29006414 132
f97c9854 133 XtManageChild (listWidget);
29006414 134
e1aae528
MB
135 wxSize best = GetBestSize();
136 if( size.x != -1 ) best.x = size.x;
137 if( size.y != -1 ) best.y = size.y;
29006414 138
ef41d80c
MB
139 XtAddCallback (listWidget,
140 XmNbrowseSelectionCallback,
141 (XtCallbackProc) wxListBoxCallback,
142 (XtPointer) this);
143 XtAddCallback (listWidget,
144 XmNextendedSelectionCallback,
145 (XtCallbackProc) wxListBoxCallback,
146 (XtPointer) this);
147 XtAddCallback (listWidget,
148 XmNmultipleSelectionCallback,
149 (XtCallbackProc) wxListBoxCallback,
150 (XtPointer) this);
151 XtAddCallback (listWidget,
152 XmNdefaultActionCallback,
153 (XtCallbackProc) wxListBoxCallback,
154 (XtPointer) this);
29006414 155
ef41d80c 156 AttachWidget (parent, m_mainWidget, (WXWidget) NULL,
e1aae528 157 pos.x, pos.y, best.x, best.y);
29006414 158
0d57be45 159 ChangeBackgroundColour();
29006414 160
f97c9854 161 return TRUE;
4bb6408c
JS
162}
163
584ad2a3
MB
164bool wxListBox::Create(wxWindow *parent, wxWindowID id,
165 const wxPoint& pos,
166 const wxSize& size,
167 const wxArrayString& choices,
168 long style,
169 const wxValidator& validator,
170 const wxString& name)
171{
172 wxCArrayString chs(choices);
173 return Create(parent, id, pos, size, chs.GetCount(), chs.GetStrings(),
174 style, validator, name);
175}
176
4bb6408c
JS
177wxListBox::~wxListBox()
178{
99ab3e3f
MB
179 if( HasClientObjectData() )
180 m_clientDataDict.DestroyData();
181}
182
183void wxListBox::SetSelectionPolicy()
184{
185 Widget listBox = (Widget)m_mainWidget;
186 Arg args[3];
187
188 XtSetArg( args[0], XmNlistSizePolicy, XmCONSTANT );
189
190 XtSetArg( args[1], XmNselectionPolicy,
191 ( m_windowStyle & wxLB_MULTIPLE ) ? XmMULTIPLE_SELECT :
192 ( m_windowStyle & wxLB_EXTENDED ) ? XmEXTENDED_SELECT :
193 XmBROWSE_SELECT );
194
195 XtSetValues( listBox, args, 2 );
4bb6408c
JS
196}
197
ef41d80c 198void wxListBox::DoSetFirstItem( int N )
4bb6408c 199{
2d120f83 200 int count, length;
29006414 201
2d120f83
JS
202 if (N < 0)
203 return;
204 XtVaGetValues ((Widget) m_mainWidget,
29006414
VZ
205 XmNvisibleItemCount, &count,
206 XmNitemCount, &length,
207 NULL);
2d120f83
JS
208 if ((N + count) >= length)
209 N = length - count;
210 XmListSetPos ((Widget) m_mainWidget, N + 1);
4bb6408c
JS
211}
212
4bb6408c
JS
213void wxListBox::Delete(int N)
214{
99ab3e3f 215 wxSizeKeeper sk( this );
2d120f83 216 Widget listBox = (Widget) m_mainWidget;
29006414 217
2d120f83 218 bool managed = XtIsManaged(listBox);
29006414 219
2d120f83
JS
220 if (managed)
221 XtUnmanageChild (listBox);
29006414 222
2d120f83 223 XmListDeletePos (listBox, N + 1);
29006414 224
2d120f83
JS
225 if (managed)
226 XtManageChild (listBox);
29006414 227
99ab3e3f
MB
228 sk.Restore();
229 m_clientDataDict.Delete(N, HasClientObjectData());
2d120f83 230 m_noItems --;
4bb6408c
JS
231}
232
ef41d80c 233int wxListBox::DoAppend(const wxString& item)
4bb6408c 234{
99ab3e3f 235 wxSizeKeeper sk( this );
2d120f83 236 Widget listBox = (Widget) m_mainWidget;
29006414 237
2d120f83 238 bool managed = XtIsManaged(listBox);
29006414 239
2d120f83
JS
240 if (managed)
241 XtUnmanageChild (listBox);
242 int n;
243 XtVaGetValues (listBox, XmNitemCount, &n, NULL);
99ab3e3f 244 wxXmString text( item );
2d120f83 245 // XmListAddItem(listBox, text, n + 1);
99ab3e3f 246 XmListAddItemUnselected (listBox, text(), 0);
29006414 247
2d120f83
JS
248 // It seems that if the list is cleared, we must re-ask for
249 // selection policy!!
99ab3e3f 250 SetSelectionPolicy();
29006414 251
2d120f83
JS
252 if (managed)
253 XtManageChild (listBox);
29006414 254
99ab3e3f 255 sk.Restore();
2d120f83 256 m_noItems ++;
ef41d80c
MB
257
258 return GetCount() - 1;
4bb6408c
JS
259}
260
ef41d80c 261void wxListBox::DoSetItems(const wxArrayString& items, void** clientData)
4bb6408c 262{
99ab3e3f 263 wxSizeKeeper sk( this );
2d120f83 264 Widget listBox = (Widget) m_mainWidget;
99ab3e3f
MB
265
266 if( HasClientObjectData() )
267 m_clientDataDict.DestroyData();
ef41d80c
MB
268
269 bool managed = XtIsManaged(listBox);
29006414 270
2d120f83
JS
271 if (managed)
272 XtUnmanageChild (listBox);
ef41d80c
MB
273 XmString *text = new XmString[items.GetCount()];
274 size_t i;
275 for (i = 0; i < items.GetCount(); ++i)
d3a80c92 276 text[i] = wxStringToXmString (items[i]);
29006414 277
ef41d80c
MB
278 if ( clientData )
279 for (i = 0; i < items.GetCount(); ++i)
99ab3e3f 280 m_clientDataDict.Set(i, (wxClientData*)clientData[i], FALSE);
ef41d80c
MB
281
282 XmListAddItems (listBox, text, items.GetCount(), 0);
283 for (i = 0; i < items.GetCount(); i++)
284 XmStringFree (text[i]);
285 delete[] text;
29006414 286
2d120f83
JS
287 // It seems that if the list is cleared, we must re-ask for
288 // selection policy!!
99ab3e3f 289 SetSelectionPolicy();
29006414 290
2d120f83
JS
291 if (managed)
292 XtManageChild (listBox);
29006414 293
99ab3e3f 294 sk.Restore();
29006414 295
ef41d80c 296 m_noItems = items.GetCount();
4bb6408c
JS
297}
298
9b1bd0c6 299int wxDoFindStringInList(Widget w, const wxString& s)
4bb6408c 300{
99ab3e3f 301 wxXmString str( s );
2d120f83
JS
302 int *positions = NULL;
303 int no_positions = 0;
9b1bd0c6 304 bool success = XmListGetMatchPos (w, str(),
ef41d80c 305 &positions, &no_positions);
99ab3e3f 306
2d120f83 307 if (success)
f97c9854 308 {
2d120f83
JS
309 int pos = positions[0];
310 if (positions)
311 XtFree ((char *) positions);
312 return pos - 1;
f97c9854 313 }
2d120f83
JS
314 else
315 return -1;
4bb6408c
JS
316}
317
9b1bd0c6
MB
318int wxListBox::FindString(const wxString& s) const
319{
320 return wxDoFindStringInList( (Widget)m_mainWidget, s );
321}
322
4bb6408c
JS
323void wxListBox::Clear()
324{
2d120f83
JS
325 if (m_noItems <= 0)
326 return;
29006414 327
99ab3e3f 328 wxSizeKeeper sk( this );
2d120f83 329 Widget listBox = (Widget) m_mainWidget;
29006414 330
2d120f83 331 XmListDeleteAllItems (listBox);
99ab3e3f
MB
332 if( HasClientObjectData() )
333 m_clientDataDict.DestroyData();
29006414 334
99ab3e3f 335 sk.Restore();
29006414 336
2d120f83 337 m_noItems = 0;
4bb6408c
JS
338}
339
340void wxListBox::SetSelection(int N, bool select)
341{
2d120f83
JS
342 m_inSetValue = TRUE;
343 if (select)
f97c9854 344 {
29006414
VZ
345#if 0
346 if (m_windowStyle & wxLB_MULTIPLE)
347 {
348 int *selections = NULL;
349 int n = GetSelections (&selections);
350
ef41d80c
MB
351 // This hack is supposed to work, to make it possible
352 // to select more than one item, but it DOESN'T under Motif 1.1.
29006414 353
ef41d80c
MB
354 XtVaSetValues ((Widget) m_mainWidget,
355 XmNselectionPolicy, XmMULTIPLE_SELECT,
356 NULL);
29006414
VZ
357
358 int i;
359 for (i = 0; i < n; i++)
ef41d80c
MB
360 XmListSelectPos ((Widget) m_mainWidget,
361 selections[i] + 1, FALSE);
29006414
VZ
362
363 XmListSelectPos ((Widget) m_mainWidget, N + 1, FALSE);
364
ef41d80c
MB
365 XtVaSetValues ((Widget) m_mainWidget,
366 XmNselectionPolicy, XmEXTENDED_SELECT,
367 NULL);
29006414
VZ
368 }
369 else
370#endif // 0
2d120f83 371 XmListSelectPos ((Widget) m_mainWidget, N + 1, FALSE);
29006414 372
f97c9854 373 }
2d120f83
JS
374 else
375 XmListDeselectPos ((Widget) m_mainWidget, N + 1);
29006414 376
2d120f83 377 m_inSetValue = FALSE;
4bb6408c
JS
378}
379
d7d38ea4 380bool wxListBox::IsSelected(int N) const
4bb6408c 381{
2d120f83
JS
382 // In Motif, no simple way to determine if the item is selected.
383 wxArrayInt theSelections;
384 int count = GetSelections (theSelections);
385 if (count == 0)
386 return FALSE;
387 else
388 {
389 int j;
390 for (j = 0; j < count; j++)
391 if (theSelections[j] == N)
392 return TRUE;
393 }
4bb6408c
JS
394 return FALSE;
395}
396
ef41d80c
MB
397void wxListBox::DoSetItemClientObject(int n, wxClientData* clientData)
398{
99ab3e3f 399 m_clientDataDict.Set(n, clientData, FALSE);
ef41d80c
MB
400}
401
402wxClientData* wxListBox::DoGetItemClientObject(int n) const
4bb6408c 403{
99ab3e3f 404 return m_clientDataDict.Get(n);
4bb6408c
JS
405}
406
ef41d80c 407void *wxListBox::DoGetItemClientData(int N) const
4bb6408c 408{
99ab3e3f 409 return (void*)m_clientDataDict.Get(N);
4bb6408c
JS
410}
411
ef41d80c 412void wxListBox::DoSetItemClientData(int N, void *Client_data)
4bb6408c 413{
99ab3e3f 414 m_clientDataDict.Set(N, (wxClientData*)Client_data, FALSE);
4bb6408c
JS
415}
416
417// Return number of selections and an array of selected integers
418int wxListBox::GetSelections(wxArrayInt& aSelections) const
419{
2d120f83 420 aSelections.Empty();
29006414 421
2d120f83
JS
422 Widget listBox = (Widget) m_mainWidget;
423 int *posList = NULL;
424 int posCnt = 0;
425 bool flag = XmListGetSelectedPos (listBox, &posList, &posCnt);
426 if (flag)
427 {
428 if (posCnt > 0)
429 {
430 aSelections.Alloc(posCnt);
29006414 431
2d120f83
JS
432 int i;
433 for (i = 0; i < posCnt; i++)
434 aSelections.Add(posList[i] - 1);
29006414 435
2d120f83
JS
436 XtFree ((char *) posList);
437 return posCnt;
438 }
439 else
440 return 0;
4bb6408c 441 }
2d120f83
JS
442 else
443 return 0;
4bb6408c
JS
444}
445
446// Get single selection, for single choice list items
9b1bd0c6 447int wxDoGetSelectionInList(Widget listBox)
4bb6408c 448{
f97c9854
JS
449 int *posList = NULL;
450 int posCnt = 0;
451 bool flag = XmListGetSelectedPos (listBox, &posList, &posCnt);
452 if (flag)
453 {
454 int id = -1;
455 if (posCnt > 0)
456 id = posList[0] - 1;
457 XtFree ((char *) posList);
458 return id;
459 }
460 else
461 return -1;
4bb6408c
JS
462}
463
9b1bd0c6
MB
464int wxListBox::GetSelection() const
465{
466 return wxDoGetSelectionInList((Widget) m_mainWidget);
467}
468
4bb6408c 469// Find string for position
e1aae528 470wxString wxDoGetStringInList( Widget listBox, int n )
4bb6408c 471{
f97c9854 472 XmString *strlist;
e1aae528
MB
473 int count;
474 XtVaGetValues( listBox,
475 XmNitemCount, &count,
476 XmNitems, &strlist,
477 NULL );
d40708e2 478 if( n < count && n >= 0 )
e1aae528 479 return wxXmStringToString( strlist[n] );
f97c9854
JS
480 else
481 return wxEmptyString;
4bb6408c
JS
482}
483
e1aae528
MB
484wxString wxListBox::GetString( int n ) const
485{
486 return wxDoGetStringInList( (Widget)m_mainWidget, n );
487}
488
ef41d80c 489void wxListBox::DoInsertItems(const wxArrayString& items, int pos)
4bb6408c 490{
99ab3e3f 491 wxSizeKeeper sk( this );
f97c9854 492 Widget listBox = (Widget) m_mainWidget;
29006414 493
f97c9854 494 bool managed = XtIsManaged(listBox);
29006414 495
f97c9854
JS
496 if (managed)
497 XtUnmanageChild(listBox);
29006414 498
ef41d80c
MB
499 XmString *text = new XmString[items.GetCount()];
500 size_t i;
2d120f83
JS
501 // Steve Hammes: Motif 1.1 compatibility
502 // #if XmVersion > 1100
503 // Corrected by Sergey Krasnov from Steve Hammes' code
f97c9854 504#if XmVersion > 1001
ef41d80c 505 for (i = 0; i < items.GetCount(); i++)
d3a80c92 506 text[i] = wxStringToXmString(items[i]);
ef41d80c 507 XmListAddItemsUnselected(listBox, text, items.GetCount(), pos+1);
f97c9854 508#else
ef41d80c 509 for (i = 0; i < items.GetCount(); i++)
f97c9854 510 {
d3a80c92 511 text[i] = wxStringToXmString(items[i]);
ef41d80c
MB
512 // Another Sergey correction
513 XmListAddItemUnselected(listBox, text[i], pos+i+1);
f97c9854
JS
514 }
515#endif
ef41d80c 516 for (i = 0; i < items.GetCount(); i++)
f97c9854 517 XmStringFree(text[i]);
f97c9854 518 delete[] text;
29006414 519
f97c9854
JS
520 // It seems that if the list is cleared, we must re-ask for
521 // selection policy!!
99ab3e3f 522 SetSelectionPolicy();
29006414 523
f97c9854
JS
524 if (managed)
525 XtManageChild(listBox);
29006414 526
99ab3e3f 527 sk.Restore();
29006414 528
ef41d80c 529 m_noItems += items.GetCount();
4bb6408c
JS
530}
531
532void wxListBox::SetString(int N, const wxString& s)
533{
99ab3e3f 534 wxSizeKeeper sk( this );
f97c9854 535 Widget listBox = (Widget) m_mainWidget;
29006414 536
99ab3e3f 537 wxXmString text( s );
29006414
VZ
538
539 // delete the item and add it again.
540 // FIXME isn't there a way to change it in place?
f97c9854 541 XmListDeletePos (listBox, N+1);
99ab3e3f 542 XmListAddItem (listBox, text(), N+1);
29006414 543
99ab3e3f 544 sk.Restore();
4bb6408c
JS
545}
546
4bb6408c
JS
547void wxListBox::Command (wxCommandEvent & event)
548{
f97c9854
JS
549 if (event.m_extraLong)
550 SetSelection (event.m_commandInt);
551 else
552 {
553 Deselect (event.m_commandInt);
554 return;
555 }
556 ProcessCommand (event);
557}
558
af111fc3 559void wxListBoxCallback (Widget WXUNUSED(w), XtPointer clientData,
2d120f83 560 XmListCallbackStruct * cbs)
f97c9854 561{
f97c9854 562 wxListBox *item = (wxListBox *) clientData;
29006414 563
a4294b78 564 if (item->InSetValue())
f97c9854 565 return;
29006414 566
ef41d80c
MB
567 wxEventType evtType;
568
569 if( cbs->reason == XmCR_DEFAULT_ACTION )
570 evtType = wxEVT_COMMAND_LISTBOX_DOUBLECLICKED;
571 else
572 evtType = wxEVT_COMMAND_LISTBOX_SELECTED;
573
574 int n = cbs->item_position - 1;
575 wxCommandEvent event (evtType, item->GetId());
576 if ( item->HasClientObjectData() )
577 event.SetClientObject( item->GetClientObject(n) );
578 else if ( item->HasClientUntypedData() )
579 event.SetClientData( item->GetClientData(n) );
580 event.m_commandInt = n;
581 event.m_extraLong = TRUE;
582 event.SetEventObject(item);
583 event.SetString( item->GetString( n ) );
584
585 int x = -1;
2b5f62a0 586 if( NULL != cbs->event && cbs->event->type == ButtonRelease )
ef41d80c
MB
587 {
588 XButtonEvent* evt = (XButtonEvent*)cbs->event;
589
590 x = evt->x;
591 }
592
f97c9854 593 switch (cbs->reason)
4bb6408c 594 {
2d120f83
JS
595 case XmCR_MULTIPLE_SELECT:
596 case XmCR_BROWSE_SELECT:
ef41d80c
MB
597#if wxUSE_CHECKLISTBOX
598 item->DoToggleItem( n, x );
599#endif
600 case XmCR_DEFAULT_ACTION:
601 item->GetEventHandler()->ProcessEvent(event);
602 break;
2d120f83 603 case XmCR_EXTENDED_SELECT:
ef41d80c 604 switch (cbs->selection_type)
f97c9854 605 {
ef41d80c
MB
606 case XmINITIAL:
607 case XmADDITION:
608 case XmMODIFICATION:
609 item->DoToggleItem( n, x );
610 item->GetEventHandler()->ProcessEvent(event);
f97c9854
JS
611 break;
612 }
ef41d80c 613 break;
4bb6408c 614 }
4bb6408c
JS
615}
616
89c7e962
JS
617WXWidget wxListBox::GetTopWidget() const
618{
2d120f83 619 return (WXWidget) XtParent( (Widget) m_mainWidget );
89c7e962 620}
0d57be45 621
0d57be45
JS
622void wxListBox::ChangeBackgroundColour()
623{
321db4b6 624 wxWindow::ChangeBackgroundColour();
29006414 625
02800301
JS
626 Widget parent = XtParent ((Widget) m_mainWidget);
627 Widget hsb, vsb;
29006414 628
02800301 629 XtVaGetValues (parent,
2d120f83
JS
630 XmNhorizontalScrollBar, &hsb,
631 XmNverticalScrollBar, &vsb,
632 NULL);
29006414 633
a91b47e8
JS
634 /* TODO: should scrollbars be affected? Should probably have separate
635 * function to change them (by default, taken from wxSystemSettings)
2d120f83 636 */
a756f210 637 wxColour backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE);
a8680e3e
MB
638 wxDoChangeBackgroundColour((WXWidget) hsb, backgroundColour, TRUE);
639 wxDoChangeBackgroundColour((WXWidget) vsb, backgroundColour, TRUE);
15d5ab67
JS
640
641 XtVaSetValues (hsb,
642 XmNtroughColor, backgroundColour.AllocColour(XtDisplay(hsb)),
643 NULL);
644 XtVaSetValues (vsb,
645 XmNtroughColor, backgroundColour.AllocColour(XtDisplay(vsb)),
646 NULL);
29006414 647
73d33f1a 648 // MBN: why change parent's background? It looks really ugly.
a8680e3e 649 // wxDoChangeBackgroundColour((WXWidget) parent, m_backgroundColour, TRUE);
0d57be45
JS
650}
651
652void wxListBox::ChangeForegroundColour()
653{
321db4b6 654 wxWindow::ChangeForegroundColour();
29006414 655
02800301
JS
656 Widget parent = XtParent ((Widget) m_mainWidget);
657 Widget hsb, vsb;
29006414
VZ
658
659 XtVaGetValues(parent,
660 XmNhorizontalScrollBar, &hsb,
661 XmNverticalScrollBar, &vsb,
662 NULL);
663
664 /* TODO: should scrollbars be affected? Should probably have separate
665 function to change them (by default, taken from wxSystemSettings)
666
a8680e3e
MB
667 wxDoChangeForegroundColour((WXWidget) hsb, m_foregroundColour);
668 wxDoChangeForegroundColour((WXWidget) vsb, m_foregroundColour);
669 wxDoChangeForegroundColour((WXWidget) parent, m_foregroundColour);
02800301 670 */
0d57be45
JS
671}
672
6adaedf0
JS
673int wxListBox::GetCount() const
674{
ef41d80c 675 return m_noItems;
6adaedf0 676}
e1aae528
MB
677
678#define LIST_SCROLL_SPACING 6
679
680wxSize wxDoGetListBoxBestSize( Widget listWidget, const wxWindow* window )
681{
682 int max;
683 Dimension spacing, highlight, xmargin, ymargin, shadow;
684 int width = 0;
685 int x, y;
686
687 XtVaGetValues( listWidget,
688 XmNitemCount, &max,
689 XmNlistSpacing, &spacing,
690 XmNhighlightThickness, &highlight,
691 XmNlistMarginWidth, &xmargin,
692 XmNlistMarginHeight, &ymargin,
693 XmNshadowThickness, &shadow,
694 NULL );
695
696 for( size_t i = 0; i < (size_t)max; ++i )
697 {
698 window->GetTextExtent( wxDoGetStringInList( listWidget, i ), &x, &y );
699 width = wxMax( width, x );
700 }
701
702 // use some arbitrary value if there are no strings
703 if( width == 0 )
704 width = 100;
705
706 // get my
707 window->GetTextExtent( "v", &x, &y );
708
709 // make it a little larger than widest string, plus the scrollbar
710 width += wxSystemSettings::GetMetric( wxSYS_VSCROLL_X )
711 + 2 * highlight + LIST_SCROLL_SPACING + 2 * xmargin + 2 * shadow;
712
713 // at least 3 items, at most 10
714 int height = wxMax( 3, wxMin( 10, max ) ) *
715 ( y + spacing + 2 * highlight ) + 2 * ymargin + 2 * shadow;
716
717 return wxSize( width, height );
718}
719
720wxSize wxListBox::DoGetBestSize() const
721{
722 return wxDoGetListBoxBestSize( (Widget)m_mainWidget, this );
723}
724