Merge pull request #518 from campagnola/treewidget-updates
Treewidget updates
This commit is contained in:
commit
952c679a61
@ -1,8 +1,6 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
from weakref import *
|
|
||||||
from ..Qt import QtGui, QtCore
|
from ..Qt import QtGui, QtCore
|
||||||
from ..python2_3 import xrange
|
from weakref import *
|
||||||
|
|
||||||
|
|
||||||
__all__ = ['TreeWidget', 'TreeWidgetItem']
|
__all__ = ['TreeWidget', 'TreeWidgetItem']
|
||||||
|
|
||||||
@ -13,6 +11,9 @@ class TreeWidget(QtGui.QTreeWidget):
|
|||||||
This class demonstrates the absurd lengths one must go to to make drag/drop work."""
|
This class demonstrates the absurd lengths one must go to to make drag/drop work."""
|
||||||
|
|
||||||
sigItemMoved = QtCore.Signal(object, object, object) # (item, parent, index)
|
sigItemMoved = QtCore.Signal(object, object, object) # (item, parent, index)
|
||||||
|
sigItemCheckStateChanged = QtCore.Signal(object, object)
|
||||||
|
sigItemTextChanged = QtCore.Signal(object, object)
|
||||||
|
sigColumnCountChanged = QtCore.Signal(object, object) # self, count
|
||||||
|
|
||||||
def __init__(self, parent=None):
|
def __init__(self, parent=None):
|
||||||
QtGui.QTreeWidget.__init__(self, parent)
|
QtGui.QTreeWidget.__init__(self, parent)
|
||||||
@ -22,6 +23,7 @@ class TreeWidget(QtGui.QTreeWidget):
|
|||||||
self.setEditTriggers(QtGui.QAbstractItemView.EditKeyPressed|QtGui.QAbstractItemView.SelectedClicked)
|
self.setEditTriggers(QtGui.QAbstractItemView.EditKeyPressed|QtGui.QAbstractItemView.SelectedClicked)
|
||||||
self.placeholders = []
|
self.placeholders = []
|
||||||
self.childNestingLimit = None
|
self.childNestingLimit = None
|
||||||
|
self.itemClicked.connect(self._itemClicked)
|
||||||
|
|
||||||
def setItemWidget(self, item, col, wid):
|
def setItemWidget(self, item, col, wid):
|
||||||
"""
|
"""
|
||||||
@ -42,7 +44,7 @@ class TreeWidget(QtGui.QTreeWidget):
|
|||||||
|
|
||||||
def itemWidget(self, item, col):
|
def itemWidget(self, item, col):
|
||||||
w = QtGui.QTreeWidget.itemWidget(self, item, col)
|
w = QtGui.QTreeWidget.itemWidget(self, item, col)
|
||||||
if w is not None:
|
if w is not None and hasattr(w, 'realChild'):
|
||||||
w = w.realChild
|
w = w.realChild
|
||||||
return w
|
return w
|
||||||
|
|
||||||
@ -141,7 +143,6 @@ class TreeWidget(QtGui.QTreeWidget):
|
|||||||
QtGui.QTreeWidget.dropEvent(self, ev)
|
QtGui.QTreeWidget.dropEvent(self, ev)
|
||||||
self.updateDropFlags()
|
self.updateDropFlags()
|
||||||
|
|
||||||
|
|
||||||
def updateDropFlags(self):
|
def updateDropFlags(self):
|
||||||
### intended to put a limit on how deep nests of children can go.
|
### intended to put a limit on how deep nests of children can go.
|
||||||
### self.childNestingLimit is upheld when moving items without children, but if the item being moved has children/grandchildren, the children/grandchildren
|
### self.childNestingLimit is upheld when moving items without children, but if the item being moved has children/grandchildren, the children/grandchildren
|
||||||
@ -165,11 +166,9 @@ class TreeWidget(QtGui.QTreeWidget):
|
|||||||
def informTreeWidgetChange(item):
|
def informTreeWidgetChange(item):
|
||||||
if hasattr(item, 'treeWidgetChanged'):
|
if hasattr(item, 'treeWidgetChanged'):
|
||||||
item.treeWidgetChanged()
|
item.treeWidgetChanged()
|
||||||
else:
|
for i in range(item.childCount()):
|
||||||
for i in xrange(item.childCount()):
|
|
||||||
TreeWidget.informTreeWidgetChange(item.child(i))
|
TreeWidget.informTreeWidgetChange(item.child(i))
|
||||||
|
|
||||||
|
|
||||||
def addTopLevelItem(self, item):
|
def addTopLevelItem(self, item):
|
||||||
QtGui.QTreeWidget.addTopLevelItem(self, item)
|
QtGui.QTreeWidget.addTopLevelItem(self, item)
|
||||||
self.informTreeWidgetChange(item)
|
self.informTreeWidgetChange(item)
|
||||||
@ -210,21 +209,61 @@ class TreeWidget(QtGui.QTreeWidget):
|
|||||||
#for item in items:
|
#for item in items:
|
||||||
#self.informTreeWidgetChange(item)
|
#self.informTreeWidgetChange(item)
|
||||||
|
|
||||||
|
def invisibleRootItem(self):
|
||||||
|
# wrap this item so that we can propagate tree change information
|
||||||
|
# to children.
|
||||||
|
return InvisibleRootItem(QtGui.QTreeWidget.invisibleRootItem(self))
|
||||||
|
|
||||||
|
def itemFromIndex(self, index):
|
||||||
|
"""Return the item and column corresponding to a QModelIndex.
|
||||||
|
"""
|
||||||
|
col = index.column()
|
||||||
|
rows = []
|
||||||
|
while index.row() >= 0:
|
||||||
|
rows.insert(0, index.row())
|
||||||
|
index = index.parent()
|
||||||
|
item = self.topLevelItem(rows[0])
|
||||||
|
for row in rows[1:]:
|
||||||
|
item = item.child(row)
|
||||||
|
return item, col
|
||||||
|
|
||||||
|
def setColumnCount(self, c):
|
||||||
|
QtGui.QTreeWidget.setColumnCount(self, c)
|
||||||
|
self.sigColumnCountChanged.emit(self, c)
|
||||||
|
|
||||||
|
def _itemClicked(self, item, col):
|
||||||
|
if hasattr(item, 'itemClicked'):
|
||||||
|
item.itemClicked(col)
|
||||||
|
|
||||||
|
|
||||||
class TreeWidgetItem(QtGui.QTreeWidgetItem):
|
class TreeWidgetItem(QtGui.QTreeWidgetItem):
|
||||||
"""
|
"""
|
||||||
TreeWidgetItem that keeps track of its own widgets.
|
TreeWidgetItem that keeps track of its own widgets and expansion state.
|
||||||
Widgets may be added to columns before the item is added to a tree.
|
|
||||||
|
* Widgets may be added to columns before the item is added to a tree.
|
||||||
|
* Expanded state may be set before item is added to a tree.
|
||||||
|
* Adds setCheked and isChecked methods.
|
||||||
|
* Adds addChildren, insertChildren, and takeChildren methods.
|
||||||
"""
|
"""
|
||||||
def __init__(self, *args):
|
def __init__(self, *args):
|
||||||
QtGui.QTreeWidgetItem.__init__(self, *args)
|
QtGui.QTreeWidgetItem.__init__(self, *args)
|
||||||
self._widgets = {} # col: widget
|
self._widgets = {} # col: widget
|
||||||
self._tree = None
|
self._tree = None
|
||||||
|
self._expanded = False
|
||||||
|
|
||||||
def setChecked(self, column, checked):
|
def setChecked(self, column, checked):
|
||||||
self.setCheckState(column, QtCore.Qt.Checked if checked else QtCore.Qt.Unchecked)
|
self.setCheckState(column, QtCore.Qt.Checked if checked else QtCore.Qt.Unchecked)
|
||||||
|
|
||||||
|
def isChecked(self, col):
|
||||||
|
return self.checkState(col) == QtCore.Qt.Checked
|
||||||
|
|
||||||
|
def setExpanded(self, exp):
|
||||||
|
self._expanded = exp
|
||||||
|
QtGui.QTreeWidgetItem.setExpanded(self, exp)
|
||||||
|
|
||||||
|
def isExpanded(self):
|
||||||
|
return self._expanded
|
||||||
|
|
||||||
def setWidget(self, column, widget):
|
def setWidget(self, column, widget):
|
||||||
if column in self._widgets:
|
if column in self._widgets:
|
||||||
self.removeWidget(column)
|
self.removeWidget(column)
|
||||||
@ -251,6 +290,10 @@ class TreeWidgetItem(QtGui.QTreeWidgetItem):
|
|||||||
return
|
return
|
||||||
for col, widget in self._widgets.items():
|
for col, widget in self._widgets.items():
|
||||||
tree.setItemWidget(self, col, widget)
|
tree.setItemWidget(self, col, widget)
|
||||||
|
QtGui.QTreeWidgetItem.setExpanded(self, self._expanded)
|
||||||
|
|
||||||
|
def childItems(self):
|
||||||
|
return [self.child(i) for i in range(self.childCount())]
|
||||||
|
|
||||||
def addChild(self, child):
|
def addChild(self, child):
|
||||||
QtGui.QTreeWidgetItem.addChild(self, child)
|
QtGui.QTreeWidgetItem.addChild(self, child)
|
||||||
@ -285,4 +328,67 @@ class TreeWidgetItem(QtGui.QTreeWidgetItem):
|
|||||||
TreeWidget.informTreeWidgetChange(child)
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
return childs
|
return childs
|
||||||
|
|
||||||
|
def setData(self, column, role, value):
|
||||||
|
# credit: ekhumoro
|
||||||
|
# http://stackoverflow.com/questions/13662020/how-to-implement-itemchecked-and-itemunchecked-signals-for-qtreewidget-in-pyqt4
|
||||||
|
checkstate = self.checkState(column)
|
||||||
|
text = self.text(column)
|
||||||
|
QtGui.QTreeWidgetItem.setData(self, column, role, value)
|
||||||
|
|
||||||
|
treewidget = self.treeWidget()
|
||||||
|
if treewidget is None:
|
||||||
|
return
|
||||||
|
if (role == QtCore.Qt.CheckStateRole and checkstate != self.checkState(column)):
|
||||||
|
treewidget.sigItemCheckStateChanged.emit(self, column)
|
||||||
|
elif (role in (QtCore.Qt.DisplayRole, QtCore.Qt.EditRole) and text != self.text(column)):
|
||||||
|
treewidget.sigItemTextChanged.emit(self, column)
|
||||||
|
|
||||||
|
def itemClicked(self, col):
|
||||||
|
"""Called when this item is clicked on.
|
||||||
|
|
||||||
|
Override this method to react to user clicks.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class InvisibleRootItem(QtGui.QTreeWidgetItem):
|
||||||
|
"""Wrapper around a TreeWidget's invisible root item that calls
|
||||||
|
TreeWidget.informTreeWidgetChange when child items are added/removed.
|
||||||
|
"""
|
||||||
|
def __init__(self, item):
|
||||||
|
self._real_item = item
|
||||||
|
|
||||||
|
def addChild(self, child):
|
||||||
|
self._real_item.addChild(child)
|
||||||
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
|
|
||||||
|
def addChildren(self, childs):
|
||||||
|
self._real_item.addChildren(childs)
|
||||||
|
for child in childs:
|
||||||
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
|
|
||||||
|
def insertChild(self, index, child):
|
||||||
|
self._real_item.insertChild(index, child)
|
||||||
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
|
|
||||||
|
def insertChildren(self, index, childs):
|
||||||
|
self._real_item.addChildren(index, childs)
|
||||||
|
for child in childs:
|
||||||
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
|
|
||||||
|
def removeChild(self, child):
|
||||||
|
self._real_item.removeChild(child)
|
||||||
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
|
|
||||||
|
def takeChild(self, index):
|
||||||
|
child = self._real_item.takeChild(index)
|
||||||
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
|
return child
|
||||||
|
|
||||||
|
def takeChildren(self):
|
||||||
|
childs = self._real_item.takeChildren()
|
||||||
|
for child in childs:
|
||||||
|
TreeWidget.informTreeWidgetChange(child)
|
||||||
|
return childs
|
||||||
|
|
||||||
|
def __getattr__(self, attr):
|
||||||
|
return getattr(self._real_item, attr)
|
||||||
|
Loading…
Reference in New Issue
Block a user