使用PyQ的组合框中的复选框

2024-05-12 21:35:21 发布

您现在位置:Python中文网/ 问答频道 /正文

我需要实现一个包含复选框的下拉列表,就像组合框中的条目是复选框一样。但是QComboBox不接受QCheckBox作为其成员,我找不到任何替代解决方案。Q^ wiki上的C++中的i ^ a1},但不知道如何将其移植到Python。


Tags: 列表a1wiki条目成员解决方案复选框qcombobox
3条回答

使用组合框项模型,因为项支持复选框 您只需将项目标记为可由用户选中,并设置初始选中状态以使复选框出现(仅当存在有效状态时才显示)

item.setFlags(QtCore.Qt.ItemIsUserCheckable | QtCore.Qt.ItemIsEnabled)
item.setCheckState(QtCore.Qt.Unchecked)  # causes checkBox to show

下面是一个最小的子类示例:

from PyQt5 import QtGui, QtCore, QtWidgets
import sys, os

# subclass
class CheckableComboBox(QtWidgets.QComboBox):
    # once there is a checkState set, it is rendered
    # here we assume default Unchecked
    def addItem(self, item):
        super(CheckableComboBox, self).addItem(item)
        item = self.model().item(self.count()-1,0)
        item.setFlags(QtCore.Qt.ItemIsUserCheckable | QtCore.Qt.ItemIsEnabled)
        item.setCheckState(QtCore.Qt.Unchecked)

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

# the basic main()
app = QtWidgets.QApplication(sys.argv)
dialog = QtWidgets.QMainWindow()
mainWidget = QtWidgets.QWidget()
dialog.setCentralWidget(mainWidget)
ComboBox = CheckableComboBox(mainWidget)
for i in range(6):
    ComboBox.addItem("Combobox Item " + str(i))

dialog.show()
sys.exit(app.exec_())

我在How do I create a tree view (with checkbox) inside a combo box - PyQt回答了一个类似的问题,但无论如何,为了完整起见,我将您粘贴在这里:

您应该创建一个模型,在data和SetData方法中支持Qt.CheckStateRole,在flags方法中支持标记Qt.ItemIsUserCheckable。

我在这里粘贴一个我在一个项目中使用的示例,这是一个QSortFilterProxyModel泛型实现,可以在任何模型中使用,但是您可以在模型实现中使用相同的思想,显然我在这个子类中使用的内部结构不是PyQt中直接使用的,而是附加到我的内部实现(self.booleanSet和self.readOnlySet)。

def flags(self, index):
    if not index.isValid():
        return Qt.ItemIsEnabled

    if index.column() in self.booleanSet:
        return Qt.ItemIsUserCheckable | Qt.ItemIsSelectable | Qt.ItemIsEnabled
    elif index.column() in self.readOnlySet:
        return Qt.ItemIsSelectable | Qt.ItemIsEnabled
    else:
        return QSortFilterProxyModel.flags(self, index)

def data(self, index, role):
    if not index.isValid():
        return QVariant()

    if index.column() in self.booleanSet and role in (Qt.CheckStateRole, Qt.DisplayRole):
        if role == Qt.CheckStateRole:
            value = QVariant(Qt.Checked) if index.data(Qt.EditRole).toBool() else QVariant(Qt.Unchecked)
            return value
        else: #if role == Qt.DisplayRole:
            return QVariant()
    else:
        return QSortFilterProxyModel.data(self, index, role)

def setData(self, index, data, role):
    if not index.isValid():
        return False

    if index.column() in self.booleanSet and role == Qt.CheckStateRole:
        value = QVariant(True) if data.toInt()[0] == Qt.Checked else QVariant(False)
        return QSortFilterProxyModel.setData(self, index, value, Qt.EditRole)
    else:
        return QSortFilterProxyModel.setData(self, index, data, role)

当我需要这样做时,我会想出一个更简单的解决方案(至少不需要将QCombobox子类化)。这对我有效。 也就是说,创建一个带有选中操作的菜单,并将其设置为一个按钮。然后将菜单或操作连接到插槽。

Qt中的代码(还没有使用PyQt,对不起,我希望您可以移植那个代码,对我来说似乎更简单)是这样的:

QMenu *menu = new QMenu;
QAction *Act1 = new QAction("Action 1", menu);
Act1->setCheckable(true);
QAction *Act2 = new QAction("Action 2", menu);
Act2->setCheckable(true);
menu->addAction(Act1);
menu->addAction(Act2);

QPushButton *btn = new QPushButton("Btn");    
btn->setMenu(menu);

希望这有帮助

相关问题 更多 >