1 /////////////////////////////////////////////////////////////////////////////
2 // Name: univ/listbox.cpp
3 // Purpose: wxListBox implementation
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 2000 SciTech Software, Inc. (www.scitechsoft.com)
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "univlistbox.h"
24 #include "wx/wxprec.h"
35 #include "wx/dcclient.h"
36 #include "wx/listbox.h"
37 #include "wx/validate.h"
40 #include "wx/univ/renderer.h"
41 #include "wx/univ/inphand.h"
42 #include "wx/univ/theme.h"
44 // ============================================================================
45 // implementation of wxListBox
46 // ============================================================================
48 IMPLEMENT_DYNAMIC_CLASS(wxListBox
, wxControl
)
50 BEGIN_EVENT_TABLE(wxListBox
, wxListBoxBase
)
51 EVT_SIZE(wxListBox::OnSize
)
53 EVT_IDLE(wxListBox::OnIdle
)
56 // ----------------------------------------------------------------------------
58 // ----------------------------------------------------------------------------
60 void wxListBox::Init()
62 // will be calculated later when needed
69 // no items hence no current item
72 m_currentChanged
= FALSE
;
74 // no need to update anything initially
77 // no scrollbars to show nor update
81 m_showScrollbarY
= FALSE
;
84 bool wxListBox::Create(wxWindow
*parent
,
89 const wxString choices
[],
91 const wxValidator
& validator
,
94 // for compatibility accept both the new and old styles - they mean the
96 if ( style
& wxLB_ALWAYS_SB
)
97 style
|= wxALWAYS_SHOW_SB
;
99 // if we don't have neither multiple nor extended flag, we must have the
100 // single selection listbox
101 if ( !(style
& (wxLB_MULTIPLE
| wxLB_EXTENDED
)) )
102 style
|= wxLB_SINGLE
;
104 #if wxUSE_TWO_WINDOWS
105 style
|= wxVSCROLL
|wxHSCROLL
;
106 if ((style
& wxBORDER_MASK
) == 0)
107 style
|= wxBORDER_SUNKEN
;
110 if ( !wxControl::Create(parent
, id
, pos
, size
, style
,
111 wxDefaultValidator
, name
) )
116 if ( style
& wxLB_SORT
)
117 m_strings
= wxArrayString(TRUE
/* auto sort */);
123 CreateInputHandler(wxINP_HANDLER_LISTBOX
);
128 wxListBox::~wxListBox()
130 // call this just to free the client data -- and avoid leaking memory
134 // ----------------------------------------------------------------------------
135 // adding/inserting strings
136 // ----------------------------------------------------------------------------
138 int wxListBox::DoAppend(const wxString
& item
)
140 size_t index
= m_strings
.Add(item
);
141 m_itemsClientData
.Insert(NULL
, index
);
143 m_updateScrollbarY
= TRUE
;
145 if ( HasHorzScrollbar() )
147 // has the max width increased?
149 GetTextExtent(item
, &width
, NULL
);
150 if ( width
> m_maxWidth
)
153 m_maxWidthItem
= index
;
154 m_updateScrollbarX
= TRUE
;
158 RefreshFromItemToEnd(index
);
163 void wxListBox::DoInsertItems(const wxArrayString
& items
, int pos
)
165 // the position of the item being added to a sorted listbox can't be
167 wxCHECK_RET( !IsSorted(), _T("can't insert items into sorted listbox") );
169 size_t count
= items
.GetCount();
170 for ( size_t n
= 0; n
< count
; n
++ )
172 m_strings
.Insert(items
[n
], pos
+ n
);
173 m_itemsClientData
.Insert(NULL
, pos
+ n
);
176 // the number of items has changed so we might have to show the scrollbar
177 m_updateScrollbarY
= TRUE
;
179 // the max width also might have changed - just recalculate it instead of
180 // keeping track of it here, this is probably more efficient for a typical
182 RefreshHorzScrollbar();
184 // note that we have to refresh all the items after the ones we inserted,
185 // not just these items
186 RefreshFromItemToEnd(pos
);
189 void wxListBox::DoSetItems(const wxArrayString
& items
, void **clientData
)
193 size_t count
= items
.GetCount();
197 m_strings
.Alloc(count
);
198 m_itemsClientData
.Alloc(count
);
199 for ( size_t n
= 0; n
< count
; n
++ )
201 size_t index
= m_strings
.Add(items
[n
]);
202 m_itemsClientData
.Insert(clientData
? clientData
[n
] : NULL
, index
);
205 m_updateScrollbarY
= TRUE
;
210 void wxListBox::SetString(int n
, const wxString
& s
)
212 if ( HasHorzScrollbar() )
214 // we need to update m_maxWidth as changing the string may cause the
215 // horz scrollbar [dis]appear
218 GetTextExtent(s
, &width
, NULL
);
220 // it might have increased if the new string is long
221 if ( width
> m_maxWidth
)
225 m_updateScrollbarX
= TRUE
;
227 // or also decreased if the old string was the longest one
228 else if ( n
== m_maxWidthItem
)
230 RefreshHorzScrollbar();
233 else // no horz scrollbar
241 // ----------------------------------------------------------------------------
243 // ----------------------------------------------------------------------------
245 void wxListBox::DoClear()
249 if ( HasClientObjectData() )
251 size_t count
= m_itemsClientData
.GetCount();
252 for ( size_t n
= 0; n
< count
; n
++ )
254 delete (wxClientData
*) m_itemsClientData
[n
];
258 m_itemsClientData
.Clear();
259 m_selections
.Clear();
264 void wxListBox::Clear()
268 m_updateScrollbarY
= TRUE
;
270 RefreshHorzScrollbar();
275 void wxListBox::Delete(int n
)
277 wxCHECK_RET( n
< GetCount(), _T("invalid index in wxListBox::Delete") );
279 // do it before removing the index as otherwise the last item will not be
280 // refreshed (as GetCount() will be decremented)
281 RefreshFromItemToEnd(n
);
283 m_strings
.RemoveAt(n
);
285 if ( HasClientObjectData() )
287 delete (wxClientData
*)m_itemsClientData
[n
];
290 m_itemsClientData
.RemoveAt(n
);
292 // when the item disappears we must not keep using its index
293 if ( n
== m_current
)
297 else if ( n
< m_current
)
301 //else: current item may stay
303 // update the selections array: the indices of all seletected items after
304 // the one being deleted must change and the item itselfm ust be removed
305 int index
= wxNOT_FOUND
;
306 size_t count
= m_selections
.GetCount();
307 for ( size_t item
= 0; item
< count
; item
++ )
309 if ( m_selections
[item
] == n
)
311 // remember to delete it later
314 else if ( m_selections
[item
] > n
)
316 // to account for the index shift
317 m_selections
[item
]--;
319 //else: nothing changed for this one
322 if ( index
!= wxNOT_FOUND
)
324 m_selections
.RemoveAt(index
);
327 // the number of items has changed, hence the scrollbar may disappear
328 m_updateScrollbarY
= TRUE
;
330 // finally, if the longest item was deleted the scrollbar may disappear
331 if ( n
== m_maxWidthItem
)
333 RefreshHorzScrollbar();
337 // ----------------------------------------------------------------------------
338 // client data handling
339 // ----------------------------------------------------------------------------
341 void wxListBox::DoSetItemClientData(int n
, void* clientData
)
343 m_itemsClientData
[n
] = clientData
;
346 void *wxListBox::DoGetItemClientData(int n
) const
348 return m_itemsClientData
[n
];
351 void wxListBox::DoSetItemClientObject(int n
, wxClientData
* clientData
)
353 m_itemsClientData
[n
] = clientData
;
356 wxClientData
* wxListBox::DoGetItemClientObject(int n
) const
358 return (wxClientData
*)m_itemsClientData
[n
];
361 // ----------------------------------------------------------------------------
363 // ----------------------------------------------------------------------------
365 void wxListBox::SetSelection(int n
, bool select
)
369 if ( m_selections
.Index(n
) == wxNOT_FOUND
)
371 if ( !HasMultipleSelection() )
373 // selecting an item in a single selection listbox deselects
382 //else: already selected
386 int index
= m_selections
.Index(n
);
387 if ( index
!= wxNOT_FOUND
)
389 m_selections
.RemoveAt(index
);
396 // sanity check: a single selection listbox can't have more than one item
398 wxASSERT_MSG( HasMultipleSelection() || (m_selections
.GetCount() < 2),
399 _T("multiple selected items in single selection lbox?") );
403 // the newly selected item becomes the current one
408 int wxListBox::GetSelection() const
410 wxCHECK_MSG( !HasMultipleSelection(), -1,
411 _T("use wxListBox::GetSelections for ths listbox") );
413 return m_selections
.IsEmpty() ? -1 : m_selections
[0];
416 int wxCMPFUNC_CONV
wxCompareInts(int *n
, int *m
)
421 int wxListBox::GetSelections(wxArrayInt
& selections
) const
423 // always return sorted array to the user
424 selections
= m_selections
;
425 size_t count
= m_selections
.GetCount();
427 // don't call sort on an empty array
430 selections
.Sort(wxCompareInts
);
436 // ----------------------------------------------------------------------------
437 // refresh logic: we use delayed refreshing which allows to avoid multiple
438 // refreshes (and hence flicker) in case when several listbox items are
439 // added/deleted/changed subsequently
440 // ----------------------------------------------------------------------------
442 void wxListBox::RefreshFromItemToEnd(int from
)
444 RefreshItems(from
, GetCount() - from
);
447 void wxListBox::RefreshItems(int from
, int count
)
449 switch ( m_updateCount
)
453 m_updateCount
= count
;
457 // we refresh everything anyhow
461 // add these items to the others which we have to refresh
462 if ( m_updateFrom
< from
)
464 count
+= from
- m_updateFrom
;
465 if ( m_updateCount
< count
)
466 m_updateCount
= count
;
468 else // m_updateFrom >= from
470 int updateLast
= wxMax(m_updateFrom
+ m_updateCount
,
473 m_updateCount
= updateLast
- m_updateFrom
;
478 void wxListBox::RefreshItem(int n
)
480 switch ( m_updateCount
)
483 // refresh this item only
489 // we refresh everything anyhow
493 // add this item to the others which we have to refresh
494 if ( m_updateFrom
< n
)
496 if ( m_updateCount
< n
- m_updateFrom
+ 1 )
497 m_updateCount
= n
- m_updateFrom
+ 1;
499 else // n <= m_updateFrom
501 m_updateCount
+= m_updateFrom
- n
;
507 void wxListBox::RefreshAll()
512 void wxListBox::RefreshHorzScrollbar()
514 m_maxWidth
= 0; // recalculate it
515 m_updateScrollbarX
= TRUE
;
518 void wxListBox::UpdateScrollbars()
520 wxSize size
= GetClientSize();
522 // is our height enough to show all items?
523 int nLines
= GetCount();
524 wxCoord lineHeight
= GetLineHeight();
525 bool showScrollbarY
= nLines
*lineHeight
> size
.y
;
527 // check the width too if required
528 wxCoord charWidth
, maxWidth
;
530 if ( HasHorzScrollbar() )
532 charWidth
= GetCharWidth();
533 maxWidth
= GetMaxWidth();
534 showScrollbarX
= maxWidth
> size
.x
;
536 else // never show it
538 charWidth
= maxWidth
= 0;
539 showScrollbarX
= FALSE
;
542 // what should be the scrollbar range now?
543 int scrollRangeX
= showScrollbarX
544 ? (maxWidth
+ charWidth
- 1) / charWidth
+ 2 // FIXME
546 int scrollRangeY
= showScrollbarY
548 (size
.y
% lineHeight
+ lineHeight
- 1) / lineHeight
551 // reset scrollbars if something changed: either the visibility status
552 // or the range of a scrollbar which is shown
553 if ( (showScrollbarY
!= m_showScrollbarY
) ||
554 (showScrollbarX
!= m_showScrollbarX
) ||
555 (showScrollbarY
&& (scrollRangeY
!= m_scrollRangeY
)) ||
556 (showScrollbarX
&& (scrollRangeX
!= m_scrollRangeX
)) )
559 GetViewStart(&x
, &y
);
560 SetScrollbars(charWidth
, lineHeight
,
561 scrollRangeX
, scrollRangeY
,
564 m_showScrollbarX
= showScrollbarX
;
565 m_showScrollbarY
= showScrollbarY
;
567 m_scrollRangeX
= scrollRangeX
;
568 m_scrollRangeY
= scrollRangeY
;
572 void wxListBox::UpdateItems()
574 // only refresh the items which must be refreshed
575 if ( m_updateCount
== -1 )
578 wxLogTrace(_T("listbox"), _T("Refreshing all"));
584 wxSize size
= GetClientSize();
587 rect
.height
= size
.y
;
588 rect
.y
+= m_updateFrom
*GetLineHeight();
589 rect
.height
= m_updateCount
*GetLineHeight();
591 // we don't need to calculate x position as we always refresh the
593 CalcScrolledPosition(0, rect
.y
, NULL
, &rect
.y
);
595 wxLogTrace(_T("listbox"), _T("Refreshing items %d..%d (%d-%d)"),
596 m_updateFrom
, m_updateFrom
+ m_updateCount
- 1,
597 rect
.GetTop(), rect
.GetBottom());
599 Refresh(TRUE
, &rect
);
603 void wxListBox::OnIdle(wxIdleEvent
& event
)
605 if ( m_updateScrollbarY
|| m_updateScrollbarX
)
610 m_updateScrollbarY
= FALSE
;
613 if ( m_currentChanged
)
615 DoEnsureVisible(m_current
);
617 m_currentChanged
= FALSE
;
630 // ----------------------------------------------------------------------------
632 // ----------------------------------------------------------------------------
634 wxBorder
wxListBox::GetDefaultBorder() const
636 return wxBORDER_SUNKEN
;
639 void wxListBox::DoDraw(wxControlRenderer
*renderer
)
641 // adjust the DC to account for scrolling
642 wxDC
& dc
= renderer
->GetDC();
644 dc
.SetFont(GetFont());
646 // get the update rect
647 wxRect rectUpdate
= GetUpdateClientRect();
650 CalcUnscrolledPosition(0, rectUpdate
.GetTop(), NULL
, &yTop
);
651 CalcUnscrolledPosition(0, rectUpdate
.GetBottom(), NULL
, &yBottom
);
653 // get the items which must be redrawn
654 wxCoord lineHeight
= GetLineHeight();
655 size_t itemFirst
= yTop
/ lineHeight
,
656 itemLast
= (yBottom
+ lineHeight
- 1) / lineHeight
,
657 itemMax
= m_strings
.GetCount();
659 if ( itemFirst
>= itemMax
)
662 if ( itemLast
> itemMax
)
666 wxLogTrace(_T("listbox"), _T("Repainting items %d..%d"),
667 itemFirst
, itemLast
);
669 DoDrawRange(renderer
, itemFirst
, itemLast
);
672 void wxListBox::DoDrawRange(wxControlRenderer
*renderer
,
673 int itemFirst
, int itemLast
)
675 renderer
->DrawItems(this, itemFirst
, itemLast
);
678 // ----------------------------------------------------------------------------
680 // ----------------------------------------------------------------------------
682 bool wxListBox::SetFont(const wxFont
& font
)
684 if ( !wxControl::SetFont(font
) )
694 void wxListBox::CalcItemsPerPage()
696 m_lineHeight
= GetRenderer()->GetListboxItemHeight(GetCharHeight());
697 m_itemsPerPage
= GetClientSize().y
/ m_lineHeight
;
700 int wxListBox::GetItemsPerPage() const
702 if ( !m_itemsPerPage
)
704 wxConstCast(this, wxListBox
)->CalcItemsPerPage();
707 return m_itemsPerPage
;
710 wxCoord
wxListBox::GetLineHeight() const
714 wxConstCast(this, wxListBox
)->CalcItemsPerPage();
720 wxCoord
wxListBox::GetMaxWidth() const
722 if ( m_maxWidth
== 0 )
724 wxListBox
*self
= wxConstCast(this, wxListBox
);
726 size_t count
= m_strings
.GetCount();
727 for ( size_t n
= 0; n
< count
; n
++ )
729 GetTextExtent(m_strings
[n
], &width
, NULL
);
730 if ( width
> m_maxWidth
)
732 self
->m_maxWidth
= width
;
733 self
->m_maxWidthItem
= n
;
741 void wxListBox::OnSize(wxSizeEvent
& event
)
743 // recalculate the number of items per page
746 // the scrollbars might [dis]appear
748 m_updateScrollbarY
= TRUE
;
753 void wxListBox::DoSetFirstItem(int n
)
758 void wxListBox::DoSetSize(int x
, int y
,
759 int width
, int height
,
762 if ( GetWindowStyle() & wxLB_INT_HEIGHT
)
764 // we must round up the height to an entire number of rows
766 // the client area must contain an int number of rows, so take borders
768 wxRect rectBorders
= GetRenderer()->GetBorderDimensions(GetBorder());
769 wxCoord hBorders
= rectBorders
.y
+ rectBorders
.height
;
771 wxCoord hLine
= GetLineHeight();
772 height
= ((height
- hBorders
+ hLine
- 1) / hLine
)*hLine
+ hBorders
;
775 wxListBoxBase::DoSetSize(x
, y
, width
, height
);
778 wxSize
wxListBox::DoGetBestClientSize() const
783 size_t count
= m_strings
.GetCount();
784 for ( size_t n
= 0; n
< count
; n
++ )
787 GetTextExtent(m_strings
[n
], &w
, &h
);
795 // if the listbox is empty, still give it some non zero (even if
796 // arbitrary) size - otherwise, leave small margin around the strings
800 width
+= 3*GetCharWidth();
803 height
= GetCharHeight();
805 // we need the height of the entire listbox, not just of one line
806 height
*= wxMax(count
, 7);
808 return wxSize(width
, height
);
811 // ----------------------------------------------------------------------------
813 // ----------------------------------------------------------------------------
815 bool wxListBox::SendEvent(wxEventType type
, int item
)
817 wxCommandEvent
event(type
, m_windowId
);
818 event
.SetEventObject(this);
820 // use the current item by default
826 // client data and string parameters only make sense if we have an item
829 if ( HasClientObjectData() )
830 event
.SetClientObject(GetClientObject(item
));
831 else if ( HasClientUntypedData() )
832 event
.SetClientData(GetClientData(item
));
834 event
.SetString(GetString(item
));
837 event
.m_commandInt
= item
;
839 return GetEventHandler()->ProcessEvent(event
);
842 void wxListBox::SetCurrentItem(int n
)
844 if ( n
!= m_current
)
846 if ( m_current
!= -1 )
847 RefreshItem(m_current
);
851 if ( m_current
!= -1 )
853 m_currentChanged
= TRUE
;
855 RefreshItem(m_current
);
858 //else: nothing to do
861 bool wxListBox::FindItem(const wxString
& prefix
, bool strictlyAfter
)
863 int count
= GetCount();
866 // empty listbox, we can't find anything in it
870 // start either from the current item or from the next one if strictlyAfter
875 // the following line will set first correctly to 0 if there is no
876 // selection (m_current == -1)
877 first
= m_current
== count
- 1 ? 0 : m_current
+ 1;
879 else // start with the current
881 first
= m_current
== -1 ? 0 : m_current
;
884 int last
= first
== 0 ? count
- 1 : first
- 1;
886 // if this is not true we'd never exit from the loop below!
887 wxASSERT_MSG( first
< count
&& last
< count
, _T("logic error") );
889 // precompute it outside the loop
890 size_t len
= prefix
.length();
892 // loop over all items in the listbox
893 for ( int item
= first
; item
!= last
; item
< count
- 1 ? item
++ : item
= 0 )
895 if ( wxStrnicmp(m_strings
[item
], prefix
, len
) == 0 )
897 SetCurrentItem(item
);
899 if ( !(GetWindowStyle() & wxLB_MULTIPLE
) )
902 SelectAndNotify(item
);
904 if ( GetWindowStyle() & wxLB_EXTENDED
)
905 AnchorSelection(item
);
916 void wxListBox::EnsureVisible(int n
)
918 if ( m_updateScrollbarY
)
923 m_updateScrollbarY
= FALSE
;
929 void wxListBox::DoEnsureVisible(int n
)
931 if ( !m_showScrollbarY
)
933 // nothing to do - everything is shown anyhow
938 GetViewStart(0, &first
);
941 // we need to scroll upwards, so make the current item appear on top
942 // of the shown range
947 int last
= first
+ GetClientSize().y
/ GetLineHeight() - 1;
950 // scroll down: the current item appears at the bottom of the
952 Scroll(0, n
- (last
- first
));
957 void wxListBox::ChangeCurrent(int diff
)
959 int current
= m_current
== -1 ? 0 : m_current
;
963 int last
= GetCount() - 1;
966 else if ( current
> last
)
969 SetCurrentItem(current
);
972 void wxListBox::ExtendSelection(int itemTo
)
974 // if we don't have the explicit values for selection start/end, make them
976 if ( m_selAnchor
== -1 )
977 m_selAnchor
= m_current
;
982 // swap the start/end of selection range if necessary
983 int itemFrom
= m_selAnchor
;
984 if ( itemFrom
> itemTo
)
986 int itemTmp
= itemFrom
;
991 // the selection should now include all items in the range between the
992 // anchor and the specified item and only them
995 for ( n
= 0; n
< itemFrom
; n
++ )
1000 for ( ; n
<= itemTo
; n
++ )
1005 int count
= GetCount();
1006 for ( ; n
< count
; n
++ )
1012 void wxListBox::DoSelect(int item
, bool sel
)
1016 // go to this item first
1017 SetCurrentItem(item
);
1020 // the current item is the one we want to change: either it was just
1021 // changed above to be the same as item or item == -1 in which we case we
1022 // are supposed to use the current one anyhow
1023 if ( m_current
!= -1 )
1026 SetSelection(m_current
, sel
);
1030 void wxListBox::SelectAndNotify(int item
)
1034 SendEvent(wxEVT_COMMAND_LISTBOX_SELECTED
);
1037 void wxListBox::Activate(int item
)
1040 SetCurrentItem(item
);
1044 if ( !(GetWindowStyle() & wxLB_MULTIPLE
) )
1053 SendEvent(wxEVT_COMMAND_LISTBOX_DOUBLECLICKED
);
1057 // ----------------------------------------------------------------------------
1059 // ----------------------------------------------------------------------------
1062 The numArg here is the listbox item index while the strArg is used
1063 differently for the different actions:
1065 a) for wxACTION_LISTBOX_FIND it has the natural meaning: this is the string
1068 b) for wxACTION_LISTBOX_SELECT and wxACTION_LISTBOX_EXTENDSEL it is used
1069 to decide if the listbox should send the notification event (it is empty)
1070 or not (it is not): this allows us to reuse the same action for when the
1071 user is dragging the mouse when it has been released although in the
1072 first case no notification is sent while in the second it is sent.
1074 bool wxListBox::PerformAction(const wxControlAction
& action
,
1076 const wxString
& strArg
)
1078 int item
= (int)numArg
;
1080 if ( action
== wxACTION_LISTBOX_SETFOCUS
)
1082 SetCurrentItem(item
);
1084 else if ( action
== wxACTION_LISTBOX_ACTIVATE
)
1088 else if ( action
== wxACTION_LISTBOX_TOGGLE
)
1093 if ( IsSelected(item
) )
1096 SelectAndNotify(item
);
1098 else if ( action
== wxACTION_LISTBOX_SELECT
)
1102 if ( strArg
.empty() )
1103 SelectAndNotify(item
);
1107 else if ( action
== wxACTION_LISTBOX_SELECTADD
)
1109 else if ( action
== wxACTION_LISTBOX_UNSELECT
)
1111 else if ( action
== wxACTION_LISTBOX_MOVEDOWN
)
1113 else if ( action
== wxACTION_LISTBOX_MOVEUP
)
1115 else if ( action
== wxACTION_LISTBOX_PAGEDOWN
)
1116 ChangeCurrent(GetItemsPerPage());
1117 else if ( action
== wxACTION_LISTBOX_PAGEUP
)
1118 ChangeCurrent(-GetItemsPerPage());
1119 else if ( action
== wxACTION_LISTBOX_START
)
1121 else if ( action
== wxACTION_LISTBOX_END
)
1122 SetCurrentItem(GetCount() - 1);
1123 else if ( action
== wxACTION_LISTBOX_UNSELECTALL
)
1125 else if ( action
== wxACTION_LISTBOX_EXTENDSEL
)
1126 ExtendSelection(item
);
1127 else if ( action
== wxACTION_LISTBOX_FIND
)
1128 FindNextItem(strArg
);
1129 else if ( action
== wxACTION_LISTBOX_ANCHOR
)
1130 AnchorSelection(item
== -1 ? m_current
: item
);
1131 else if ( action
== wxACTION_LISTBOX_SELECTALL
||
1132 action
== wxACTION_LISTBOX_SELTOGGLE
)
1133 wxFAIL_MSG(_T("unimplemented yet"));
1135 return wxControl::PerformAction(action
, numArg
, strArg
);
1140 // ============================================================================
1141 // implementation of wxStdListboxInputHandler
1142 // ============================================================================
1144 wxStdListboxInputHandler::wxStdListboxInputHandler(wxInputHandler
*handler
,
1145 bool toggleOnPressAlways
)
1146 : wxStdInputHandler(handler
)
1149 m_toggleOnPressAlways
= toggleOnPressAlways
;
1150 m_actionMouse
= wxACTION_NONE
;
1151 m_trackMouseOutside
= TRUE
;
1154 int wxStdListboxInputHandler::HitTest(const wxListBox
*lbox
,
1155 const wxMouseEvent
& event
)
1157 int item
= HitTestUnsafe(lbox
, event
);
1159 return FixItemIndex(lbox
, item
);
1162 int wxStdListboxInputHandler::HitTestUnsafe(const wxListBox
*lbox
,
1163 const wxMouseEvent
& event
)
1165 wxPoint pt
= event
.GetPosition();
1166 pt
-= lbox
->GetClientAreaOrigin();
1168 lbox
->CalcUnscrolledPosition(0, pt
.y
, NULL
, &y
);
1169 return y
/ lbox
->GetLineHeight();
1172 int wxStdListboxInputHandler::FixItemIndex(const wxListBox
*lbox
,
1177 // mouse is above the first item
1180 else if ( item
>= lbox
->GetCount() )
1182 // mouse is below the last item
1183 item
= lbox
->GetCount() - 1;
1189 bool wxStdListboxInputHandler::IsValidIndex(const wxListBox
*lbox
, int item
)
1191 return item
>= 0 && item
< lbox
->GetCount();
1195 wxStdListboxInputHandler::SetupCapture(wxListBox
*lbox
,
1196 const wxMouseEvent
& event
,
1199 // we currently only allow selecting with the left mouse button, if we
1200 // do need to allow using other buttons too we might use the code
1203 m_btnCapture
= event
.LeftDown()
1212 wxControlAction action
;
1213 if ( lbox
->HasMultipleSelection() )
1215 if ( lbox
->GetWindowStyle() & wxLB_MULTIPLE
)
1217 if ( m_toggleOnPressAlways
)
1219 // toggle the item right now
1220 action
= wxACTION_LISTBOX_TOGGLE
;
1224 m_actionMouse
= wxACTION_LISTBOX_SETFOCUS
;
1226 else // wxLB_EXTENDED listbox
1228 // simple click in an extended sel listbox clears the old
1229 // selection and adds the clicked item to it then, ctrl-click
1230 // toggles an item to it and shift-click adds a range between
1231 // the old selection anchor and the clicked item
1232 if ( event
.ControlDown() )
1234 lbox
->PerformAction(wxACTION_LISTBOX_ANCHOR
, item
);
1236 action
= wxACTION_LISTBOX_TOGGLE
;
1238 else if ( event
.ShiftDown() )
1240 action
= wxACTION_LISTBOX_EXTENDSEL
;
1242 else // simple click
1244 lbox
->PerformAction(wxACTION_LISTBOX_ANCHOR
, item
);
1246 action
= wxACTION_LISTBOX_SELECT
;
1249 m_actionMouse
= wxACTION_LISTBOX_EXTENDSEL
;
1252 else // single selection
1255 action
= wxACTION_LISTBOX_SELECT
;
1258 // by default we always do track it
1259 m_trackMouseOutside
= TRUE
;
1264 bool wxStdListboxInputHandler::HandleKey(wxInputConsumer
*consumer
,
1265 const wxKeyEvent
& event
,
1268 // we're only interested in the key press events
1269 if ( pressed
&& !event
.AltDown() )
1271 bool isMoveCmd
= TRUE
;
1272 int style
= consumer
->GetInputWindow()->GetWindowStyle();
1274 wxControlAction action
;
1277 int keycode
= event
.GetKeyCode();
1282 action
= wxACTION_LISTBOX_MOVEUP
;
1286 action
= wxACTION_LISTBOX_MOVEDOWN
;
1292 action
= wxACTION_LISTBOX_PAGEUP
;
1298 action
= wxACTION_LISTBOX_PAGEDOWN
;
1302 action
= wxACTION_LISTBOX_START
;
1306 action
= wxACTION_LISTBOX_END
;
1311 if ( style
& wxLB_MULTIPLE
)
1313 action
= wxACTION_LISTBOX_TOGGLE
;
1319 action
= wxACTION_LISTBOX_ACTIVATE
;
1324 if ( (keycode
< 255) && wxIsalnum(keycode
) )
1326 action
= wxACTION_LISTBOX_FIND
;
1327 strArg
= (wxChar
)keycode
;
1333 consumer
->PerformAction(action
, -1, strArg
);
1337 if ( style
& wxLB_SINGLE
)
1339 // the current item is always the one selected
1340 consumer
->PerformAction(wxACTION_LISTBOX_SELECT
);
1342 else if ( style
& wxLB_EXTENDED
)
1344 if ( event
.ShiftDown() )
1345 consumer
->PerformAction(wxACTION_LISTBOX_EXTENDSEL
);
1348 // select the item and make it the new selection anchor
1349 consumer
->PerformAction(wxACTION_LISTBOX_SELECT
);
1350 consumer
->PerformAction(wxACTION_LISTBOX_ANCHOR
);
1353 //else: nothing to do for multiple selection listboxes
1360 return wxStdInputHandler::HandleKey(consumer
, event
, pressed
);
1363 bool wxStdListboxInputHandler::HandleMouse(wxInputConsumer
*consumer
,
1364 const wxMouseEvent
& event
)
1366 wxListBox
*lbox
= wxStaticCast(consumer
->GetInputWindow(), wxListBox
);
1367 int item
= HitTest(lbox
, event
);
1368 wxControlAction action
;
1370 // when the left mouse button is pressed, capture the mouse and track the
1371 // item under mouse (if the mouse leaves the window, we will still be
1372 // getting the mouse move messages generated by wxScrollWindow)
1373 if ( event
.LeftDown() )
1375 // capture the mouse to track the selected item
1376 lbox
->CaptureMouse();
1378 action
= SetupCapture(lbox
, event
, item
);
1380 else if ( m_btnCapture
&& event
.ButtonUp(m_btnCapture
) )
1382 // when the left mouse button is released, release the mouse too
1383 wxWindow
*winCapture
= wxWindow::GetCapture();
1386 winCapture
->ReleaseMouse();
1389 action
= m_actionMouse
;
1391 //else: the mouse wasn't presed over the listbox, only released here
1393 else if ( event
.LeftDClick() )
1395 action
= wxACTION_LISTBOX_ACTIVATE
;
1400 lbox
->PerformAction(action
, item
);
1405 return wxStdInputHandler::HandleMouse(consumer
, event
);
1408 bool wxStdListboxInputHandler::HandleMouseMove(wxInputConsumer
*consumer
,
1409 const wxMouseEvent
& event
)
1411 wxWindow
*winCapture
= wxWindow::GetCapture();
1412 if ( winCapture
&& (event
.GetEventObject() == winCapture
) )
1414 wxListBox
*lbox
= wxStaticCast(consumer
->GetInputWindow(), wxListBox
);
1416 if ( !m_btnCapture
|| !m_trackMouseOutside
)
1418 // someone captured the mouse for us (we always set m_btnCapture
1419 // when we do it ourselves): in this case we only react to
1420 // the mouse messages when they happen inside the listbox
1421 if ( lbox
->HitTest(event
.GetPosition()) != wxHT_WINDOW_INSIDE
)
1425 int item
= HitTest(lbox
, event
);
1426 if ( !m_btnCapture
)
1428 // now that we have the mouse inside the listbox, do capture it
1429 // normally - but ensure that we will still ignore the outside
1431 SetupCapture(lbox
, event
, item
);
1433 m_trackMouseOutside
= FALSE
;
1436 if ( IsValidIndex(lbox
, item
) )
1438 // pass something into strArg to tell the listbox that it shouldn't
1439 // send the notification message: see PerformAction() above
1440 lbox
->PerformAction(m_actionMouse
, item
, _T("no"));
1442 // else: don't pass invalid index to the listbox
1444 else // we don't have capture any more
1448 // if we lost capture unexpectedly (someone else took the capture
1449 // from us), return to a consistent state
1454 return wxStdInputHandler::HandleMouseMove(consumer
, event
);
1457 #endif // wxUSE_LISTBOX