首页 文章

在QTreeView中显示QAbstractTableModel的数据

提问于
浏览
2

Short version

我试图在 QTreeView 中显示来自 QAbstracctTableModel 的数据 . 当我这样做时,我最终将整个表显示为每个节点(和孙子等等)的子节点 . 如何显示抽象表模型的树视图?

Details

我试图在 QAbstractTableModel 中的 QAbstractTableModel 中显示一些数据 . 在Model-View Tutorial中,在展示了一个示例 QAbstractTableModel 之后,它似乎就像用 QTreeView 替换_2817339一样简单:

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

当我做这个替换时,我最终得到一个树显示,但是如果我点击任何图标来展开它(它应该什么也不做,因为没有内置的层次结构),Python崩溃了 Python.exe has stopped working . 这个has been brought up before],但没有可行的解决方案 .

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

bad tree

我似乎处于某种递归的噩梦中,不知道如何逃避 . 下面的相关问题表明我可能必须转到 QAbstractItemModel ,但上面的教程引用另有说明(其中说明,不需要对模型进行任何更改) .

Related question

QTreeView always displaying the same data

Full working example

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

2 回答

  • 4

    来自official documentation

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

    columnCount() 也是如此 . 为完整起见,如果父级有效, data() 应返回 None .


    发生了什么事:

    • 您单击"Stinkberries"旁边的'+'符号 .

    • QTreeView认为,"I need to expand the view. I wonder how many rows exist under 'Stinkberries'?"为了找到答案,QTreeView调用 rowCount() ,并将"Stinkberries"单元格的索引作为父级传递 .

    • FavoritesTableModel::rowCount() 返回5,所以QTreeView认为,"Ah, there are 5 rows under 'Stinkberries'."

    • 对列进行相同的处理 .

    • QTreeView决定检索"Stinkberries"下的第一个项目 . 它调用 data() ,传递第0行,第0列,以及"Stinkberries"单元格的索引作为父级 .

    • FavoritesTableModel::data() 返回"Apples",所以QTreeView认为,"Ah, the first item under 'Stinkberries' is 'Apples'."

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


    最后,为确保'+'符号根本不显示,请使 hasChildren() 为每个单元格返回false .

  • 3

    Solution

    理论上,Model-View框架的一个很好的特性是您可以拥有同一模型的多个视图 . 但实际上, QAbstractTableModel 实际上是为了帮助您查看表格,而不是树木 . The documentation for QAbstractTableModel 说:

    由于模型提供了比QAbstractItemModel更专业的接口,因此它不适合与树视图一起使用

    然而,即使给出了警告,也有办法让它发挥作用 . 首先,正如JKSH所指出的,你必须修复 rowCount (并注意它的第二个参数是父索引):

    def rowCount(self, parent=QtCore.QModelIndex()):
        if parent.isValid():
            return 0
        return len(self.foods)
    

    其次,删除 index 的不切实际的重新实现,这使得选择行为的行为非常奇怪,原因我坦率地不明白 .

    但是,一般情况下,如果你想要一个通用模型,那么从 QAbstractItemModel 开始安全和子类化,而不是一个预先制作的模型 .

    Discussion

    在表模型中,忽略 rowCount 中的父级是可接受的 . 在官方Qt书中,它们遵循标准程序,即 rowCount 仅返回表中显示的行数 . Blanchette和Summerfield注:

    父参数对表模型没有意义;因为rowCount()和columnCount()是从更通用的QAbstractItemModel基类继承的,它支持层次结构 . (第255页)

    在他的PyQt书中,Summerfield注意到:

    [T]他的父母QModelIndex p仅对树模型有用(第434页)

    基本上, rowCount 告诉您在父项下面显示多少行 . 因为在表中所有项都具有相同的父项,所以在 QTableView 中不使用父项 . 但是由于JKSH在他的回答中指出非常好的原因,这种策略不适用于树木 .

    因此,声明应该修改父参数"has no meaning for a table model",并且具有此限定条件仅当数据将由 QTableView 显示时才会为true(这通常是一个非常好的假设) .

相关问题