gpt4 book ai didi

qt - 模型数据未更改时刷新 View (Qt/PySide/PyQt)?

转载 作者:行者123 更新时间:2023-12-04 12:43:04 26 4
gpt4 key购买 nike

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

enter image description here

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

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

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

在实践中,我发现使用 layoutChanged 作品:

QtGui.QStandardItemModel.layoutChanged.emit()

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

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

无论如何,网上有很多关于更改模型中的数据时该怎么做的讨论(请参阅相关帖子),但我没有找到关于当您只想简单地刷新 View 以获得纯粹的装饰性时该怎么做原因。

跨帖

我在 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()

    最佳答案

    解决方案

    原则性解决方案是发出 QAbstractItemDelegate.sizeHintChanged当 spinbox 值改变时。这是因为您只想调用 sizeHint你的委托(delegate),这正是这个方法所做的。

    在 OP 中的示例中,大小提示旨在在旋转框中的值更改时更改。您可以连接valueChanged从 spinbox 到代表的 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 时有效,它的原则性较差,因为从技术上讲,它是用来告诉 View 模型的项目已重新排列,而它们没有。

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

    有原则比快速好吗?

    请注意,当您按照原则方式进行操作时,它实际上比使用 layoutChanged 慢一点。诡计。专门运行 layoutChanged大约需要 70 微秒,同时发出 sizeHintChanged大约需要 100 微秒。这不取决于我测试的模型的大小(最多 1000 行)。这种 30 微秒的差异非常小,在大多数应用程序中可以忽略不计,但如果有人真的想完全优化速度,他们可能会选择 layoutChanged。诡计。
    layoutChanged技巧还具有更简单的好处:它不涉及弄乱委托(delegate),而是在主窗口类上使用直观简单的方法。同样因为它不依赖于委托(delegate)中实现的任何方法,所以这个技巧(可以说)似乎更加模块化。 “正确”的方式取决于在委托(delegate)和主窗口之间创建更脆弱的依赖关系,这意味着当开发人员修改其中一个或另一个时,更容易破坏应用程序。

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

    致谢

    我开始意识到 sizeHintChanged 的存在来自我在 QtCentre 交叉发布的同一个问题的答案:

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

    关于qt - 模型数据未更改时刷新 View (Qt/PySide/PyQt)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33061249/

    26 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com