]> git.saurik.com Git - wxWidgets.git/blame - src/common/lboxcmn.cpp
Fix out of bounds string access in wxMSW wxDirDialog.
[wxWidgets.git] / src / common / lboxcmn.cpp
CommitLineData
2ee3ee1b 1///////////////////////////////////////////////////////////////////////////////
aa61d352 2// Name: src/common/lboxcmn.cpp
2ee3ee1b
VZ
3// Purpose: wxListBox class methods common to all platforms
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 22.10.99
7// RCS-ID: $Id$
77ffb593 8// Copyright: (c) wxWidgets team
65571936 9// Licence: wxWindows licence
2ee3ee1b
VZ
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
2ee3ee1b
VZ
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
1e6feb95
VZ
27#if wxUSE_LISTBOX
28
2a673eb1
WS
29#include "wx/listbox.h"
30
2ee3ee1b 31#ifndef WX_PRECOMP
ed39ff57 32 #include "wx/dynarray.h"
a9711a4d 33 #include "wx/arrstr.h"
f313deaa 34 #include "wx/log.h"
2ee3ee1b
VZ
35#endif
36
f313deaa 37extern WXDLLEXPORT_DATA(const char) wxListBoxNameStr[] = "listBox";
22892dea 38
2ee3ee1b
VZ
39// ============================================================================
40// implementation
41// ============================================================================
42
799ea011
GD
43wxListBoxBase::~wxListBoxBase()
44{
45 // this destructor is required for Darwin
46}
47
28953245
SC
48// ----------------------------------------------------------------------------
49// XTI
50// ----------------------------------------------------------------------------
51
52wxDEFINE_FLAGS( wxListBoxStyle )
53wxBEGIN_FLAGS( wxListBoxStyle )
54// new style border flags, we put them first to
55// use them for streaming out
56wxFLAGS_MEMBER(wxBORDER_SIMPLE)
57wxFLAGS_MEMBER(wxBORDER_SUNKEN)
58wxFLAGS_MEMBER(wxBORDER_DOUBLE)
59wxFLAGS_MEMBER(wxBORDER_RAISED)
60wxFLAGS_MEMBER(wxBORDER_STATIC)
61wxFLAGS_MEMBER(wxBORDER_NONE)
62
63// old style border flags
64wxFLAGS_MEMBER(wxSIMPLE_BORDER)
65wxFLAGS_MEMBER(wxSUNKEN_BORDER)
66wxFLAGS_MEMBER(wxDOUBLE_BORDER)
67wxFLAGS_MEMBER(wxRAISED_BORDER)
68wxFLAGS_MEMBER(wxSTATIC_BORDER)
69wxFLAGS_MEMBER(wxBORDER)
70
71// standard window styles
72wxFLAGS_MEMBER(wxTAB_TRAVERSAL)
73wxFLAGS_MEMBER(wxCLIP_CHILDREN)
74wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW)
75wxFLAGS_MEMBER(wxWANTS_CHARS)
76wxFLAGS_MEMBER(wxFULL_REPAINT_ON_RESIZE)
77wxFLAGS_MEMBER(wxALWAYS_SHOW_SB )
78wxFLAGS_MEMBER(wxVSCROLL)
79wxFLAGS_MEMBER(wxHSCROLL)
80
81wxFLAGS_MEMBER(wxLB_SINGLE)
82wxFLAGS_MEMBER(wxLB_MULTIPLE)
83wxFLAGS_MEMBER(wxLB_EXTENDED)
84wxFLAGS_MEMBER(wxLB_HSCROLL)
85wxFLAGS_MEMBER(wxLB_ALWAYS_SB)
86wxFLAGS_MEMBER(wxLB_NEEDED_SB)
87wxFLAGS_MEMBER(wxLB_SORT)
88wxEND_FLAGS( wxListBoxStyle )
89
37788684 90wxIMPLEMENT_DYNAMIC_CLASS_XTI(wxListBox, wxControl, "wx/listbox.h")
28953245
SC
91
92wxBEGIN_PROPERTIES_TABLE(wxListBox)
93wxEVENT_PROPERTY( Select, wxEVT_COMMAND_LISTBOX_SELECTED, wxCommandEvent )
94wxEVENT_PROPERTY( DoubleClick, wxEVT_COMMAND_LISTBOX_DOUBLECLICKED, wxCommandEvent )
95
96wxPROPERTY( Font, wxFont, SetFont, GetFont , wxEMPTY_PARAMETER_VALUE, 0 /*flags*/, \
97 wxT("Helpstring"), wxT("group"))
98wxPROPERTY_COLLECTION( Choices, wxArrayString, wxString, AppendString, \
99 GetStrings, 0 /*flags*/, wxT("Helpstring"), wxT("group") )
100wxPROPERTY( Selection, int, SetSelection, GetSelection, wxEMPTY_PARAMETER_VALUE, \
101 0 /*flags*/, wxT("Helpstring"), wxT("group") )
102
103wxPROPERTY_FLAGS( WindowStyle, wxListBoxStyle, long, SetWindowStyleFlag, \
104 GetWindowStyleFlag, wxEMPTY_PARAMETER_VALUE, 0 /*flags*/, \
105 wxT("Helpstring"), wxT("group")) // style
106wxEND_PROPERTIES_TABLE()
107
108wxEMPTY_HANDLERS_TABLE(wxListBox)
109
110wxCONSTRUCTOR_4( wxListBox, wxWindow*, Parent, wxWindowID, Id, \
111 wxPoint, Position, wxSize, Size )
112
113/*
114 TODO PROPERTIES
115 selection
116 content
117 item
118 */
119
2ee3ee1b
VZ
120// ----------------------------------------------------------------------------
121// selection
122// ----------------------------------------------------------------------------
123
2ee3ee1b
VZ
124bool wxListBoxBase::SetStringSelection(const wxString& s, bool select)
125{
95668975
VZ
126 const int sel = FindString(s);
127 if ( sel == wxNOT_FOUND )
128 return false;
2ee3ee1b
VZ
129
130 SetSelection(sel, select);
131
f644b28c 132 return true;
2ee3ee1b
VZ
133}
134
24ee1bef
VZ
135void wxListBoxBase::SetSelection(int n)
136{
137 if ( !HasMultipleSelection() )
138 DoChangeSingleSelection(n);
139
140 DoSetSelection(n, true);
141}
142
1e6feb95
VZ
143void wxListBoxBase::DeselectAll(int itemToLeaveSelected)
144{
145 if ( HasMultipleSelection() )
146 {
147 wxArrayInt selections;
148 GetSelections(selections);
149
150 size_t count = selections.GetCount();
151 for ( size_t n = 0; n < count; n++ )
152 {
153 int item = selections[n];
154 if ( item != itemToLeaveSelected )
155 Deselect(item);
156 }
157 }
158 else // single selection
159 {
160 int sel = GetSelection();
f644b28c 161 if ( sel != wxNOT_FOUND && sel != itemToLeaveSelected )
1e6feb95
VZ
162 {
163 Deselect(sel);
164 }
165 }
166}
167
05d790f8
RR
168void wxListBoxBase::UpdateOldSelections()
169{
a614ffae
VS
170 // We need to remember the selection even in single-selection case on
171 // Windows, so that we don't send an event when the user clicks on an
172 // already selected item.
173#ifndef __WXMSW__
05d790f8 174 if (HasFlag(wxLB_MULTIPLE) || HasFlag(wxLB_EXTENDED))
a614ffae
VS
175#endif
176 {
05d790f8 177 GetSelections( m_oldSelections );
a614ffae 178 }
05d790f8
RR
179}
180
53f60d4a 181bool wxListBoxBase::SendEvent(wxEventType evtType, int item, bool selected)
05d790f8 182{
53f60d4a
VZ
183 wxCommandEvent event(evtType, GetId());
184 event.SetEventObject(this);
185
186 event.SetInt(item);
187 event.SetString(GetString(item));
188 event.SetExtraLong(selected);
189
190 if ( HasClientObjectData() )
191 event.SetClientObject(GetClientObject(item));
192 else if ( HasClientUntypedData() )
193 event.SetClientData(GetClientData(item));
194
195 return HandleWindowEvent(event);
05d790f8
RR
196}
197
24ee1bef
VZ
198bool wxListBoxBase::DoChangeSingleSelection(int item)
199{
200 // As we don't use m_oldSelections in single selection mode, we store the
201 // last item that we notified the user about in it in this case because we
202 // need to remember it to be able to filter out the dummy selection changes
203 // that we get when the user clicks on an already selected item.
204 if ( !m_oldSelections.empty() && *m_oldSelections.begin() == item )
205 {
206 // Same item as the last time.
207 return false;
208 }
209
210 m_oldSelections.clear();
211 m_oldSelections.push_back(item);
212
213 return true;
214}
215
53f60d4a 216bool wxListBoxBase::CalcAndSendEvent()
05d790f8 217{
05d790f8 218 wxArrayInt selections;
7ead3845 219 GetSelections(selections);
53f60d4a 220 bool selected = true;
7ead3845
VZ
221
222 if ( selections.empty() && m_oldSelections.empty() )
223 {
224 // nothing changed, just leave
53f60d4a 225 return false;
7ead3845
VZ
226 }
227
228 const size_t countSel = selections.size(),
229 countSelOld = m_oldSelections.size();
230 if ( countSel == countSelOld )
231 {
232 bool changed = false;
233 for ( size_t idx = 0; idx < countSel; idx++ )
05d790f8 234 {
7ead3845 235 if (selections[idx] != m_oldSelections[idx])
05d790f8 236 {
7ead3845
VZ
237 changed = true;
238 break;
05d790f8 239 }
05d790f8
RR
240 }
241
7ead3845
VZ
242 // nothing changed, just leave
243 if ( !changed )
53f60d4a 244 return false;
7ead3845
VZ
245 }
246
247 int item = wxNOT_FOUND;
248 if ( selections.empty() )
249 {
53f60d4a 250 selected = false;
7ead3845
VZ
251 item = m_oldSelections[0];
252 }
253 else // we [still] have some selections
254 {
05d790f8
RR
255 // Now test if any new item is selected
256 bool any_new_selected = false;
7ead3845 257 for ( size_t idx = 0; idx < countSel; idx++ )
05d790f8
RR
258 {
259 item = selections[idx];
7ead3845 260 if ( m_oldSelections.Index(item) == wxNOT_FOUND )
05d790f8
RR
261 {
262 any_new_selected = true;
263 break;
264 }
265 }
7ead3845 266
53f60d4a 267 if ( !any_new_selected )
05d790f8 268 {
53f60d4a 269 // No new items selected, now test if any new item is deselected
7ead3845
VZ
270 bool any_new_deselected = false;
271 for ( size_t idx = 0; idx < countSelOld; idx++ )
05d790f8 272 {
7ead3845
VZ
273 item = m_oldSelections[idx];
274 if ( selections.Index(item) == wxNOT_FOUND )
275 {
276 any_new_deselected = true;
277 break;
278 }
279 }
280
281 if ( any_new_deselected )
282 {
283 // indicate that this is a selection
53f60d4a 284 selected = false;
7ead3845
VZ
285 }
286 else
287 {
288 item = wxNOT_FOUND; // this should be impossible
05d790f8
RR
289 }
290 }
7ead3845
VZ
291 }
292
293 wxASSERT_MSG( item != wxNOT_FOUND,
294 "Logic error in wxListBox selection event generation code" );
295
296 m_oldSelections = selections;
297
53f60d4a 298 return SendEvent(wxEVT_COMMAND_LISTBOX_SELECTED, item, selected);
05d790f8
RR
299}
300
2ee3ee1b 301// ----------------------------------------------------------------------------
6c8a980f 302// misc
2ee3ee1b
VZ
303// ----------------------------------------------------------------------------
304
6c8a980f 305void wxListBoxBase::Command(wxCommandEvent& event)
2ee3ee1b 306{
687706f5 307 SetSelection(event.GetInt(), event.GetExtraLong() != 0);
004867db 308 (void)GetEventHandler()->ProcessEvent(event);
2ee3ee1b
VZ
309}
310
1e6feb95
VZ
311// ----------------------------------------------------------------------------
312// SetFirstItem() and such
313// ----------------------------------------------------------------------------
314
2ee3ee1b
VZ
315void wxListBoxBase::SetFirstItem(const wxString& s)
316{
317 int n = FindString(s);
318
f644b28c 319 wxCHECK_RET( n != wxNOT_FOUND, wxT("invalid string in wxListBox::SetFirstItem") );
2ee3ee1b
VZ
320
321 DoSetFirstItem(n);
322}
1e6feb95
VZ
323
324void wxListBoxBase::AppendAndEnsureVisible(const wxString& s)
325{
326 Append(s);
327 EnsureVisible(GetCount() - 1);
328}
329
330void wxListBoxBase::EnsureVisible(int WXUNUSED(n))
331{
332 // the base class version does nothing (the only alternative would be to
333 // call SetFirstItem() but this is probably even more stupid)
334}
335
336#endif // wxUSE_LISTBOX