1 /////////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxVListBox is a virtual listbox with lines of variable height 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 2003 Vadim Zeitlin <vadim@wxwidgets.org> 
   9 // Licence:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  15 #include "wx/vscroll.h"         // base class 
  16 #include "wx/bitmap.h" 
  18 class WXDLLIMPEXP_FWD_CORE wxSelectionStore
; 
  20 #define wxVListBoxNameStr _T("wxVListBox") 
  22 // ---------------------------------------------------------------------------- 
  24 // ---------------------------------------------------------------------------- 
  27     This class has two main differences from a regular listbox: it can have an 
  28     arbitrarily huge number of items because it doesn't store them itself but 
  29     uses OnDrawItem() callback to draw them and its items can have variable 
  30     height as determined by OnMeasureItem(). 
  32     It emits the same events as wxListBox and the same event macros may be used 
  35 class WXDLLIMPEXP_CORE wxVListBox 
: public wxVScrolledWindow
 
  38     // constructors and such 
  39     // --------------------- 
  41     // default constructor, you must call Create() later 
  42     wxVListBox() { Init(); } 
  44     // normal constructor which calls Create() internally 
  45     wxVListBox(wxWindow 
*parent
, 
  46                wxWindowID id 
= wxID_ANY
, 
  47                const wxPoint
& pos 
= wxDefaultPosition
, 
  48                const wxSize
& size 
= wxDefaultSize
, 
  50                const wxString
& name 
= wxVListBoxNameStr
) 
  54         (void)Create(parent
, id
, pos
, size
, style
, name
); 
  57     // really creates the control and sets the initial number of items in it 
  58     // (which may be changed later with SetItemCount()) 
  60     // the only special style which may be specified here is wxLB_MULTIPLE 
  62     // returns true on success or false if the control couldn't be created 
  63     bool Create(wxWindow 
*parent
, 
  64                 wxWindowID id 
= wxID_ANY
, 
  65                 const wxPoint
& pos 
= wxDefaultPosition
, 
  66                 const wxSize
& size 
= wxDefaultSize
, 
  68                 const wxString
& name 
= wxVListBoxNameStr
); 
  70     // dtor does some internal cleanup (deletes m_selStore if any) 
  71     virtual ~wxVListBox(); 
  77     // get the number of items in the control 
  78     size_t GetItemCount() const { return GetRowCount(); } 
  80     // does this control use multiple selection? 
  81     bool HasMultipleSelection() const { return m_selStore 
!= NULL
; } 
  83     // get the currently selected item or wxNOT_FOUND if there is no selection 
  85     // this method is only valid for the single selection listboxes 
  86     int GetSelection() const 
  88         wxASSERT_MSG( !HasMultipleSelection(), 
  89                         _T("GetSelection() can't be used with wxLB_MULTIPLE") ); 
  94     // is this item the current one? 
  95     bool IsCurrent(size_t item
) const { return item 
== (size_t)m_current
; } 
  96     #ifdef __WXUNIVERSAL__ 
  97     bool IsCurrent() const { return wxVScrolledWindow::IsCurrent(); } 
 100     // is this item selected? 
 101     bool IsSelected(size_t item
) const; 
 103     // get the number of the selected items (maybe 0) 
 105     // this method is valid for both single and multi selection listboxes 
 106     size_t GetSelectedCount() const; 
 108     // get the first selected item, returns wxNOT_FOUND if none 
 110     // cookie is an opaque parameter which should be passed to 
 111     // GetNextSelected() later 
 113     // this method is only valid for the multi selection listboxes 
 114     int GetFirstSelected(unsigned long& cookie
) const; 
 116     // get next selection item, return wxNOT_FOUND if no more 
 118     // cookie must be the same parameter that was passed to GetFirstSelected() 
 121     // this method is only valid for the multi selection listboxes 
 122     int GetNextSelected(unsigned long& cookie
) const; 
 124     // get the margins around each item 
 125     wxPoint 
GetMargins() const { return m_ptMargins
; } 
 127     // get the background colour of selected cells 
 128     const wxColour
& GetSelectionBackground() const { return m_colBgSel
; } 
 134     // set the number of items to be shown in the control 
 136     // this is just a synonym for wxVScrolledWindow::SetRowCount() 
 137     virtual void SetItemCount(size_t count
); 
 139     // delete all items from the control 
 140     void Clear() { SetItemCount(0); } 
 142     // set the selection to the specified item, if it is wxNOT_FOUND the 
 143     // selection is unset 
 145     // this function is only valid for the single selection listboxes 
 146     void SetSelection(int selection
); 
 148     // selects or deselects the specified item which must be valid (i.e. not 
 149     // equal to wxNOT_FOUND) 
 151     // return true if the items selection status has changed or false 
 154     // this function is only valid for the multiple selection listboxes 
 155     bool Select(size_t item
, bool select 
= true); 
 157     // selects the items in the specified range whose end points may be given 
 160     // return true if any items selection status has changed, false otherwise 
 162     // this function is only valid for the single selection listboxes 
 163     bool SelectRange(size_t from
, size_t to
); 
 165     // toggle the selection of the specified item (must be valid) 
 167     // this function is only valid for the multiple selection listboxes 
 168     void Toggle(size_t item
) { Select(item
, !IsSelected(item
)); } 
 170     // select all items in the listbox 
 172     // the return code indicates if any items were affected by this operation 
 173     // (true) or if nothing has changed (false) 
 174     bool SelectAll() { return DoSelectAll(true); } 
 176     // unselect all items in the listbox 
 178     // the return code has the same meaning as for SelectAll() 
 179     bool DeselectAll() { return DoSelectAll(false); } 
 181     // set the margins: horizontal margin is the distance between the window 
 182     // border and the item contents while vertical margin is half of the 
 183     // distance between items 
 185     // by default both margins are 0 
 186     void SetMargins(const wxPoint
& pt
); 
 187     void SetMargins(wxCoord x
, wxCoord y
) { SetMargins(wxPoint(x
, y
)); } 
 189     // change the background colour of the selected cells 
 190     void SetSelectionBackground(const wxColour
& col
); 
 192     // refreshes only the selected items 
 193     void RefreshSelected(); 
 196     virtual wxVisualAttributes 
GetDefaultAttributes() const 
 198         return GetClassDefaultAttributes(GetWindowVariant()); 
 201     static wxVisualAttributes
 
 202     GetClassDefaultAttributes(wxWindowVariant variant 
= wxWINDOW_VARIANT_NORMAL
); 
 205     virtual wxBorder 
GetDefaultBorder() const { return wxBORDER_THEME
; } 
 207     // the derived class must implement this function to actually draw the item 
 208     // with the given index on the provided DC 
 209     virtual void OnDrawItem(wxDC
& dc
, const wxRect
& rect
, size_t n
) const = 0; 
 211     // the derived class must implement this method to return the height of the 
 213     virtual wxCoord 
OnMeasureItem(size_t n
) const = 0; 
 215     // this method may be used to draw separators between the lines; note that 
 216     // the rectangle may be modified, typically to deflate it a bit before 
 217     // passing to OnDrawItem() 
 219     // the base class version doesn't do anything 
 220     virtual void OnDrawSeparator(wxDC
& dc
, wxRect
& rect
, size_t n
) const; 
 222     // this method is used to draw the items background and, maybe, a border 
 225     // the base class version implements a reasonable default behaviour which 
 226     // consists in drawing the selected item with the standard background 
 227     // colour and drawing a border around the item if it is either selected or 
 229     virtual void OnDrawBackground(wxDC
& dc
, const wxRect
& rect
, size_t n
) const; 
 231     // we implement OnGetRowHeight() in terms of OnMeasureItem() because this 
 232     // allows us to add borders to the items easily 
 234     // this function is not supposed to be overridden by the derived classes 
 235     virtual wxCoord 
OnGetRowHeight(size_t line
) const; 
 239     void OnPaint(wxPaintEvent
& event
); 
 240     void OnKeyDown(wxKeyEvent
& event
); 
 241     void OnLeftDown(wxMouseEvent
& event
); 
 242     void OnLeftDClick(wxMouseEvent
& event
); 
 243     void OnSetOrKillFocus(wxFocusEvent
& event
); 
 245     // common part of all ctors 
 248     // send the wxEVT_COMMAND_LISTBOX_SELECTED event 
 249     void SendSelectedEvent(); 
 251     // common implementation of SelectAll() and DeselectAll() 
 252     bool DoSelectAll(bool select
); 
 254     // change the current item (in single selection listbox it also implicitly 
 255     // changes the selection); current may be wxNOT_FOUND in which case there 
 256     // will be no current item any more 
 258     // return true if the current item changed, false otherwise 
 259     bool DoSetCurrent(int current
); 
 261     // flags for DoHandleItemClick 
 264         ItemClick_Shift 
= 1,        // item shift-clicked 
 265         ItemClick_Ctrl  
= 2,        //       ctrl 
 266         ItemClick_Kbd   
= 4         // item selected from keyboard 
 269     // common part of keyboard and mouse handling processing code 
 270     void DoHandleItemClick(int item
, int flags
); 
 272     // paint the background of the given item using the provided colour if it's 
 273     // valid, otherwise just return false and do nothing (this is used by 
 274     // OnDrawBackground()) 
 275     bool DoDrawSolidBackground(const wxColour
& col
, 
 281     // the current item or wxNOT_FOUND 
 283     // if m_selStore == NULL this is also the selected item, otherwise the 
 284     // selections are managed by m_selStore 
 287     // the anchor of the selection for the multiselection listboxes: 
 288     // shift-clicking an item extends the selection from m_anchor to the item 
 289     // clicked, for example 
 291     // always wxNOT_FOUND for single selection listboxes 
 294     // the object managing our selected items if not NULL 
 295     wxSelectionStore 
*m_selStore
; 
 300     // the selection bg colour 
 303     DECLARE_EVENT_TABLE() 
 304     DECLARE_NO_COPY_CLASS(wxVListBox
) 
 305     DECLARE_ABSTRACT_CLASS(wxVListBox
) 
 308 #endif // _WX_VLBOX_H_