Patch from Andrea that fixes the following problems/issues:

a) ZeroDivisionError when using the Vista selection style and calling
SelectItem; for some strange reason, sometimes the item rect is not
initialized and that generates the ZeroDivisionError when painting the
selection rectangle;
b) Added a DeleteWindow method to GenericTreeItem class, for items
that hold a widget next to them;
c) Renamed CustomTreeCtrl method IsEnabled to IsItemEnabled, otherwise
it conflicts with wx.Window.IsEnabled;
d) Now CustomTreeCtrl behaves correctly when the widget attached to an
item is narrower (in height) than the item text;


git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@45520 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Robin Dunn 2007-04-17 16:26:28 +00:00
parent 59953bf4ba
commit be3eff6271

View File

@ -3,7 +3,7 @@
# Inspired By And Heavily Based On wxGenericTreeCtrl. # Inspired By And Heavily Based On wxGenericTreeCtrl.
# #
# Andrea Gavana, @ 17 May 2006 # Andrea Gavana, @ 17 May 2006
# Latest Revision: 01 Apr 2007, 22.30 CET # Latest Revision: 16 Apr 2007, 11.00 CET
# #
# #
# TODO List # TODO List
@ -134,7 +134,7 @@ CustomTreeCtrl has been tested on the following platforms:
* Mac OS (Thanks to John Jackson). * Mac OS (Thanks to John Jackson).
Latest Revision: Andrea Gavana @ 01 Apr 2007, 22.30 CET Latest Revision: Andrea Gavana @ 16 Apr 2007, 11.00 CET
Version 1.0 Version 1.0
""" """
@ -1164,28 +1164,7 @@ class GenericTreeItem:
self._wnd = wnd # are we holding a window? self._wnd = wnd # are we holding a window?
if wnd: if wnd:
if wnd.GetSizer(): # the window is a complex one hold by a sizer self.SetWindow(wnd)
size = wnd.GetBestSize()
else: # simple window, without sizers
size = wnd.GetSize()
# We have to bind the wx.EVT_SET_FOCUS for the associated window
# No other solution to handle the focus changing from an item in
# CustomTreeCtrl and the window associated to an item
# Do better strategies exist?
self._wnd.Bind(wx.EVT_SET_FOCUS, self.OnSetFocus)
self._height = size.GetHeight() + 2
self._width = size.GetWidth()
self._windowsize = size
# We don't show the window if the item is collapsed
if self._isCollapsed:
self._wnd.Show(False)
# The window is enabled only if the item is enabled
self._wnd.Enable(self._enabled)
self._windowenabled = self._enabled
def IsOk(self): def IsOk(self):
@ -1310,6 +1289,29 @@ class GenericTreeItem:
self._wnd = wnd self._wnd = wnd
if wnd.GetSizer(): # the window is a complex one hold by a sizer
size = wnd.GetBestSize()
else: # simple window, without sizers
size = wnd.GetSize()
# We have to bind the wx.EVT_SET_FOCUS for the associated window
# No other solution to handle the focus changing from an item in
# CustomTreeCtrl and the window associated to an item
# Do better strategies exist?
self._wnd.Bind(wx.EVT_SET_FOCUS, self.OnSetFocus)
self._height = size.GetHeight() + 2
self._width = size.GetWidth()
self._windowsize = size
# We don't show the window if the item is collapsed
if self._isCollapsed:
self._wnd.Show(False)
# The window is enabled only if the item is enabled
self._wnd.Enable(self._enabled)
self._windowenabled = self._enabled
def GetWindow(self): def GetWindow(self):
"""Returns the window associated to the item.""" """Returns the window associated to the item."""
@ -1317,6 +1319,14 @@ class GenericTreeItem:
return self._wnd return self._wnd
def DeleteWindow(self):
"""Deletes the window associated to the item (if any)."""
if self._wnd:
self._wnd.Destroy()
self._wnd = None
def GetWindowEnabled(self): def GetWindowEnabled(self):
"""Returns whether the associated window is enabled or not.""" """Returns whether the associated window is enabled or not."""
@ -2055,7 +2065,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
self.RefreshLine(item) self.RefreshLine(item)
def IsEnabled(self, item): def IsItemEnabled(self, item):
"""Returns whether an item is enabled or disabled.""" """Returns whether an item is enabled or disabled."""
if not item: if not item:
@ -4202,6 +4212,8 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
r2, g2, b2 = int(bottom.Red()), int(bottom.Green()), int(bottom.Blue()) r2, g2, b2 = int(bottom.Red()), int(bottom.Green()), int(bottom.Blue())
flrect = float(filRect.height) flrect = float(filRect.height)
if flrect < 1:
flrect = self._lineHeight
rstep = float((r2 - r1)) / flrect rstep = float((r2 - r1)) / flrect
gstep = float((g2 - g1)) / flrect gstep = float((g2 - g1)) / flrect
@ -4244,7 +4256,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
dc.SetTextForeground(self.GetHyperTextVisitedColour()) dc.SetTextForeground(self.GetHyperTextVisitedColour())
else: else:
dc.SetTextForeground(self.GetHyperTextNewColour()) dc.SetTextForeground(self.GetHyperTextNewColour())
text_w, text_h, dummy = dc.GetMultiLineTextExtent(item.GetText()) text_w, text_h, dummy = dc.GetMultiLineTextExtent(item.GetText())
image = item.GetCurrentImage() image = item.GetCurrentImage()
@ -4316,7 +4328,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
wx.RendererNative.Get().DrawItemSelectionRect(self, dc, itemrect, flags) wx.RendererNative.Get().DrawItemSelectionRect(self, dc, itemrect, flags)
else: else:
dc.DrawRectangleRect(itemrect) dc.DrawRectangleRect(itemrect)
else: else:
if item.IsSelected(): if item.IsSelected():
@ -4415,6 +4427,9 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
wndx = wcheck + image_w + item.GetX() + text_w + 4 wndx = wcheck + image_w + item.GetX() + text_w + 4
xa, ya = self.CalcScrolledPosition((0, item.GetY())) xa, ya = self.CalcScrolledPosition((0, item.GetY()))
wndx += xa wndx += xa
if item.GetHeight() > item.GetWindowSize()[1]:
ya += (item.GetHeight() - item.GetWindowSize()[1])/2
if not wnd.IsShown(): if not wnd.IsShown():
wnd.Show() wnd.Show()
if wnd.GetPosition() != (wndx, ya): if wnd.GetPosition() != (wndx, ya):
@ -4742,11 +4757,11 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
keyCode = event.GetKeyCode() keyCode = event.GetKeyCode()
if keyCode in [ord("+"), wx.WXK_ADD]: # "+" if keyCode in [ord("+"), wx.WXK_ADD]: # "+"
if self._current.HasPlus() and not self.IsExpanded(self._current) and self.IsEnabled(self._current): if self._current.HasPlus() and not self.IsExpanded(self._current) and self.IsItemEnabled(self._current):
self.Expand(self._current) self.Expand(self._current)
elif keyCode in [ord("*"), wx.WXK_MULTIPLY]: # "*" elif keyCode in [ord("*"), wx.WXK_MULTIPLY]: # "*"
if not self.IsExpanded(self._current) and self.IsEnabled(self._current): if not self.IsExpanded(self._current) and self.IsItemEnabled(self._current):
# expand all # expand all
self.ExpandAll(self._current) self.ExpandAll(self._current)
@ -4766,7 +4781,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
elif keyCode in [wx.WXK_RETURN, wx.WXK_SPACE]: elif keyCode in [wx.WXK_RETURN, wx.WXK_SPACE]:
if not self.IsEnabled(self._current): if not self.IsItemEnabled(self._current):
event.Skip() event.Skip()
return return
@ -4799,7 +4814,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
if prev: if prev:
current = self._key_current current = self._key_current
# TODO: Huh? If we get here, we'd better be the first child of our parent. How else could it be? # TODO: Huh? If we get here, we'd better be the first child of our parent. How else could it be?
if current == self.GetFirstChild(prev)[0] and self.IsEnabled(prev): if current == self.GetFirstChild(prev)[0] and self.IsItemEnabled(prev):
# otherwise we return to where we came from # otherwise we return to where we came from
self.DoSelectItem(prev, unselect_others, extended_select) self.DoSelectItem(prev, unselect_others, extended_select)
self._key_current = prev self._key_current = prev
@ -4815,13 +4830,13 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
current = prev current = prev
# Try to get the previous siblings and see if they are active # Try to get the previous siblings and see if they are active
while prev and not self.IsEnabled(prev): while prev and not self.IsItemEnabled(prev):
prev = self.GetPrevSibling(prev) prev = self.GetPrevSibling(prev)
if not prev: if not prev:
# No previous siblings active: go to the parent and up # No previous siblings active: go to the parent and up
prev = self.GetItemParent(current) prev = self.GetItemParent(current)
while prev and not self.IsEnabled(prev): while prev and not self.IsItemEnabled(prev):
prev = self.GetItemParent(prev) prev = self.GetItemParent(prev)
if prev: if prev:
@ -4839,7 +4854,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
if self.IsExpanded(self._current): if self.IsExpanded(self._current):
self.Collapse(self._current) self.Collapse(self._current)
else: else:
if prev and self.IsEnabled(prev): if prev and self.IsItemEnabled(prev):
self.DoSelectItem(prev, unselect_others, extended_select) self.DoSelectItem(prev, unselect_others, extended_select)
elif keyCode == wx.WXK_RIGHT: elif keyCode == wx.WXK_RIGHT:
@ -4847,7 +4862,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
# also expand the item if it wasn't expanded yet # also expand the item if it wasn't expanded yet
if self.IsExpanded(self._current) and self.HasChildren(self._current): if self.IsExpanded(self._current) and self.HasChildren(self._current):
child, cookie = self.GetFirstChild(self._key_current) child, cookie = self.GetFirstChild(self._key_current)
if self.IsEnabled(child): if self.IsItemEnabled(child):
self.DoSelectItem(child, unselect_others, extended_select) self.DoSelectItem(child, unselect_others, extended_select)
self._key_current = child self._key_current = child
else: else:
@ -4873,11 +4888,11 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
current = self.GetItemParent(current) current = self.GetItemParent(current)
if current: if current:
next = self.GetNextSibling(current) next = self.GetNextSibling(current)
if not next or not self.IsEnabled(next): if not next or not self.IsItemEnabled(next):
next = None next = None
else: else:
while next and not self.IsEnabled(next): while next and not self.IsItemEnabled(next):
next = self.GetNext(next) next = self.GetNext(next)
if next: if next:
@ -4902,7 +4917,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
last = lastChild last = lastChild
if last and self.IsEnabled(last): if last and self.IsItemEnabled(last):
self.DoSelectItem(last, unselect_others, extended_select) self.DoSelectItem(last, unselect_others, extended_select)
@ -4919,7 +4934,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
if not prev: if not prev:
return return
if self.IsEnabled(prev): if self.IsItemEnabled(prev):
self.DoSelectItem(prev, unselect_others, extended_select) self.DoSelectItem(prev, unselect_others, extended_select)
else: else:
@ -4936,7 +4951,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
# no such item # no such item
return return
if self.IsEnabled(id): if self.IsItemEnabled(id):
self.SelectItem(id) self.SelectItem(id)
self._findPrefix += ch self._findPrefix += ch
@ -4968,7 +4983,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
while 1: while 1:
child = sibling(item) child = sibling(item)
if (child and self.IsEnabled(child)) or not child: if (child and self.IsItemEnabled(child)) or not child:
break break
item = child item = child
@ -4976,10 +4991,10 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
# Tha's not a radiobutton... but some of its children can be # Tha's not a radiobutton... but some of its children can be
# inactive # inactive
child, cookie = self.GetFirstChild(item) child, cookie = self.GetFirstChild(item)
while child and not self.IsEnabled(child): while child and not self.IsItemEnabled(child):
child, cookie = self.GetNextChild(item, cookie) child, cookie = self.GetNextChild(item, cookie)
if child and self.IsEnabled(child): if child and self.IsItemEnabled(child):
return child return child
return None return None
@ -5033,7 +5048,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
flags = TREE_HITTEST_NOWHERE flags = TREE_HITTEST_NOWHERE
return None, flags return None, flags
if not self.IsEnabled(hit): if not self.IsItemEnabled(hit):
return None, flags return None, flags
return hit, flags return hit, flags
@ -5552,6 +5567,7 @@ class CustomTreeCtrl(wx.PyScrolledWindow):
item.SetHeight(total_h) item.SetHeight(total_h)
else: else:
item.SetWidth(item.GetWindowSize()[0]+image_w+text_w+wcheck+2) item.SetWidth(item.GetWindowSize()[0]+image_w+text_w+wcheck+2)
item.SetHeight(max(total_h, item.GetWindowSize()[1]))
def CalculateLevel(self, item, dc, level, y): def CalculateLevel(self, item, dc, level, y):