d14a1e2856
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@24541 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
1142 lines
39 KiB
Python
1142 lines
39 KiB
Python
"""
|
|
wxMVCTree is a control which handles hierarchical data. It is constructed
|
|
in model-view-controller architecture, so the display of that data, and
|
|
the content of the data can be changed greatly without affecting the other parts.
|
|
|
|
wxMVCTree actually is even more configurable than MVC normally implies, because
|
|
almost every aspect of it is pluggable:
|
|
wxMVCTree - Overall controller, and the window that actually gets placed
|
|
in the GUI.
|
|
Painter - Paints the control. The 'view' part of MVC.
|
|
NodePainter - Paints just the nodes
|
|
LinePainter - Paints just the lines between the nodes
|
|
TextConverter - Figures out what text to print for each node
|
|
Editor - Edits the contents of a node, if the model is editable.
|
|
LayoutEngine - Determines initial placement of nodes
|
|
Transform - Adjusts positions of nodes for movement or special effects.
|
|
TreeModel - Contains the data which the rest of the control acts
|
|
on. The 'model' part of MVC.
|
|
|
|
Author/Maintainer - Bryn Keller <xoltar@starship.python.net>
|
|
|
|
|
|
NOTE: This module is *not* supported in any way. Use it however you
|
|
wish, but be warned that dealing with any consequences is
|
|
entirly up to you.
|
|
--Robin
|
|
"""
|
|
|
|
#------------------------------------------------------------------------
|
|
from wxPython.wx import *
|
|
import os, sys, traceback
|
|
#------------------------------------------------------------------------
|
|
|
|
class MVCTreeNode:
|
|
"""
|
|
Used internally by wxMVCTree to manage its data. Contains information about
|
|
screen placement, the actual data associated with it, and more. These are
|
|
the nodes passed to all the other helper parts to do their work with.
|
|
"""
|
|
def __init__(self, data=None, parent = None, kids = None, x = 0, y = 0):
|
|
self.x = 0
|
|
self.y = 0
|
|
self.projx = 0
|
|
self.projy = 0
|
|
self.parent = parent
|
|
self.kids = kids
|
|
if self.kids is None:
|
|
self.kids = []
|
|
self.data = data
|
|
self.expanded = False
|
|
self.selected = False
|
|
self.built = False
|
|
self.scale = 0
|
|
|
|
def GetChildren(self):
|
|
return self.kids
|
|
|
|
def GetParent(self):
|
|
return self.parent
|
|
|
|
def Remove(self, node):
|
|
try:
|
|
self.kids.remove(node)
|
|
except:
|
|
pass
|
|
def Add(self, node):
|
|
self.kids.append(node)
|
|
node.SetParent(self)
|
|
|
|
def SetParent(self, parent):
|
|
if self.parent and not (self.parent is parent):
|
|
self.parent.Remove(self)
|
|
self.parent = parent
|
|
def __str__(self):
|
|
return "Node: " + str(self.data) + " (" + str(self.x) + ", " + str(self.y) + ")"
|
|
def __repr__(self):
|
|
return str(self.data)
|
|
def GetTreeString(self, tabs=0):
|
|
s = tabs * '\t' + str(self) + '\n'
|
|
for kid in self.kids:
|
|
s = s + kid.GetTreeString(tabs + 1)
|
|
return s
|
|
|
|
|
|
class Editor:
|
|
def __init__(self, tree):
|
|
self.tree = tree
|
|
def Edit(self, node):
|
|
raise NotImplementedError
|
|
def EndEdit(self, node, commit):
|
|
raise NotImplementedError
|
|
def CanEdit(self, node):
|
|
raise NotImplementedError
|
|
|
|
class LayoutEngine:
|
|
"""
|
|
Interface for layout engines.
|
|
"""
|
|
def __init__(self, tree):
|
|
self.tree = tree
|
|
def Layout(self, node):
|
|
raise NotImplementedError
|
|
def GetNodeList(self):
|
|
raise NotImplementedError
|
|
|
|
class Transform:
|
|
"""
|
|
Transform interface.
|
|
"""
|
|
def __init__(self, tree):
|
|
self.tree = tree
|
|
def Transform(self, node, offset, rotation):
|
|
"""
|
|
This method should only change the projx and projy attributes of
|
|
the node. These represent the position of the node as it should
|
|
be drawn on screen. Adjusting the x and y attributes can and
|
|
should cause havoc.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
def GetSize(self):
|
|
"""
|
|
Returns the size of the entire tree as laid out and transformed
|
|
as a tuple
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
class Painter:
|
|
"""
|
|
This is the interface that wxMVCTree expects from painters. All painters should
|
|
be Painter subclasses.
|
|
"""
|
|
def __init__(self, tree):
|
|
self.tree = tree
|
|
self.textcolor = wxNamedColour("BLACK")
|
|
self.bgcolor = wxNamedColour("WHITE")
|
|
self.fgcolor = wxNamedColour("BLUE")
|
|
self.linecolor = wxNamedColour("GREY")
|
|
self.font = wxFont(9, wxDEFAULT, wxNORMAL, wxNORMAL, False)
|
|
self.bmp = None
|
|
|
|
def GetFont(self):
|
|
return self.font
|
|
|
|
def SetFont(self, font):
|
|
self.font = font
|
|
self.tree.Refresh()
|
|
def GetBuffer(self):
|
|
return self.bmp
|
|
def ClearBuffer(self):
|
|
self.bmp = None
|
|
def Paint(self, dc, node, doubleBuffered=1, paintBackground=1):
|
|
raise NotImplementedError
|
|
def GetTextColour(self):
|
|
return self.textcolor
|
|
def SetTextColour(self, color):
|
|
self.textcolor = color
|
|
self.textbrush = wxBrush(color)
|
|
self.textpen = wxPen(color, 1, wxSOLID)
|
|
def GetBackgroundColour(self):
|
|
return self.bgcolor
|
|
def SetBackgroundColour(self, color):
|
|
self.bgcolor = color
|
|
self.bgbrush = wxBrush(color)
|
|
self.bgpen = wxPen(color, 1, wxSOLID)
|
|
def GetForegroundColour(self):
|
|
return self.fgcolor
|
|
def SetForegroundColour(self, color):
|
|
self.fgcolor = color
|
|
self.fgbrush = wxBrush(color)
|
|
self.fgpen = wxPen(color, 1, wxSOLID)
|
|
def GetLineColour(self):
|
|
return self.linecolor
|
|
def SetLineColour(self, color):
|
|
self.linecolor = color
|
|
self.linebrush = wxBrush(color)
|
|
self.linepen = wxPen( color, 1, wxSOLID)
|
|
def GetForegroundPen(self):
|
|
return self.fgpen
|
|
def GetBackgroundPen(self):
|
|
return self.bgpen
|
|
def GetTextPen(self):
|
|
return self.textpen
|
|
def GetForegroundBrush(self):
|
|
return self.fgbrush
|
|
def GetBackgroundBrush(self):
|
|
return self.bgbrush
|
|
def GetTextBrush(self):
|
|
return self.textbrush
|
|
def GetLinePen(self):
|
|
return self.linepen
|
|
def GetLineBrush(self):
|
|
return self.linebrush
|
|
def OnMouse(self, evt):
|
|
if evt.LeftDClick():
|
|
x, y = self.tree.CalcUnscrolledPosition(evt.GetX(), evt.GetY())
|
|
for item in self.rectangles:
|
|
if item[1].Contains((x,y)):
|
|
self.tree.Edit(item[0].data)
|
|
self.tree.OnNodeClick(item[0], evt)
|
|
return
|
|
elif evt.ButtonDown():
|
|
x, y = self.tree.CalcUnscrolledPosition(evt.GetX(), evt.GetY())
|
|
for item in self.rectangles:
|
|
if item[1].Contains((x, y)):
|
|
self.tree.OnNodeClick(item[0], evt)
|
|
return
|
|
for item in self.knobs:
|
|
if item[1].Contains((x, y)):
|
|
self.tree.OnKnobClick(item[0])
|
|
return
|
|
evt.Skip()
|
|
|
|
|
|
class wxTreeModel:
|
|
"""
|
|
Interface for tree models
|
|
"""
|
|
def GetRoot(self):
|
|
raise NotImplementedError
|
|
def SetRoot(self, root):
|
|
raise NotImplementedError
|
|
def GetChildCount(self, node):
|
|
raise NotImplementedError
|
|
def GetChildAt(self, node, index):
|
|
raise NotImplementedError
|
|
def GetParent(self, node):
|
|
raise NotImplementedError
|
|
def AddChild(self, parent, child):
|
|
if hasattr(self, 'tree') and self.tree:
|
|
self.tree.NodeAdded(parent, child)
|
|
def RemoveNode(self, child):
|
|
if hasattr(self, 'tree') and self.tree:
|
|
self.tree.NodeRemoved(child)
|
|
def InsertChild(self, parent, child, index):
|
|
if hasattr(self, 'tree') and self.tree:
|
|
self.tree.NodeInserted(parent, child, index)
|
|
def IsLeaf(self, node):
|
|
raise NotImplementedError
|
|
|
|
def IsEditable(self, node):
|
|
return False
|
|
|
|
def SetEditable(self, node):
|
|
return False
|
|
|
|
class NodePainter:
|
|
"""
|
|
This is the interface expected of a nodepainter.
|
|
"""
|
|
def __init__(self, painter):
|
|
self.painter = painter
|
|
def Paint(self, node, dc, location = None):
|
|
"""
|
|
location should be provided only to draw in an unusual position
|
|
(not the node's normal position), otherwise the node's projected x and y
|
|
coordinates will be used.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
class LinePainter:
|
|
"""
|
|
The linepainter interface.
|
|
"""
|
|
def __init__(self, painter):
|
|
self.painter = painter
|
|
def Paint(self, parent, child, dc):
|
|
raise NotImplementedError
|
|
|
|
class TextConverter:
|
|
"""
|
|
TextConverter interface.
|
|
"""
|
|
def __init__(self, painter):
|
|
self.painter = painter
|
|
def Convert(node):
|
|
"""
|
|
Should return a string. The node argument will be an
|
|
MVCTreeNode.
|
|
"""
|
|
raise NotImplementedError
|
|
|
|
|
|
class BasicTreeModel(wxTreeModel):
|
|
"""
|
|
A very simple treemodel implementation, but flexible enough for many needs.
|
|
"""
|
|
def __init__(self):
|
|
self.children = {}
|
|
self.parents = {}
|
|
self.root = None
|
|
def GetRoot(self):
|
|
return self.root
|
|
def SetRoot(self, root):
|
|
self.root = root
|
|
def GetChildCount(self, node):
|
|
if self.children.has_key(node):
|
|
return len(self.children[node])
|
|
else:
|
|
return 0
|
|
def GetChildAt(self, node, index):
|
|
return self.children[node][index]
|
|
|
|
def GetParent(self, node):
|
|
return self.parents[node]
|
|
|
|
def AddChild(self, parent, child):
|
|
self.parents[child]=parent
|
|
if not self.children.has_key(parent):
|
|
self.children[parent]=[]
|
|
self.children[parent].append(child)
|
|
wxTreeModel.AddChild(self, parent, child)
|
|
return child
|
|
|
|
def RemoveNode(self, node):
|
|
parent = self.parents[node]
|
|
del self.parents[node]
|
|
self.children[parent].remove(node)
|
|
wxTreeModel.RemoveNode(self, node)
|
|
|
|
def InsertChild(self, parent, child, index):
|
|
self.parents[child]=parent
|
|
if not self.children.has_key(parent):
|
|
self.children[parent]=[]
|
|
self.children[parent].insert(child, index)
|
|
wxTreeModel.InsertChild(self, parent, child, index)
|
|
return child
|
|
|
|
def IsLeaf(self, node):
|
|
return not self.children.has_key(node)
|
|
|
|
def IsEditable(self, node):
|
|
return False
|
|
|
|
def SetEditable(self, node, bool):
|
|
return False
|
|
|
|
|
|
class FileEditor(Editor):
|
|
def Edit(self, node):
|
|
treenode = self.tree.nodemap[node]
|
|
self.editcomp = wxTextCtrl(self.tree, -1)
|
|
for rect in self.tree.painter.rectangles:
|
|
if rect[0] == treenode:
|
|
self.editcomp.SetPosition((rect[1][0], rect[1][1]))
|
|
break
|
|
self.editcomp.SetValue(node.fileName)
|
|
self.editcomp.SetSelection(0, len(node.fileName))
|
|
self.editcomp.SetFocus()
|
|
self.treenode = treenode
|
|
# EVT_KEY_DOWN(self.editcomp, self._key)
|
|
EVT_KEY_UP(self.editcomp, self._key)
|
|
EVT_LEFT_DOWN(self.editcomp, self._mdown)
|
|
self.editcomp.CaptureMouse()
|
|
|
|
def CanEdit(self, node):
|
|
return isinstance(node, FileWrapper)
|
|
|
|
def EndEdit(self, commit):
|
|
if not self.tree._EditEnding(self.treenode.data):
|
|
return
|
|
if commit:
|
|
node = self.treenode.data
|
|
try:
|
|
os.rename(node.path + os.sep + node.fileName, node.path + os.sep + self.editcomp.GetValue())
|
|
node.fileName = self.editcomp.GetValue()
|
|
except:
|
|
traceback.print_exc()
|
|
self.editcomp.ReleaseMouse()
|
|
self.editcomp.Destroy()
|
|
del self.editcomp
|
|
self.tree.Refresh()
|
|
|
|
|
|
def _key(self, evt):
|
|
if evt.KeyCode() == WXK_RETURN:
|
|
self.EndEdit(True)
|
|
elif evt.KeyCode() == WXK_ESCAPE:
|
|
self.EndEdit(False)
|
|
else:
|
|
evt.Skip()
|
|
|
|
def _mdown(self, evt):
|
|
if evt.IsButton():
|
|
pos = evt.GetPosition()
|
|
edsize = self.editcomp.GetSize()
|
|
if pos.x < 0 or pos.y < 0 or pos.x > edsize.width or pos.y > edsize.height:
|
|
self.EndEdit(False)
|
|
|
|
|
|
class FileWrapper:
|
|
"""
|
|
Node class for FSTreeModel.
|
|
"""
|
|
def __init__(self, path, fileName):
|
|
self.path = path
|
|
self.fileName = fileName
|
|
|
|
def __str__(self):
|
|
return self.fileName
|
|
|
|
class FSTreeModel(BasicTreeModel):
|
|
"""
|
|
This treemodel models the filesystem starting from a given path.
|
|
"""
|
|
def __init__(self, path):
|
|
BasicTreeModel.__init__(self)
|
|
fw = FileWrapper(path, path.split(os.sep)[-1])
|
|
self._Build(path, fw)
|
|
self.SetRoot(fw)
|
|
self._editable = True
|
|
def _Build(self, path, fileWrapper):
|
|
for name in os.listdir(path):
|
|
fw = FileWrapper(path, name)
|
|
self.AddChild(fileWrapper, fw)
|
|
childName = path + os.sep + name
|
|
if os.path.isdir(childName):
|
|
self._Build(childName, fw)
|
|
|
|
def IsEditable(self, node):
|
|
return self._editable
|
|
|
|
def SetEditable(self, node, bool):
|
|
self._editable = bool
|
|
|
|
class LateFSTreeModel(FSTreeModel):
|
|
"""
|
|
This treemodel models the filesystem starting from a given path.
|
|
It retrieves the directory list as requested.
|
|
"""
|
|
def __init__(self, path):
|
|
BasicTreeModel.__init__(self)
|
|
name = path.split(os.sep)[-1]
|
|
pathpart = path[:-len(name)]
|
|
fw = FileWrapper(pathpart, name)
|
|
self._Build(path, fw)
|
|
self.SetRoot(fw)
|
|
self._editable = True
|
|
self.children = {}
|
|
self.parents = {}
|
|
def _Build(self, path, parent):
|
|
ppath = parent.path + os.sep + parent.fileName
|
|
if not os.path.isdir(ppath):
|
|
return
|
|
for name in os.listdir(ppath):
|
|
fw = FileWrapper(ppath, name)
|
|
self.AddChild(parent, fw)
|
|
def GetChildCount(self, node):
|
|
if self.children.has_key(node):
|
|
return FSTreeModel.GetChildCount(self, node)
|
|
else:
|
|
self._Build(node.path, node)
|
|
return FSTreeModel.GetChildCount(self, node)
|
|
|
|
def IsLeaf(self, node):
|
|
return not os.path.isdir(node.path + os.sep + node.fileName)
|
|
|
|
class StrTextConverter(TextConverter):
|
|
def Convert(self, node):
|
|
return str(node.data)
|
|
|
|
class NullTransform(Transform):
|
|
def GetSize(self):
|
|
return tuple(self.size)
|
|
|
|
def Transform(self, node, offset, rotation):
|
|
self.size = [0,0]
|
|
list = self.tree.GetLayoutEngine().GetNodeList()
|
|
for node in list:
|
|
node.projx = node.x + offset[0]
|
|
node.projy = node.y + offset[1]
|
|
if node.projx > self.size[0]:
|
|
self.size[0] = node.projx
|
|
if node.projy > self.size[1]:
|
|
self.size[1] = node.projy
|
|
|
|
class Rect:
|
|
def __init__(self, x, y, width, height):
|
|
self.x = x
|
|
self.y = y
|
|
self.width = width
|
|
self.height = height
|
|
def __getitem__(self, index):
|
|
return (self.x, self.y, self.width, self.height)[index]
|
|
|
|
def __setitem__(self, index, value):
|
|
name = ['x', 'y', 'width', 'height'][index]
|
|
setattr(self, name, value)
|
|
|
|
def Contains(self, other):
|
|
if type(other) == type(()):
|
|
other = Rect(other[0], other[1], 0, 0)
|
|
if other.x >= self.x:
|
|
if other.y >= self.y:
|
|
if other.width + other.x <= self.width + self.x:
|
|
if other.height + other.y <= self.height + self.y:
|
|
return True
|
|
return False
|
|
|
|
def __str__(self):
|
|
return "Rect: " + str([self.x, self.y, self.width, self.height])
|
|
|
|
class TreeLayout(LayoutEngine):
|
|
def SetHeight(self, num):
|
|
self.NODE_HEIGHT = num
|
|
|
|
def __init__(self, tree):
|
|
LayoutEngine.__init__(self, tree)
|
|
self.NODE_STEP = 20
|
|
self.NODE_HEIGHT = 20
|
|
self.nodelist = []
|
|
|
|
def Layout(self, node):
|
|
self.nodelist = []
|
|
self.NODE_HEIGHT = self.tree.GetFont().GetPointSize() * 2
|
|
self.layoutwalk(node)
|
|
|
|
def GetNodeList(self):
|
|
return self.nodelist
|
|
|
|
def layoutwalk(self, node):
|
|
if node == self.tree.currentRoot:
|
|
node.level = 1
|
|
self.lastY = (-self.NODE_HEIGHT)
|
|
node.x = self.NODE_STEP * node.level
|
|
node.y = self.lastY + self.NODE_HEIGHT
|
|
self.lastY = node.y
|
|
self.nodelist.append(node)
|
|
if node.expanded:
|
|
for kid in node.kids:
|
|
kid.level = node.level + 1
|
|
self.layoutwalk(kid)
|
|
|
|
class TreePainter(Painter):
|
|
"""
|
|
The default painter class. Uses double-buffering, delegates the painting of nodes and
|
|
lines to helper classes deriving from NodePainter and LinePainter.
|
|
"""
|
|
def __init__(self, tree, nodePainter = None, linePainter = None, textConverter = None):
|
|
Painter.__init__(self, tree)
|
|
if not nodePainter:
|
|
nodePainter = TreeNodePainter(self)
|
|
self.nodePainter = nodePainter
|
|
if not linePainter:
|
|
linePainter = TreeLinePainter(self)
|
|
self.linePainter = linePainter
|
|
if not textConverter:
|
|
textConverter = StrTextConverter(self)
|
|
self.textConverter = textConverter
|
|
self.charWidths = []
|
|
|
|
def Paint(self, dc, node, doubleBuffered=1, paintBackground=1):
|
|
if not self.charWidths:
|
|
self.charWidths = []
|
|
for i in range(25):
|
|
self.charWidths.append(dc.GetTextExtent("D")[0] * i)
|
|
self.charHeight = dc.GetTextExtent("D")[1]
|
|
self.textpen = wxPen(self.GetTextColour(), 1, wxSOLID)
|
|
self.fgpen = wxPen(self.GetForegroundColour(), 1, wxSOLID)
|
|
self.bgpen = wxPen(self.GetBackgroundColour(), 1, wxSOLID)
|
|
self.linepen = wxPen(self.GetLineColour(), 1, wxSOLID)
|
|
self.dashpen = wxPen(self.GetLineColour(), 1, wxDOT)
|
|
self.textbrush = wxBrush(self.GetTextColour(), wxSOLID)
|
|
self.fgbrush = wxBrush(self.GetForegroundColour(), wxSOLID)
|
|
self.bgbrush = wxBrush(self.GetBackgroundColour(), wxSOLID)
|
|
self.linebrush = wxPen(self.GetLineColour(), 1, wxSOLID)
|
|
treesize = self.tree.GetSize()
|
|
size = self.tree.transform.GetSize()
|
|
size = (max(treesize.width, size[0]+50), max(treesize.height, size[1]+50))
|
|
dc.BeginDrawing()
|
|
if doubleBuffered:
|
|
mem_dc = wxMemoryDC()
|
|
if not self.GetBuffer():
|
|
self.knobs = []
|
|
self.rectangles = []
|
|
self.bmp = wxEmptyBitmap(size[0], size[1])
|
|
mem_dc.SelectObject(self.GetBuffer())
|
|
mem_dc.SetPen(self.GetBackgroundPen())
|
|
mem_dc.SetBrush(self.GetBackgroundBrush())
|
|
mem_dc.DrawRectangle(0, 0, size[0], size[1])
|
|
mem_dc.SetFont(self.tree.GetFont())
|
|
self.paintWalk(node, mem_dc)
|
|
else:
|
|
mem_dc.SelectObject(self.GetBuffer())
|
|
xstart, ystart = self.tree.CalcUnscrolledPosition(0,0)
|
|
size = self.tree.GetClientSizeTuple()
|
|
dc.Blit(xstart, ystart, size[0], size[1], mem_dc, xstart, ystart)
|
|
else:
|
|
if node == self.tree.currentRoot:
|
|
self.knobs = []
|
|
self.rectangles = []
|
|
dc.SetPen(self.GetBackgroundPen())
|
|
dc.SetBrush(self.GetBackgroundBrush())
|
|
dc.SetFont(self.tree.GetFont())
|
|
if paintBackground:
|
|
dc.DrawRectangle(0, 0, size[0], size[1])
|
|
if node:
|
|
#Call with not paintBackground because if we are told not to paint the
|
|
#whole background, we have to paint in parts to undo selection coloring.
|
|
pb = paintBackground
|
|
self.paintWalk(node, dc, not pb)
|
|
dc.EndDrawing()
|
|
|
|
def GetDashPen(self):
|
|
return self.dashpen
|
|
|
|
def SetLinePen(self, pen):
|
|
Painter.SetLinePen(self, pen)
|
|
self.dashpen = wxPen(pen.GetColour(), 1, wxDOT)
|
|
|
|
def paintWalk(self, node, dc, paintRects=0):
|
|
self.linePainter.Paint(node.parent, node, dc)
|
|
self.nodePainter.Paint(node, dc, drawRects = paintRects)
|
|
if node.expanded:
|
|
for kid in node.kids:
|
|
if not self.paintWalk(kid, dc, paintRects):
|
|
return False
|
|
for kid in node.kids:
|
|
px = (kid.projx - self.tree.layout.NODE_STEP) + 5
|
|
py = kid.projy + kid.height/2
|
|
if (not self.tree.model.IsLeaf(kid.data)) or ((kid.expanded or self.tree._assumeChildren) and len(kid.kids)):
|
|
dc.SetPen(self.linepen)
|
|
dc.SetBrush(self.bgbrush)
|
|
dc.DrawRectangle(px -4, py-4, 9, 9)
|
|
self.knobs.append( (kid, Rect(px -4, py -4, 9, 9)) )
|
|
dc.SetPen(self.textpen)
|
|
if not kid.expanded:
|
|
dc.DrawLine(px, py -2, px, py + 3)
|
|
dc.DrawLine(px -2, py, px + 3, py)
|
|
if node == self.tree.currentRoot:
|
|
px = (node.projx - self.tree.layout.NODE_STEP) + 5
|
|
py = node.projy + node.height/2
|
|
dc.SetPen(self.linepen)
|
|
dc.SetBrush(self.bgbrush)
|
|
dc.DrawRectangle(px -4, py-4, 9, 9)
|
|
self.knobs.append( (node, Rect(px -4, py -4, 9, 9)) )
|
|
dc.SetPen(self.textpen)
|
|
if not node.expanded:
|
|
dc.DrawLine(px, py -2, px, py + 3)
|
|
dc.DrawLine(px -2, py, px + 3, py)
|
|
return True
|
|
|
|
def OnMouse(self, evt):
|
|
Painter.OnMouse(self, evt)
|
|
|
|
class TreeNodePainter(NodePainter):
|
|
def Paint(self, node, dc, location = None, drawRects = 0):
|
|
text = self.painter.textConverter.Convert(node)
|
|
extent = dc.GetTextExtent(text)
|
|
node.width = extent[0]
|
|
node.height = extent[1]
|
|
if node.selected:
|
|
dc.SetPen(self.painter.GetLinePen())
|
|
dc.SetBrush(self.painter.GetForegroundBrush())
|
|
dc.SetTextForeground(wxNamedColour("WHITE"))
|
|
dc.DrawRectangle(node.projx -1, node.projy -1, node.width + 3, node.height + 3)
|
|
else:
|
|
if drawRects:
|
|
dc.SetBrush(self.painter.GetBackgroundBrush())
|
|
dc.SetPen(self.painter.GetBackgroundPen())
|
|
dc.DrawRectangle(node.projx -1, node.projy -1, node.width + 3, node.height + 3)
|
|
dc.SetTextForeground(self.painter.GetTextColour())
|
|
dc.DrawText(text, node.projx, node.projy)
|
|
self.painter.rectangles.append((node, Rect(node.projx, node.projy, node.width, node.height)))
|
|
|
|
class TreeLinePainter(LinePainter):
|
|
def Paint(self, parent, child, dc):
|
|
dc.SetPen(self.painter.GetDashPen())
|
|
px = py = cx = cy = 0
|
|
if parent is None or child == self.painter.tree.currentRoot:
|
|
px = (child.projx - self.painter.tree.layout.NODE_STEP) + 5
|
|
py = child.projy + self.painter.tree.layout.NODE_HEIGHT/2 -2
|
|
cx = child.projx
|
|
cy = py
|
|
dc.DrawLine(px, py, cx, cy)
|
|
else:
|
|
px = parent.projx + 5
|
|
py = parent.projy + parent.height
|
|
cx = child.projx -5
|
|
cy = child.projy + self.painter.tree.layout.NODE_HEIGHT/2 -3
|
|
dc.DrawLine(px, py, px, cy)
|
|
dc.DrawLine(px, cy, cx, cy)
|
|
|
|
|
|
wxEVT_MVCTREE_BEGIN_EDIT = 20204 #Start editing. Vetoable.
|
|
wxEVT_MVCTREE_END_EDIT = 20205 #Stop editing. Vetoable.
|
|
wxEVT_MVCTREE_DELETE_ITEM = 20206 #Item removed from model.
|
|
wxEVT_MVCTREE_ITEM_EXPANDED = 20209
|
|
wxEVT_MVCTREE_ITEM_EXPANDING = 20210
|
|
wxEVT_MVCTREE_ITEM_COLLAPSED = 20211
|
|
wxEVT_MVCTREE_ITEM_COLLAPSING = 20212
|
|
wxEVT_MVCTREE_SEL_CHANGED = 20213
|
|
wxEVT_MVCTREE_SEL_CHANGING = 20214 #Vetoable.
|
|
wxEVT_MVCTREE_KEY_DOWN = 20215
|
|
wxEVT_MVCTREE_ADD_ITEM = 20216 #Item added to model.
|
|
|
|
def EVT_MVCTREE_SEL_CHANGED(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_SEL_CHANGED, func)
|
|
|
|
def EVT_MVCTREE_SEL_CHANGING(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_SEL_CHANGING, func)
|
|
|
|
def EVT_MVCTREE_ITEM_EXPANDED(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_ITEM_EXPANDED, func)
|
|
|
|
def EVT_MVCTREE_ITEM_EXPANDING(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_ITEM_EXPANDING, func)
|
|
|
|
def EVT_MVCTREE_ITEM_COLLAPSED(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_ITEM_COLLAPSED, func)
|
|
|
|
def EVT_MVCTREE_ITEM_COLLAPSING(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_ITEM_COLLAPSING, func)
|
|
|
|
def EVT_MVCTREE_ADD_ITEM(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_ADD_ITEM, func)
|
|
|
|
def EVT_MVCTREE_DELETE_ITEM(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_DELETE_ITEM, func)
|
|
|
|
def EVT_MVCTREE_KEY_DOWN(win, id, func):
|
|
win.Connect(id, -1, wxEVT_MVCTREE_KEY_DOWN, func)
|
|
|
|
|
|
class wxMVCTreeEvent(wxPyCommandEvent):
|
|
def __init__(self, type, id, node = None, nodes = None, keyEvent = None, **kwargs):
|
|
apply(wxPyCommandEvent.__init__, (self, type, id), kwargs)
|
|
self.node = node
|
|
self.nodes = nodes
|
|
self.keyEvent = keyEvent
|
|
def GetNode(self):
|
|
return self.node
|
|
def GetNodes(self):
|
|
return self.nodes
|
|
def getKeyEvent(self):
|
|
return self.keyEvent
|
|
|
|
class wxMVCTreeNotifyEvent(wxMVCTreeEvent):
|
|
def __init__(self, type, id, node = None, nodes = None, **kwargs):
|
|
apply(wxMVCTreeEvent.__init__, (self, type, id, node, nodes), kwargs)
|
|
self.notify = wxNotifyEvent(type, id)
|
|
def getNotifyEvent(self):
|
|
return self.notify
|
|
|
|
class wxMVCTree(wxScrolledWindow):
|
|
"""
|
|
The main mvc tree class.
|
|
"""
|
|
def __init__(self, parent, id, model = None, layout = None, transform = None,
|
|
painter = None, *args, **kwargs):
|
|
apply(wxScrolledWindow.__init__, (self, parent, id), kwargs)
|
|
self.nodemap = {}
|
|
self._multiselect = False
|
|
self._selections = []
|
|
self._assumeChildren = False
|
|
self._scrollx = False
|
|
self._scrolly = False
|
|
self.doubleBuffered = False
|
|
self._lastPhysicalSize = self.GetSize()
|
|
self._editors = []
|
|
if not model:
|
|
model = BasicTreeModel()
|
|
model.SetRoot("Root")
|
|
self.SetModel(model)
|
|
if not layout:
|
|
layout = TreeLayout(self)
|
|
self.layout = layout
|
|
if not transform:
|
|
transform = NullTransform(self)
|
|
self.transform = transform
|
|
if not painter:
|
|
painter = TreePainter(self)
|
|
self.painter = painter
|
|
self.SetFont(wxFont(9, wxDEFAULT, wxNORMAL, wxNORMAL, False))
|
|
EVT_MOUSE_EVENTS(self, self.OnMouse)
|
|
EVT_KEY_DOWN(self, self.OnKeyDown)
|
|
self.doubleBuffered = True
|
|
EVT_SIZE(self, self.OnSize)
|
|
EVT_ERASE_BACKGROUND(self, self.OnEraseBackground)
|
|
EVT_PAINT(self, self.OnPaint)
|
|
|
|
|
|
def Refresh(self):
|
|
if self.doubleBuffered:
|
|
self.painter.ClearBuffer()
|
|
wxScrolledWindow.Refresh(self, False)
|
|
|
|
def GetPainter(self):
|
|
return self.painter
|
|
|
|
def GetLayoutEngine(self):
|
|
return self.layout
|
|
|
|
def GetTransform(self):
|
|
return self.transform
|
|
|
|
def __repr__(self):
|
|
return "<wxMVCTree instance at %s>" % str(hex(id(self)))
|
|
|
|
def __str__(self):
|
|
return self.__repr__()
|
|
|
|
def NodeAdded(self, parent, child):
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_ADD_ITEM, self.GetId(), node = child, nodes = [parent, child])
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
self.painter.ClearBuffer()
|
|
|
|
def NodeInserted(self, parent, child, index):
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_ADD_ITEM, self.GetId(), node = child, nodes = [parent, child])
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
self.painter.ClearBuffer()
|
|
|
|
def NodeRemoved(self, node):
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_DELETE_ITEM, self.GetId(), node = child, nodes = [parent, child])
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
self.painter.ClearBuffer()
|
|
|
|
def OnKeyDown(self, evt):
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_KEY_DOWN, self.GetId(), keyEvent = evt)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
|
|
def SetFont(self, font):
|
|
self.painter.SetFont(font)
|
|
dc = wxClientDC(self)
|
|
dc.SetFont(font)
|
|
self.layout.SetHeight(dc.GetTextExtent("")[1] + 18)
|
|
self.painter.ClearBuffer()
|
|
|
|
def GetFont(self):
|
|
return self.painter.GetFont()
|
|
|
|
def AddEditor(self, editor):
|
|
self._editors.append(editor)
|
|
|
|
def RemoveEditor(self, editor):
|
|
self._editors.remove(editor)
|
|
|
|
def OnMouse(self, evt):
|
|
self.painter.OnMouse(evt)
|
|
|
|
def OnNodeClick(self, node, mouseEvent):
|
|
if node.selected and (self.IsMultiSelect() and mouseEvent.ControlDown()):
|
|
self.RemoveFromSelection(node.data)
|
|
else:
|
|
self.AddToSelection(node.data, mouseEvent.ControlDown(), mouseEvent.ShiftDown())
|
|
|
|
def OnKnobClick(self, node):
|
|
self.SetExpanded(node.data, not node.expanded)
|
|
|
|
def GetDisplayText(self, node):
|
|
treenode = self.nodemap[node]
|
|
return self.painter.textConverter.Convert(treenode)
|
|
|
|
def IsDoubleBuffered(self):
|
|
return self.doubleBuffered
|
|
|
|
def SetDoubleBuffered(self, bool):
|
|
"""
|
|
By default wxMVCTree is double-buffered.
|
|
"""
|
|
self.doubleBuffered = bool
|
|
|
|
def GetModel(self):
|
|
return self.model
|
|
|
|
def SetModel(self, model):
|
|
"""
|
|
Completely change the data to be displayed.
|
|
"""
|
|
self.model = model
|
|
model.tree = self
|
|
self.laidOut = 0
|
|
self.transformed = 0
|
|
self._selections = []
|
|
self.layoutRoot = MVCTreeNode()
|
|
self.layoutRoot.data = self.model.GetRoot()
|
|
self.layoutRoot.expanded = True
|
|
self.LoadChildren(self.layoutRoot)
|
|
self.currentRoot = self.layoutRoot
|
|
self.offset = [0,0]
|
|
self.rotation = 0
|
|
self._scrollset = None
|
|
self.Refresh()
|
|
|
|
def GetCurrentRoot(self):
|
|
return self.currentRoot
|
|
|
|
def LoadChildren(self, layoutNode):
|
|
if layoutNode.built:
|
|
return
|
|
else:
|
|
self.nodemap[layoutNode.data]=layoutNode
|
|
for i in range(self.GetModel().GetChildCount(layoutNode.data)):
|
|
p = MVCTreeNode("RAW", layoutNode, [])
|
|
layoutNode.Add(p)
|
|
p.data = self.GetModel().GetChildAt(layoutNode.data, i)
|
|
self.nodemap[p.data]=p
|
|
layoutNode.built = True
|
|
if not self._assumeChildren:
|
|
for kid in layoutNode.kids:
|
|
self.LoadChildren(kid)
|
|
|
|
def OnEraseBackground(self, evt):
|
|
pass
|
|
|
|
def OnSize(self, evt):
|
|
size = self.GetSize()
|
|
self.center = (size.width/2, size.height/2)
|
|
if self._lastPhysicalSize.width < size.width or self._lastPhysicalSize.height < size.height:
|
|
self.painter.ClearBuffer()
|
|
self._lastPhysicalSize = size
|
|
|
|
def GetSelection(self):
|
|
"Returns a tuple of selected nodes."
|
|
return tuple(self._selections)
|
|
|
|
def SetSelection(self, nodeTuple):
|
|
if type(nodeTuple) != type(()):
|
|
nodeTuple = (nodeTuple,)
|
|
e = wxMVCTreeNotifyEvent(wxEVT_MVCTREE_SEL_CHANGING, self.GetId(), nodeTuple[0], nodes = nodeTuple)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
if not e.notify.IsAllowed():
|
|
return
|
|
for node in nodeTuple:
|
|
treenode = self.nodemap[node]
|
|
treenode.selected = True
|
|
for node in self._selections:
|
|
treenode = self.nodemap[node]
|
|
node.selected = False
|
|
self._selections = list(nodeTuple)
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_SEL_CHANGED, self.GetId(), nodeTuple[0], nodes = nodeTuple)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
|
|
def IsMultiSelect(self):
|
|
return self._multiselect
|
|
|
|
def SetMultiSelect(self, bool):
|
|
self._multiselect = bool
|
|
|
|
def IsSelected(self, node):
|
|
return self.nodemap[node].selected
|
|
|
|
def Edit(self, node):
|
|
if not self.model.IsEditable(node):
|
|
return
|
|
for ed in self._editors:
|
|
if ed.CanEdit(node):
|
|
e = wxMVCTreeNotifyEvent(wxEVT_MVCTREE_BEGIN_EDIT, self.GetId(), node)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
if not e.notify.IsAllowed():
|
|
return
|
|
ed.Edit(node)
|
|
self._currentEditor = ed
|
|
break
|
|
|
|
def EndEdit(self):
|
|
if self._currentEditor:
|
|
self._currentEditor.EndEdit
|
|
self._currentEditor = None
|
|
|
|
def _EditEnding(self, node):
|
|
e = wxMVCTreeNotifyEvent(wxEVT_MVCTREE_END_EDIT, self.GetId(), node)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
if not e.notify.IsAllowed():
|
|
return False
|
|
self._currentEditor = None
|
|
return True
|
|
|
|
|
|
def SetExpanded(self, node, bool):
|
|
treenode = self.nodemap[node]
|
|
if bool:
|
|
e = wxMVCTreeNotifyEvent(wxEVT_MVCTREE_ITEM_EXPANDING, self.GetId(), node)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
if not e.notify.IsAllowed():
|
|
return
|
|
if not treenode.built:
|
|
self.LoadChildren(treenode)
|
|
else:
|
|
e = wxMVCTreeNotifyEvent(wxEVT_MVCTREE_ITEM_COLLAPSING, self.GetId(), node)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
if not e.notify.IsAllowed():
|
|
return
|
|
treenode.expanded = bool
|
|
e = None
|
|
if treenode.expanded:
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_ITEM_EXPANDED, self.GetId(), node)
|
|
else:
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_ITEM_COLLAPSED, self.GetId(), node)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
self.layout.Layout(self.currentRoot)
|
|
self.transform.Transform(self.currentRoot, self.offset, self.rotation)
|
|
self.Refresh()
|
|
|
|
def IsExpanded(self, node):
|
|
return self.nodemap[node].expanded
|
|
|
|
def AddToSelection(self, nodeOrTuple, enableMulti = True, shiftMulti = False):
|
|
nodeTuple = nodeOrTuple
|
|
if type(nodeOrTuple)!= type(()):
|
|
nodeTuple = (nodeOrTuple,)
|
|
e = wxMVCTreeNotifyEvent(wxEVT_MVCTREE_SEL_CHANGING, self.GetId(), nodeTuple[0], nodes = nodeTuple)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
if not e.notify.IsAllowed():
|
|
return
|
|
changeparents = []
|
|
if not (self.IsMultiSelect() and (enableMulti or shiftMulti)):
|
|
for node in self._selections:
|
|
treenode = self.nodemap[node]
|
|
treenode.selected = False
|
|
changeparents.append(treenode)
|
|
node = nodeTuple[0]
|
|
self._selections = [node]
|
|
treenode = self.nodemap[node]
|
|
changeparents.append(treenode)
|
|
treenode.selected = True
|
|
else:
|
|
if shiftMulti:
|
|
for node in nodeTuple:
|
|
treenode = self.nodemap[node]
|
|
oldtreenode = self.nodemap[self._selections[0]]
|
|
if treenode.parent == oldtreenode.parent:
|
|
found = 0
|
|
for kid in oldtreenode.parent.kids:
|
|
if kid == treenode or kid == oldtreenode:
|
|
found = not found
|
|
kid.selected = True
|
|
self._selections.append(kid.data)
|
|
changeparents.append(kid)
|
|
elif found:
|
|
kid.selected = True
|
|
self._selections.append(kid.data)
|
|
changeparents.append(kid)
|
|
else:
|
|
for node in nodeTuple:
|
|
try:
|
|
self._selections.index(node)
|
|
except ValueError:
|
|
self._selections.append(node)
|
|
treenode = self.nodemap[node]
|
|
treenode.selected = True
|
|
changeparents.append(treenode)
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_SEL_CHANGED, self.GetId(), nodeTuple[0], nodes = nodeTuple)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
dc = wxClientDC(self)
|
|
self.PrepareDC(dc)
|
|
for node in changeparents:
|
|
if node:
|
|
self.painter.Paint(dc, node, doubleBuffered = 0, paintBackground = 0)
|
|
self.painter.ClearBuffer()
|
|
|
|
def RemoveFromSelection(self, nodeTuple):
|
|
if type(nodeTuple) != type(()):
|
|
nodeTuple = (nodeTuple,)
|
|
changeparents = []
|
|
for node in nodeTuple:
|
|
self._selections.remove(node)
|
|
treenode = self.nodemap[node]
|
|
changeparents.append(treenode)
|
|
treenode.selected = False
|
|
e = wxMVCTreeEvent(wxEVT_MVCTREE_SEL_CHANGED, self.GetId(), node, nodes = nodeTuple)
|
|
self.GetEventHandler().ProcessEvent(e)
|
|
dc = wxClientDC(self)
|
|
self.PrepareDC(dc)
|
|
for node in changeparents:
|
|
if node:
|
|
self.painter.Paint(dc, node, doubleBuffered = 0, paintBackground = 0)
|
|
self.painter.ClearBuffer()
|
|
|
|
|
|
def GetBackgroundColour(self):
|
|
if hasattr(self, 'painter') and self.painter:
|
|
return self.painter.GetBackgroundColour()
|
|
else:
|
|
return wxWindow.GetBackgroundColour(self)
|
|
def SetBackgroundColour(self, color):
|
|
if hasattr(self, 'painter') and self.painter:
|
|
self.painter.SetBackgroundColour(color)
|
|
else:
|
|
wxWindow.SetBackgroundColour(self, color)
|
|
def GetForegroundColour(self):
|
|
if hasattr(self, 'painter') and self.painter:
|
|
return self.painter.GetForegroundColour()
|
|
else:
|
|
return wxWindow.GetBackgroundColour(self)
|
|
def SetForegroundColour(self, color):
|
|
if hasattr(self, 'painter') and self.painter:
|
|
self.painter.SetForegroundColour(color)
|
|
else:
|
|
wxWindow.SetBackgroundColour(self, color)
|
|
|
|
def SetAssumeChildren(self, bool):
|
|
self._assumeChildren = bool
|
|
|
|
def GetAssumeChildren(self):
|
|
return self._assumeChildren
|
|
|
|
def OnPaint(self, evt):
|
|
"""
|
|
Ensures that the tree has been laid out and transformed, then calls the painter
|
|
to paint the control.
|
|
"""
|
|
try:
|
|
self.EnableScrolling(False, False)
|
|
if not self.laidOut:
|
|
self.layout.Layout(self.currentRoot)
|
|
self.laidOut = True
|
|
self.transformed = False
|
|
if not self.transformed:
|
|
self.transform.Transform(self.currentRoot, self.offset, self.rotation)
|
|
self.transformed = True
|
|
tsize = None
|
|
tsize = list(self.transform.GetSize())
|
|
tsize[0] = tsize[0] + 50
|
|
tsize[1] = tsize[1] + 50
|
|
size = self.GetSizeTuple()
|
|
if tsize[0] > size[0] or tsize[1] > size[1]:
|
|
if not hasattr(self, '_oldsize') or (tsize[0] > self._oldsize[0] or tsize[1] > self._oldsize[1]):
|
|
self._oldsize = tsize
|
|
oldstart = self.ViewStart()
|
|
self._lastPhysicalSize = self.GetSize()
|
|
self.SetScrollbars(10, 10, tsize[0]/10, tsize[1]/10)
|
|
self.Scroll(oldstart[0], oldstart[1])
|
|
dc = wxPaintDC(self)
|
|
self.PrepareDC(dc)
|
|
dc.SetFont(self.GetFont())
|
|
self.painter.Paint(dc, self.currentRoot, self.doubleBuffered)
|
|
except:
|
|
traceback.print_exc()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|