检查影响 QListWidget 中特定项目集的项目

Posted

技术标签:

【中文标题】检查影响 QListWidget 中特定项目集的项目【英文标题】:Check an item that effects on a certain set of items within QListWidget 【发布时间】:2019-07-12 01:33:10 【问题描述】:

我有一个我创建并添加到 QListWidget 的项目列表,其中这些项目被“分类”。

在我的以下代码中,我有 2 个类别 - -- All Nums ---- All Letters --。在每个类别中,它都有其子项。

由于我已将所有项目设为可勾选,是否可以勾选/取消勾选这些标题项目并影响其子项目?

例如。如果我检查-- All Nums,所有num_items 也会被检查。如果我取消选中它,它们也会取消选中。


class TestDialog(QtGui.QDialog):
    def __init__(self, parent=None):
        super(TestDialog, self).__init__()
        self.listWidget = QtGui.QListWidget()

        all_num = QtGui.QListWidgetItem('-- All Nums --')
        self.listWidget.addItem(all_num)

        num_items = ['One', 'Two', 'Three']
        for num in num_items:
            self.listWidget.addItem(num)

        all_letters = QtGui.QListWidgetItem('-- All Letters --')
        self.listWidget.addItem(all_letters)

        letter_items = ['One', 'Two', 'Three']
        for letter in letter_items:
            self.listWidget.addItem(letter)

        for index in range(self.listWidget.count()):
            item = self.listWidget.item(index)
            item.setFlags(
                item.flags() |
                QtCore.Qt.ItemIsUserCheckable |
                QtCore.Qt.ItemIsEditable
            )
            item.setCheckState(QtCore.Qt.Unchecked)

        layout = QtGui.QVBoxLayout(self)
        layout.addWidget(self.listWidget)


if __name__ == '__main__':
    import sys
    app = QtGui.QApplication(sys.argv)
    w = TestDialog()
    w.show()
    sys.exit(app.exec_())

【问题讨论】:

【参考方案1】:

解决思路是使用一个角色来存储项目是否为类别的信息,并使用另一个角色来指示项目之间的父级关系,然后使用委托对其进行修改是否满足要求。

from PyQt4 import QtCore, QtGui


IsCategoryRole = QtCore.Qt.UserRole
ParentRole = QtCore.Qt.UserRole + 1


class CategoryDelegate(QtGui.QStyledItemDelegate):
    def editorEvent(self, event, model, option, index):
        old_state = model.data(index, QtCore.Qt.CheckStateRole)
        res = super(CategoryDelegate, self).editorEvent(
            event, model, option, index
        )
        current_state = model.data(index, QtCore.Qt.CheckStateRole)
        if old_state != current_state:
            if index.data(IsCategoryRole):
                pix = QtCore.QPersistentModelIndex(index)
                for i in range(model.rowCount()):
                    ix = model.index(i, 0)
                    if pix == ix.data(ParentRole):
                        model.setData(
                            ix, current_state, QtCore.Qt.CheckStateRole
                        )
        return res


class ListWidget(QtGui.QListWidget):
    def __init__(self, parent=None):
        super(ListWidget, self).__init__(parent)
        delegate = CategoryDelegate(self)
        self.setItemDelegate(delegate)

    def addCategory(self, text):
        item = ListWidget.create_checkable_item(text)
        item.setData(IsCategoryRole, True)
        self.addItem(item)
        return item

    def addItemToCategory(self, category_item, text):
        item = ListWidget.create_checkable_item(text)
        item.setData(IsCategoryRole, False)
        ix = self.indexFromItem(category_item)
        pix = QtCore.QPersistentModelIndex(ix)
        item.setData(ParentRole, pix)
        self.addItem(item)
        return item

    @staticmethod
    def create_checkable_item(text):
        item = QtGui.QListWidgetItem(text)
        item.setFlags(
            item.flags()
            | QtCore.Qt.ItemIsUserCheckable
            | QtCore.Qt.ItemIsEditable
        )
        item.setCheckState(QtCore.Qt.Unchecked)
        return item


class TestDialog(QtGui.QDialog):
    def __init__(self, parent=None):
        super(TestDialog, self).__init__()
        self.listWidget = ListWidget()

        all_num = self.listWidget.addCategory("-- All Nums --")
        num_items = ["One", "Two", "Three"]
        for num in num_items:
            self.listWidget.addItemToCategory(all_num, num)

        all_letters = self.listWidget.addCategory("-- All Letters --")

        letter_items = ["One", "Two", "Three"]
        for letter in letter_items:
            self.listWidget.addItemToCategory(all_letters, letter)

        layout = QtGui.QVBoxLayout(self)
        layout.addWidget(self.listWidget)


if __name__ == "__main__":
    import sys

    app = QtGui.QApplication(sys.argv)
    w = TestDialog()
    w.show()
    sys.exit(app.exec_())

【讨论】:

谢谢。我可以为此创建另一个问题,当我尝试为ListWidget class() 下的类别菜单创建上下文菜单时,我尝试使用setCheckState - 它确实检查/取消选中类别,但它似乎没有做同样的事情对于该类别中的项目,除非我直接单击该选项。

以上是关于检查影响 QListWidget 中特定项目集的项目的主要内容,如果未能解决你的问题,请参考以下文章

Apriori / Market Basket Analysis - 对特定长度项目集的限制分析

QCombobox & QListWidget - 检查内容

如何获取Qt QListWidget中列出的已检查项目

如何在 PyQt5 中返回 QlistWidget 中项目的值

PyQt5 检查项目是不是已经在 QListWidget

PyQt4 - 从 QListWidget 中删除项目小部件