PyQt5 可检查组合框:显示已检查项目列表

Posted

技术标签:

【中文标题】PyQt5 可检查组合框:显示已检查项目列表【英文标题】:PyQt5 checkable combo box: display list of checked items 【发布时间】:2020-05-03 04:13:13 【问题描述】:

基于https://***.com/a/22775990/7894940checkable combo box实现,我想更进一步,能够在主QComboBox标签上直接显示选中项的列表,即当显示文本时QComboBox 没有“展开”。

到目前为止,我可以打印检查项目的列表,但我不知道如何用前者更改主 QComboBox 标签文本:

from PyQt5.QtWidgets import QApplication, QComboBox, QMainWindow, QWidget, QVBoxLayout
from PyQt5.QtGui import QStandardItemModel
from PyQt5.QtCore import Qt
import sys


class CheckableComboBox(QComboBox):
    def __init__(self):
        super(CheckableComboBox, self).__init__()
        self.view().pressed.connect(self.handle_item_pressed)
        self.setModel(QStandardItemModel(self))

    def handle_item_pressed(self, index):
        item = self.model().itemFromIndex(index)
        if item.checkState() == Qt.Checked:
            item.setCheckState(Qt.Unchecked)
            # print(item.text() + " was unselected.")
        else:
            item.setCheckState(Qt.Checked)
            # print(item.text() + " was selected.")
        self.check_items()

    def item_checked(self, index):
        item = self.model().item(index, 0)
        return item.checkState() == Qt.Checked

    def check_items(self):
        checkedItems = []
        for i in range(self.count()):
            if self.item_checked(i):
                checkedItems.append(self.model().item(i, 0).text())
        print(checkedItems)


class Dialog_01(QMainWindow):
    def __init__(self):
        super(QMainWindow, self).__init__()
        myQWidget = QWidget()
        myBoxLayout = QVBoxLayout()
        myQWidget.setLayout(myBoxLayout)
        self.setCentralWidget(myQWidget)
        self.ComboBox = CheckableComboBox()
        for i in range(3):
            self.ComboBox.addItem("Combobox Item " + str(i))
            item = self.ComboBox.model().item(i, 0)
            item.setCheckState(Qt.Unchecked)
        myBoxLayout.addWidget(self.ComboBox)

if __name__ == '__main__':
    app = QApplication(sys.argv)
    dialog_1 = Dialog_01()
    dialog_1.show()
    dialog_1.resize(480, 320)
    sys.exit(app.exec_())

一张解释我想要什么的图片:

【问题讨论】:

“到目前为止,我可以打印检查项目的列表,但我不知道如何用前者更改主 QComboBox 标签文本”是什么意思?当您从下拉菜单中选中该框时,列表顺序是否应该在外观上发生变化?如果更容易举例,您可以绘制一张图片并通过更新将其添加到您的问题中。 感谢您的反馈:我上传了一张说明性图片。我希望它现在更清楚了:) @rubebop 我发现将另一个问题标记为正确是矛盾的,也许我误解了你的问题。据我了解,QComboBox 标签将显示选中项目的文本,这就是我的解决方案所做的与其他解决方案不同的是,每次选择或取消选择一个项目时都会创建一个新项目,假设你有 1000 有时然后你将有许多不可检查的选项而不是少数可检查的选项。 @ZF007 请不要添加不必要的标签,这里python3没有相关性 @eyllanesc 复制/粘贴错误问题。它现在已修复。我确实添加了 py3.x,因为打印语句构造显示了它的 py3.x。没什么大不了的。 【参考方案1】:

你可以重写paintEvent方法:

class CheckableComboBox(QComboBox):
    def __init__(self):
        super(CheckableComboBox, self).__init__()
        self.view().pressed.connect(self.handle_item_pressed)
        self.setModel(QStandardItemModel(self))

    def handle_item_pressed(self, index):
        item = self.model().itemFromIndex(index)
        if item.checkState() == Qt.Checked:
            item.setCheckState(Qt.Unchecked)
        else:
            item.setCheckState(Qt.Checked)

    def item_checked(self, index):
        item = self.model().item(index, 0)
        return item.checkState() == Qt.Checked

    def check_items(self):
        checkedItems = []
        for i in range(self.count()):
            if self.item_checked(i):
                checkedItems.append(self.model().item(i, 0).text())
        return checkedItems

    def paintEvent(self, event):
        painter = QStylePainter(self)
        painter.setPen(self.palette().color(QPalette.Text))
        opt = QStyleOptionComboBox()
        self.initStyleOption(opt)
        opt.currentText = ",".join(self.check_items())
        painter.drawComplexControl(QStyle.CC_ComboBox, opt)
        painter.drawControl(QStyle.CE_ComboBoxLabel, opt)

【讨论】:

在我正在实施的程序中,此解决方案存在一个大问题。每次运行 check_items 函数时,我都会记录 checkItems 列表。使用您的解决方案,该功能已经通过将光标放在 CheckableComboBox 上自动运行,这会创建不需要的日志。如果这不是由 QPushButton 上的“按下”操作通过 handle_item_pressed 函数触发的,我不想运行 check_items。【参考方案2】:

只需对原始代码稍作改动,即可轻松实现以下代码。当您打开组合框选项卡时,您将看到所有项目标签都更新为当前选定的选项。

例子:

    如果选择了所有选项,您将获得以下信息:

“组合框项目 0 - 选定项目:0, 1, 2”

    如果例如选项 0 被删除:

组合框项目 2 - 选定项目:1、2

代码:

    def check_items(self):
        checkedItems = []
        for i in range(self.count()):
            if self.item_checked(i):
                checkedItems.append(i)

        self.update_labels(checkedItems)

    def update_labels(self, item_list):

        n = ''
        count = 0
        for i in item_list:
            if count == 0:
                n += ' %s' % i
            else:
                n += ', %s' % i

            count += 1

#        print('n : "%s".' % n)

        for i in range(self.count()):

            text_label = self.model().item(i, 0).text()

#            print('Current (index %s) text_label "%s"' % (i, text_label))
#            sys.stdout.flush()

            if text_label.find('-') >= 0:
                text_label = text_label.split('-')[0]

            item_new_text_label = text_label + ' - selected item(s): ' + n

#            self.model().item(i).setText(item_new_text_label)
             self.setItemText(i, item_new_text_label)  # copy/paste error corrected.

#        print(item_list)
        sys.stdout.flush()

【讨论】:

我相信 "text_label.rstrip(' %s' % i)" 可以被删除,因为输出未使用;) 您是对的,可以毫无问题地将其取出,或者使用带有解释说明的#。它可以帮助您更快地获得“CopyEditor”徽章;-) 感谢您的提示:我确实删除了该行 ^^

以上是关于PyQt5 可检查组合框:显示已检查项目列表的主要内容,如果未能解决你的问题,请参考以下文章

将可检查组合框的文本显示到 QTableWidget

尝试在 mousepress 事件中使用来自另一个组合框的值加载自定义组合框

ComboBox 下拉列表隐藏

带有用户表单上的建议列表的可搜索组合框

在MFC中:静态文本类;按纽类;编辑框类;滚动条类;列表框类;组合框类等所对应的类名以及其功能

如何在组合框中获取已选中的单选按钮?