在模型数据未更改时刷新视图 (Qt/PySide/PyQt)?
Refresh view when model data has not changed (Qt/PySide/PyQt)?
我有一个标准项目模型的树视图,我可以在其中使用旋转框更改视图中的行高(请参阅下面的 SSCCE)。这不会改变视图的内容,只会改变它的外观,有点像调整主视图的大小 window 除了我必须自己做:
我在委托的 sizeHint
方法中更改了行高。在 sizeHint
内,我从旋转框获取值并将行高设置为该值。为确保实际调用大小提示,我在旋转框值更改时刷新视图。
我的问题是:在这种纯粹的外观变化的情况下,建议的刷新视图的方法是什么?是否有专门针对此类情况构建的方法?
显然,这个问题假设我调整行高的一般策略是合理的,我也愿意更正。
有几种方法可以告诉视图是时候重新获取数据并重绘了:layoutChanged
、reset
、setModel
、dataChanged
。该死,我发现即使只是在树上调用 expandAll
也足以更新我的视图以显示新的行高。
在实践中,我发现使用 layoutChanged 有效:
QtGui.QStandardItemModel.layoutChanged.emit()
这是一种不常见的用法,因为它更适用于重新排列数据(例如,通过排序)的情况。这是我在下面的 SSCCE 中包含的内容,因为它有效。我还尝试遵循更常见的发出 dataChanged
:
的建议做法
QtGui.QStandardItemModel.dataChanged.emit(QtCore.QModelIndex(), QtCore.QModelIndex())
这对我不起作用。即使它这样做了,它也可能是一种 hack,因为它告诉视图模型中的数据已更改。当它还没有。
无论如何,网上有很多关于更改模型中的数据时要做什么的讨论(请参阅相关帖子),但是 none 我发现当您只是纯粹出于美观原因,只想刷新视图。
交叉post
我在 Qt 中心 post 编辑了同样的问题:
http://www.qtcentre.org/threads/63982-Best-way-to-refresh-view-for-cosmetic-%28non-model%29-changes
我在那里得到了一个答案,该答案已包含在下面已接受的答案中。
相关posts
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
。这是因为您只想调用委托的 sizeHint
,而这正是此方法的作用。
在 OP 中的示例中,大小提示旨在在旋转框中的值更改时更改。您可以将旋转框的 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
技巧还有一个好处是更简单:它不涉及与委托打交道,而是在主要 window class 上使用直观简单的方法。也因为它不依赖于在委托中实现的任何方法,这个技巧(可以说)似乎更模块化。 "proper" 方式依赖于在委托和主要 window 之间创建更脆弱的依赖关系,这意味着当开发人员修改一个或另一个时,更容易破坏应用程序。
总而言之,可以证明,在几乎所有可衡量的方式中,来自 OP 的 hack 都比此处的 "principled" 解决方案更好。
致谢
我从我在 QtCentre 交叉发布的同一个问题的答案中了解到 sizeHintChanged
的存在:
http://www.qtcentre.org/threads/63982-Best-way-to-refresh-view-for-cosmetic-%28non-model%29-changes
我有一个标准项目模型的树视图,我可以在其中使用旋转框更改视图中的行高(请参阅下面的 SSCCE)。这不会改变视图的内容,只会改变它的外观,有点像调整主视图的大小 window 除了我必须自己做:
我在委托的 sizeHint
方法中更改了行高。在 sizeHint
内,我从旋转框获取值并将行高设置为该值。为确保实际调用大小提示,我在旋转框值更改时刷新视图。
我的问题是:在这种纯粹的外观变化的情况下,建议的刷新视图的方法是什么?是否有专门针对此类情况构建的方法? 显然,这个问题假设我调整行高的一般策略是合理的,我也愿意更正。
有几种方法可以告诉视图是时候重新获取数据并重绘了:layoutChanged
、reset
、setModel
、dataChanged
。该死,我发现即使只是在树上调用 expandAll
也足以更新我的视图以显示新的行高。
在实践中,我发现使用 layoutChanged 有效:
QtGui.QStandardItemModel.layoutChanged.emit()
这是一种不常见的用法,因为它更适用于重新排列数据(例如,通过排序)的情况。这是我在下面的 SSCCE 中包含的内容,因为它有效。我还尝试遵循更常见的发出 dataChanged
:
QtGui.QStandardItemModel.dataChanged.emit(QtCore.QModelIndex(), QtCore.QModelIndex())
这对我不起作用。即使它这样做了,它也可能是一种 hack,因为它告诉视图模型中的数据已更改。当它还没有。
无论如何,网上有很多关于更改模型中的数据时要做什么的讨论(请参阅相关帖子),但是 none 我发现当您只是纯粹出于美观原因,只想刷新视图。
交叉post
我在 Qt 中心 post 编辑了同样的问题:
http://www.qtcentre.org/threads/63982-Best-way-to-refresh-view-for-cosmetic-%28non-model%29-changes
我在那里得到了一个答案,该答案已包含在下面已接受的答案中。
相关posts
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
。这是因为您只想调用委托的 sizeHint
,而这正是此方法的作用。
在 OP 中的示例中,大小提示旨在在旋转框中的值更改时更改。您可以将旋转框的 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
技巧还有一个好处是更简单:它不涉及与委托打交道,而是在主要 window class 上使用直观简单的方法。也因为它不依赖于在委托中实现的任何方法,这个技巧(可以说)似乎更模块化。 "proper" 方式依赖于在委托和主要 window 之间创建更脆弱的依赖关系,这意味着当开发人员修改一个或另一个时,更容易破坏应用程序。
总而言之,可以证明,在几乎所有可衡量的方式中,来自 OP 的 hack 都比此处的 "principled" 解决方案更好。
致谢
我从我在 QtCentre 交叉发布的同一个问题的答案中了解到 sizeHintChanged
的存在:
http://www.qtcentre.org/threads/63982-Best-way-to-refresh-view-for-cosmetic-%28non-model%29-changes