Source code for edifice.base_components.check_box
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 QtCore, QtWidgets
else:
from PySide6 import QtCore, QtWidgets
class EdCheckBox(QtWidgets.QCheckBox):
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 button checked state to
be set directly by the user.
We always want the checked state to be set by the checked prop.
The user can request a state change, but only the checked prop can
set the state.
"""
pass
[docs]
class CheckBox(QtWidgetElement[EdCheckBox]):
"""Checkbox widget.
A CheckBox allows the user to select a boolean.
.. highlights::
- Underlying Qt Widget
`QCheckBox <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QCheckBox.html>`_
.. rubric:: Props
All **props** from :class:`QtWidgetElement`, plus:
Args:
checked:
Whether or not the CheckBox is checked.
text:
Text for the label of the CheckBox.
on_change:
Event handler for when the checked value changes, but
only when the user checks or unchecks, not when the checked prop
changes.
The :code:`checked` prop determines the check-state of the widget.
When the user toggles the check state, the :code:`on_change` callback is called
with the requested new check state.
.. rubric:: Usage
.. figure:: /image/checkbox_dropdown.png
:width: 300
CheckBox on the left.
.. code-block:: python
checked, checked_set = ed.use_state(False)
CheckBox(
checked=checked,
text="Check me",
on_change=checked_set,
)
"""
[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 = EdCheckBox(str(self.props.text))
size = self.underlying.font().pointSize()
self._set_size(size * len(self.props.text), size)
self.underlying.setObjectName(str(id(self)))
self.underlying.clicked.connect(self._on_clicked)
def _on_clicked(self, checked):
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(EdCheckBox, self.underlying)
widget.blockSignals(True)
check_state = QtCore.Qt.CheckState.Checked if checked else QtCore.Qt.CheckState.Unchecked
widget.setCheckState(check_state)
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(EdCheckBox, self.underlying)
commands = super()._qt_update_commands_super(widget_trees, newprops, self.underlying)
if "text" in newprops:
commands.append(CommandType(widget.setText, str(newprops.text)))
if "checked" in newprops:
commands.append(CommandType(self._set_checked, newprops.checked))
return commands