模型数据未更改时刷新视图(Qt/PySide/PyQt)?

Posted

技术标签:

【中文标题】模型数据未更改时刷新视图(Qt/PySide/PyQt)?【英文标题】:Refresh view when model data has not changed (Qt/PySide/PyQt)? 【发布时间】:2016-01-08 18:12:58 【问题描述】:

我有一个标准项目模型的树视图,我可以在其中使用旋转框来更改视图中的行高(请参阅下面的 SSCCE)。这不会改变视图的内容,只会改变它的外观,有点像调整主窗口的大小,除了我必须自己做:

我从委托的sizeHint 方法中更改行高。在sizeHint 内,我从旋转框中获取值并将行高设置为该值。为了确保实际调用了大小提示,我会在微调框值更改时刷新视图。

我的问题是:在这种纯粹的外观更改的情况下,告诉视图刷新的推荐方法是什么?有没有专门为这种情况构建的方法? 显然,这个问题假设我调整行高的一般策略是合理的,我也愿意更正。

有几种方法可以告诉视图是时候重新获取数据并重新绘制内容了:layoutChangedresetsetModeldataChanged。见鬼,我发现即使只是在树上调用expandAll 也足以更新我的视图以显示新的行高。

在实践中,我发现使用 layoutChanged 可以工作:

QtGui.QStandardItemModel.layoutChanged.emit()

这是一种不常见的用法,因为它更适用于重新排列数据(例如,通过排序)时。这就是我在下面的 SSCCE 中包含的内容,因为它有效。我还尝试遵循更常见的建议做法,即发送dataChanged

 QtGui.QStandardItemModel.dataChanged.emit(QtCore.QModelIndex(), QtCore.QModelIndex())

这对我不起作用。即使是这样,它也将是一种 hack,因为它告诉视图模型中的数据已更改。当它没有时。

无论如何,网上有很多关于更改模型中的数据时该怎么做的讨论(请参阅相关帖子),但我没有发现当您只想刷新视图时该怎么做纯粹是为了美观。

跨帖

我在 Qt 中心发布了同样的问题:

http://www.qtcentre.org/threads/63982-Best-way-to-refresh-view-for-cosmetic-%28non-model%29-changes

我在那里得到了一个答案,该答案已包含在下面接受的答案中。

相关帖子

Qt Model-View update view?

PyQt - Automatically refresh a custom view when the model is updated?

http://www.qtcentre.org/threads/48230-QTreeView-How-to-refresh-the-view

http://www.qtcentre.org/threads/3145-QTableView-How-to-refresh-the-view-after-an-update-of-the-model

SSCCE

import sys
from PySide import QtGui, QtCore

class MainTree(QtGui.QMainWindow):
    def __init__(self, parent = None):
        QtGui.QMainWindow.__init__(self)
        self.setAttribute(QtCore.Qt.WA_DeleteOnClose)
        self.createRowHeightSpinbox() #create first otherwise get errors
        self.tree = SimpleTree(self)
        self.setCentralWidget(self.tree)       
        #Add spinbox to toolbar
        self.rowHeightAction = QtGui.QAction("Change row height", self)
        self.toolbar = self.addToolBar("rowHeight")
        self.toolbar.addWidget(QtGui.QLabel("Row height "))
        self.toolbar.addWidget(self.rowHeightSpinBox)  
        #Expand and resize tree
        self.tree.expandAll()
        self.tree.resizeColumnToContents(0) 
        self.tree.resizeColumnToContents(1) 

    def createRowHeightSpinbox(self):
        self.rowHeightSpinBox = QtGui.QSpinBox()
        self.rowHeightSpinBox.setRange(10, 50)
        self.rowHeightSpinBox.setValue(18)
        self.rowHeightSpinBox.valueChanged.connect(self.refreshView)  #showimage uses the spinbox attribute to scale image

    def refreshView(self):
        self.tree.model.layoutChanged.emit()


class SimpleTree(QtGui.QTreeView):
    def __init__(self, parent = None):    
        QtGui.QTreeView.__init__(self, parent)
        self.setUniformRowHeights(True) #optimize
        self.model = QtGui.QStandardItemModel()
        self.rootItem = self.model.invisibleRootItem()
        item0 = [QtGui.QStandardItem('Sneeze'), QtGui.QStandardItem('You have been blocked up')]
        item00 = [QtGui.QStandardItem('Tickle nose'), QtGui.QStandardItem('Key first step')]
        item1 = [QtGui.QStandardItem('Get a job'), QtGui.QStandardItem('Do not blow it')]
        self.rootItem.appendRow(item0)
        item0[0].appendRow(item00) 
        self.rootItem.appendRow(item1)
        self.setModel(self.model)
        self.setItemDelegate(ExpandableRows(self))


class ExpandableRows(QtGui.QStyledItemDelegate):
    def __init__(self, parent=None):
        QtGui.QStyledItemDelegate.__init__(self, parent)
        self.parent = parent

    def sizeHint(self, option, index):
        rowHeight = self.parent.window().rowHeightSpinBox.value()
        text = index.model().data(index)
        document = QtGui.QTextDocument()
        document.setDefaultFont(option.font)
        document.setPlainText(text)  #for html use setHtml
        return QtCore.QSize(document.idealWidth() + 5,  rowHeight) 


def main():
    app = QtGui.QApplication(sys.argv)
    #myTree = SimpleTree()
    myMainTree = MainTree()
    myMainTree.show()
    sys.exit(app.exec_())


if __name__ == "__main__":
    main()

【问题讨论】:

我通常只使用dataChanged @figs 奇怪的是,这是一件事没有奏效,当我尝试 self.tree.model.dataChanged.emit(QtCore.QModelIndex(), QtCore.QModelIndex()) 而不是 layoutchanged 时,实际上什么也没发生。也许我做错了?我已经更新了我原来的 Q 以明确我尝试了什么并且它没有工作。 如果我能让它发挥作用,至少我可以开始做比较/指标。我有 4 种其他的工作方式,简单地比较它们对于不同尺寸模型的速度会很有趣,并以此为基础进行回答。毕竟,代理并不关心数据是否已更改:它使用paint 重新绘制数据,即使您只是进行外观更改,例如更改背景颜色、行高等等.这可能是正确的答案。因此,如果我们让dataChanged 工作,我将能够进行适当的度量。 @figs 很奇怪:dataChanged 导致我的委托重新绘制,但不调用 sizeHint。当我使用layoutChanged sizeHint 确实被调用时,这表明这就是它起作用的原因。无论如何,这变得越来越复杂,目前还不清楚发生了什么。如果有人可以让dataChanged 在我的 SSCCE 工作,我会非常好奇。 @figs 如果你看这里 (qtcentre.org/threads/…),你会发现对 dataChanged 非常奇怪的行为的描述。稍后将添加更多详细信息,可能会单独提出问题,只是想突出显示它。 【参考方案1】:

解决方案

原则上的解决方案是在 spinbox 值更改时发出QAbstractItemDelegate.sizeHintChanged。这是因为您只想调用委托的sizeHint,而这正是此方法的作用。

在 OP 中的示例中,大小提示旨在在旋转框中的值更改时更改。您可以将来自 spinbox 的 valueChanged 信号连接到代理的 sizeHintChanged 信号,如下所示:

class ExpandableRows(QtGui.QStyledItemDelegate):
    def __init__(self, parent=None):
        QtGui.QStyledItemDelegate.__init__(self, parent)
        self.parent = parent
        self.parent.window().rowHeightSpinBox.valueChanged.connect(self.emitSizeChange)

    def emitSizeChange(self):
        self.sizeHintChanged.emit(QtCore.QModelIndex())

分析

如 OP 中所述,您不想调用 dataChanged,因为它实际上不起作用,并且您的数据实际上并没有改变。此外,虽然调用 layoutChanged 有效,但它的原则性较差,因为从技术上讲,它是用来告诉视图模型的项目已重新排列,而它们没有。

警告:我相信sizeHintChanged 期望使用有效索引,但我的解决方案是使用无效索引。因为它有效,所以我将其保留为无效的QtCore.QModelIndex()。也许有人可以对此进行改进并编辑此答案。

有原则比快速好吗?

请注意,当您按照原则方式进行操作时,它实际上比使用 layoutChanged 技巧要慢一些。具体运行layoutChanged 大约需要70 微秒,而发射sizeHintChanged 大约需要100 微秒。这不取决于我测试的模型的大小(最多 1000 行)。这种 30 微秒的差异非常小,在大多数应用程序中可以忽略不计,但如果有人真的想完全优化速度,他们可能会采用 layoutChanged 技巧。

layoutChanged 技巧还具有更简单的优点:它不涉及搞乱委托,而是在主窗口类上使用直观简单的方法。也因为它不依赖于委托中实现的任何方法,这个技巧(可以说)似乎更加模块化。 “正确”的方式取决于在委托和主窗口之间创建更脆弱的依赖关系,这意味着当开发人员修改其中一个或另一个时,更容易破坏应用程序。

总之,可以证明,在几乎所有可衡量的方式中,来自 OP 的 hack 都比这里的“有原则”解决方案更好。

致谢

我从我在 QtCentre 交叉发布的同一个问题的答案中了解到 sizeHintChanged 的存在:

http://www.qtcentre.org/threads/63982-Best-way-to-refresh-view-for-cosmetic-%28non-model%29-changes

【讨论】:

哇,我不会错过使用 pyqt 深入杂草丛生的机会。

以上是关于模型数据未更改时刷新视图(Qt/PySide/PyQt)?的主要内容,如果未能解决你的问题,请参考以下文章

Angular 2组件模型刷新视图,无需模型更改

在 Android 视图模型中的内部网络更改回调时 LiveData 未触发 - Kotlin

由于事件导致模型更改后AngularJS不刷新视图

模型更改时刷新 ViewModel 的所有属性的数据绑定的好方法

对 SwiftUI FetchRequest 的更改不会触发视图刷新?

Ruby on Rails:视图更改后,网站未更新