Source code for edifice.base_components.spin_input

from __future__ import annotations

import typing as tp

from edifice.qt import QT_VERSION

if QT_VERSION == "PyQt6" and not tp.TYPE_CHECKING:
    from PyQt6.QtGui import QValidator
    from PyQt6.QtWidgets import QSpinBox
else:
    from PySide6.QtGui import QValidator
    from PySide6.QtWidgets import QSpinBox

from .base_components import CommandType, Element, QtWidgetElement, _ensure_future, _WidgetTree


class EdSpinBox(QSpinBox):
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        self._textFromValue: tp.Callable[[int], str] | None = None
        self._valueFromText: (
            tp.Callable[[str], int | tp.Literal[QValidator.State.Intermediate, QValidator.State.Invalid]] | None
        ) = None

    def textFromValue(self, val: int) -> str:
        if self._textFromValue is not None:
            return self._textFromValue(val)
        return super().textFromValue(val)

    def valueFromText(self, text: str) -> int:
        if self._valueFromText is not None:
            match self._valueFromText(text):
                case QValidator.State.Intermediate:
                    return 0  # unreachable case
                case QValidator.State.Invalid:
                    return 0  # unreachable case
                case val:
                    return val
        else:
            return super().valueFromText(text)

    def validate(self, input: str, pos: int) -> object:  # noqa: A002
        if self._valueFromText is not None:
            match self._valueFromText(input):
                case QValidator.State.Intermediate:
                    return QValidator.State.Intermediate
                case QValidator.State.Invalid:
                    return QValidator.State.Invalid
                case _:
                    return QValidator.State.Acceptable
        else:
            return super().validate(input, pos)


[docs] class SpinInput(QtWidgetElement[EdSpinBox]): """Widget for a :code:`int` input value with up/down buttons. Allows the user to choose an :code:`int` by clicking the up/down buttons or pressing up/down on the keyboard to increase/decrease the value currently displayed. The user can also type the value in manually. .. highlights:: - Underlying Qt Widget `QSpinBox <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QSpinBox.html>`_ .. rubric:: Props All **props** for :class:`QtWidgetElement` plus: Args: value: Value of the text input. min_value: Minimum value of the text input. max_value: Maximum value of the text input. on_change: Callback for when the value changes. The callback is passed the changed value. value_to_text: Function to convert the value to a text. If not provided, the default text conversion is used. text_to_value: Function to convert the text to a value. If not provided, the default text conversion is used. The function should return one of * :code:`int` value if the text is valid. * :code:`QValidator.State.Intermediate` if the text might be valid with more input. * :code:`QValidator.State.Invalid` if the text is invalid. See `QValidator.State <https://doc.qt.io/qtforpython-6/PySide6/QtGui/QValidator.html#PySide6.QtGui.QValidator.State>`_. (At the time of this writing, the PySide6.QValidator.State Enum cannot be correctly typechecked as a Literal by Pyright for some reason.) single_step: The value step size for the up/down buttons. enable_mouse_scroll: Whether mouse scroll events should be able to change the value. """ # TODO Note that you can set an optional Completer, giving the dropdown for completion.
[docs] def __init__( self, value: int = 0, min_value: int = 0, max_value: int = 100, on_change: tp.Callable[[int], None | tp.Awaitable[None]] | None = None, value_to_text: tp.Callable[[int], str] | None = None, text_to_value: tp.Callable[ [str], int | tp.Literal[QValidator.State.Intermediate, QValidator.State.Invalid], ] | None = None, single_step: int = 1, enable_mouse_scroll: bool = True, **kwargs, ): super().__init__(**kwargs) self._register_props( { "value": value, "min_value": min_value, "max_value": max_value, "on_change": on_change, "value_to_text": value_to_text, "text_to_value": text_to_value, "single_step": single_step, "enable_mouse_scroll": enable_mouse_scroll, }, )
def _initialize(self): self.underlying = EdSpinBox() self.underlying.setObjectName(str(id(self))) self.underlying.valueChanged.connect(self._on_change_handler) def _on_change_handler(self, value: int): if self.props.on_change is not None: _ensure_future(self.props.on_change)(value) def _set_value(self, value: int): assert self.underlying is not None self.underlying.blockSignals(True) self.underlying.setValue(value) self.underlying.blockSignals(False) def _set_min_value(self, value: int): assert self.underlying is not None self.underlying.blockSignals(True) self.underlying.setMinimum(value) self.underlying.blockSignals(False) def _set_max_value(self, value: int): assert self.underlying is not None self.underlying.blockSignals(True) self.underlying.setMaximum(value) self.underlying.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 commands = super()._qt_update_commands_super(widget_trees, newprops, self.underlying) if "value_to_text" in newprops: commands.append(CommandType(setattr, self.underlying, "_textFromValue", newprops.value_to_text)) if "text_to_value" in newprops: commands.append(CommandType(setattr, self.underlying, "_valueFromText", newprops.text_to_value)) if "min_value" in newprops: commands.append(CommandType(self._set_min_value, newprops.min_value)) if "max_value" in newprops: commands.append(CommandType(self._set_max_value, newprops.max_value)) if "value" in newprops: commands.append(CommandType(self._set_value, newprops.value)) if "single_step" in newprops: commands.append(CommandType(self.underlying.setSingleStep, newprops.single_step)) if "enable_mouse_scroll" in newprops: # Doing it this way means that if the user tries to attach an # on_mouse_wheel event handler then that won't work. But I think # that's okay for now. if newprops.enable_mouse_scroll: commands.append(CommandType(self._set_mouse_wheel, self.underlying, None)) else: commands.append(CommandType(self._set_mouse_wheel, self.underlying, lambda e: e.ignore())) return commands