PyQt:如何设置 QTreeView 的行高
PyQt: How Can I set row heights of QTreeView
在 PyQt 中,我正在寻找一种在 QTreeView
中设置行高的方法(类似于 QTableView.setRowHeight(row, row_height)
,但 QTreeView
没有此功能)。 QAbstractItemModel
用于设置树模型。我阅读了一些建议 here 使用和子类化 QAbstractItemDelegate.sizeHint(option, index)
但我不知道如何在我的树模型中正确调用它们。
任何最少的代码或建议将不胜感激。谢谢。
不知道 Python,但 C++ 代码应该是这样的:
model->setData(model->index(/*your index*/), QSize(20, 20), Qt::SizeHintRole);
并且您需要为树中的所有项目设置它。
如果你想使用 QItemDelegate - 你不需要调用这个函数,你只需设置你的委托来查看,就像这样(再次使用 C++ 代码,但主要思想是相同的):
treeView->setItemDelegate(new MyDelegate(this));
然后视图会在需要时使用它。
QTreeView
根据每个项目的数据和大小提示 return 计算出每一行的高度。我认为您可能只需要 return 大小提示,无论是对所有项目,还是至少对第一行(如果您有 setUniformRowHeights(True)
)。顺便说一下,这可以显着提高性能,所以如果可以的话,你应该设置它。
因此,您只需实施 AbstractItemModel.data()
方法即可 return SizeHintRole
中的尺寸提示。像这样:
def data(self, index, role = QtCore.Qt.DisplayRole):
# Check the index, possibly return None
if role == QtCore.Qt.DisplayRole:
# Return the data
elif role == QtCore.Qt.SizeHintRole:
return QtCore.QSize(item_width,item_height)
# Other roles - maybe return None if you don't use them.
编辑:大例子
您说您仍然遇到问题,所以这里是一个完整的工作示例,基于标准的 QT 项目视图示例。尝试改变数据方法中的 QSize returned 以查看视图如何变化:
import sys
from PySide import QtCore,QtGui
class TreeItem(object):
def __init__(self, data, parent=None):
self.parentItem = parent
self.data = data
self.childItems = []
def appendChild(self, item):
self.childItems.append(item)
def row(self):
if self.parentItem:
return self.parentItem.childItems.index(self)
return 0
class TreeModel(QtCore.QAbstractItemModel):
def __init__(self, parent=None):
super(TreeModel, self).__init__(parent)
self.rootItem = TreeItem(None)
for i,c in enumerate("abcdefg"):
child = TreeItem([i,c],self.rootItem)
self.rootItem.appendChild(child)
parent = self.rootItem.childItems[1]
child = TreeItem(["down","down"],parent)
parent.appendChild(child)
def columnCount(self, parent):
return 2
def data(self, index, role):
if not index.isValid():
return None
if role == QtCore.Qt.DisplayRole:
item = index.internalPointer()
return item.data[index.column()]
elif role == QtCore.Qt.SizeHintRole:
print "giving size hint"
return QtCore.QSize(40,40)
return None
def flags(self, index):
if not index.isValid():
return QtCore.Qt.NoItemFlags
return QtCore.Qt.ItemIsEnabled | QtCore.Qt.ItemIsSelectable
def headerData(self, section, orientation, role):
if orientation == QtCore.Qt.Horizontal and role == QtCore.Qt.DisplayRole:
return ["A","B"][section]
return None
def index(self, row, column, parent):
if not self.hasIndex(row, column, parent):
return QtCore.QModelIndex()
if not parent.isValid():
parentItem = self.rootItem
else:
parentItem = parent.internalPointer()
childItem = parentItem.childItems[row]
if childItem:
return self.createIndex(row, column, childItem)
else:
return QtCore.QModelIndex()
def parent(self, index):
if not index.isValid():
return QtCore.QModelIndex()
parentItem = index.internalPointer().parentItem
if parentItem == self.rootItem:
return QtCore.QModelIndex()
return self.createIndex(parentItem.row(), 0, parentItem)
def rowCount(self, parent):
if parent.column() > 0:
return 0
if not parent.isValid():
parentItem = self.rootItem
else:
parentItem = parent.internalPointer()
return len(parentItem.childItems)
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
model = TreeModel()
view = QtGui.QTreeView()
view.setModel(model)
view.setWindowTitle("Simple Tree Model")
view.show()
sys.exit(app.exec_())
对于 PySide 绑定,这里是 AbstractItemModel 上自定义委托的 Python 代码。
from PySide import QtCore, QtGui
tree = QTreeView()
tree.model = QtGui.QAbstractItemModel()
tree.setModel(tree.model)
size = QtCore.QSize(20, 20)
index = tree.model.index(row, col) # row, col are your own
tree.model.setData(index, size, QtCore.Qt.SizeHintRole)
delegate = MyDelegate()
tree.setItemDelegate(delegate)
这只是将代码从@RazrFalcon 翻译成Python,并应另一位用户的请求。
使用样式表来实现这一点非常简单:
self.setStyleSheet("QTreeView::item { padding: 10px }")
使用 Qt 5.9.2,我无法同时获取 QTreeView 自动调整列大小和自定义行高。使用 SizeHintRole 对高度有效,但似乎破坏了自动宽度。我的解决方案是将行设置为具有统一的高度,然后在 Python 3.7:
中执行类似的操作
self._tree = QTreeView()
self._tree.setAlternatingRowColors(True)
self._tree.setUniformRowHeights(True)
...
fi = QFontInfo(self.font())
self._tree.setStyleSheet(
f'''
QTreeView {{
alternate-background-color: #E6FFFF;
}}
QTreeView::Item{{
height:{fi.pixelSize() * 2}px;
}}
''')
我推测以这种方式设置颜色和高度会稍微提高性能。
在 PyQt 中,我正在寻找一种在 QTreeView
中设置行高的方法(类似于 QTableView.setRowHeight(row, row_height)
,但 QTreeView
没有此功能)。 QAbstractItemModel
用于设置树模型。我阅读了一些建议 here 使用和子类化 QAbstractItemDelegate.sizeHint(option, index)
但我不知道如何在我的树模型中正确调用它们。
任何最少的代码或建议将不胜感激。谢谢。
不知道 Python,但 C++ 代码应该是这样的:
model->setData(model->index(/*your index*/), QSize(20, 20), Qt::SizeHintRole);
并且您需要为树中的所有项目设置它。
如果你想使用 QItemDelegate - 你不需要调用这个函数,你只需设置你的委托来查看,就像这样(再次使用 C++ 代码,但主要思想是相同的):
treeView->setItemDelegate(new MyDelegate(this));
然后视图会在需要时使用它。
QTreeView
根据每个项目的数据和大小提示 return 计算出每一行的高度。我认为您可能只需要 return 大小提示,无论是对所有项目,还是至少对第一行(如果您有 setUniformRowHeights(True)
)。顺便说一下,这可以显着提高性能,所以如果可以的话,你应该设置它。
因此,您只需实施 AbstractItemModel.data()
方法即可 return SizeHintRole
中的尺寸提示。像这样:
def data(self, index, role = QtCore.Qt.DisplayRole):
# Check the index, possibly return None
if role == QtCore.Qt.DisplayRole:
# Return the data
elif role == QtCore.Qt.SizeHintRole:
return QtCore.QSize(item_width,item_height)
# Other roles - maybe return None if you don't use them.
编辑:大例子
您说您仍然遇到问题,所以这里是一个完整的工作示例,基于标准的 QT 项目视图示例。尝试改变数据方法中的 QSize returned 以查看视图如何变化:
import sys
from PySide import QtCore,QtGui
class TreeItem(object):
def __init__(self, data, parent=None):
self.parentItem = parent
self.data = data
self.childItems = []
def appendChild(self, item):
self.childItems.append(item)
def row(self):
if self.parentItem:
return self.parentItem.childItems.index(self)
return 0
class TreeModel(QtCore.QAbstractItemModel):
def __init__(self, parent=None):
super(TreeModel, self).__init__(parent)
self.rootItem = TreeItem(None)
for i,c in enumerate("abcdefg"):
child = TreeItem([i,c],self.rootItem)
self.rootItem.appendChild(child)
parent = self.rootItem.childItems[1]
child = TreeItem(["down","down"],parent)
parent.appendChild(child)
def columnCount(self, parent):
return 2
def data(self, index, role):
if not index.isValid():
return None
if role == QtCore.Qt.DisplayRole:
item = index.internalPointer()
return item.data[index.column()]
elif role == QtCore.Qt.SizeHintRole:
print "giving size hint"
return QtCore.QSize(40,40)
return None
def flags(self, index):
if not index.isValid():
return QtCore.Qt.NoItemFlags
return QtCore.Qt.ItemIsEnabled | QtCore.Qt.ItemIsSelectable
def headerData(self, section, orientation, role):
if orientation == QtCore.Qt.Horizontal and role == QtCore.Qt.DisplayRole:
return ["A","B"][section]
return None
def index(self, row, column, parent):
if not self.hasIndex(row, column, parent):
return QtCore.QModelIndex()
if not parent.isValid():
parentItem = self.rootItem
else:
parentItem = parent.internalPointer()
childItem = parentItem.childItems[row]
if childItem:
return self.createIndex(row, column, childItem)
else:
return QtCore.QModelIndex()
def parent(self, index):
if not index.isValid():
return QtCore.QModelIndex()
parentItem = index.internalPointer().parentItem
if parentItem == self.rootItem:
return QtCore.QModelIndex()
return self.createIndex(parentItem.row(), 0, parentItem)
def rowCount(self, parent):
if parent.column() > 0:
return 0
if not parent.isValid():
parentItem = self.rootItem
else:
parentItem = parent.internalPointer()
return len(parentItem.childItems)
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
model = TreeModel()
view = QtGui.QTreeView()
view.setModel(model)
view.setWindowTitle("Simple Tree Model")
view.show()
sys.exit(app.exec_())
对于 PySide 绑定,这里是 AbstractItemModel 上自定义委托的 Python 代码。
from PySide import QtCore, QtGui
tree = QTreeView()
tree.model = QtGui.QAbstractItemModel()
tree.setModel(tree.model)
size = QtCore.QSize(20, 20)
index = tree.model.index(row, col) # row, col are your own
tree.model.setData(index, size, QtCore.Qt.SizeHintRole)
delegate = MyDelegate()
tree.setItemDelegate(delegate)
这只是将代码从@RazrFalcon 翻译成Python,并应另一位用户的请求。
使用样式表来实现这一点非常简单:
self.setStyleSheet("QTreeView::item { padding: 10px }")
使用 Qt 5.9.2,我无法同时获取 QTreeView 自动调整列大小和自定义行高。使用 SizeHintRole 对高度有效,但似乎破坏了自动宽度。我的解决方案是将行设置为具有统一的高度,然后在 Python 3.7:
中执行类似的操作self._tree = QTreeView()
self._tree.setAlternatingRowColors(True)
self._tree.setUniformRowHeights(True)
...
fi = QFontInfo(self.font())
self._tree.setStyleSheet(
f'''
QTreeView {{
alternate-background-color: #E6FFFF;
}}
QTreeView::Item{{
height:{fi.pixelSize() * 2}px;
}}
''')
我推测以这种方式设置颜色和高度会稍微提高性能。