1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/listbox.cpp 
   4 // Author:      Robert Roebling 
   5 // Modified By: Ryan Norton (GtkTreeView implementation) 
   7 // Copyright:   (c) 1998 Robert Roebling 
   8 // Licence:     wxWindows licence 
   9 ///////////////////////////////////////////////////////////////////////////// 
  11 // For compilers that support precompilation, includes "wx.h". 
  12 #include "wx/wxprec.h" 
  16 #include "wx/listbox.h" 
  19     #include "wx/dynarray.h" 
  23     #include "wx/settings.h" 
  24     #include "wx/checklst.h" 
  25     #include "wx/arrstr.h" 
  28 #include "wx/gtk/private.h" 
  29 #include "wx/gtk/treeentry_gtk.h" 
  32     #include "wx/tooltip.h" 
  37 #include <gdk/gdkkeysyms.h> 
  39 //----------------------------------------------------------------------------- 
  41 //----------------------------------------------------------------------------- 
  43 extern bool           g_blockEventsOnDrag
; 
  44 extern bool           g_blockEventsOnScroll
; 
  48 //----------------------------------------------------------------------------- 
  49 // Macro to tell which row the strings are in (1 if native checklist, 0 if not) 
  50 //----------------------------------------------------------------------------- 
  52 #if wxUSE_CHECKLISTBOX 
  53 #   define WXLISTBOX_DATACOLUMN_ARG(x)  (x->m_hasCheckBoxes ? 1 : 0) 
  55 #   define WXLISTBOX_DATACOLUMN_ARG(x)  (0) 
  56 #endif // wxUSE_CHECKLISTBOX 
  58 #define WXLISTBOX_DATACOLUMN    WXLISTBOX_DATACOLUMN_ARG(this) 
  60 //----------------------------------------------------------------------------- 
  62 //----------------------------------------------------------------------------- 
  66 gtk_listbox_row_activated_callback(GtkTreeView        
*treeview
, 
  68                                    GtkTreeViewColumn  
*col
, 
  71     if (g_isIdle
) wxapp_install_idle_handler(); 
  73     if (g_blockEventsOnDrag
) return; 
  74     if (g_blockEventsOnScroll
) return; 
  76     // This is triggered by either a double-click or a space press 
  78     int sel 
= gtk_tree_path_get_indices(path
)[0]; 
  80     wxCommandEvent 
event(wxEVT_COMMAND_LISTBOX_DOUBLECLICKED
, listbox
->GetId() ); 
  81     event
.SetEventObject( listbox 
); 
  83     if (listbox
->IsSelected(sel
)) 
  85         GtkTreeEntry
* entry 
= listbox
->GtkGetEntry(sel
); 
  90             event
.SetString(wxConvUTF8
.cMB2WX(gtk_tree_entry_get_label(entry
))); 
  92             if ( listbox
->HasClientObjectData() ) 
  93                 event
.SetClientObject( (wxClientData
*) gtk_tree_entry_get_userdata(entry
) ); 
  94             else if ( listbox
->HasClientUntypedData() ) 
  95                 event
.SetClientData( gtk_tree_entry_get_userdata(entry
) ); 
  97             g_object_unref (entry
); 
 101             wxLogSysError(wxT("Internal error - could not get entry for double-click")); 
 110     listbox
->GetEventHandler()->ProcessEvent( event 
); 
 114 //----------------------------------------------------------------------------- 
 116 //----------------------------------------------------------------------------- 
 120 gtk_listbox_key_press_callback( GtkWidget 
*widget
, 
 121                                 GdkEventKey 
*gdk_event
, 
 124     if (g_blockEventsOnDrag
) return FALSE
; 
 126     if ((gdk_event
->keyval 
== GDK_Tab
) || (gdk_event
->keyval 
== GDK_ISO_Left_Tab
)) 
 128         wxNavigationKeyEvent new_event
; 
 129         /* GDK reports GDK_ISO_Left_Tab for SHIFT-TAB */ 
 130         new_event
.SetDirection( (gdk_event
->keyval 
== GDK_Tab
) ); 
 131         /* CTRL-TAB changes the (parent) window, i.e. switch notebook page */ 
 132         new_event
.SetWindowChange( (gdk_event
->state 
& GDK_CONTROL_MASK
) ); 
 133         new_event
.SetCurrentFocus( listbox 
); 
 134         if (listbox
->GetEventHandler()->ProcessEvent( new_event 
)) 
 142 //----------------------------------------------------------------------------- 
 144 //----------------------------------------------------------------------------- 
 148 gtk_listitem_changed_callback( GtkTreeSelection
* selection
, wxListBox 
*listbox 
) 
 150     if (g_blockEventsOnDrag
) return; 
 152     if (listbox
->m_blockEvent
) return; 
 154     wxCommandEvent 
event(wxEVT_COMMAND_LISTBOX_SELECTED
, listbox
->GetId() ); 
 155     event
.SetEventObject( listbox 
); 
 157     if (listbox
->HasFlag(wxLB_MULTIPLE
) || listbox
->HasFlag(wxLB_EXTENDED
)) 
 159         wxArrayInt selections
; 
 160         listbox
->GetSelections( selections 
); 
 162         if (selections
.GetCount() == 0) 
 164             // indicate that this is a deselection 
 165             event
.SetExtraLong( 0 ); 
 168             listbox
->GetEventHandler()->ProcessEvent( event 
); 
 174             // indicate that this is a selection 
 175             event
.SetExtraLong( 1 ); 
 176             event
.SetInt( selections
[0] ); 
 178             listbox
->GetEventHandler()->ProcessEvent( event 
); 
 183         int index 
= listbox
->GetSelection(); 
 184         if (index 
== wxNOT_FOUND
) 
 186             // indicate that this is a deselection 
 187             event
.SetExtraLong( 0 ); 
 190             listbox
->GetEventHandler()->ProcessEvent( event 
); 
 196             GtkTreeEntry
* entry 
= listbox
->GtkGetEntry( index 
); 
 198             // indicate that this is a selection 
 199             event
.SetExtraLong( 1 ); 
 201             event
.SetInt( index 
); 
 202             event
.SetString(wxConvUTF8
.cMB2WX(gtk_tree_entry_get_label(entry
))); 
 204             if ( listbox
->HasClientObjectData() ) 
 205                 event
.SetClientObject( 
 206                     (wxClientData
*) gtk_tree_entry_get_userdata(entry
) 
 208             else if ( listbox
->HasClientUntypedData() ) 
 209                 event
.SetClientData( gtk_tree_entry_get_userdata(entry
) ); 
 211             listbox
->GetEventHandler()->ProcessEvent( event 
); 
 213             g_object_unref (entry
); 
 219 //----------------------------------------------------------------------------- 
 220 // GtkTreeEntry destruction (to destroy client data) 
 221 //----------------------------------------------------------------------------- 
 224 static void gtk_tree_entry_destroy_cb(GtkTreeEntry
* entry
, 
 227     if (listbox
->HasClientObjectData()) 
 229         gpointer userdata 
= gtk_tree_entry_get_userdata(entry
); 
 231             delete (wxClientData 
*)userdata
; 
 236 //----------------------------------------------------------------------------- 
 237 // Sorting callback (standard CmpNoCase return value) 
 238 //----------------------------------------------------------------------------- 
 241 static gint 
gtk_listbox_sort_callback(GtkTreeModel 
*model
, 
 247     GtkTreeEntry
* entry2
; 
 249     gtk_tree_model_get(GTK_TREE_MODEL(listbox
->m_liststore
), 
 251                              WXLISTBOX_DATACOLUMN_ARG(listbox
), 
 253     gtk_tree_model_get(GTK_TREE_MODEL(listbox
->m_liststore
), 
 255                              WXLISTBOX_DATACOLUMN_ARG(listbox
), 
 257     wxCHECK_MSG(entry
, 0, wxT("Could not get entry")); 
 258     wxCHECK_MSG(entry2
, 0, wxT("Could not get entry2")); 
 260     //We compare collate keys here instead of calling g_utf8_collate 
 261     //as it is rather slow (and even the docs reccommend this) 
 262     int ret 
= strcasecmp(gtk_tree_entry_get_collate_key(entry
), 
 263                          gtk_tree_entry_get_collate_key(entry2
)); 
 265     g_object_unref (entry
); 
 266     g_object_unref (entry2
); 
 272 //----------------------------------------------------------------------------- 
 273 // Searching callback (TRUE == not equal, FALSE == equal) 
 274 //----------------------------------------------------------------------------- 
 277 static gboolean 
gtk_listbox_searchequal_callback(GtkTreeModel
* model
, 
 285     gtk_tree_model_get(GTK_TREE_MODEL(listbox
->m_liststore
), 
 287                              WXLISTBOX_DATACOLUMN_ARG(listbox
), 
 289     wxCHECK_MSG(entry
, 0, wxT("Could not get entry")); 
 290     wxGtkString 
keycollatekey(g_utf8_collate_key(key
, -1)); 
 292     int ret 
= strcasecmp(keycollatekey
, 
 293                          gtk_tree_entry_get_collate_key(entry
)); 
 295     g_object_unref (entry
); 
 301 //----------------------------------------------------------------------------- 
 303 //----------------------------------------------------------------------------- 
 305 IMPLEMENT_DYNAMIC_CLASS(wxListBox
, wxControl
) 
 307 // ---------------------------------------------------------------------------- 
 309 // ---------------------------------------------------------------------------- 
 311 void wxListBox::Init() 
 313     m_treeview 
= (GtkTreeView
*) NULL
; 
 314 #if wxUSE_CHECKLISTBOX 
 315     m_hasCheckBoxes 
= false; 
 316 #endif // wxUSE_CHECKLISTBOX 
 319 bool wxListBox::Create( wxWindow 
*parent
, wxWindowID id
, 
 320                         const wxPoint 
&pos
, const wxSize 
&size
, 
 321                         const wxArrayString
& choices
, 
 322                         long style
, const wxValidator
& validator
, 
 323                         const wxString 
&name 
) 
 325     wxCArrayString 
chs(choices
); 
 327     return Create( parent
, id
, pos
, size
, chs
.GetCount(), chs
.GetStrings(), 
 328                    style
, validator
, name 
); 
 331 bool wxListBox::Create( wxWindow 
*parent
, wxWindowID id
, 
 332                         const wxPoint 
&pos
, const wxSize 
&size
, 
 333                         int n
, const wxString choices
[], 
 334                         long style
, const wxValidator
& validator
, 
 335                         const wxString 
&name 
) 
 338     m_acceptsFocus 
= true; 
 339     m_blockEvent 
= false; 
 341     if (!PreCreation( parent
, pos
, size 
) || 
 342         !CreateBase( parent
, id
, pos
, size
, style
, validator
, name 
)) 
 344         wxFAIL_MSG( wxT("wxListBox creation failed") ); 
 348     m_widget 
= gtk_scrolled_window_new( (GtkAdjustment
*) NULL
, (GtkAdjustment
*) NULL 
); 
 349     if (style 
& wxLB_ALWAYS_SB
) 
 351       gtk_scrolled_window_set_policy( GTK_SCROLLED_WINDOW(m_widget
), 
 352         GTK_POLICY_AUTOMATIC
, GTK_POLICY_ALWAYS 
); 
 356       gtk_scrolled_window_set_policy( GTK_SCROLLED_WINDOW(m_widget
), 
 357         GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC 
); 
 361     GtkScrolledWindowSetBorder(m_widget
, style
); 
 363     m_treeview 
= GTK_TREE_VIEW( gtk_tree_view_new( ) ); 
 365     //wxListBox doesn't have a header :) 
 366     //NB: If enabled SetFirstItem doesn't work correctly 
 367     gtk_tree_view_set_headers_visible(m_treeview
, FALSE
); 
 369 #if wxUSE_CHECKLISTBOX 
 371         ((wxCheckListBox
*)this)->DoCreateCheckList(); 
 372 #endif // wxUSE_CHECKLISTBOX 
 374     // Create the data column 
 375     gtk_tree_view_insert_column_with_attributes(m_treeview
, -1, "", 
 376                                                 gtk_cell_renderer_text_new(), 
 378                                                 WXLISTBOX_DATACOLUMN
, NULL
); 
 380     // Now create+set the model (GtkListStore) - first argument # of columns 
 381 #if wxUSE_CHECKLISTBOX 
 383         m_liststore 
= gtk_list_store_new(2, G_TYPE_BOOLEAN
, 
 384                                             GTK_TYPE_TREE_ENTRY
); 
 387         m_liststore 
= gtk_list_store_new(1, GTK_TYPE_TREE_ENTRY
); 
 389     gtk_tree_view_set_model(m_treeview
, GTK_TREE_MODEL(m_liststore
)); 
 391     g_object_unref (m_liststore
); //free on treeview destruction 
 393     // Disable the pop-up textctrl that enables searching - note that 
 394     // the docs specify that even if this disabled (which we are doing) 
 395     // the user can still have it through the start-interactive-search 
 396     // key binding...either way we want to provide a searchequal callback 
 397     // NB: If this is enabled a doubleclick event (activate) gets sent 
 398     //     on a successful search 
 399     gtk_tree_view_set_search_column(m_treeview
, WXLISTBOX_DATACOLUMN
); 
 400     gtk_tree_view_set_search_equal_func(m_treeview
, 
 401        (GtkTreeViewSearchEqualFunc
) gtk_listbox_searchequal_callback
, 
 405     gtk_tree_view_set_enable_search(m_treeview
, FALSE
); 
 408     GtkTreeSelection
* selection 
= gtk_tree_view_get_selection( m_treeview 
); 
 410     g_signal_connect_after (selection
, "changed", 
 411                             G_CALLBACK (gtk_listitem_changed_callback
), this); 
 413     GtkSelectionMode mode
; 
 414     if (style 
& wxLB_MULTIPLE
) 
 416         mode 
= GTK_SELECTION_MULTIPLE
; 
 418     else if (style 
& wxLB_EXTENDED
) 
 420         mode 
= GTK_SELECTION_EXTENDED
; 
 424         // if style was 0 set single mode 
 425         m_windowStyle 
|= wxLB_SINGLE
; 
 426         mode 
= GTK_SELECTION_SINGLE
; 
 429     gtk_tree_selection_set_mode( selection
, mode 
); 
 431     // Handle sortable stuff 
 432     if(style 
& wxLB_SORT
) 
 434         // Setup sorting in ascending (wx) order 
 435         gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(m_liststore
), 
 436                                              WXLISTBOX_DATACOLUMN
, 
 439         // Set the sort callback 
 440         gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(m_liststore
), 
 441                                         WXLISTBOX_DATACOLUMN
, 
 442                    (GtkTreeIterCompareFunc
) gtk_listbox_sort_callback
, 
 444                                         NULL 
//"destroy notifier" 
 449     gtk_container_add (GTK_CONTAINER (m_widget
), GTK_WIDGET(m_treeview
) ); 
 451     gtk_widget_show( GTK_WIDGET(m_treeview
) ); 
 452     m_focusWidget 
= GTK_WIDGET(m_treeview
); 
 454     wxListBox::DoInsertItems(wxArrayString(n
, choices
), 0); // insert initial items 
 456     // generate dclick events 
 457     g_signal_connect_after(m_treeview
, "row-activated", 
 458                      G_CALLBACK(gtk_listbox_row_activated_callback
), this); 
 460     // for panel navigation 
 461     g_signal_connect (m_treeview
, "key_press_event", 
 462                       G_CALLBACK (gtk_listbox_key_press_callback
), 
 465     m_parent
->DoAddChild( this ); 
 468     SetInitialSize(size
); // need this too because this is a wxControlWithItems 
 473 wxListBox::~wxListBox() 
 480 // ---------------------------------------------------------------------------- 
 482 // ---------------------------------------------------------------------------- 
 484 void wxListBox::GtkInsertItems(const wxArrayString
& items
, 
 485                                void** clientData
, unsigned int pos
) 
 487     wxCHECK_RET( m_treeview 
!= NULL
, wxT("invalid listbox") ); 
 489     InvalidateBestSize(); 
 491     // Create and set column ids and GValues 
 493     unsigned int nNum 
= items
.GetCount(); 
 494     unsigned int nCurCount 
= wxListBox::GetCount(); 
 495     wxASSERT_MSG(pos 
<= nCurCount
, wxT("Invalid index passed to wxListBox")); 
 497     GtkTreeIter
* pIter 
= NULL
; // append by default 
 499     if (pos 
!= nCurCount
) 
 501         gboolean res 
= gtk_tree_model_iter_nth_child( 
 502                         GTK_TREE_MODEL(m_liststore
), 
 503                         &iter
, NULL
, //NULL = parent = get first 
 507             wxLogSysError(wxT("internal wxListBox error in insertion")); 
 514     for (unsigned int i 
= 0; i 
< nNum
; ++i
) 
 516         wxString label 
= items
[i
]; 
 518         GtkTreeEntry
* entry 
= gtk_tree_entry_new(); 
 519         gtk_tree_entry_set_label(entry
, wxGTK_CONV(label
)); 
 520         gtk_tree_entry_set_destroy_func(entry
, 
 521                 (GtkTreeEntryDestroy
)gtk_tree_entry_destroy_cb
, 
 525             gtk_tree_entry_set_userdata(entry
, clientData
[i
]); 
 528         gtk_list_store_insert_before(m_liststore
, &itercur
, pIter
); 
 530 #if wxUSE_CHECKLISTBOX 
 533             gtk_list_store_set(m_liststore
, &itercur
, 
 534                                  0, FALSE
, //FALSE == not toggled 
 539             gtk_list_store_set(m_liststore
, &itercur
, 
 542         g_object_unref (entry
); //liststore always refs :) 
 546 void wxListBox::DoInsertItems(const wxArrayString
& items
, unsigned int pos
) 
 548     wxCHECK_RET( IsValidInsert(pos
), wxT("invalid index in wxListBox::InsertItems") ); 
 550     GtkInsertItems(items
, NULL
, pos
); 
 553 int wxListBox::DoAppend( const wxString
& item 
) 
 555     // Call DoInsertItems 
 556     unsigned int nWhere 
= wxListBox::GetCount(); 
 557     wxArrayString aItems
; 
 559     wxListBox::DoInsertItems(aItems
, nWhere
); 
 563 void wxListBox::DoSetItems( const wxArrayString
& items
, 
 567     GtkInsertItems(items
, clientData
, 0); 
 570 // ---------------------------------------------------------------------------- 
 572 // ---------------------------------------------------------------------------- 
 574 void wxListBox::Clear() 
 576     wxCHECK_RET( m_treeview 
!= NULL
, wxT("invalid listbox") ); 
 578     InvalidateBestSize(); 
 580     gtk_list_store_clear( m_liststore 
); /* well, THAT was easy :) */ 
 583 void wxListBox::Delete(unsigned int n
) 
 585     wxCHECK_RET( m_treeview 
!= NULL
, wxT("invalid listbox") ); 
 587     InvalidateBestSize(); 
 590     gboolean res 
= gtk_tree_model_iter_nth_child( 
 591                         GTK_TREE_MODEL(m_liststore
), 
 592                         &iter
, NULL
, //NULL = parent = get first 
 596     wxCHECK_RET( res
, wxT("wrong listbox index") ); 
 598     //this returns false if iter is invalid (i.e. deleting item 
 599     //at end) but since we don't use iter, well... :) 
 600     gtk_list_store_remove(m_liststore
, &iter
); 
 603 // ---------------------------------------------------------------------------- 
 604 // get GtkTreeEntry from position (note: you need to g_unref it if valid) 
 605 // ---------------------------------------------------------------------------- 
 607 struct _GtkTreeEntry
* wxListBox::GtkGetEntry(int n
) const 
 610     gboolean res 
= gtk_tree_model_iter_nth_child( 
 611                         GTK_TREE_MODEL(m_liststore
), 
 612                         &iter
, NULL
, //NULL = parent = get first 
 617         wxLogDebug(wxT("gtk_tree_model_iter_nth_child failed\n") 
 618                    wxT("Passed in value was:[%i]  List size:[%u]"), 
 619                    n
, wxListBox::GetCount() ); 
 624     GtkTreeEntry
* entry 
= NULL
; 
 625     gtk_tree_model_get(GTK_TREE_MODEL(m_liststore
), &iter
, 
 626                        WXLISTBOX_DATACOLUMN
, &entry
, -1); 
 631 // ---------------------------------------------------------------------------- 
 633 // ---------------------------------------------------------------------------- 
 635 void* wxListBox::DoGetItemClientData(unsigned int n
) const 
 637     wxCHECK_MSG( IsValid(n
), NULL
, 
 638                  wxT("Invalid index passed to GetItemClientData") ); 
 640     GtkTreeEntry
* entry 
= GtkGetEntry(n
); 
 641     wxCHECK_MSG(entry
, NULL
, wxT("could not get entry")); 
 643     void* userdata 
= gtk_tree_entry_get_userdata( entry 
); 
 644     g_object_unref (entry
); 
 648 wxClientData
* wxListBox::DoGetItemClientObject(unsigned int n
) const 
 650     return (wxClientData
*) wxListBox::DoGetItemClientData(n
); 
 653 void wxListBox::DoSetItemClientData(unsigned int n
, void* clientData
) 
 655     wxCHECK_RET( IsValid(n
), 
 656                  wxT("Invalid index passed to SetItemClientData") ); 
 658     GtkTreeEntry
* entry 
= GtkGetEntry(n
); 
 659     wxCHECK_RET(entry
, wxT("could not get entry")); 
 661     gtk_tree_entry_set_userdata( entry
, clientData 
); 
 662     g_object_unref (entry
); 
 665 void wxListBox::DoSetItemClientObject(unsigned int n
, wxClientData
* clientData
) 
 667     // wxItemContainer already deletes data for us 
 668     wxListBox::DoSetItemClientData(n
, (void*) clientData
); 
 671 // ---------------------------------------------------------------------------- 
 672 // string list access 
 673 // ---------------------------------------------------------------------------- 
 675 void wxListBox::SetString(unsigned int n
, const wxString 
&string
) 
 677     wxCHECK_RET( IsValid(n
), wxT("invalid index in wxListBox::SetString") ); 
 678     wxCHECK_RET( m_treeview 
!= NULL
, wxT("invalid listbox") ); 
 680     GtkTreeEntry
* entry 
= GtkGetEntry(n
); 
 681     wxCHECK_RET( entry
, wxT("wrong listbox index") ); 
 683     wxString label 
= string
; 
 685     // RN: This may look wierd but the problem is that the TreeView 
 686     // doesn't resort or update when changed above and there is no real 
 687     // notification function... 
 688     void* userdata 
= gtk_tree_entry_get_userdata(entry
); 
 689     gtk_tree_entry_set_userdata(entry
, NULL
); //don't delete on destroy 
 690     g_object_unref (entry
); 
 692     bool bWasSelected 
= wxListBox::IsSelected(n
); 
 693     wxListBox::Delete(n
); 
 695     wxArrayString aItems
; 
 697     GtkInsertItems(aItems
, &userdata
, n
); 
 699         wxListBox::GtkSetSelection(n
, true, true); 
 702 wxString 
wxListBox::GetString(unsigned int n
) const 
 704     wxCHECK_MSG( m_treeview 
!= NULL
, wxEmptyString
, wxT("invalid listbox") ); 
 706     GtkTreeEntry
* entry 
= GtkGetEntry(n
); 
 707     wxCHECK_MSG( entry
, wxEmptyString
, wxT("wrong listbox index") ); 
 709     wxString label 
= wxGTK_CONV_BACK( gtk_tree_entry_get_label(entry
) ); 
 711     g_object_unref (entry
); 
 715 unsigned int wxListBox::GetCount() const 
 717     wxCHECK_MSG( m_treeview 
!= NULL
, 0, wxT("invalid listbox") ); 
 719     return (unsigned int)gtk_tree_model_iter_n_children(GTK_TREE_MODEL(m_liststore
), NULL
); 
 722 int wxListBox::FindString( const wxString 
&item
, bool bCase 
) const 
 724     wxCHECK_MSG( m_treeview 
!= NULL
, wxNOT_FOUND
, wxT("invalid listbox") ); 
 726     //Sort of hackish - maybe there is a faster way 
 727     unsigned int nCount 
= wxListBox::GetCount(); 
 729     for(unsigned int i 
= 0; i 
< nCount
; ++i
) 
 731         if( item
.IsSameAs( wxListBox::GetString(i
), bCase 
) ) 
 736     // it's not an error if the string is not found -> no wxCHECK 
 740 // ---------------------------------------------------------------------------- 
 742 // ---------------------------------------------------------------------------- 
 744 int wxListBox::GetSelection() const 
 746     wxCHECK_MSG( m_treeview 
!= NULL
, wxNOT_FOUND
, wxT("invalid listbox")); 
 747     wxCHECK_MSG( HasFlag(wxLB_SINGLE
), wxNOT_FOUND
, 
 748                     wxT("must be single selection listbox")); 
 751     GtkTreeSelection
* selection 
= gtk_tree_view_get_selection(m_treeview
); 
 753     // only works on single-sel 
 754     if (!gtk_tree_selection_get_selected(selection
, NULL
, &iter
)) 
 758         gtk_tree_model_get_path(GTK_TREE_MODEL(m_liststore
), &iter
); 
 760     int sel 
= gtk_tree_path_get_indices(path
)[0]; 
 762     gtk_tree_path_free(path
); 
 767 int wxListBox::GetSelections( wxArrayInt
& aSelections 
) const 
 769     wxCHECK_MSG( m_treeview 
!= NULL
, wxNOT_FOUND
, wxT("invalid listbox") ); 
 775     GtkTreeSelection
* selection 
= gtk_tree_view_get_selection(m_treeview
); 
 777     if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(m_liststore
), &iter
)) 
 778     { //gtk_tree_selection_get_selected_rows is GTK 2.2+ so iter instead 
 781             if (gtk_tree_selection_iter_is_selected(selection
, &iter
)) 
 785         } while(gtk_tree_model_iter_next(GTK_TREE_MODEL(m_liststore
), &iter
)); 
 788     return aSelections
.GetCount(); 
 791 bool wxListBox::IsSelected( int n 
) const 
 793     wxCHECK_MSG( m_treeview 
!= NULL
, false, wxT("invalid listbox") ); 
 795     GtkTreeSelection
* selection 
= gtk_tree_view_get_selection(m_treeview
); 
 798     gboolean res 
= gtk_tree_model_iter_nth_child( 
 799                         GTK_TREE_MODEL(m_liststore
), 
 800                         &iter
, NULL
, //NULL = parent = get first 
 803     wxCHECK_MSG( res
, false, wxT("Invalid index") ); 
 805     return gtk_tree_selection_iter_is_selected(selection
, &iter
); 
 808 void wxListBox::DoSetSelection( int n
, bool select 
) 
 810     // passing -1 to SetSelection() is documented to deselect all items 
 811     if ( n 
== wxNOT_FOUND 
) 
 813         // ... and not generate any events in the process 
 818     wxCHECK_RET( IsValid(n
), wxT("invalid index in wxListBox::SetSelection") ); 
 820     // don't generate the selection event 
 821     GtkSetSelection(n
, select
, true); 
 824 void wxListBox::GtkDeselectAll() 
 826     wxCHECK_RET( m_treeview 
!= NULL
, wxT("invalid listbox") ); 
 828     GtkTreeSelection
* selection 
= gtk_tree_view_get_selection(m_treeview
); 
 832     gtk_tree_selection_unselect_all(selection
); 
 834     m_blockEvent 
= false; 
 837 void wxListBox::GtkSetSelection(int n
, const bool select
, const bool blockEvent
) 
 839     wxCHECK_RET( m_treeview 
!= NULL
, wxT("invalid listbox") ); 
 841     GtkTreeSelection
* selection 
= gtk_tree_view_get_selection(m_treeview
); 
 844     gboolean res 
= gtk_tree_model_iter_nth_child( 
 845                         GTK_TREE_MODEL(m_liststore
), 
 846                         &iter
, NULL
, //NULL = parent = get first 
 849     wxCHECK_RET( res
, wxT("Invalid index") ); 
 851     m_blockEvent 
= blockEvent
; 
 854         gtk_tree_selection_select_iter(selection
, &iter
); 
 856         gtk_tree_selection_unselect_iter(selection
, &iter
); 
 858     m_blockEvent 
= false; 
 861 void wxListBox::DoSetFirstItem( int n 
) 
 863     wxCHECK_RET( m_treeview
, wxT("invalid listbox") ); 
 864     wxCHECK_RET( IsValid(n
), wxT("invalid index")); 
 866     //RN: I have no idea why this line is needed... 
 867     if (gdk_pointer_is_grabbed () && GTK_WIDGET_HAS_GRAB (m_treeview
)) 
 871     gtk_tree_model_iter_nth_child( 
 872                                     GTK_TREE_MODEL(m_liststore
), 
 874                                     NULL
, //NULL = parent = get first 
 878     GtkTreePath
* path 
= gtk_tree_model_get_path( 
 879                             GTK_TREE_MODEL(m_liststore
), &iter
); 
 881     // Scroll to the desired cell (0.0 == topleft alignment) 
 882     gtk_tree_view_scroll_to_cell(m_treeview
, path
, NULL
, 
 885     gtk_tree_path_free(path
); 
 888 // ---------------------------------------------------------------------------- 
 890 // ---------------------------------------------------------------------------- 
 892 int wxListBox::DoListHitTest(const wxPoint
& point
) const 
 894     // gtk_tree_view_get_path_at_pos() also gets items that are not visible and 
 895     // we only want visible items we need to check for it manually here 
 896     if ( !GetClientRect().Contains(point
) ) 
 899     // need to translate from master window since it is in client coords 
 901     gdk_window_get_geometry(gtk_tree_view_get_bin_window(m_treeview
), 
 902                             &binx
, &biny
, NULL
, NULL
, NULL
); 
 905     if ( !gtk_tree_view_get_path_at_pos
 
 911             NULL
,   // [out] column (always 0 here) 
 912             NULL
,   // [out] x-coord relative to the cell (not interested) 
 913             NULL    
// [out] y-coord relative to the cell 
 919     int index 
= gtk_tree_path_get_indices(path
)[0]; 
 920     gtk_tree_path_free(path
); 
 925 // ---------------------------------------------------------------------------- 
 927 // ---------------------------------------------------------------------------- 
 930 void wxListBox::ApplyToolTip( GtkTooltips 
*tips
, const wxChar 
*tip 
) 
 932     // RN: Is this needed anymore? 
 933     gtk_tooltips_set_tip( tips
, GTK_WIDGET( m_treeview 
), wxGTK_CONV(tip
), (gchar
*) NULL 
); 
 935 #endif // wxUSE_TOOLTIPS 
 937 GtkWidget 
*wxListBox::GetConnectWidget() 
 939     // the correct widget for listbox events (such as mouse clicks for example) 
 940     // is m_treeview, not the parent scrolled window 
 941     return GTK_WIDGET(m_treeview
); 
 944 GdkWindow 
*wxListBox::GTKGetWindow(wxArrayGdkWindows
& WXUNUSED(windows
)) const 
 946     return gtk_tree_view_get_bin_window(m_treeview
); 
 949 void wxListBox::DoApplyWidgetStyle(GtkRcStyle 
*style
) 
 951     if (m_hasBgCol 
&& m_backgroundColour
.Ok()) 
 953         GdkWindow 
*window 
= gtk_tree_view_get_bin_window(m_treeview
); 
 956             m_backgroundColour
.CalcPixel( gdk_drawable_get_colormap( window 
) ); 
 957             gdk_window_set_background( window
, m_backgroundColour
.GetColor() ); 
 958             gdk_window_clear( window 
); 
 962     gtk_widget_modify_style( GTK_WIDGET(m_treeview
), style 
); 
 965 wxSize 
wxListBox::DoGetBestSize() const 
 967     wxCHECK_MSG(m_treeview
, wxDefaultSize
, wxT("invalid tree view")); 
 969     // Start with a minimum size that's not too small 
 971     GetTextExtent( wxT("X"), &cx
, &cy
); 
 972     int lbWidth 
= 3 * cx
; 
 975     // Get the visible area of the tree view (limit to the 10th item 
 976     // so that it isn't too big) 
 977     unsigned int count 
= GetCount(); 
 982         // Find the widest line 
 983         for(unsigned int i 
= 0; i 
< count
; i
++) { 
 984             wxString 
str(GetString(i
)); 
 985             GetTextExtent(str
, &wLine
, NULL
); 
 986             lbWidth 
= wxMax(lbWidth
, wLine
); 
 991         // And just a bit more for the checkbox if present and then some 
 992         // (these are rough guesses) 
 993 #if wxUSE_CHECKLISTBOX 
 994         if ( m_hasCheckBoxes 
) 
 997             cy 
= cy 
> 25 ? cy 
: 25; // rough height of checkbox 
1001         // don't make the listbox too tall (limit height to around 10 items) but don't 
1002         // make it too small neither 
1003         lbHeight 
= (cy
+4) * wxMin(wxMax(count
, 3), 10); 
1006     // Add room for the scrollbar 
1007     lbWidth 
+= wxSystemSettings::GetMetric(wxSYS_VSCROLL_X
); 
1009     wxSize 
best(lbWidth
, lbHeight
); 
1010     CacheBestSize(best
); 
1016 wxListBox::GetClassDefaultAttributes(wxWindowVariant 
WXUNUSED(variant
)) 
1018     return GetDefaultAttributesFromGTKWidget(gtk_tree_view_new
, true); 
1021 #endif // wxUSE_LISTBOX