1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/generic/vlbox.cpp 
   3 // Purpose:     implementation of wxVListBox 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 2003 Vadim Zeitlin <vadim@wxwindows.org> 
   9 // License:     wxWindows license 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  32     #include "wx/settings.h" 
  33     #include "wx/dcclient.h" 
  34     #include "wx/listbox.h" 
  37 #include "wx/dcbuffer.h" 
  38 #include "wx/selstore.h" 
  39 #include "wx/renderer.h" 
  41 // ---------------------------------------------------------------------------- 
  43 // ---------------------------------------------------------------------------- 
  45 BEGIN_EVENT_TABLE(wxVListBox
, wxVScrolledWindow
) 
  46     EVT_PAINT(wxVListBox::OnPaint
) 
  48     EVT_KEY_DOWN(wxVListBox::OnKeyDown
) 
  49     EVT_LEFT_DOWN(wxVListBox::OnLeftDown
) 
  50     EVT_LEFT_DCLICK(wxVListBox::OnLeftDClick
) 
  52     EVT_SET_FOCUS(wxVListBox::OnSetOrKillFocus
) 
  53     EVT_KILL_FOCUS(wxVListBox::OnSetOrKillFocus
) 
  55     EVT_SIZE(wxVListBox::OnSize
) 
  58 // ============================================================================ 
  60 // ============================================================================ 
  62 IMPLEMENT_ABSTRACT_CLASS(wxVListBox
, wxVScrolledWindow
) 
  63 const char wxVListBoxNameStr
[] = "wxVListBox"; 
  65 // ---------------------------------------------------------------------------- 
  66 // wxVListBox creation 
  67 // ---------------------------------------------------------------------------- 
  69 void wxVListBox::Init() 
  72     m_anchor 
= wxNOT_FOUND
; 
  76 bool wxVListBox::Create(wxWindow 
*parent
, 
  84     if ( (style 
& wxBORDER_MASK
) == wxDEFAULT 
) 
  85         style 
|= wxBORDER_THEME
; 
  88     style 
|= wxWANTS_CHARS 
| wxFULL_REPAINT_ON_RESIZE
; 
  89     if ( !wxVScrolledWindow::Create(parent
, id
, pos
, size
, style
, name
) ) 
  92     if ( style 
& wxLB_MULTIPLE 
) 
  93         m_selStore 
= new wxSelectionStore
; 
  95     // make sure the native widget has the right colour since we do 
  96     // transparent drawing by default 
  97     SetBackgroundColour(GetBackgroundColour()); 
  99     // leave m_colBgSel in an invalid state: it means for OnDrawBackground() 
 100     // to use wxRendererNative instead of painting selection bg ourselves 
 101     m_colBgSel 
= wxNullColour
; 
 103     // flicker-free drawing requires this 
 104     SetBackgroundStyle(wxBG_STYLE_CUSTOM
); 
 109 wxVListBox::~wxVListBox() 
 114 void wxVListBox::SetItemCount(size_t count
) 
 116     // don't leave the current index invalid 
 117     if ( m_current 
!= wxNOT_FOUND 
&& (size_t)m_current 
>= count 
) 
 118         m_current 
= count 
- 1; // also ok when count == 0 as wxNOT_FOUND == -1 
 122         // tell the selection store that our number of items has changed 
 123         m_selStore
->SetItemCount(count
); 
 129 // ---------------------------------------------------------------------------- 
 130 // selection handling 
 131 // ---------------------------------------------------------------------------- 
 133 bool wxVListBox::IsSelected(size_t line
) const 
 135     return m_selStore 
? m_selStore
->IsSelected(line
) : (int)line 
== m_current
; 
 138 bool wxVListBox::Select(size_t item
, bool select
) 
 140     wxCHECK_MSG( m_selStore
, false, 
 141                  wxT("Select() may only be used with multiselection listbox") ); 
 143     wxCHECK_MSG( item 
< GetItemCount(), false, 
 144                  wxT("Select(): invalid item index") ); 
 146     bool changed 
= m_selStore
->SelectItem(item
, select
); 
 149         // selection really changed 
 158 bool wxVListBox::SelectRange(size_t from
, size_t to
) 
 160     wxCHECK_MSG( m_selStore
, false, 
 161                  wxT("SelectRange() may only be used with multiselection listbox") ); 
 163     // make sure items are in correct order 
 171     wxCHECK_MSG( to 
< GetItemCount(), false, 
 172                     wxT("SelectRange(): invalid item index") ); 
 175     if ( !m_selStore
->SelectRange(from
, to
, true, &changed
) ) 
 177         // too many items have changed, we didn't record them in changed array 
 178         // so we have no choice but to refresh everything between from and to 
 179         RefreshRows(from
, to
); 
 181     else // we've got the indices of the changed items 
 183         const size_t count 
= changed
.GetCount(); 
 190         // refresh just the lines which have really changed 
 191         for ( size_t n 
= 0; n 
< count
; n
++ ) 
 193             RefreshRow(changed
[n
]); 
 201 bool wxVListBox::DoSelectAll(bool select
) 
 203     wxCHECK_MSG( m_selStore
, false, 
 204                  wxT("SelectAll may only be used with multiselection listbox") ); 
 206     size_t count 
= GetItemCount(); 
 210         if ( !m_selStore
->SelectRange(0, count 
- 1, select
) || 
 223 bool wxVListBox::DoSetCurrent(int current
) 
 225     wxASSERT_MSG( current 
== wxNOT_FOUND 
|| 
 226                     (current 
>= 0 && (size_t)current 
< GetItemCount()), 
 227                   wxT("wxVListBox::DoSetCurrent(): invalid item index") ); 
 229     if ( current 
== m_current 
) 
 235     if ( m_current 
!= wxNOT_FOUND 
) 
 236         RefreshRow(m_current
); 
 240     if ( m_current 
!= wxNOT_FOUND 
) 
 242         // if the line is not visible at all, we scroll it into view but we 
 243         // don't need to refresh it -- it will be redrawn anyhow 
 244         if ( !IsVisible(m_current
) ) 
 246             ScrollToRow(m_current
); 
 248         else // line is at least partly visible 
 250             // it is, indeed, only partly visible, so scroll it into view to 
 251             // make it entirely visible 
 252             while ( (size_t)m_current 
+ 1 == GetVisibleRowsEnd() && 
 253                     ScrollToRow(GetVisibleBegin() + 1) ) ; 
 255             // but in any case refresh it as even if it was only partly visible 
 256             // before we need to redraw it entirely as its background changed 
 257             RefreshRow(m_current
); 
 264 void wxVListBox::InitEvent(wxCommandEvent
& event
, int n
) 
 266     event
.SetEventObject(this); 
 270 void wxVListBox::SendSelectedEvent() 
 272     wxASSERT_MSG( m_current 
!= wxNOT_FOUND
, 
 273                     wxT("SendSelectedEvent() shouldn't be called") ); 
 275     wxCommandEvent 
event(wxEVT_COMMAND_LISTBOX_SELECTED
, GetId()); 
 276     InitEvent(event
, m_current
); 
 277     (void)GetEventHandler()->ProcessEvent(event
); 
 280 void wxVListBox::SetSelection(int selection
) 
 282     wxCHECK_RET( selection 
== wxNOT_FOUND 
|| 
 283                   (selection 
>= 0 && (size_t)selection 
< GetItemCount()), 
 284                   wxT("wxVListBox::SetSelection(): invalid item index") ); 
 286     if ( HasMultipleSelection() ) 
 288         if (selection 
!= wxNOT_FOUND
) 
 292         m_anchor 
= selection
; 
 295     DoSetCurrent(selection
); 
 298 size_t wxVListBox::GetSelectedCount() const 
 300     return m_selStore 
? m_selStore
->GetSelectedCount() 
 301                       : m_current 
== wxNOT_FOUND 
? 0 : 1; 
 304 int wxVListBox::GetFirstSelected(unsigned long& cookie
) const 
 308     return GetNextSelected(cookie
); 
 311 int wxVListBox::GetNextSelected(unsigned long& cookie
) const 
 313     wxCHECK_MSG( m_selStore
, wxNOT_FOUND
, 
 314                   wxT("GetFirst/NextSelected() may only be used with multiselection listboxes") ); 
 316     while ( cookie 
< GetItemCount() ) 
 318         if ( IsSelected(cookie
++) ) 
 325 void wxVListBox::RefreshSelected() 
 327     // only refresh those items which are currently visible and selected: 
 328     for ( size_t n 
= GetVisibleBegin(), end 
= GetVisibleEnd(); n 
< end
; n
++ ) 
 335 wxRect 
wxVListBox::GetItemRect(size_t n
) const 
 339     // check that this item is visible 
 340     const size_t lineMax 
= GetVisibleEnd(); 
 343     size_t line 
= GetVisibleBegin(); 
 349         itemrect
.y 
+= itemrect
.height
; 
 350         itemrect
.height 
= OnGetRowHeight(line
); 
 355     itemrect
.width 
= GetClientSize().x
; 
 360 // ---------------------------------------------------------------------------- 
 361 // wxVListBox appearance parameters 
 362 // ---------------------------------------------------------------------------- 
 364 void wxVListBox::SetMargins(const wxPoint
& pt
) 
 366     if ( pt 
!= m_ptMargins 
) 
 374 void wxVListBox::SetSelectionBackground(const wxColour
& col
) 
 379 // ---------------------------------------------------------------------------- 
 380 // wxVListBox painting 
 381 // ---------------------------------------------------------------------------- 
 383 wxCoord 
wxVListBox::OnGetRowHeight(size_t line
) const 
 385     return OnMeasureItem(line
) + 2*m_ptMargins
.y
; 
 388 void wxVListBox::OnDrawSeparator(wxDC
& WXUNUSED(dc
), 
 389                                  wxRect
& WXUNUSED(rect
), 
 390                                  size_t WXUNUSED(n
)) const 
 395 wxVListBox::DoDrawSolidBackground(const wxColour
& col
, 
 403     // we need to render selected and current items differently 
 404     const bool isSelected 
= IsSelected(n
), 
 405                isCurrent 
= IsCurrent(n
); 
 406     if ( isSelected 
|| isCurrent 
) 
 410             dc
.SetBrush(wxBrush(col
, wxBRUSHSTYLE_SOLID
)); 
 414             dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
 416         dc
.SetPen(*(isCurrent 
? wxBLACK_PEN 
: wxTRANSPARENT_PEN
)); 
 417         dc
.DrawRectangle(rect
); 
 419     //else: do nothing for the normal items 
 424 void wxVListBox::OnDrawBackground(wxDC
& dc
, const wxRect
& rect
, size_t n
) const 
 426     // use wxRendererNative for more native look unless we use custom bg colour 
 427     if ( !DoDrawSolidBackground(m_colBgSel
, dc
, rect
, n
) ) 
 431             flags 
|= wxCONTROL_SELECTED
; 
 433             flags 
|= wxCONTROL_CURRENT
; 
 434         if ( wxWindow::FindFocus() == const_cast<wxVListBox
*>(this) ) 
 435             flags 
|= wxCONTROL_FOCUSED
; 
 437         wxRendererNative::Get().DrawItemSelectionRect( 
 438             const_cast<wxVListBox 
*>(this), dc
, rect
, flags
); 
 442 void wxVListBox::OnPaint(wxPaintEvent
& WXUNUSED(event
)) 
 444     wxSize clientSize 
= GetClientSize(); 
 446     wxAutoBufferedPaintDC 
dc(this); 
 448     // the update rectangle 
 449     wxRect rectUpdate 
= GetUpdateClientRect(); 
 451     // fill it with background colour 
 452     dc
.SetBackground(GetBackgroundColour()); 
 455     // the bounding rectangle of the current line 
 457     rectRow
.width 
= clientSize
.x
; 
 459     // iterate over all visible lines 
 460     const size_t lineMax 
= GetVisibleEnd(); 
 461     for ( size_t line 
= GetVisibleBegin(); line 
< lineMax
; line
++ ) 
 463         const wxCoord hRow 
= OnGetRowHeight(line
); 
 465         rectRow
.height 
= hRow
; 
 467         // and draw the ones which intersect the update rect 
 468         if ( rectRow
.Intersects(rectUpdate
) ) 
 470             // don't allow drawing outside of the lines rectangle 
 471             wxDCClipper 
clip(dc
, rectRow
); 
 473             wxRect rect 
= rectRow
; 
 474             OnDrawBackground(dc
, rect
, line
); 
 476             OnDrawSeparator(dc
, rect
, line
); 
 478             rect
.Deflate(m_ptMargins
.x
, m_ptMargins
.y
); 
 479             OnDrawItem(dc
, rect
, line
); 
 481         else // no intersection 
 483             if ( rectRow
.GetTop() > rectUpdate
.GetBottom() ) 
 485                 // we are already below the update rect, no need to continue 
 489             //else: the next line may intersect the update rect 
 496 void wxVListBox::OnSetOrKillFocus(wxFocusEvent
& WXUNUSED(event
)) 
 498     // we need to repaint the selection when we get the focus since 
 499     // wxRendererNative in general draws the focused selection differently 
 500     // from the unfocused selection (see OnDrawItem): 
 504 void wxVListBox::OnSize(wxSizeEvent
& event
) 
 510 // ============================================================================ 
 511 // wxVListBox keyboard/mouse handling 
 512 // ============================================================================ 
 514 void wxVListBox::DoHandleItemClick(int item
, int flags
) 
 516     // has anything worth telling the client code about happened? 
 519     if ( HasMultipleSelection() ) 
 521         // select the iteem clicked? 
 524         // NB: the keyboard interface we implement here corresponds to 
 525         //     wxLB_EXTENDED rather than wxLB_MULTIPLE but this one makes more 
 527         if ( flags 
& ItemClick_Shift 
) 
 529             if ( m_current 
!= wxNOT_FOUND 
) 
 531                 if ( m_anchor 
== wxNOT_FOUND 
) 
 532                     m_anchor 
= m_current
; 
 536                 // only the range from the selection anchor to new m_current 
 541                 if ( SelectRange(m_anchor
, item
) ) 
 544             //else: treat it as ordinary click/keypress 
 546         else // Shift not pressed 
 550             if ( flags 
& ItemClick_Ctrl 
) 
 554                 if ( !(flags 
& ItemClick_Kbd
) ) 
 558                     // the status of the item has definitely changed 
 561                 //else: Ctrl-arrow pressed, don't change selection 
 563             //else: behave as in single selection case 
 568             // make the clicked item the only selection 
 577     // in any case the item should become the current one 
 578     if ( DoSetCurrent(item
) ) 
 580         if ( !HasMultipleSelection() ) 
 582             // this has also changed the selection for single selection case 
 589         // notify the user about the selection change 
 592     //else: nothing changed at all 
 595 // ---------------------------------------------------------------------------- 
 597 // ---------------------------------------------------------------------------- 
 599 void wxVListBox::OnKeyDown(wxKeyEvent
& event
) 
 601     // flags for DoHandleItemClick() 
 602     int flags 
= ItemClick_Kbd
; 
 605     switch ( event
.GetKeyCode() ) 
 608         case WXK_NUMPAD_HOME
: 
 614             current 
= GetRowCount() - 1; 
 618         case WXK_NUMPAD_DOWN
: 
 619             if ( m_current 
== (int)GetRowCount() - 1 ) 
 622             current 
= m_current 
+ 1; 
 627             if ( m_current 
== wxNOT_FOUND 
) 
 628                 current 
= GetRowCount() - 1; 
 629             else if ( m_current 
!= 0 ) 
 630                 current 
= m_current 
- 1; 
 631             else // m_current == 0 
 636         case WXK_NUMPAD_PAGEDOWN
: 
 638             current 
= GetVisibleBegin(); 
 642         case WXK_NUMPAD_PAGEUP
: 
 643             if ( m_current 
== (int)GetVisibleBegin() ) 
 648             current 
= GetVisibleBegin(); 
 652             // hack: pressing space should work like a mouse click rather than 
 653             // like a keyboard arrow press, so trick DoHandleItemClick() in 
 654             // thinking we were clicked 
 655             flags 
&= ~ItemClick_Kbd
; 
 661             // Since we are using wxWANTS_CHARS we need to send navigation 
 662             // events for the tabs on MSW 
 663             HandleAsNavigationKey(event
); 
 664             // fall through to default 
 668             current 
= 0; // just to silent the stupid compiler warnings 
 669             wxUnusedVar(current
); 
 673     if ( event
.ShiftDown() ) 
 674        flags 
|= ItemClick_Shift
; 
 675     if ( event
.ControlDown() ) 
 676         flags 
|= ItemClick_Ctrl
; 
 678     DoHandleItemClick(current
, flags
); 
 681 // ---------------------------------------------------------------------------- 
 682 // wxVListBox mouse handling 
 683 // ---------------------------------------------------------------------------- 
 685 void wxVListBox::OnLeftDown(wxMouseEvent
& event
) 
 689     int item 
= VirtualHitTest(event
.GetPosition().y
); 
 691     if ( item 
!= wxNOT_FOUND 
) 
 694         if ( event
.ShiftDown() ) 
 695            flags 
|= ItemClick_Shift
; 
 697         // under Mac Apple-click is used in the same way as Ctrl-click 
 700         if ( event
.MetaDown() ) 
 702         if ( event
.ControlDown() ) 
 704             flags 
|= ItemClick_Ctrl
; 
 706         DoHandleItemClick(item
, flags
); 
 710 void wxVListBox::OnLeftDClick(wxMouseEvent
& eventMouse
) 
 712     int item 
= VirtualHitTest(eventMouse
.GetPosition().y
); 
 713     if ( item 
!= wxNOT_FOUND 
) 
 716         // if item double-clicked was not yet selected, then treat 
 717         // this event as a left-click instead 
 718         if ( item 
== m_current 
) 
 720             wxCommandEvent 
event(wxEVT_COMMAND_LISTBOX_DOUBLECLICKED
, GetId()); 
 721             InitEvent(event
, item
); 
 722             (void)GetEventHandler()->ProcessEvent(event
); 
 726             OnLeftDown(eventMouse
); 
 733 // ---------------------------------------------------------------------------- 
 734 // use the same default attributes as wxListBox 
 735 // ---------------------------------------------------------------------------- 
 739 wxVListBox::GetClassDefaultAttributes(wxWindowVariant variant
) 
 741     return wxListBox::GetClassDefaultAttributes(variant
);