Проверяемое поле со списком PyQt5: отображать список отмеченных элементов - PullRequest
1 голос
/ 16 января 2020

Основываясь на { ссылка } проверяемой реализации поля со списком, я хочу go сделать еще один шаг и иметь возможность отображать список отмеченных элементов непосредственно на главной метке 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_())

Картинка, объясняющая, что я хочу: Diagram

Ответы [ 2 ]

2 голосов
/ 16 января 2020

С небольшими изменениями в исходном коде вы можете легко реализовать приведенный ниже код. Когда вы откроете вкладку comboBox, вы увидите, что все метки элементов обновлены до текущих выбранных параметров.

Пример:

  1. Если выбраны все опции, вы получите следующее:

"Combobox Item 0 - selected item (s): 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()
2 голосов
/ 16 января 2020

Вы можете переопределить метод 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)
...