继承QGroupBox,使其可以成为QButtonGroup的成员

问题描述 投票:0回答:1

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

对于某些UI来说,能够有一些带有专有复选框的QGroupBox真是太好了。也就是说,您选中了一个,其他QGroupBoxes被自动取消选中。

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

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_())

一旦您尝试将超级组添加到按钮组,此代码就会失败。 PyQt5显式不支持多重继承。但是有一些examples out in the wild, like from this blog

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

python pyqt pyqt5 qbuttongroup
1个回答
4
投票

不必创建从QGroupBox和QAbstractButton继承的类(此外,在pyqt或Qt / C ++中是不可能的)。解决方案是创建一个QObject,当检查任何QGroupBox时,它会处理另一个QGroupBox的状态,我为Qt / C ++的old answer实现了此功能,因此此答案仅是翻译:

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_())
© www.soinside.com 2019 - 2024. All rights reserved.