1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/checklst.cpp 
   3 // Purpose:     implementation of wxCheckListBox class 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 // Licence:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  21 #pragma implementation "checklst.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  34     #include "wx/object.h" 
  35     #include "wx/colour.h" 
  37     #include "wx/bitmap.h" 
  38     #include "wx/window.h" 
  39     #include "wx/listbox.h" 
  40     #include "wx/dcmemory.h" 
  42     #include "wx/settings.h" 
  47 #include "wx/ownerdrw.h" 
  48 #include "wx/checklst.h" 
  50 #include "wx/msw/wrapwin.h" 
  53 #include "wx/msw/private.h" 
  55 #if defined(__GNUWIN32_OLD__) 
  56     #include "wx/msw/gnuwin32/extra.h" 
  59 // ---------------------------------------------------------------------------- 
  61 // ---------------------------------------------------------------------------- 
  63 // get item (converted to right type) 
  64 #define GetItem(n)    ((wxCheckListBoxItem *)(GetItem(n))) 
  66 // ============================================================================ 
  68 // ============================================================================ 
  70 // TODO: wxCONSTRUCTOR 
  71 #if 0 // wxUSE_EXTENDED_RTTI 
  72 WX_DEFINE_FLAGS( wxCheckListBoxStyle 
) 
  74 wxBEGIN_FLAGS( wxCheckListBoxStyle 
) 
  75     // new style border flags, we put them first to 
  76     // use them for streaming out 
  77     wxFLAGS_MEMBER(wxBORDER_SIMPLE
) 
  78     wxFLAGS_MEMBER(wxBORDER_SUNKEN
) 
  79     wxFLAGS_MEMBER(wxBORDER_DOUBLE
) 
  80     wxFLAGS_MEMBER(wxBORDER_RAISED
) 
  81     wxFLAGS_MEMBER(wxBORDER_STATIC
) 
  82     wxFLAGS_MEMBER(wxBORDER_NONE
) 
  84     // old style border flags 
  85     wxFLAGS_MEMBER(wxSIMPLE_BORDER
) 
  86     wxFLAGS_MEMBER(wxSUNKEN_BORDER
) 
  87     wxFLAGS_MEMBER(wxDOUBLE_BORDER
) 
  88     wxFLAGS_MEMBER(wxRAISED_BORDER
) 
  89     wxFLAGS_MEMBER(wxSTATIC_BORDER
) 
  90     wxFLAGS_MEMBER(wxNO_BORDER
) 
  92     // standard window styles 
  93     wxFLAGS_MEMBER(wxTAB_TRAVERSAL
) 
  94     wxFLAGS_MEMBER(wxCLIP_CHILDREN
) 
  95     wxFLAGS_MEMBER(wxTRANSPARENT_WINDOW
) 
  96     wxFLAGS_MEMBER(wxWANTS_CHARS
) 
  97     wxFLAGS_MEMBER(wxNO_FULL_REPAINT_ON_RESIZE
) 
  98     wxFLAGS_MEMBER(wxALWAYS_SHOW_SB 
) 
  99     wxFLAGS_MEMBER(wxVSCROLL
) 
 100     wxFLAGS_MEMBER(wxHSCROLL
) 
 102     wxFLAGS_MEMBER(wxLB_SINGLE
) 
 103     wxFLAGS_MEMBER(wxLB_MULTIPLE
) 
 104     wxFLAGS_MEMBER(wxLB_EXTENDED
) 
 105     wxFLAGS_MEMBER(wxLB_HSCROLL
) 
 106     wxFLAGS_MEMBER(wxLB_ALWAYS_SB
) 
 107     wxFLAGS_MEMBER(wxLB_NEEDED_SB
) 
 108     wxFLAGS_MEMBER(wxLB_SORT
) 
 110 wxEND_FLAGS( wxCheckListBoxStyle 
) 
 112 IMPLEMENT_DYNAMIC_CLASS_XTI(wxCheckListBox
, wxListBox
,"wx/checklst.h") 
 114 wxBEGIN_PROPERTIES_TABLE(wxCheckListBox
) 
 115         wxEVENT_PROPERTY( Toggle 
, wxEVT_COMMAND_CHECKLISTBOX_TOGGLED 
, wxCommandEvent 
) 
 117     wxPROPERTY_FLAGS( WindowStyle 
, wxCheckListBoxStyle 
, long , SetWindowStyleFlag 
, GetWindowStyleFlag 
, , 0 /*flags*/ , wxT("Helpstring") , wxT("group")) // style 
 118 wxEND_PROPERTIES_TABLE() 
 121 IMPLEMENT_DYNAMIC_CLASS(wxCheckListBox
, wxListBox
) 
 129 // ---------------------------------------------------------------------------- 
 130 // declaration and implementation of wxCheckListBoxItem class 
 131 // ---------------------------------------------------------------------------- 
 133 class wxCheckListBoxItem 
: public wxOwnerDrawn
 
 135 friend class WXDLLEXPORT wxCheckListBox
; 
 138   wxCheckListBoxItem(wxCheckListBox 
*pParent
, size_t nIndex
); 
 141   virtual bool OnDrawItem(wxDC
& dc
, const wxRect
& rc
, wxODAction act
, wxODStatus stat
); 
 143   // simple accessors and operations 
 144   bool IsChecked() const { return m_bChecked
; } 
 146   void Check(bool bCheck
); 
 147   void Toggle() { Check(!IsChecked()); } 
 153     DECLARE_NO_COPY_CLASS(wxCheckListBoxItem
) 
 155   wxCheckListBox 
*m_pParent
; 
 159 wxCheckListBoxItem::wxCheckListBoxItem(wxCheckListBox 
*pParent
, size_t nIndex
) 
 160                   : wxOwnerDrawn(wxEmptyString
, TRUE
)   // checkable 
 166   // we don't initialize m_nCheckHeight/Width vars because it's 
 167   // done in OnMeasure while they are used only in OnDraw and we 
 168   // know that there will always be OnMeasure before OnDraw 
 171   SetMarginWidth(GetDefaultMarginWidth()); 
 175  * JACS - I've got the owner-draw stuff partially working with WIN16, 
 176  * with a really horrible-looking cross for wxCheckListBox instead of a 
 177  * check - could use a bitmap check-mark instead, defined in wx.rc. 
 178  * Also there's a refresh problem whereby it doesn't always draw the 
 179  * check until you click to the right of it, which is OK for WIN32. 
 182 bool wxCheckListBoxItem::OnDrawItem(wxDC
& dc
, const wxRect
& rc
, 
 183                                     wxODAction act
, wxODStatus stat
) 
 186     stat 
= (wxOwnerDrawn::wxODStatus
)(stat 
| wxOwnerDrawn::wxODChecked
); 
 188   if ( wxOwnerDrawn::OnDrawItem(dc
, rc
, act
, stat
) ) { 
 189     // ## using native API for performance and precision 
 190     size_t nCheckWidth  
= GetDefaultMarginWidth(), 
 191          nCheckHeight 
= m_pParent
->GetItemHeight(); 
 196     HDC hdc 
= (HDC
)dc
.GetHDC(); 
 199     HPEN hpenBack 
= CreatePen(PS_SOLID
, 0, GetSysColor(COLOR_WINDOW
)), 
 200          hpenGray 
= CreatePen(PS_SOLID
, 0, RGB(128, 128, 128)), 
 201          hpenPrev 
= (HPEN
)SelectObject(hdc
, hpenBack
); 
 203     // we erase the 1-pixel border 
 204     Rectangle(hdc
, x
, y
, x 
+ nCheckWidth
, y 
+ nCheckHeight
); 
 206     // shift check mark 1 pixel to the right (it looks better like this) 
 210       // first create a monochrome bitmap in a memory DC 
 211       HDC hdcMem 
= CreateCompatibleDC(hdc
); 
 212       HBITMAP hbmpCheck 
= CreateBitmap(nCheckWidth
, nCheckHeight
, 1, 1, 0); 
 213       HBITMAP hbmpOld 
= (HBITMAP
)SelectObject(hdcMem
, hbmpCheck
); 
 215       // then draw a check mark into it 
 220       rect
.right  
= nCheckWidth
; 
 221       rect
.bottom 
= nCheckHeight
; 
 224       DrawFrameControl(hdcMem
, &rect
, DFC_BUTTON
, DFCS_BUTTONCHECK
); 
 226       DrawFrameControl(hdcMem
, &rect
, DFC_MENU
, DFCS_MENUCHECK
); 
 229       // finally copy it to screen DC and clean up 
 230       BitBlt(hdc
, x
, y
, nCheckWidth 
- 1, nCheckHeight
, 
 231              hdcMem
, 0, 0, SRCCOPY
); 
 233       SelectObject(hdcMem
, hbmpOld
); 
 234       DeleteObject(hbmpCheck
); 
 238     // now we draw the smaller rectangle 
 243     // draw hollow gray rectangle 
 244     (void)SelectObject(hdc
, hpenGray
); 
 245     HBRUSH hbrPrev  
= (HBRUSH
)SelectObject(hdc
, GetStockObject(NULL_BRUSH
)); 
 246     Rectangle(hdc
, x
, y
, x 
+ nCheckWidth
, y 
+ nCheckHeight
); 
 249     (void)SelectObject(hdc
, hpenPrev
); 
 250     (void)SelectObject(hdc
, hbrPrev
); 
 252     DeleteObject(hpenBack
); 
 253     DeleteObject(hpenGray
); 
 256     dc.DrawRectangle(x, y, nCheckWidth, nCheckHeight); 
 259       dc.DrawLine(x, y, x + nCheckWidth, y + nCheckHeight); 
 260       dc.DrawLine(x, y + nCheckHeight, x + nCheckWidth, y); 
 270 // change the state of the item and redraw it 
 271 void wxCheckListBoxItem::Check(bool check
) 
 275     // index may be changed because new items were added/deleted 
 276     if ( m_pParent
->GetItemIndex(this) != (int)m_nIndex 
) 
 279         int index 
= m_pParent
->GetItemIndex(this); 
 281         wxASSERT_MSG( index 
!= wxNOT_FOUND
, wxT("what does this item do here?") ); 
 283         m_nIndex 
= (size_t)index
; 
 286     HWND hwndListbox 
= (HWND
)m_pParent
->GetHWND(); 
 291         if ( ::SendMessage(hwndListbox
, LB_GETITEMRECT
, 
 292                            m_nIndex
, (LPARAM
)&rcUpdate
) == LB_ERR 
) 
 294             wxLogDebug(wxT("LB_GETITEMRECT failed")); 
 297         // FIXME this doesn't work if the listbox is scrolled! 
 298         size_t nHeight 
= m_pParent
->GetItemHeight(); 
 299         size_t y 
= m_nIndex 
* nHeight
; 
 303         rcUpdate
.right  
= GetDefaultMarginWidth() ; 
 304         rcUpdate
.bottom 
= y 
+ nHeight 
; 
 307     InvalidateRect(hwndListbox
, &rcUpdate
, FALSE
); 
 310 // send an "item checked" event 
 311 void wxCheckListBoxItem::SendEvent() 
 313     wxCommandEvent 
event(wxEVT_COMMAND_CHECKLISTBOX_TOGGLED
, m_pParent
->GetId()); 
 314     event
.SetInt(m_nIndex
); 
 315     event
.SetEventObject(m_pParent
); 
 316     m_pParent
->ProcessCommand(event
); 
 319 // ---------------------------------------------------------------------------- 
 320 // implementation of wxCheckListBox class 
 321 // ---------------------------------------------------------------------------- 
 323 // define event table 
 324 // ------------------ 
 325 BEGIN_EVENT_TABLE(wxCheckListBox
, wxListBox
) 
 326   EVT_KEY_DOWN(wxCheckListBox::OnKeyDown
) 
 327   EVT_LEFT_DOWN(wxCheckListBox::OnLeftClick
) 
 333 // def ctor: use Create() to really create the control 
 334 wxCheckListBox::wxCheckListBox() 
 338 // ctor which creates the associated control 
 339 wxCheckListBox::wxCheckListBox(wxWindow 
*parent
, wxWindowID id
, 
 340                                const wxPoint
& pos
, const wxSize
& size
, 
 341                                int nStrings
, const wxString choices
[], 
 342                                long style
, const wxValidator
& val
, 
 343                                const wxString
& name
) 
 345     Create(parent
, id
, pos
, size
, nStrings
, choices
, style
, val
, name
); 
 348 bool wxCheckListBox::Create(wxWindow 
*parent
, wxWindowID id
, 
 349                             const wxPoint
& pos
, const wxSize
& size
, 
 350                             int n
, const wxString choices
[], 
 352                             const wxValidator
& validator
, const wxString
& name
) 
 354     return wxListBox::Create(parent
, id
, pos
, size
, n
, choices
, 
 355                              style 
| wxLB_OWNERDRAW
, validator
, name
); 
 358 // misc overloaded methods 
 359 // ----------------------- 
 361 void wxCheckListBox::Delete(int N
) 
 363     wxCHECK_RET( N 
>= 0 && N 
< m_noItems
, 
 364                  wxT("invalid index in wxListBox::Delete") ); 
 366     wxListBox::Delete(N
); 
 371     m_aItems
.RemoveAt(N
); 
 374 bool wxCheckListBox::SetFont( const wxFont 
&font 
) 
 377     for ( i 
= 0; i 
< m_aItems
.GetCount(); i
++ ) 
 378         m_aItems
[i
]->SetFont(font
); 
 380     wxListBox::SetFont(font
); 
 385 // create/retrieve item 
 386 // -------------------- 
 388 // create a check list box item 
 389 wxOwnerDrawn 
*wxCheckListBox::CreateLboxItem(size_t nIndex
) 
 391   wxCheckListBoxItem 
*pItem 
= new wxCheckListBoxItem(this, nIndex
); 
 397 bool wxCheckListBox::MSWOnMeasure(WXMEASUREITEMSTRUCT 
*item
) 
 399   if ( wxListBox::MSWOnMeasure(item
) ) { 
 400     MEASUREITEMSTRUCT 
*pStruct 
= (MEASUREITEMSTRUCT 
*)item
; 
 403     m_nItemHeight 
= pStruct
->itemHeight
; 
 405     // add place for the check mark 
 406     pStruct
->itemWidth 
+= wxOwnerDrawn::GetDefaultMarginWidth(); 
 417 bool wxCheckListBox::IsChecked(size_t uiIndex
) const 
 419     wxCHECK_MSG( uiIndex 
< (size_t)GetCount(), FALSE
, _T("bad wxCheckListBox index") ); 
 421     return GetItem(uiIndex
)->IsChecked(); 
 424 void wxCheckListBox::Check(size_t uiIndex
, bool bCheck
) 
 426     wxCHECK_RET( uiIndex 
< (size_t)GetCount(), _T("bad wxCheckListBox index") ); 
 428     GetItem(uiIndex
)->Check(bCheck
); 
 434 void wxCheckListBox::OnKeyDown(wxKeyEvent
& event
) 
 445     switch ( event
.GetKeyCode() ) 
 456         case WXK_NUMPAD_SUBTRACT
: 
 467         wxArrayInt selections
; 
 469         if ( HasMultipleSelection() ) 
 471             count 
= GetSelections(selections
); 
 475             int sel 
= GetSelection(); 
 483         for ( int i 
= 0; i 
< count
; i
++ ) 
 485             wxCheckListBoxItem 
*item 
= GetItem(selections
[i
]); 
 488                 wxFAIL_MSG( _T("no wxCheckListBoxItem?") ); 
 500                     item
->Check( oper 
== Set 
); 
 504                     wxFAIL_MSG( _T("what should this key do?") ); 
 507             // we should send an event as this has been done by the user and 
 508             // not by the program 
 512     else // nothing to do 
 518 void wxCheckListBox::OnLeftClick(wxMouseEvent
& event
) 
 520   // clicking on the item selects it, clicking on the checkmark toggles 
 521   if ( event
.GetX() <= wxOwnerDrawn::GetDefaultMarginWidth() ) { 
 522     int nItem 
= HitTest(event
.GetX(), event
.GetY()); 
 524     if ( nItem 
!= wxNOT_FOUND 
) { 
 525       wxCheckListBoxItem 
*item 
= GetItem(nItem
); 
 529     //else: it's not an error, just click outside of client zone 
 532     // implement default behaviour: clicking on the item selects it 
 537 int wxCheckListBox::DoHitTestItem(wxCoord x
, wxCoord y
) const 
 540     int nItem 
= (int)::SendMessage
 
 548     // FIXME this doesn't work when the listbox is scrolled! 
 549     int nItem 
= y 
/ m_nItemHeight
; 
 552   return nItem 
>= m_noItems 
? wxNOT_FOUND 
: nItem
;