PyQt: Как я могу установить высоту строки QTreeView

В PyQt я ищу способ установить высоту строк в QTreeView (аналогично QTableView.setRowHeight(row, row_height), но QTreeView не имеет этой функции). QAbstractItemModel используется для установки модели дерева. Я прочитал некоторые предложения здесь, используя и подклассы QAbstractItemDelegate.sizeHint(option, index) но я не знаю точно, как правильно их вызывать в рамках модели дерева.

Любой минимальный код или предложение будет принята с благодарностью. Благодарю.

3 ответа

Решение

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. Попробуйте изменить 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_())

Сделать это с помощью таблицы стилей довольно просто:

self.setStyleSheet("QTreeView::item { padding: 10px }")

Не знаю насчет Python, но код C++ будет выглядеть так:

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

И вам нужно установить его для всех элементов вашего дерева.

Если вы хотите использовать QItemDelegate - вам не нужно вызывать эту функцию, вы просто настраиваете свой делегат на просмотр, вот так (снова код C++, но основная идея та же):

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

Представление будет использовать его, когда это будет необходимо.

Для привязок PySide вот код Python для пользовательского делегата в AbstractItemModel.

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

Я предполагаю, что установка цветов и высоты таким образом дает небольшое улучшение производительности.

Другие вопросы по тегам