1 #----------------------------------------------------------------------------
2 # Name: wxPython.lib.mixins.listctrl
3 # Purpose: Helpful mix-in classes for wxListCtrl
9 # Copyright: (c) 2001 by Total Control Software
10 # Licence: wxWindows license
11 #----------------------------------------------------------------------------
12 # 12/14/2003 - Jeff Grimmett (grimmtooth@softhome.net)
14 # o 2.5 compatability update.
15 # o ListCtrlSelectionManagerMix untested.
17 # 12/21/2003 - Jeff Grimmett (grimmtooth@softhome.net)
19 # o wxColumnSorterMixin -> ColumnSorterMixin
20 # o wxListCtrlAutoWidthMixin -> ListCtrlAutoWidthMixin
22 # 13/10/2004 - Pim Van Heuven (pim@think-wize.com)
23 # o wxTextEditMixin: Support Horizontal scrolling when TAB is pressed on long
24 # ListCtrls, support for WXK_DOWN, WXK_UP, performance improvements on
25 # very long ListCtrls, Support for virtual ListCtrls
27 # 15-Oct-2004 - Robin Dunn
28 # o wxTextEditMixin: Added Shift-TAB support
34 #----------------------------------------------------------------------------
36 class ColumnSorterMixin
:
38 A mixin class that handles sorting of a wx.ListCtrl in REPORT mode when
39 the column header is clicked on.
41 There are a few requirments needed in order for this to work genericly:
43 1. The combined class must have a GetListCtrl method that
44 returns the wx.ListCtrl to be sorted, and the list control
45 must exist at the time the wx.ColumnSorterMixin.__init__
46 method is called because it uses GetListCtrl.
48 2. Items in the list control must have a unique data value set
49 with list.SetItemData.
51 3. The combined class must have an attribute named itemDataMap
52 that is a dictionary mapping the data values to a sequence of
53 objects representing the values in each column. These values
54 are compared in the column sorter to determine sort order.
56 Interesting methods to override are GetColumnSorter,
57 GetSecondarySortValues, and GetSortImages. See below for details.
60 def __init__(self
, numColumns
):
61 self
.SetColumnCount(numColumns
)
62 list = self
.GetListCtrl()
64 raise ValueError, "No wx.ListCtrl available"
65 list.Bind(wx
.EVT_LIST_COL_CLICK
, self
.__OnColClick
, list)
68 def SetColumnCount(self
, newNumColumns
):
69 self
._colSortFlag
= [0] * newNumColumns
73 def SortListItems(self
, col
=-1, ascending
=1):
74 """Sort the list on demand. Can also be used to set the sort column and order."""
78 self
._colSortFlag
[col
] = ascending
79 self
.GetListCtrl().SortItems(self
.GetColumnSorter())
80 self
.__updateImages
(oldCol
)
83 def GetColumnWidths(self
):
85 Returns a list of column widths. Can be used to help restore the current
88 list = self
.GetListCtrl()
90 for x
in range(len(self
._colSortFlag
)):
91 rv
.append(list.GetColumnWidth(x
))
95 def GetSortImages(self
):
97 Returns a tuple of image list indexesthe indexes in the image list for an image to be put on the column
98 header when sorting in descending order.
100 return (-1, -1) # (decending, ascending) image IDs
103 def GetColumnSorter(self
):
104 """Returns a callable object to be used for comparing column values when sorting."""
105 return self
.__ColumnSorter
108 def GetSecondarySortValues(self
, col
, key1
, key2
):
109 """Returns a tuple of 2 values to use for secondary sort values when the
110 items in the selected column match equal. The default just returns the
115 def __OnColClick(self
, evt
):
117 self
._col
= col
= evt
.GetColumn()
118 self
._colSortFlag
[col
] = not self
._colSortFlag
[col
]
119 self
.GetListCtrl().SortItems(self
.GetColumnSorter())
120 self
.__updateImages
(oldCol
)
124 def __ColumnSorter(self
, key1
, key2
):
126 ascending
= self
._colSortFlag
[col
]
127 item1
= self
.itemDataMap
[key1
][col
]
128 item2
= self
.itemDataMap
[key2
][col
]
130 #--- Internationalization of string sorting with locale module
131 if type(item1
) == type('') or type(item2
) == type(''):
132 cmpVal
= locale
.strcoll(str(item1
), str(item2
))
134 cmpVal
= cmp(item1
, item2
)
137 # If the items are equal then pick something else to make the sort value unique
139 cmpVal
= apply(cmp, self
.GetSecondarySortValues(col
, key1
, key2
))
147 def __updateImages(self
, oldCol
):
148 sortImages
= self
.GetSortImages()
149 if self
._col
!= -1 and sortImages
[0] != -1:
150 img
= sortImages
[self
._colSortFlag
[self
._col
]]
151 list = self
.GetListCtrl()
153 list.ClearColumnImage(oldCol
)
154 list.SetColumnImage(self
._col
, img
)
157 #----------------------------------------------------------------------------
158 #----------------------------------------------------------------------------
160 class ListCtrlAutoWidthMixin
:
161 """ A mix-in class that automatically resizes the last column to take up
162 the remaining width of the wx.ListCtrl.
164 This causes the wx.ListCtrl to automatically take up the full width of
165 the list, without either a horizontal scroll bar (unless absolutely
166 necessary) or empty space to the right of the last column.
168 NOTE: This only works for report-style lists.
170 WARNING: If you override the EVT_SIZE event in your wx.ListCtrl, make
171 sure you call event.Skip() to ensure that the mixin's
172 _OnResize method is called.
174 This mix-in class was written by Erik Westra <ewestra@wave.co.nz>
177 """ Standard initialiser.
179 self
._resizeColMinWidth
= None
180 self
._resizeColStyle
= "LAST"
182 self
.Bind(wx
.EVT_SIZE
, self
._onResize
)
183 self
.Bind(wx
.EVT_LIST_COL_END_DRAG
, self
._onResize
, self
)
186 def setResizeColumn(self
, col
):
188 Specify which column that should be autosized. Pass either
189 'LAST' or the column number. Default is 'LAST'.
192 self
._resizeColStyle
= "LAST"
194 self
._resizeColStyle
= "COL"
195 self
._resizeCol
= col
198 def resizeLastColumn(self
, minWidth
):
199 """ Resize the last column appropriately.
201 If the list's columns are too wide to fit within the window, we use
202 a horizontal scrollbar. Otherwise, we expand the right-most column
203 to take up the remaining free space in the list.
205 This method is called automatically when the wx.ListCtrl is resized;
206 you can also call it yourself whenever you want the last column to
207 be resized appropriately (eg, when adding, removing or resizing
210 'minWidth' is the preferred minimum width for the last column.
212 self
.resizeColumn(minWidth
)
215 def resizeColumn(self
, minWidth
):
216 self
._resizeColMinWidth
= minWidth
220 # =====================
221 # == Private Methods ==
222 # =====================
224 def _onResize(self
, event
):
225 """ Respond to the wx.ListCtrl being resized.
227 We automatically resize the last column in the list.
229 if 'gtk2' in wx
.PlatformInfo
:
232 wx
.CallAfter(self
._doResize
)
237 """ Resize the last column as appropriate.
239 If the list's columns are too wide to fit within the window, we use
240 a horizontal scrollbar. Otherwise, we expand the right-most column
241 to take up the remaining free space in the list.
243 We remember the current size of the last column, before resizing,
244 as the preferred minimum width if we haven't previously been given
245 or calculated a minimum width. This ensure that repeated calls to
246 _doResize() don't cause the last column to size itself too large.
249 if not self
: # avoid a PyDeadObject error
252 numCols
= self
.GetColumnCount()
253 if numCols
== 0: return # Nothing to resize.
255 if(self
._resizeColStyle
== "LAST"):
256 resizeCol
= self
.GetColumnCount()
258 resizeCol
= self
._resizeCol
260 if self
._resizeColMinWidth
== None:
261 self
._resizeColMinWidth
= self
.GetColumnWidth(resizeCol
- 1)
263 # We're showing the vertical scrollbar -> allow for scrollbar width
264 # NOTE: on GTK, the scrollbar is included in the client size, but on
265 # Windows it is not included
266 listWidth
= self
.GetClientSize().width
267 if wx
.Platform
!= '__WXMSW__':
268 if self
.GetItemCount() > self
.GetCountPerPage():
269 scrollWidth
= wx
.SystemSettings_GetMetric(wx
.SYS_VSCROLL_X
)
270 listWidth
= listWidth
- scrollWidth
272 totColWidth
= 0 # Width of all columns except last one.
273 for col
in range(numCols
):
274 if col
!= (resizeCol
-1):
275 totColWidth
= totColWidth
+ self
.GetColumnWidth(col
)
277 resizeColWidth
= self
.GetColumnWidth(resizeCol
- 1)
279 if totColWidth
+ self
._resizeColMinWidth
> listWidth
:
280 # We haven't got the width to show the last column at its minimum
281 # width -> set it to its minimum width and allow the horizontal
283 self
.SetColumnWidth(resizeCol
-1, self
._resizeColMinWidth
)
286 # Resize the last column to take up the remaining available space.
288 self
.SetColumnWidth(resizeCol
-1, listWidth
- totColWidth
)
293 #----------------------------------------------------------------------------
294 #----------------------------------------------------------------------------
296 SEL_FOC
= wx
.LIST_STATE_SELECTED | wx
.LIST_STATE_FOCUSED
297 def selectBeforePopup(event
):
298 """Ensures the item the mouse is pointing at is selected before a popup.
300 Works with both single-select and multi-select lists."""
301 ctrl
= event
.GetEventObject()
302 if isinstance(ctrl
, wx
.ListCtrl
):
303 n
, flags
= ctrl
.HitTest(event
.GetPosition())
305 if not ctrl
.GetItemState(n
, wx
.LIST_STATE_SELECTED
):
306 for i
in range(ctrl
.GetItemCount()):
307 ctrl
.SetItemState(i
, 0, SEL_FOC
)
308 #for i in getListCtrlSelection(ctrl, SEL_FOC):
309 # ctrl.SetItemState(i, 0, SEL_FOC)
310 ctrl
.SetItemState(n
, SEL_FOC
, SEL_FOC
)
313 def getListCtrlSelection(listctrl
, state
=wx
.LIST_STATE_SELECTED
):
314 """ Returns list of item indexes of given state (selected by defaults) """
318 idx
= listctrl
.GetNextItem(idx
, wx
.LIST_NEXT_ALL
, state
)
324 wxEVT_DOPOPUPMENU
= wx
.NewEventType()
325 EVT_DOPOPUPMENU
= wx
.PyEventBinder(wxEVT_DOPOPUPMENU
, 0)
328 class ListCtrlSelectionManagerMix
:
329 """Mixin that defines a platform independent selection policy
331 As selection single and multi-select list return the item index or a
332 list of item indexes respectively.
337 self
.Bind(wx
.EVT_RIGHT_DOWN
, self
.OnLCSMRightDown
)
338 self
.Bind(EVT_DOPOPUPMENU
, self
.OnLCSMDoPopup
)
339 # self.Connect(-1, -1, self.wxEVT_DOPOPUPMENU, self.OnLCSMDoPopup)
342 def getPopupMenu(self
):
343 """ Override to implement dynamic menus (create) """
347 def setPopupMenu(self
, menu
):
348 """ Must be set for default behaviour """
352 def afterPopupMenu(self
, menu
):
353 """ Override to implement dynamic menus (destroy) """
357 def getSelection(self
):
358 res
= getListCtrlSelection(self
)
359 if self
.GetWindowStyleFlag() & wx
.LC_SINGLE_SEL
:
368 def OnLCSMRightDown(self
, event
):
369 selectBeforePopup(event
)
371 menu
= self
.getPopupMenu()
374 evt
.SetEventType(wxEVT_DOPOPUPMENU
)
376 evt
.pos
= event
.GetPosition()
377 wx
.PostEvent(self
, evt
)
380 def OnLCSMDoPopup(self
, event
):
381 self
.PopupMenu(event
.menu
, event
.pos
)
382 self
.afterPopupMenu(event
.menu
)
385 #----------------------------------------------------------------------------
386 #----------------------------------------------------------------------------
387 from bisect
import bisect
392 A mixin class that enables any text in any column of a
393 multi-column listctrl to be edited by clicking on the given row
394 and column. You close the text editor by hitting the ENTER key or
395 clicking somewhere else on the listctrl. You switch to the next
396 column by hiting TAB.
398 To use the mixin you have to include it in the class definition
399 and call the __init__ function::
401 class TestListCtrl(wx.ListCtrl, TextEditMixin):
402 def __init__(self, parent, ID, pos=wx.DefaultPosition,
403 size=wx.DefaultSize, style=0):
404 wx.ListCtrl.__init__(self, parent, ID, pos, size, style)
405 TextEditMixin.__init__(self)
408 Authors: Steve Zatz, Pim Van Heuven (pim@think-wize.com)
411 editorBgColour
= wx
.Colour(255,255,175) # Yellow
412 editorFgColour
= wx
.Colour(0,0,0) # black
415 #editor = wx.TextCtrl(self, -1, pos=(-1,-1), size=(-1,-1),
416 # style=wx.TE_PROCESS_ENTER|wx.TE_PROCESS_TAB \
420 self
.Bind(wx
.EVT_TEXT_ENTER
, self
.CloseEditor
)
421 self
.Bind(wx
.EVT_LEFT_DOWN
, self
.OnLeftDown
)
422 self
.Bind(wx
.EVT_LEFT_DCLICK
, self
.OnLeftDown
)
423 self
.Bind(wx
.EVT_LIST_ITEM_SELECTED
, self
.OnItemSelected
)
426 def make_editor(self
, col_style
=wx
.LIST_FORMAT_LEFT
):
427 editor
= wx
.PreTextCtrl()
429 style
=wx
.TE_PROCESS_ENTER|wx
.TE_PROCESS_TAB|wx
.TE_RICH2
430 style |
= {wx
.LIST_FORMAT_LEFT
: wx
.TE_LEFT
,
431 wx
.LIST_FORMAT_RIGHT
: wx
.TE_RIGHT
,
432 wx
.LIST_FORMAT_CENTRE
: wx
.TE_CENTRE
435 editor
.Create(self
, -1, style
=style
)
436 editor
.SetBackgroundColour(self
.editorBgColour
)
437 editor
.SetForegroundColour(self
.editorFgColour
)
438 font
= self
.GetFont()
447 self
.col_style
= col_style
448 self
.editor
.Bind(wx
.EVT_CHAR
, self
.OnChar
)
449 self
.editor
.Bind(wx
.EVT_KILL_FOCUS
, self
.CloseEditor
)
452 def OnItemSelected(self
, evt
):
453 self
.curRow
= evt
.GetIndex()
457 def OnChar(self
, event
):
458 ''' Catch the TAB, Shift-TAB, cursor DOWN/UP key code
459 so we can open the editor at the next column (if any).'''
461 keycode
= event
.GetKeyCode()
462 if keycode
== wx
.WXK_TAB
and event
.ShiftDown():
464 if self
.curCol
-1 >= 0:
465 self
.OpenEditor(self
.curCol
-1, self
.curRow
)
467 elif keycode
== wx
.WXK_TAB
:
469 if self
.curCol
+1 < self
.GetColumnCount():
470 self
.OpenEditor(self
.curCol
+1, self
.curRow
)
472 elif keycode
== wx
.WXK_ESCAPE
:
475 elif keycode
== wx
.WXK_DOWN
:
477 if self
.curRow
+1 < self
.GetItemCount():
478 self
._SelectIndex
(self
.curRow
+1)
479 self
.OpenEditor(self
.curCol
, self
.curRow
)
481 elif keycode
== wx
.WXK_UP
:
484 self
._SelectIndex
(self
.curRow
-1)
485 self
.OpenEditor(self
.curCol
, self
.curRow
)
491 def OnLeftDown(self
, evt
=None):
492 ''' Examine the click and double
493 click events to see if a row has been click on twice. If so,
494 determine the current row and columnn and open the editor.'''
496 if self
.editor
.IsShown():
499 x
,y
= evt
.GetPosition()
500 row
,flags
= self
.HitTest((x
,y
))
502 if row
!= self
.curRow
: # self.curRow keeps track of the current row
506 # the following should really be done in the mixin's init but
507 # the wx.ListCtrl demo creates the columns after creating the
508 # ListCtrl (generally not a good idea) on the other hand,
509 # doing this here handles adjustable column widths
513 for n
in range(self
.GetColumnCount()):
514 loc
= loc
+ self
.GetColumnWidth(n
)
515 self
.col_locs
.append(loc
)
518 col
= bisect(self
.col_locs
, x
+self
.GetScrollPos(wx
.HORIZONTAL
)) - 1
519 self
.OpenEditor(col
, row
)
522 def OpenEditor(self
, col
, row
):
523 ''' Opens an editor at the current position. '''
525 # give the derived class a chance to Allow/Veto this edit.
526 evt
= wx
.ListEvent(wx
.wxEVT_COMMAND_LIST_BEGIN_LABEL_EDIT
, self
.GetId())
527 evt
.m_itemIndex
= row
529 item
= self
.GetItem(row
, col
)
530 evt
.m_item
.SetId(item
.GetId())
531 evt
.m_item
.SetColumn(item
.GetColumn())
532 evt
.m_item
.SetData(item
.GetData())
533 evt
.m_item
.SetText(item
.GetText())
534 ret
= self
.GetEventHandler().ProcessEvent(evt
)
535 if ret
and not evt
.IsAllowed():
536 return # user code doesn't allow the edit.
538 if self
.GetColumn(col
).m_format
!= self
.col_style
:
539 self
.make_editor(self
.GetColumn(col
).m_format
)
541 x0
= self
.col_locs
[col
]
542 x1
= self
.col_locs
[col
+1] - x0
544 scrolloffset
= self
.GetScrollPos(wx
.HORIZONTAL
)
547 if x0
+x1
-scrolloffset
> self
.GetSize()[0]:
548 if wx
.Platform
== "__WXMSW__":
549 # don't start scrolling unless we really need to
550 offset
= x0
+x1
-self
.GetSize()[0]-scrolloffset
551 # scroll a bit more than what is minimum required
552 # so we don't have to scroll everytime the user presses TAB
553 # which is very tireing to the eye
554 addoffset
= self
.GetSize()[0]/4
555 # but be careful at the end of the list
556 if addoffset
+ scrolloffset
< self
.GetSize()[0]:
559 self
.ScrollList(offset
, 0)
560 scrolloffset
= self
.GetScrollPos(wx
.HORIZONTAL
)
562 # Since we can not programmatically scroll the ListCtrl
563 # close the editor so the user can scroll and open the editor
565 self
.editor
.SetValue(self
.GetItem(row
, col
).GetText())
571 y0
= self
.GetItemRect(row
)[1]
574 editor
.SetDimensions(x0
-scrolloffset
,y0
, x1
,-1)
576 editor
.SetValue(self
.GetItem(row
, col
).GetText())
579 editor
.SetSelection(-1,-1)
586 # FIXME: this function is usually called twice - second time because
587 # it is binded to wx.EVT_KILL_FOCUS. Can it be avoided? (MW)
588 def CloseEditor(self
, evt
=None):
589 ''' Close the editor and save the new value to the ListCtrl. '''
590 if not self
.editor
.IsShown():
592 text
= self
.editor
.GetValue()
596 # post wxEVT_COMMAND_LIST_END_LABEL_EDIT
597 # Event can be vetoed. It doesn't has SetEditCanceled(), what would
598 # require passing extra argument to CloseEditor()
599 evt
= wx
.ListEvent(wx
.wxEVT_COMMAND_LIST_END_LABEL_EDIT
, self
.GetId())
600 evt
.m_itemIndex
= self
.curRow
601 evt
.m_col
= self
.curCol
602 item
= self
.GetItem(self
.curRow
, self
.curCol
)
603 evt
.m_item
.SetId(item
.GetId())
604 evt
.m_item
.SetColumn(item
.GetColumn())
605 evt
.m_item
.SetData(item
.GetData())
606 evt
.m_item
.SetText(text
) #should be empty string if editor was canceled
607 ret
= self
.GetEventHandler().ProcessEvent(evt
)
608 if not ret
or evt
.IsAllowed():
610 # replace by whather you use to populate the virtual ListCtrl
612 self
.SetVirtualData(self
.curRow
, self
.curCol
, text
)
614 self
.SetStringItem(self
.curRow
, self
.curCol
, text
)
615 self
.RefreshItem(self
.curRow
)
617 def _SelectIndex(self
, row
):
618 listlen
= self
.GetItemCount()
619 if row
< 0 and not listlen
:
621 if row
> (listlen
-1):
624 self
.SetItemState(self
.curRow
, ~wx
.LIST_STATE_SELECTED
,
625 wx
.LIST_STATE_SELECTED
)
626 self
.EnsureVisible(row
)
627 self
.SetItemState(row
, wx
.LIST_STATE_SELECTED
,
628 wx
.LIST_STATE_SELECTED
)
632 #----------------------------------------------------------------------------
633 #----------------------------------------------------------------------------
636 FILENAME: CheckListCtrlMixin.py
637 AUTHOR: Bruce Who (bruce.who.hk at gmail.com)
641 This script provide a mixin for ListCtrl which add a checkbox in the first
642 column of each row. It is inspired by limodou's CheckList.py(which can be
643 got from his NewEdit) and improved:
644 - You can just use InsertStringItem() to insert new items;
645 - Once a checkbox is checked/unchecked, the corresponding item is not
647 - You can use SetItemData() and GetItemData();
648 - Interfaces are changed to OnCheckItem(), IsChecked(), CheckItem().
650 You should not set a imagelist for the ListCtrl once this mixin is used.
653 1.3 - You can check/uncheck a group of sequential items by <Shift-click>:
654 First click(or <Shift-Click>) item1 to check/uncheck it, then
655 Shift-click item2 to check/uncheck it, and you'll find that all
656 items between item1 and item2 are check/unchecked!
657 1.2 - Add ToggleItem()
658 1.1 - Initial version
661 from wx
import ImageFromStream
, BitmapFromImage
662 import cStringIO
, zlib
664 def getUncheckData():
665 return zlib
.decompress(
666 "x\xda\xeb\x0c\xf0s\xe7\xe5\x92\xe2b``\xe0\xf5\xf4p\t\x02\xd2\x02 \xcc\xc1\
667 \x06$\xe5?\xffO\x04R,\xc5N\x9e!\x1c@P\xc3\x91\xd2\x01\xe4\xbb{\xba8\x86X\xf4\
668 &\xa7\xa4$\xa5-`1\x08\\2\xbb\xb1\xb1\x91\xf5\xd8\x84o\xeb\xff\xfaw\x1d[.=[2\
669 \x90'\x01\x08v\xec]\xd3\xa3qvU`l\x81\xd9\xd18\t\xd3\x84+\x0cll[\xa6t\xcc9\
670 \xd4\xc1\xda\xc3<O\x9a1\xc3\x88\xc3j\xfa\x86_\xee@#\x19<]\xfd\\\xd69%4\x01\
673 def getUncheckBitmap():
674 return BitmapFromImage(getUncheckImage())
676 def getUncheckImage():
677 stream
= cStringIO
.StringIO(getUncheckData())
678 return ImageFromStream(stream
)
681 return zlib
.decompress(
682 'x\xda\xeb\x0c\xf0s\xe7\xe5\x92\xe2b``\xe0\xf5\xf4p\t\x02\xd2\x02 \xcc\xc1\
683 \x06$\xe5?\xffO\x04R,\xc5N\x9e!\x1c@P\xc3\x91\xd2\x01\xe47{\xba8\x86X\xf4&\
684 \xa7\xa4$\xa5-`1\x08\\2\xbb\xb1\xb1\x91\xf5\xd8\x84o\xeb\xff\xfaw\x1d[.=[2\
685 \x90\'\x01\x08v\xec\\2C\xe3\xec+\xc3\xbd\x05fG\xe3\x14n1\xcc5\xad\x8a8\x1a\
686 \xb9\xa1\xeb\xd1\x853-\xaa\xc76\xecb\xb8i\x16c&\\\xc2\xb8\xe9Xvbx\xa1T\xc3U\
687 \xd6p\'\xbd\x85\x19\xff\xbe\xbf\xd7\xe7R\xcb`\xd8\xa5\xf8\x83\xe1^\xc4\x0e\
688 \xa1"\xce\xc3n\x93x\x14\xd8\x16\xb0(\x15q)\x8b\x19\xf0U\xe4\xb10\x08V\xa8\
689 \x99\xf3\xdd\xde\xad\x06t\x0e\x83\xa7\xab\x9f\xcb:\xa7\x84&\x00\xe0HE\xab' )
691 def getCheckBitmap():
692 return BitmapFromImage(getCheckImage())
695 stream
= cStringIO
.StringIO(getCheckData())
696 return ImageFromStream(stream
)
700 class CheckListCtrlMixin
:
702 This is a mixin for ListCtrl which add a checkbox in the first
703 column of each row. It is inspired by limodou's CheckList.py(which
704 can be got from his NewEdit) and improved:
705 - You can just use InsertStringItem() to insert new items;
706 - Once a checkbox is checked/unchecked, the corresponding item is not
708 - You can use SetItemData() and GetItemData();
709 - Interfaces are changed to OnCheckItem(), IsChecked(), CheckItem().
711 You should not set a imagelist for the ListCtrl once this mixin is used.
713 def __init__(self
, check_image
=None, uncheck_image
=None):
714 self
.__imagelist
_ = wx
.ImageList(16, 16)
716 check_image
= getCheckBitmap()
717 if not uncheck_image
:
718 uncheck_image
= getUncheckBitmap()
719 self
.uncheck_image
= self
.__imagelist
_.Add(uncheck_image
)
720 self
.check_image
= self
.__imagelist
_.Add(check_image
)
721 self
.SetImageList(self
.__imagelist
_, wx
.IMAGE_LIST_SMALL
)
722 self
.__last
_check
_ = None
724 self
.Bind(wx
.EVT_LEFT_DOWN
, self
.__OnLeftDown
_)
726 # override the default methods of ListCtrl/ListView
727 self
.InsertStringItem
= self
.__InsertStringItem
_
729 # NOTE: if you use InsertItem, InsertImageItem or InsertImageStringItem,
730 # you must set the image yourself.
731 def __InsertStringItem_(self
, index
, label
):
732 index
= self
.InsertImageStringItem(index
, label
, 0)
735 def __OnLeftDown_(self
, evt
):
736 (index
, flags
) = self
.HitTest(evt
.GetPosition())
737 if flags
== wx
.LIST_HITTEST_ONITEMICON
:
738 img_idx
= self
.GetItem(index
).GetImage()
739 flag_check
= img_idx
== 0
742 if self
.__last
_check
_ is not None \
743 and wx
.GetKeyState(wx
.WXK_SHIFT
):
744 last_index
, last_flag_check
= self
.__last
_check
_
745 if last_flag_check
== flag_check
:
746 # XXX what if the previous item is deleted or new items
748 item_count
= self
.GetItemCount()
749 if last_index
< item_count
:
750 if last_index
< index
:
751 begin_index
= last_index
753 elif last_index
> index
:
755 end_index
= last_index
758 while begin_index
<= end_index
:
759 self
.CheckItem(begin_index
, flag_check
)
761 self
.__last
_check
_ = (index
, flag_check
)
765 def OnCheckItem(self
, index
, flag
):
768 def IsChecked(self
, index
):
769 return self
.GetItem(index
).GetImage() == 1
771 def CheckItem(self
, index
, check
= True):
772 img_idx
= self
.GetItem(index
).GetImage()
773 if img_idx
== 0 and check
is True:
774 self
.SetItemImage(index
, 1)
775 self
.OnCheckItem(index
, True)
776 elif img_idx
== 1 and check
is False:
777 self
.SetItemImage(index
, 0)
778 self
.OnCheckItem(index
, False)
780 def ToggleItem(self
, index
):
781 self
.CheckItem(index
, not self
.IsChecked(index
))
784 #----------------------------------------------------------------------------