在 QTreeView 中显示来自 QAbstractTableModel 的数据

eri*_*ric 3 qt pyqt pyside

简洁版本

我正在尝试显示 aQAbstracctTableModel中的数据QTreeView。当我这样做时,我最终将整个表显示为每个节点的子节点(以及孙节点,等等)。如何显示抽象表模型的树视图?

细节

QAbstractTableModel我正在尝试在a 中显示一些数据QTreeView。在模型-视图教程中,介绍了一个示例后QAbstractTableModel,它看起来就像替换为一样QTableView简单QTreeView

您可以将上面的示例转换为具有树视图的应用程序。只需将 QTableView 替换为 QTreeView,就会生成一个读/写树。无需对模型进行任何更改。

当我进行此替换时,我最终会显示一棵树,但如果我单击任何图标来展开它(这应该不会执行任何操作,因为没有内置层次结构),Python 会崩溃并显示Python.exe has stopped working. 这个问题之前已经提出过],但没有可行的解决方案。

为了尝试修复此行为,我在子QAbstractTableModel类中重新实现了索引函数(请参阅下面的完整工作示例)。这会导致一种非常不同类型的错误。也就是说,树中的每个节点现在都包含整个表作为数据。无论我点击多少次,整个表格都会显示出来。像这样:

坏树

我似乎陷入了某种递归的噩梦中,不知道如何逃脱。下面的相关问题表明我可能必须转到QAbstractItemModel,但上面的教程引用表明不然(其中指出,无需对模型进行任何更改)。

相关问题

QTreeView总是显示相同的数据

完整的工作示例

from PySide import QtGui, QtCore

class Food(object):
    def __init__(self, name, shortDescription, note, parent = None):
        self.data = (name, shortDescription, note);
        self.parentIndex = parent

class FavoritesTableModel(QtCore.QAbstractTableModel):
    def __init__(self):
        QtCore.QAbstractTableModel.__init__(self)
        self.foods = []  
        self.loadData() 

    def data(self, index, role = QtCore.Qt.DisplayRole):
        if role == QtCore.Qt.DisplayRole:
            return self.foods[index.row()].data[index.column()]
        return None

    def rowCount(self, index=QtCore.QModelIndex()):
        return len(self.foods)

    def columnCount(self, index=QtCore.QModelIndex()):
        return 3

    def index(self, row, column, parent = QtCore.QModelIndex()):  
        return self.createIndex(row, column, parent)

    def loadData(self):   
        allFoods=("Apples", "Pears", "Grapes", "Cookies", "Stinkberries")
        allDescs = ("Red", "Green", "Purple", "Yummy", "Huh?")
        allNotes = ("Bought recently", "Kind of delicious", "Weird wine grapes",
                    "So good...eat with milk", "Don't put in your nose")
        for name, shortDescription, note in zip(allFoods, allDescs, allNotes):
            food = Food(name, shortDescription, note)                                      
            self.foods.append(food) 

def main():
    import sys
    app = QtGui.QApplication(sys.argv)

    model = FavoritesTableModel() 

    #Table view
    view1 = QtGui.QTableView()
    view1.setModel(model)
    view1.show()

    #Tree view
    view2 = QtGui.QTreeView()
    view2.setModel(model)
    view2.show()

    sys.exit(app.exec_())

if __name__ == '__main__':
    main()
Run Code Online (Sandbox Code Playgroud)

JKS*_*KSH 5

来自官方文档

实现基于表的模型时,rowCount()当父级有效时应返回 0。

也同样如此columnCount()。为了完整起见,如果父项有效,data()则应返回。None


发生的事情是这样的:

  1. 您单击“Stinkberry”旁边的“+”号。
  2. QTreeView 认为,“我需要扩展视图。我想知道‘Stinkberry’下有多少行?” 为了找出答案,QTreeView 调用rowCount(),并将“Stinkberry”单元格的索引作为父单元格传递。
  3. FavoritesTableModel::rowCount()返回 5,因此 QTreeView 认为,“啊,'Stinkberry' 下有 5 行。”
  4. 列也会发生相同的过程。
  5. QTreeView 决定检索“Stinkberry”下的第一个项目。它调用data(),传递第 0 行、第 0 列以及“Stinkberry”单元格的索引作为父单元格。
  6. FavoritesTableModel::data()返回“Apples”,因此 QTreeView 认为,“啊,‘Stinkberry’下的第一项是‘Apples’。”
  7. ETC。

为了获得正确的行为,您的代码必须为步骤 #3 和 #4 返回 0。


最后,为了确保“+”号根本不会出现,请hasChildren()为每个单元格设置 return false。