1

QButtonGroups 可以有复选框。但是您不能将它们添加到 QButtonGroup,因为它们不继承 QAbstractButton。

如果某些 UI 能够拥有一些带有专用复选框的 QGroupBox,那就太好了。也就是说,您选中一个,而其他 QGroupBox 将自动取消选中。

在理想的世界中,我可以这样做:

import sys
from PyQt5.QtCore import Qt
from PyQt5.QtWidgets import (QGroupBox, QWidget, QApplication, 
                             QAbstractButton, QButtonGroup)

class SuperGroup(QGroupBox, QAbstractButton):
    def __init__(self, title, parent=None):
        super(SuperGroup, self).__init__(title, parent)
        self.setCheckable(True)
        self.setChecked(False)

class Example(QWidget):

    def __init__(self):
        super().__init__()

        sg1 = SuperGroup(title = 'Super Group 1', parent = self)
        sg1.resize(200,200)
        sg1.move(20,20)

        sg2 = SuperGroup(title = 'Super Group 2', parent = self)
        sg2.resize(200,200)
        sg2.move(300,20)

        self.bgrp = QButtonGroup()
        self.bgrp.addButton(sg1)
        self.bgrp.addButton(sg2)


        self.setGeometry(300, 300, 650, 500)
        self.setWindowTitle('SuperGroups!')
        self.show()



if __name__ == '__main__':

    app = QApplication(sys.argv)
    ex = Example()
    sys.exit(app.exec_())

只要您尝试将 SuperGroup 添加到按钮组,此代码就会失败。PyQt5 明确不支持多重继承。但是在野外也有一些例子,比如这个博客。

在这个简单的例子中,很容易以编程方式管理点击。但是,当您添加更多组框时,它会变得更加混乱。或者如果你想要一个带有按钮、复选框和分组框的 QButtonGroup 怎么办?啊。

4

1 回答 1

4

没有必要创建一个继承自 QGroupBox 和 QAbstractButton 的类(另外在 pyqt 或 Qt/C++ 中是不可能的)。解决方案是创建一个 QObject 来在检查任何 QGroupBox 时处理另一个 QGroupBox 的状态,并且我为 Qt/C++ 的旧答案实现了这一点,所以这个答案只是一个翻译:

import sys
from PyQt5.QtCore import pyqtSlot, QObject, Qt
from PyQt5.QtWidgets import QGroupBox, QWidget, QApplication, QButtonGroup


class GroupBoxManager(QObject):
    def __init__(self, parent=None):
        super().__init__(parent)
        self._groups = []

    @property
    def groups(self):
        return self._groups

    def add_group(self, group):
        if isinstance(group, QGroupBox):
            group.toggled.connect(self.on_toggled)
            self.groups.append(group)

    @pyqtSlot(bool)
    def on_toggled(self, state):
        group = self.sender()
        if state:
            for g in self.groups:
                if g != group and g.isChecked():
                    g.blockSignals(True)
                    g.setChecked(False)
                    g.blockSignals(False)

        else:
            group.blockSignals(True)
            group.setChecked(False)
            group.blockSignals(False)


class Example(QWidget):
    def __init__(self):
        super().__init__()

        sg1 = QGroupBox(
            title="Super Group 1", parent=self, checkable=True, checked=False
        )
        sg1.resize(200, 200)
        sg1.move(20, 20)

        sg2 = QGroupBox(
            title="Super Group 2", parent=self, checkable=True, checked=False
        )
        sg2.resize(200, 200)
        sg2.move(300, 20)

        self.bgrp = GroupBoxManager()
        self.bgrp.add_group(sg1)
        self.bgrp.add_group(sg2)

        self.setGeometry(300, 300, 650, 500)
        self.setWindowTitle("SuperGroups!")
        self.show()


if __name__ == "__main__":

    app = QApplication(sys.argv)
    ex = Example()
    sys.exit(app.exec_())
于 2020-04-24T19:21:14.883 回答