Source code for edifice.base_components.radio_button
from __future__ import annotations
import typing as tp
from edifice.qt import QT_VERSION
from .base_components import CommandType, Element, QtWidgetElement, _ensure_future, _WidgetTree
if QT_VERSION == "PyQt6" and not tp.TYPE_CHECKING:
from PyQt6 import QtWidgets
else:
from PySide6 import QtWidgets
class EdRadioButton(QtWidgets.QRadioButton):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
def nextCheckState(self):
"""
https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QAbstractButton.html#PySide6.QtWidgets.QAbstractButton.nextCheckState
We override this because we never want the radio button checked state to
be set directly by the user, or by the Qt radio button group mechanism.
We always want the checked state to be set by the checked prop.
"""
pass
[docs]
class RadioButton(QtWidgetElement[EdRadioButton]):
"""Radio buttons.
.. highlights::
- Underlying Qt Widget
`QRadioButton <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QRadioButton.html>`_
Radio buttons are used to specify a single choice out of many.
.. rubric:: Props
All **props** from :class:`QtWidgetElement` plus:
Args:
checked:
Whether or not the RadioButton is checked.
text:
Text for the label of the RadioButton.
on_change:
Event handler for when the checked value changes, but
only when the user checks or unchecks, not when the checked prop
changes.
.. rubric:: Usage
.. figure:: /image/radio_button.png
:width: 300
Three RadioButtons
Because of the declarative nature of Edifice, we can ignore all of the
Qt mechanisms for radio button “groups” and “exclusivity.”
Just declare each radio button :code:`checked` prop to depend on the state.
.. code-block:: python
:caption: Exclusive RadioButtons with different parents
value, value_set = use_state(cast(Literal["op1", "op2"], "op1"))
with ed.VBoxView():
with ed.VBoxView():
ed.RadioButton(
checked = value == "op1",
on_change = lambda checked: value_set("op1") if checked else None,
text = "Option 1",
style = {} if value == "op1" else { "color": "grey" },
)
with ed.VBoxView():
ed.RadioButton(
checked = value == "op2",
on_change = lambda checked: value_set("op2") if checked else None,
text = "Option 2",
style = {} if value == "op1" else { "color": "grey" },
)
"""
[docs]
def __init__(
self,
checked: bool = False,
text: str = "",
on_change: tp.Callable[[bool], None | tp.Awaitable[None]] | None = None,
**kwargs,
):
super().__init__(**kwargs)
self._register_props(
{
"checked": checked,
"text": text,
"on_change": on_change,
},
)
self._register_props(kwargs)
def _initialize(self):
self.underlying = EdRadioButton(str(self.props.text))
size = self.underlying.font().pointSize()
self._set_size(size * len(self.props.text), size)
self.underlying.setObjectName(str(id(self)))
# We setAutoExclusive(False) because we don't want to use the Qt
# radio button group mechanism. We handle the exclusivity.
self.underlying.setAutoExclusive(False)
self.underlying.clicked.connect(self._on_clicked)
def _on_clicked(self, checked: bool):
if self.props.on_change is not None:
return _ensure_future(self.props.on_change)(not checked)
def _set_checked(self, checked: bool):
widget = tp.cast(EdRadioButton, self.underlying)
widget.blockSignals(True)
widget.setChecked(checked)
widget.blockSignals(False)
def _qt_update_commands(
self,
widget_trees: dict[Element, _WidgetTree],
newprops,
):
if self.underlying is None:
self._initialize()
assert self.underlying is not None
widget = tp.cast(EdRadioButton, self.underlying)
commands = super()._qt_update_commands_super(widget_trees, newprops, self.underlying)
if "checked" in newprops:
commands.append(CommandType(self._set_checked, newprops.checked))
if "text" in newprops:
commands.append(CommandType(widget.setText, str(newprops.text)))
return commands