]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/checklst.cpp
In wxBitmapComboBox::RecreateControl(), always get recreated control's height from...
[wxWidgets.git] / src / msw / checklst.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/checklst.cpp
3// Purpose: implementation of wxCheckListBox class
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 16.11.97
7// RCS-ID: $Id$
8// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows licence
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#if wxUSE_CHECKLISTBOX && wxUSE_OWNER_DRAWN
28
29#include "wx/checklst.h"
30
31#ifndef WX_PRECOMP
32 #include "wx/msw/wrapcctl.h"
33 #include "wx/object.h"
34 #include "wx/colour.h"
35 #include "wx/font.h"
36 #include "wx/bitmap.h"
37 #include "wx/window.h"
38 #include "wx/listbox.h"
39 #include "wx/dcmemory.h"
40 #include "wx/settings.h"
41 #include "wx/log.h"
42#endif
43
44#include "wx/ownerdrw.h"
45
46#include <windowsx.h>
47
48#include "wx/renderer.h"
49#include "wx/msw/private.h"
50#include "wx/msw/dc.h"
51
52// ----------------------------------------------------------------------------
53// private functions
54// ----------------------------------------------------------------------------
55
56// get item (converted to right type)
57#define GetItem(n) ((wxCheckListBoxItem *)(GetItem(n)))
58
59namespace
60{
61 // space around check mark bitmap in pixels
62 static const int CHECKMARK_EXTRA_SPACE = 1;
63
64 // space betwen check bitmap and text label
65 static const int CHECKMARK_LABEL_SPACE = 2;
66
67} // anonymous namespace
68
69// ============================================================================
70// implementation
71// ============================================================================
72
73
74#if wxUSE_EXTENDED_RTTI
75WX_DEFINE_FLAGS( wxCheckListBoxStyle )
76
77wxBEGIN_FLAGS( wxCheckListBoxStyle )
78 // new style border flags, we put them first to
79 // use them for streaming out
80 wxFLAGS_MEMBER(wxBORDER_SIMPLE)
81 wxFLAGS_MEMBER(wxBORDER_SUNKEN)
82 wxFLAGS_MEMBER(wxBORDER_DOUBLE)
83 wxFLAGS_MEMBER(wxBORDER_RAISED)
84 wxFLAGS_MEMBER(wxBORDER_STATIC)
85 wxFLAGS_MEMBER(wxBORDER_NONE)
86
87 // old style border flags
88 wxFLAGS_MEMBER(wxSIMPLE_BORDER)
89 wxFLAGS_MEMBER(wxSUNKEN_BORDER)
90 wxFLAGS_MEMBER(wxDOUBLE_BORDER)
91 wxFLAGS_MEMBER(wxRAISED_BORDER)
92 wxFLAGS_MEMBER(wxSTATIC_BORDER)
93 wxFLAGS_MEMBER(wxBORDER)
94
95 // standard window styles
96 wxFLAGS_MEMBER(wxTAB_TRAVERSAL)
97 wxFLAGS_MEMBER(wxCLIP_CHILDREN)
98 wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW)
99 wxFLAGS_MEMBER(wxWANTS_CHARS)
100 wxFLAGS_MEMBER(wxFULL_REPAINT_ON_RESIZE)
101 wxFLAGS_MEMBER(wxALWAYS_SHOW_SB )
102 wxFLAGS_MEMBER(wxVSCROLL)
103 wxFLAGS_MEMBER(wxHSCROLL)
104
105 wxFLAGS_MEMBER(wxLB_SINGLE)
106 wxFLAGS_MEMBER(wxLB_MULTIPLE)
107 wxFLAGS_MEMBER(wxLB_EXTENDED)
108 wxFLAGS_MEMBER(wxLB_HSCROLL)
109 wxFLAGS_MEMBER(wxLB_ALWAYS_SB)
110 wxFLAGS_MEMBER(wxLB_NEEDED_SB)
111 wxFLAGS_MEMBER(wxLB_SORT)
112 wxFLAGS_MEMBER(wxLB_OWNERDRAW)
113
114wxEND_FLAGS( wxCheckListBoxStyle )
115
116IMPLEMENT_DYNAMIC_CLASS_XTI(wxCheckListBox, wxListBox,"wx/checklst.h")
117
118wxBEGIN_PROPERTIES_TABLE(wxCheckListBox)
119 wxEVENT_PROPERTY( Toggle , wxEVT_COMMAND_CHECKLISTBOX_TOGGLED , wxCommandEvent )
120 wxPROPERTY_FLAGS( WindowStyle , wxCheckListBoxStyle , long , SetWindowStyleFlag , GetWindowStyleFlag , EMPTY_MACROVALUE , wxLB_OWNERDRAW /*flags*/ , wxT("Helpstring") , wxT("group")) // style
121wxEND_PROPERTIES_TABLE()
122
123wxBEGIN_HANDLERS_TABLE(wxCheckListBox)
124wxEND_HANDLERS_TABLE()
125
126wxCONSTRUCTOR_4( wxCheckListBox , wxWindow* , Parent , wxWindowID , Id , wxPoint , Position , wxSize , Size )
127
128#else
129IMPLEMENT_DYNAMIC_CLASS(wxCheckListBox, wxListBox)
130#endif
131
132// ----------------------------------------------------------------------------
133// declaration and implementation of wxCheckListBoxItem class
134// ----------------------------------------------------------------------------
135
136class wxCheckListBoxItem : public wxOwnerDrawn
137{
138public:
139 // ctor
140 wxCheckListBoxItem(wxCheckListBox *parent);
141
142 // drawing functions
143 virtual bool OnDrawItem(wxDC& dc, const wxRect& rc, wxODAction act, wxODStatus stat);
144
145 // simple accessors and operations
146 wxCheckListBox *GetParent() const
147 { return m_parent; }
148
149 int GetIndex() const
150 { return m_parent->GetItemIndex(const_cast<wxCheckListBoxItem*>(this)); }
151
152 wxString GetName() const
153 { return m_parent->GetString(GetIndex()); }
154
155
156 bool IsChecked() const
157 { return m_checked; }
158
159 void Check(bool bCheck)
160 { m_checked = bCheck; }
161
162 void Toggle()
163 { Check(!IsChecked()); }
164
165private:
166 wxCheckListBox *m_parent;
167 bool m_checked;
168
169 wxDECLARE_NO_COPY_CLASS(wxCheckListBoxItem);
170};
171
172wxCheckListBoxItem::wxCheckListBoxItem(wxCheckListBox *parent)
173{
174 m_parent = parent;
175 m_checked = false;
176
177 wxSize size = wxRendererNative::Get().GetCheckBoxSize(parent);
178 size.x += 2 * CHECKMARK_EXTRA_SPACE + CHECKMARK_LABEL_SPACE;
179
180 SetMarginWidth(size.GetWidth());
181 SetBackgroundColour(parent->GetBackgroundColour());
182}
183
184bool wxCheckListBoxItem::OnDrawItem(wxDC& dc, const wxRect& rc,
185 wxODAction act, wxODStatus stat)
186{
187 // first draw the label
188 if ( !wxOwnerDrawn::OnDrawItem(dc, rc, act, stat) )
189 return false;
190
191 // now draw the check mark part
192 wxMSWDCImpl *impl = (wxMSWDCImpl*) dc.GetImpl();
193 HDC hdc = GetHdcOf(*impl);
194
195 wxSize size = wxRendererNative::Get().GetCheckBoxSize(GetParent());
196
197 // first create bitmap in a memory DC
198 MemoryHDC hdcMem(hdc);
199 CompatibleBitmap hBmpCheck(hdc, size.GetWidth(), size.GetHeight());
200
201 // then draw a check mark into it
202 {
203 SelectInHDC selBmp(hdcMem, hBmpCheck);
204
205 int flags = wxCONTROL_FLAT;
206 if ( IsChecked() )
207 flags |= wxCONTROL_CHECKED;
208
209 wxDCTemp dcMem(hdcMem);
210 wxRendererNative::Get().DrawCheckBox(GetParent(), dcMem, wxRect(size), flags);
211 } // select hBmpCheck out of hdcMem
212
213 // finally draw bitmap to screen
214
215 // position of check mark bitmap
216 int x = rc.GetX() + CHECKMARK_EXTRA_SPACE;
217 int y = rc.GetY() + (rc.GetHeight() - size.GetHeight()) / 2;
218
219 UINT uState = stat & wxOwnerDrawn::wxODSelected ? wxDSB_SELECTED : wxDSB_NORMAL;
220 wxDrawStateBitmap(hdc, hBmpCheck, x, y, uState);
221
222 return true;
223}
224
225// ----------------------------------------------------------------------------
226// implementation of wxCheckListBox class
227// ----------------------------------------------------------------------------
228
229// define event table
230// ------------------
231BEGIN_EVENT_TABLE(wxCheckListBox, wxListBox)
232 EVT_KEY_DOWN(wxCheckListBox::OnKeyDown)
233 EVT_LEFT_DOWN(wxCheckListBox::OnLeftClick)
234END_EVENT_TABLE()
235
236// control creation
237// ----------------
238
239// def ctor: use Create() to really create the control
240wxCheckListBox::wxCheckListBox()
241{
242}
243
244// ctor which creates the associated control
245wxCheckListBox::wxCheckListBox(wxWindow *parent, wxWindowID id,
246 const wxPoint& pos, const wxSize& size,
247 int nStrings, const wxString choices[],
248 long style, const wxValidator& val,
249 const wxString& name)
250{
251 Create(parent, id, pos, size, nStrings, choices, style, val, name);
252}
253
254wxCheckListBox::wxCheckListBox(wxWindow *parent, wxWindowID id,
255 const wxPoint& pos, const wxSize& size,
256 const wxArrayString& choices,
257 long style, const wxValidator& val,
258 const wxString& name)
259{
260 Create(parent, id, pos, size, choices, style, val, name);
261}
262
263bool wxCheckListBox::Create(wxWindow *parent, wxWindowID id,
264 const wxPoint& pos, const wxSize& size,
265 int n, const wxString choices[],
266 long style,
267 const wxValidator& validator, const wxString& name)
268{
269 return wxListBox::Create(parent, id, pos, size, n, choices,
270 style | wxLB_OWNERDRAW, validator, name);
271}
272
273bool wxCheckListBox::Create(wxWindow *parent, wxWindowID id,
274 const wxPoint& pos, const wxSize& size,
275 const wxArrayString& choices,
276 long style,
277 const wxValidator& validator, const wxString& name)
278{
279 return wxListBox::Create(parent, id, pos, size, choices,
280 style | wxLB_OWNERDRAW, validator, name);
281}
282
283// create/retrieve item
284// --------------------
285
286// create a check list box item
287wxOwnerDrawn *wxCheckListBox::CreateLboxItem(size_t WXUNUSED(n))
288{
289 wxCheckListBoxItem *pItem = new wxCheckListBoxItem(this);
290 return pItem;
291}
292
293// return item size
294// ----------------
295bool wxCheckListBox::MSWOnMeasure(WXMEASUREITEMSTRUCT *item)
296{
297 if ( wxListBox::MSWOnMeasure(item) )
298 {
299 MEASUREITEMSTRUCT *pStruct = (MEASUREITEMSTRUCT *)item;
300
301 wxSize size = wxRendererNative::Get().GetCheckBoxSize(this);
302 size.x += 2 * CHECKMARK_EXTRA_SPACE;
303 size.y += 2 * CHECKMARK_EXTRA_SPACE;
304
305 // add place for the check mark
306 pStruct->itemWidth += size.GetWidth();
307
308 if ( pStruct->itemHeight < static_cast<unsigned int>(size.GetHeight()) )
309 pStruct->itemHeight = size.GetHeight();
310
311 return true;
312 }
313
314 return false;
315 }
316
317// check items
318// -----------
319
320bool wxCheckListBox::IsChecked(unsigned int uiIndex) const
321{
322 wxCHECK_MSG( IsValid(uiIndex), false, wxT("bad wxCheckListBox index") );
323
324 return GetItem(uiIndex)->IsChecked();
325}
326
327void wxCheckListBox::Check(unsigned int uiIndex, bool bCheck)
328{
329 wxCHECK_RET( IsValid(uiIndex), wxT("bad wxCheckListBox index") );
330
331 GetItem(uiIndex)->Check(bCheck);
332 RefreshItem(uiIndex);
333}
334
335void wxCheckListBox::Toggle(unsigned int uiIndex)
336{
337 wxCHECK_RET( IsValid(uiIndex), wxT("bad wxCheckListBox index") );
338
339 GetItem(uiIndex)->Toggle();
340 RefreshItem(uiIndex);
341}
342
343// process events
344// --------------
345
346void wxCheckListBox::OnKeyDown(wxKeyEvent& event)
347{
348 // what do we do?
349 enum
350 {
351 NONE,
352 TOGGLE,
353 SET,
354 CLEAR
355 } oper;
356
357 switch ( event.GetKeyCode() )
358 {
359 case WXK_SPACE:
360 oper = TOGGLE;
361 break;
362
363 case WXK_NUMPAD_ADD:
364 case '+':
365 oper = SET;
366 break;
367
368 case WXK_NUMPAD_SUBTRACT:
369 case '-':
370 oper = CLEAR;
371 break;
372
373 default:
374 oper = NONE;
375 }
376
377 if ( oper != NONE )
378 {
379 wxArrayInt selections;
380 int count = 0;
381 if ( HasMultipleSelection() )
382 {
383 count = GetSelections(selections);
384 }
385 else
386 {
387 int sel = GetSelection();
388 if (sel != -1)
389 {
390 count = 1;
391 selections.Add(sel);
392 }
393 }
394
395 for ( int i = 0; i < count; i++ )
396 {
397 int nItem = selections[i];
398
399 switch ( oper )
400 {
401 case TOGGLE:
402 Toggle(nItem);
403 break;
404
405 case SET:
406 case CLEAR:
407 Check(nItem, oper == SET);
408 break;
409
410 default:
411 wxFAIL_MSG( wxT("what should this key do?") );
412 }
413
414 // we should send an event as this has been done by the user and
415 // not by the program
416 SendEvent(nItem);
417 }
418 }
419 else // nothing to do
420 {
421 event.Skip();
422 }
423}
424
425void wxCheckListBox::OnLeftClick(wxMouseEvent& event)
426{
427 // clicking on the item selects it, clicking on the checkmark toggles
428
429 int nItem = HitTest(event.GetX(), event.GetY());
430
431 if ( nItem != wxNOT_FOUND )
432 {
433 wxRect rect;
434 GetItemRect(nItem, rect);
435
436 // convert item rect to check mark rect
437 wxSize size = wxRendererNative::Get().GetCheckBoxSize(this);
438 rect.x += CHECKMARK_EXTRA_SPACE;
439 rect.y += (rect.GetHeight() - size.GetHeight()) / 2;
440 rect.SetSize(size);
441
442 if ( rect.Contains(event.GetX(), event.GetY()) )
443 {
444 // people expect to get "kill focus" event for the currently
445 // focused control before getting events from the other controls
446 // and, equally importantly, they may prevent the focus change from
447 // taking place at all (e.g. because the old control contents is
448 // invalid and needs to be corrected) in which case we shouldn't
449 // generate this event at all
450 SetFocus();
451 if ( FindFocus() == this )
452 {
453 Toggle(nItem);
454 SendEvent(nItem);
455
456 // scroll one item down if the item is the last one
457 // and isn't visible at all
458 int h;
459 GetClientSize(NULL, &h);
460 if ( rect.GetBottom() > h )
461 ScrollLines(1);
462 }
463 }
464 else
465 {
466 // implement default behaviour: clicking on the item selects it
467 event.Skip();
468 }
469 }
470 else
471 {
472 // implement default behavior on click outside of client zone
473 event.Skip();
474 }
475}
476
477wxSize wxCheckListBox::DoGetBestClientSize() const
478{
479 wxSize best = wxListBox::DoGetBestClientSize();
480
481 // add room for the checkbox
482 wxSize size = wxRendererNative::Get().GetCheckBoxSize(const_cast<wxCheckListBox*>(this));
483 size.x += 2 * CHECKMARK_EXTRA_SPACE;
484 size.y += 2 * CHECKMARK_EXTRA_SPACE;
485
486 best.x += size.GetWidth();
487 if ( best.y < size.GetHeight() )
488 best.y = size.GetHeight();
489
490 CacheBestSize(best);
491 return best;
492}
493
494#endif // wxUSE_CHECKLISTBOX