Связывание нескольких групп радиокнопок вызывает странное поведение

Моя программа генерирует формы на лету на основе данных SQL. Я делаю две радиокнопки и запись QLineEdit рядом с ними. Когда переключатель справа установлен, QLineEdit включен правильно. Проблема возникает из-за того, что переключатели связываются и вызывают между собой эксклюзивные действия. Когда программа запускается, они выглядят так:

Затем, когда я нажимаю первое "Нет", это меняет мои ожидания и включает QLineEdit.

Теперь я хочу нажать "Нет" и для "Серийный номер: RC1". Здесь поведение начинает идти не так. Нажатие кнопки "Нет" снимает выделение со всей строки выше.

Если я попытаюсь снова щелкнуть "Нет" в верхней строке, "Да" во второй строке отменяется.

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

Я генерирую кнопки на лету из вспомогательных функций, которые помещают переключатели в QButtonGroups. Я думал, что этого будет достаточно, чтобы остановить такое поведение, но я ошибался. Я бы хотел, чтобы переключатели в каждой строке не реагировали на действия других переключателей в других строках.

# !/user/bin/env python
import os
import sys
from PyQt5 import uic
from PyQt5.QtWidgets import *
from PyQt5.QtGui import *


class Radio(QDialog):
    def __init__(self, app):
        super(Radio, self).__init__()
        self.bundle_dir = os.path.dirname(__file__)
        gui_path = os.path.join(self.bundle_dir, 'ui', 'radio_bt_test.ui')
        self.ui = uic.loadUi(gui_path, self)
        self.num_of_row = 4
        self.formLayout = self.findChild(QFormLayout, "formLayout")
        self.radio_bt_lineEdit_connections = dict()  # to help link the radio buttons and lineEdit
        self.add_rows()
        self.show()

    def add_rows(self):
        """
        Adds pairs of radio buttons with a lineEdit to each row of the form layout
        :return: 
        """
        for i in range(self.num_of_row):
            lbl = QLabel("Label#" + str(i))
            hbox = QHBoxLayout()
            buttons = self.new_radio_pair()
            entry = self.new_entry("Value if No")
            entry.setEnabled(False)
            self.radio_bt_lineEdit_connections[buttons[-1]] = entry
            # adding connection to dictionary for later event handling
            buttons[-1].toggled.connect(self.radio_bt_changed)
            for button in buttons:
                hbox.addWidget(button)
            hbox.addWidget(entry)
            self.formLayout.addRow(lbl, hbox)

    def new_radio_pair(self, texts=('Yes', 'No')) -> list:
        """
        Makes a pair of radio buttons in a button group for creating data entries in "Part" grouping on the fly
        :param texts: The texts that will go on the two buttons. The more texts that are added to make more radio buttons
        :return: A list with QRadioButtons that are all part of the same QButtonGroup
        """
        group = QButtonGroup()
        buttons = []
        for text in texts:
            bt = QRadioButton(text)
            bt.setFont(QFont('Roboto', 11))
            if text == texts[0]:
                bt.setChecked(True)
            group.addButton(bt)
            buttons.append(bt)

        return buttons

    def radio_bt_changed(self) -> None:
        """
        Helps the anonymous radio buttons link to the anonymous QLineEdits that are made for data fields
        :return: None
        """
        sender = self.sender()
        assert isinstance(sender, QRadioButton)
        lineEdit = self.radio_bt_lineEdit_connections[sender]
        assert isinstance(lineEdit, QLineEdit)
        if sender.isChecked():
            lineEdit.setEnabled(True)
        else:
            lineEdit.setEnabled(False)
            lineEdit.clear()

    def new_entry(self, placeholder_text: str = "") -> QLineEdit:
        """
        Makes a new QLineEdit object for creating data entries in "Part" grouping on the fly
        :return: A new QLineEdit with appropriate font and size policy
        """
        entry = QLineEdit()
        entry.setSizePolicy(QSizePolicy.Expanding, QSizePolicy.Fixed)
        entry.setFont(QFont('Roboto', 11))
        entry.setStyleSheet("background-color: rgb(239, 241, 243);")
        # with style sheets, past anything in here between the css tags
        entry.setMaxLength(15)
        entry.setPlaceholderText(placeholder_text)
        return entry


def main():
    app = QApplication(sys.argv)
    radio = Radio(app)
    sys.exit(app.exec())

main()

Может быть, потому, что я объявляю QButtonGroup, а потом забываю о них? Приходит ли сборщик мусора и стирает их, потому что мне не назначена переменная или есть еще одна проблема, которую мне не хватает.

Пользовательский интерфейс был разработан на QtDesigner и представляет собой просто диалоговое окно с макетом формы на нем.

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>Dialog</class>
 <widget class="QDialog" name="Dialog">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>491</width>
    <height>382</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>Dialog</string>
  </property>
  <property name="styleSheet">
   <string notr="true">background-color: rgb(219, 221, 223);</string>
  </property>
  <widget class="QWidget" name="formLayoutWidget">
   <property name="geometry">
    <rect>
     <x>80</x>
     <y>40</y>
     <width>301</width>
     <height>291</height>
    </rect>
   </property>
   <layout class="QFormLayout" name="formLayout"/>
  </widget>
 </widget>
 <resources/>
 <connections/>
</ui>

1 ответ

Решение

Объект, который следует использовать, чтобы сделать кнопки строк исключительными, - это "группа", но это локальная переменная, которая уничтожается, когда метод new_radio_pair завершает выполнение, в результате чего он не ведет себя так, как предполагалось ранее.

Решение состоит в том, чтобы продлить жизненный цикл, и для этого есть несколько вариантов, таких как сделать его атрибутом класса, добавить его в контейнер или класс, который имеет более длительный жизненный цикл, или, в случае QObjects, передать ему другой QObject как родитель (как self) с более длительным жизненным циклом, и это лучшее решение для этого случая:

group = QButtonGroup(self)
Другие вопросы по тегам