PyQt:我如何设置QTreeView的行高度



在PyQt中,我正在寻找一种方法来设置QTreeView中的行高度(类似于QTableView.setRowHeight(row, row_height),但QTreeView没有此功能)。使用QAbstractItemModel建立树模型。我在这里阅读了一些使用和子类化QAbstractItemDelegate.sizeHint(option, index)的建议,但我不知道如何在我的树模型中正确地调用它们。

任何最小的代码或建议将非常感激。谢谢。

QTreeView根据每个项目返回的数据和大小提示计算出每行的高度。我认为你可能只需要返回大小提示,无论是为所有的项目,或至少第一行(如果你有setUniformRowHeights(True))。顺便说一句,这可以显著提高性能,所以你应该设置它,如果你可以。

所以你只需要实现你的AbstractItemModel.data()方法在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 itemviews示例。尝试改变data方法中返回的QSize,看看视图是如何变化的:

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_())

使用stylesheet:

实现这个功能非常简单
self.setStyleSheet("QTreeView::item { padding: 10px }")

不了解Python,但c++代码应该是这样的:

model->setData(model->index(/*your index*/), QSize(20, 20), Qt::SizeHintRole);

你需要为树中的所有项目设置它。

如果你想使用QItemDelegate -你不需要调用这个函数,你只需要设置你的委托为view,像这样(c++代码,但主要思想是一样的):

treeView->setItemDelegate(new MyDelegate(this));

对于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,并由另一个用户请求。

在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;
    }}
    ''')

我推测这样设置颜色和高度可以提供一点性能改进。

最新更新