如何使用 Delegate 控制 QTableView 的行高

Posted

技术标签:

【中文标题】如何使用 Delegate 控制 QTableView 的行高【英文标题】:How to use Delegate to control QTableView's rows height 【发布时间】:2015-01-19 03:04:57 【问题描述】:

此代码创建一个QTableView,并将QSpinBoxes 设置为其委托项。我想根据此处描述的技术使用SpinBoxDelegatesizeHint() 方法返回一个固定的QSize(64,64): Adjusting the height of a QTableView Row

不幸的是,指定的QSize(64,64) 仅针对单行#1 (?) 返回。为什么不是每一行都发生这种情况?代码中的错误在哪里?

import sip
sip.setapi('QVariant', 2)

from PyQt4.QtCore import *
from PyQt4.QtGui import *


class SpinBoxDelegate(QItemDelegate):
    def createEditor(self, parent, option, index):
        editor = QSpinBox(parent)
        editor.setMinimum(0)
        editor.setMaximum(100)

        return editor

    def setEditorData(self, spinBox, index):
        value = index.model().data(index, Qt.EditRole)

        spinBox.setValue(value)

    def setModelData(self, spinBox, model, index):
        spinBox.interpretText()
        value = spinBox.value()

        model.setData(index, value, Qt.EditRole)

    def updateEditorGeometry(self, editor, option, index):
        print option
        editor.setGeometry(option.rect)
    def sizeHint(self, option, index):
        print 'sizeHint', index.row(), index.column()
        return QSize(64,64)

if __name__ == '__main__':

    import sys

    app = QApplication(sys.argv)

    model = QStandardItemModel(4, 2)
    tableView = QTableView()
    tableView.setModel(model)

    delegate = SpinBoxDelegate()
    tableView.setItemDelegate(delegate)

    for row in range(4):
        for column in range(2):
            index = model.index(row, column, QModelIndex())
            model.setData(index, (row + 1) * (column + 1))

    tableView.resizeRowToContents(True)

    tableView.show()
    sys.exit(app.exec_())

【问题讨论】:

【参考方案1】:

错误是使用tableView.resizeRowToContents(True) 而不是tableView.resizeRowsToContents() 更新所有当前可见的QModelIndexes。固定代码如下:

import sip
sip.setapi('QVariant', 2)

from PyQt4.QtCore import *
from PyQt4.QtGui import *


class SpinBoxDelegate(QItemDelegate):
    def createEditor(self, parent, option, index):
        editor = QSpinBox(parent)
        editor.setMinimum(0)
        editor.setMaximum(100)

        return editor

    def setEditorData(self, spinBox, index):
        value = index.model().data(index, Qt.EditRole)

        spinBox.setValue(value)

    def setModelData(self, spinBox, model, index):
        spinBox.interpretText()
        value = spinBox.value()

        model.setData(index, value, Qt.EditRole)

    def updateEditorGeometry(self, editor, option, index):
        print option
        editor.setGeometry(option.rect)
    def sizeHint(self, option, index):
        print 'sizeHint', index.row(), index.column()
        return QSize(64,64)

if __name__ == '__main__':

    import sys

    app = QApplication(sys.argv)

    model = QStandardItemModel(4, 2)
    tableView = QTableView()
    tableView.setModel(model)

    delegate = SpinBoxDelegate()
    tableView.setItemDelegate(delegate)

    for row in range(4):
        for column in range(2):
            index = model.index(row, column, QModelIndex())
            model.setData(index, (row + 1) * (column + 1))

    tableView.resizeRowsToContents()

    tableView.show()
    sys.exit(app.exec_())

或者这里是相同概念的骨架版本:

import sys
from PyQt4.QtCore import *
from PyQt4.QtGui import *

class ForSizeOnlyDelegate(QItemDelegate):
    def sizeHint(self, option, index):
        print 'sizeHint', index.row(), index.column()
        return QSize(64,64)

if __name__ == '__main__':
    app = QApplication(sys.argv)

    model = QStandardItemModel(14, 12)
    tableView = QTableView()
    tableView.setModel(model)

    delegate = ForSizeOnlyDelegate()

    tableView.setItemDelegate(delegate)
    tableView.resizeRowsToContents()

    tableView.show()
    sys.exit(app.exec_())

【讨论】:

以上是关于如何使用 Delegate 控制 QTableView 的行高的主要内容,如果未能解决你的问题,请参考以下文章

如何从app delegate访问NSViewController?

Cocoa:根据来自 App Delegate 的输入加载不同的视图控制器

如何从我的视图控制器引用我的 App Delegate 中的 managedObjectContext?

如何在 App Delegate 中将选项卡控制器设置为根控制器而不实例化(重新加载)它?

从 App Delegate 打开导航控制器

使用 App Delegate 访问通过 Storyboard 创建的视图控制器