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
子类中重新实现了索引函数(请参阅下面的完整工作示例) . 这导致了非常不同类型的错误 . 也就是说,树中的每个节点现在都包含整个表作为数据 . 无论我点击多少次,整个表格都会显示出来 . 像这样:
我似乎处于某种递归的噩梦中,不知道如何逃避 . 下面的相关问题表明我可能必须转到 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 回答
来自official documentation:
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 .Solution
理论上,Model-View框架的一个很好的特性是您可以拥有同一模型的多个视图 . 但实际上,
QAbstractTableModel
实际上是为了帮助您查看表格,而不是树木 . The documentation forQAbstractTableModel
说:然而,即使给出了警告,也有办法让它发挥作用 . 首先,正如JKSH所指出的,你必须修复
rowCount
(并注意它的第二个参数是父索引):其次,删除
index
的不切实际的重新实现,这使得选择行为的行为非常奇怪,原因我坦率地不明白 .但是,一般情况下,如果你想要一个通用模型,那么从
QAbstractItemModel
开始安全和子类化,而不是一个预先制作的模型 .Discussion
在表模型中,忽略
rowCount
中的父级是可接受的 . 在官方Qt书中,它们遵循标准程序,即rowCount
仅返回表中显示的行数 . Blanchette和Summerfield注:在他的PyQt书中,Summerfield注意到:
基本上,
rowCount
告诉您在父项下面显示多少行 . 因为在表中所有项都具有相同的父项,所以在QTableView
中不使用父项 . 但是由于JKSH在他的回答中指出非常好的原因,这种策略不适用于树木 .因此,声明应该修改父参数"has no meaning for a table model",并且具有此限定条件仅当数据将由
QTableView
显示时才会为true(这通常是一个非常好的假设) .