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 license 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21 #pragma implementation "checklst.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  33 #include "wx/object.h" 
  34 #include "wx/colour.h" 
  36 #include "wx/bitmap.h" 
  37 #include "wx/window.h" 
  38 #include "wx/listbox.h" 
  39 #include "wx/ownerdrw.h" 
  40 #include "wx/settings.h" 
  41 #include "wx/dcmemory.h" 
  42 #include "wx/msw/checklst.h" 
  48 #if defined(__GNUWIN32__) && !defined(wxUSE_NORLANDER_HEADERS) 
  49 #include "wx/msw/gnuwin32/extra.h" 
  52 // ---------------------------------------------------------------------------- 
  54 // ---------------------------------------------------------------------------- 
  56 // get item (converted to right type) 
  57 #define GetItem(n)    ((wxCheckListBoxItem *)(GetItem(n))) 
  59 // ============================================================================ 
  61 // ============================================================================ 
  63   IMPLEMENT_DYNAMIC_CLASS(wxCheckListBox
, wxListBox
) 
  65 // ---------------------------------------------------------------------------- 
  66 // declaration and implementation of wxCheckListBoxItem class 
  67 // ---------------------------------------------------------------------------- 
  69 class wxCheckListBoxItem 
: public wxOwnerDrawn
 
  71 friend class wxCheckListBox
; 
  74   wxCheckListBoxItem(wxCheckListBox 
*pParent
, size_t nIndex
); 
  77   virtual bool OnDrawItem(wxDC
& dc
, const wxRect
& rc
, wxODAction act
, wxODStatus stat
); 
  80   bool IsChecked() const  { return m_bChecked
;        } 
  81   void Check(bool bCheck
); 
  82   void Toggle() { Check(!IsChecked()); } 
  86   wxCheckListBox 
*m_pParent
; 
  90 wxCheckListBoxItem::wxCheckListBoxItem(wxCheckListBox 
*pParent
, size_t nIndex
) 
  91                   : wxOwnerDrawn("", TRUE
)   // checkable 
  97   // we don't initialize m_nCheckHeight/Width vars because it's 
  98   // done in OnMeasure while they are used only in OnDraw and we 
  99   // know that there will always be OnMeasure before OnDraw 
 102   SetMarginWidth(GetDefaultMarginWidth()); 
 106  * JACS - I've got the owner-draw stuff partially working with WIN16, 
 107  * with a really horrible-looking cross for wxCheckListBox instead of a 
 108  * check - could use a bitmap check-mark instead, defined in wx.rc. 
 109  * Also there's a refresh problem whereby it doesn't always draw the 
 110  * check until you click to the right of it, which is OK for WIN32. 
 113 bool wxCheckListBoxItem::OnDrawItem(wxDC
& dc
, const wxRect
& rc
, 
 114                                     wxODAction act
, wxODStatus stat
) 
 117     stat 
= (wxOwnerDrawn::wxODStatus
)(stat 
| wxOwnerDrawn::wxODChecked
); 
 119   if ( wxOwnerDrawn::OnDrawItem(dc
, rc
, act
, stat
) ) { 
 120     // ## using native API for performance and precision 
 121     size_t nCheckWidth  
= GetDefaultMarginWidth(), 
 122          nCheckHeight 
= m_pParent
->GetItemHeight(); 
 127     HDC hdc 
= (HDC
)dc
.GetHDC(); 
 130     HPEN hpenBack 
= CreatePen(PS_SOLID
, 0, GetSysColor(COLOR_WINDOW
)), 
 131          hpenGray 
= CreatePen(PS_SOLID
, 0, RGB(128, 128, 128)), 
 132          hpenPrev 
= (HPEN
)SelectObject(hdc
, hpenBack
); 
 134     // we erase the 1-pixel border 
 135     Rectangle(hdc
, x
, y
, x 
+ nCheckWidth
, y 
+ nCheckHeight
); 
 137     // shift check mark 1 pixel to the right (it looks better like this) 
 141       // first create a monochrome bitmap in a memory DC 
 142       HDC hdcMem 
= CreateCompatibleDC(hdc
); 
 143       HBITMAP hbmpCheck 
= CreateBitmap(nCheckWidth
, nCheckHeight
, 1, 1, 0); 
 144       HBITMAP hbmpOld 
= (HBITMAP
)SelectObject(hdcMem
, hbmpCheck
); 
 146       // then draw a check mark into it 
 150       rect
.right  
= nCheckWidth 
; 
 151       rect
.bottom 
= nCheckHeight 
; 
 155       DrawFrameControl(hdcMem
, &rect
, DFC_MENU
, DFCS_MENUCHECK
); 
 158       // In WIN16, draw a cross 
 159       HPEN blackPen 
= CreatePen(PS_SOLID
, 1, RGB(0, 0, 0)); 
 160       HPEN whiteBrush 
= (HPEN
)GetStockObject(WHITE_BRUSH
); 
 161       HPEN hPenOld 
= (HPEN
)::SelectObject(hdcMem
, blackPen
); 
 162       HPEN hBrushOld 
= (HPEN
)::SelectObject(hdcMem
, whiteBrush
); 
 163       ::SetROP2(hdcMem
, R2_COPYPEN
); 
 164       Rectangle(hdcMem
, 0, 0, nCheckWidth
, nCheckHeight
); 
 165       MoveTo(hdcMem
, 0, 0); 
 166       LineTo(hdcMem
, nCheckWidth
, nCheckHeight
); 
 167       MoveTo(hdcMem
, nCheckWidth
, 0); 
 168       LineTo(hdcMem
, 0, nCheckHeight
); 
 169       ::SelectObject(hdcMem
, hPenOld
); 
 170       ::SelectObject(hdcMem
, hBrushOld
); 
 171       ::DeleteObject(blackPen
); 
 174       // finally copy it to screen DC and clean up 
 175       BitBlt(hdc
, x
, y
, nCheckWidth 
- 1, nCheckHeight
, 
 176              hdcMem
, 0, 0, SRCCOPY
); 
 178       SelectObject(hdcMem
, hbmpOld
); 
 179       DeleteObject(hbmpCheck
); 
 183     // now we draw the smaller rectangle 
 188     // draw hollow gray rectangle 
 189     (void)SelectObject(hdc
, hpenGray
); 
 190     HBRUSH hbrPrev  
= (HBRUSH
)SelectObject(hdc
, GetStockObject(NULL_BRUSH
)); 
 191     Rectangle(hdc
, x
, y
, x 
+ nCheckWidth
, y 
+ nCheckHeight
); 
 194     (void)SelectObject(hdc
, hpenPrev
); 
 195     (void)SelectObject(hdc
, hbrPrev
); 
 197     DeleteObject(hpenBack
); 
 198     DeleteObject(hpenGray
); 
 201     dc.DrawRectangle(x, y, nCheckWidth, nCheckHeight); 
 204       dc.DrawLine(x, y, x + nCheckWidth, y + nCheckHeight); 
 205       dc.DrawLine(x, y + nCheckHeight, x + nCheckWidth, y); 
 215 // change the state of the item and redraw it 
 216 void wxCheckListBoxItem::Check(bool check
) 
 220     // index may be chanegd because new items were added/deleted 
 221     if ( m_pParent
->GetItemIndex(this) != (int)m_nIndex 
) 
 224         int index 
= m_pParent
->GetItemIndex(this); 
 226         wxASSERT_MSG( index 
!= wxNOT_FOUND
, wxT("what does this item do here?") ); 
 228         m_nIndex 
= (size_t)index
; 
 231     HWND hwndListbox 
= (HWND
)m_pParent
->GetHWND(); 
 236         if ( ::SendMessage(hwndListbox
, LB_GETITEMRECT
, 
 237                            m_nIndex
, (LPARAM
)&rcUpdate
) == LB_ERR 
) 
 239             wxLogDebug(wxT("LB_GETITEMRECT failed")); 
 242         // FIXME this doesn't work if the listbox is scrolled! 
 243         size_t nHeight 
= m_pParent
->GetItemHeight(); 
 244         size_t y 
= m_nIndex 
* nHeight
; 
 248         rcUpdate
.right  
= GetDefaultMarginWidth() ; 
 249         rcUpdate
.bottom 
= y 
+ nHeight 
; 
 252     InvalidateRect(hwndListbox
, &rcUpdate
, FALSE
); 
 254     wxCommandEvent 
event(wxEVT_COMMAND_CHECKLISTBOX_TOGGLED
, m_pParent
->GetId()); 
 255     event
.SetInt(m_nIndex
); 
 256     event
.SetEventObject(m_pParent
); 
 257     m_pParent
->ProcessCommand(event
); 
 260 // ---------------------------------------------------------------------------- 
 261 // implementation of wxCheckListBox class 
 262 // ---------------------------------------------------------------------------- 
 264 // define event table 
 265 // ------------------ 
 266 BEGIN_EVENT_TABLE(wxCheckListBox
, wxListBox
) 
 267   EVT_CHAR(wxCheckListBox::OnChar
) 
 268   EVT_LEFT_DOWN(wxCheckListBox::OnLeftClick
) 
 274 // def ctor: use Create() to really create the control 
 275 wxCheckListBox::wxCheckListBox() : wxListBox() 
 279 // ctor which creates the associated control 
 280 wxCheckListBox::wxCheckListBox(wxWindow 
*parent
, wxWindowID id
, 
 281                                const wxPoint
& pos
, const wxSize
& size
, 
 282                                int nStrings
, const wxString choices
[], 
 283                                long style
, const wxValidator
& val
, 
 284                                const wxString
& name
) 
 287     Create(parent
, id
, pos
, size
, nStrings
, choices
, 
 288            style 
| wxLB_OWNERDRAW
, val
, name
); 
 291 void wxCheckListBox::Delete(int N
) 
 293     wxCHECK_RET( N 
>= 0 && N 
< m_noItems
, 
 294                  wxT("invalid index in wxListBox::Delete") ); 
 296     wxListBox::Delete(N
); 
 304 void wxCheckListBox::InsertItems(int nItems
, const wxString items
[], int pos
) 
 306     wxCHECK_RET( pos 
>= 0 && pos 
<= m_noItems
, 
 307                  wxT("invalid index in wxCheckListBox::InsertItems") ); 
 309     wxListBox::InsertItems(nItems
, items
, pos
); 
 312     for ( i 
= 0; i 
< nItems
; i
++ ) { 
 313         wxOwnerDrawn 
*pNewItem 
= CreateItem((size_t)(pos 
+ i
)); 
 314         pNewItem
->SetName(items
[i
]); 
 315         pNewItem
->SetFont(GetFont()); 
 317         m_aItems
.Insert(pNewItem
, (size_t)(pos 
+ i
)); 
 319         ListBox_SetItemData((HWND
)GetHWND(), i 
+ pos
, pNewItem
); 
 324 bool wxCheckListBox::SetFont( const wxFont 
&font 
) 
 327     for ( i 
= 0; i 
< m_aItems
.GetCount(); i
++ ) 
 328         m_aItems
[i
]->SetFont(font
); 
 330     wxListBox::SetFont(font
); 
 335 // create/retrieve item 
 336 // -------------------- 
 338 // create a check list box item 
 339 wxOwnerDrawn 
*wxCheckListBox::CreateItem(size_t nIndex
) 
 341   wxCheckListBoxItem 
*pItem 
= new wxCheckListBoxItem(this, nIndex
); 
 347 bool wxCheckListBox::MSWOnMeasure(WXMEASUREITEMSTRUCT 
*item
) 
 349   if ( wxListBox::MSWOnMeasure(item
) ) { 
 350     MEASUREITEMSTRUCT 
*pStruct 
= (MEASUREITEMSTRUCT 
*)item
; 
 353     m_nItemHeight 
= pStruct
->itemHeight
; 
 355     // add place for the check mark 
 356     pStruct
->itemWidth 
+= wxOwnerDrawn::GetDefaultMarginWidth(); 
 367 bool wxCheckListBox::IsChecked(size_t uiIndex
) const 
 369   return GetItem(uiIndex
)->IsChecked(); 
 372 void wxCheckListBox::Check(size_t uiIndex
, bool bCheck
) 
 374   GetItem(uiIndex
)->Check(bCheck
); 
 380 void wxCheckListBox::OnChar(wxKeyEvent
& event
) 
 382   if ( event
.KeyCode() == WXK_SPACE 
) 
 383     GetItem(GetSelection())->Toggle(); 
 388 void wxCheckListBox::OnLeftClick(wxMouseEvent
& event
) 
 390   // clicking on the item selects it, clicking on the checkmark toggles 
 391   if ( event
.GetX() <= wxOwnerDrawn::GetDefaultMarginWidth() ) { 
 393       size_t nItem 
= (size_t)::SendMessage
 
 398                                 MAKELPARAM(event
.GetX(), event
.GetY()) 
 401         // FIXME this doesn't work when the listbox is scrolled! 
 402         size_t nItem 
= ((size_t)event
.GetY()) / m_nItemHeight
; 
 405     if ( nItem 
< (size_t)m_noItems 
) 
 406       GetItem(nItem
)->Toggle(); 
 407     //else: it's not an error, just click outside of client zone 
 410     // implement default behaviour: clicking on the item selects it