QTreeView请求无效行的索引

时间:2019-05-22 16:15:55

标签: python pyqt mvp qabstractitemmodel

看看下面的MWE。

这是一个简单的QAbstractItemModel,只有一个级别,将其项目存储在列表中。我创建了一个QTreeView来显示模型,并创建了一个按钮来删除第二项。

from PyQt5.QtCore import QModelIndex, QAbstractItemModel, Qt
from PyQt5.QtWidgets import QTreeView, QApplication, QPushButton


class Item:
    def __init__(self, title):
        self.title = title


class TreeModel(QAbstractItemModel):
    def __init__(self, parent=None):
        super().__init__(parent)
        self._items = []  # typing.List[Item]

    def addItem(self, item: Item):
        self.beginInsertRows(QModelIndex(), len(self._items), len(self._items))
        self._items.append(item)
        self.endInsertRows()

    def removeItem(self, item: Item):
        index = self._items.index(item)
        self.beginRemoveRows(QModelIndex(), index, index)
        self._items.remove(item)
        self.endRemoveRows()

    # ----- overridden methods from QAbstractItemModel -----

    # noinspection PyMethodOverriding
    def data(self, index: QModelIndex, role):
        item = index.internalPointer()
        if role == Qt.DisplayRole:
            return item.title

    # noinspection PyMethodOverriding
    def rowCount(self, parent=QModelIndex()):
        if not parent.isValid():
            return len(self._items)
        return 0

    # noinspection PyMethodOverriding
    def columnCount(self, parent=QModelIndex()):
        return 1

    # noinspection PyMethodOverriding
    def index(self, row: int, col: int, parent=QModelIndex()):
        assert not parent.isValid()
        return self.createIndex(row, 0, self._items[row])

    def parent(self, index=QModelIndex()):
        return QModelIndex()


def removeItem():
    model.removeItem(item2)


if __name__ == '__main__':
    app = QApplication([])
    model = TreeModel()
    button = QPushButton('Delete')
    button.clicked.connect(removeItem)
    button.show()
    item1 = Item('Item 1')
    model.addItem(item1)
    item2 = Item('Item 2')
    model.addItem(item2)
    treeView = QTreeView()
    treeView.setModel(model)
    treeView.show()

    app.exec()

据我所知,我模型的实现是正确的(尽管很基本)。特别是,它报告的行数和列数都是正确的,并且永远不会为无效数据创建索引。

重现我的问题的步骤:

  • 运行上面的代码。
  • 在树形视图中,选择项目2。
  • 按下删除按钮。

在我的系统上,应用程序在beginRemoveRows()中崩溃,因为视图请求第2行使用QModelIndex。自然地,第2行不存在。

有人知道为什么QTreeView会认为当模型明确报告只有2行时会有3行吗?

1 个答案:

答案 0 :(得分:1)

添加,移动,删除项目等时,模型执行的操作是验证QPersistentModelIndex有效还是无效,因此它调用QAbstractItemModel的index()方法。在这种方法中,开发人员有责任验证行或列是否有效,并且为此模型提供了您不使用的hasIndex()方法,这会导致您指出错误,因此解决方案是: / p>

def index(self, row: int, col: int, parent=QModelIndex()):
    if not self.hasIndex(row, col, parent):
        return QModelIndex()
    assert not parent.isValid()
    return self.createIndex(row, 0, self._items[row])