You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

108 lines
4.1 KiB

from typing import Any, Callable, Dict, Optional
from ..events import GenericEventArguments
from .mixins.disableable_element import DisableableElement
from .mixins.validation_element import ValidationElement
class Number(ValidationElement, DisableableElement):
LOOPBACK = False
def __init__(self,
label: Optional[str] = None, *,
placeholder: Optional[str] = None,
value: Optional[float] = None,
min: Optional[float] = None, # pylint: disable=redefined-builtin
max: Optional[float] = None, # pylint: disable=redefined-builtin
step: Optional[float] = None,
prefix: Optional[str] = None,
suffix: Optional[str] = None,
format: Optional[str] = None, # pylint: disable=redefined-builtin
on_change: Optional[Callable[..., Any]] = None,
validation: Dict[str, Callable[..., bool]] = {},
) -> None:
"""Number Input
This element is based on Quasar's `QInput <https://quasar.dev/vue-components/input>`_ component.
You can use the `validation` parameter to define a dictionary of validation rules.
The key of the first rule that fails will be displayed as an error message.
:param label: displayed name for the number input
:param placeholder: text to show if no value is entered
:param value: the initial value of the field
:param min: the minimum value allowed
:param max: the maximum value allowed
:param step: the step size for the stepper buttons
:param prefix: a prefix to prepend to the displayed value
:param suffix: a suffix to append to the displayed value
:param format: a string like "%.2f" to format the displayed value
:param on_change: callback to execute when the value changes
:param validation: dictionary of validation rules, e.g. ``{'Too large!': lambda value: value < 3}``
"""
self.format = format
super().__init__(tag='q-input', value=value, on_value_change=on_change, validation=validation)
self._props['type'] = 'number'
if label is not None:
self._props['label'] = label
if placeholder is not None:
self._props['placeholder'] = placeholder
if min is not None:
self._props['min'] = min
if max is not None:
self._props['max'] = max
if step is not None:
self._props['step'] = step
if prefix is not None:
self._props['prefix'] = prefix
if suffix is not None:
self._props['suffix'] = suffix
self.on('blur', self.sanitize, [])
@property
def min(self) -> float:
"""The minimum value allowed."""
return self._props.get('min', -float('inf'))
@min.setter
def min(self, value: float) -> None:
self._props['min'] = value
self.sanitize()
@property
def max(self) -> float:
"""The maximum value allowed."""
return self._props.get('max', float('inf'))
@max.setter
def max(self, value: float) -> None:
self._props['max'] = value
self.sanitize()
@property
def out_of_limits(self) -> bool:
"""Whether the current value is out of the allowed limits."""
return not self.min <= self.value <= self.max
def sanitize(self) -> None:
value = float(self.value or 0)
value = max(value, self.min)
value = min(value, self.max)
self.set_value(float(self.format % value) if self.format else value)
def _event_args_to_value(self, e: GenericEventArguments) -> Any:
if not e.args:
return None
return float(e.args)
def _value_to_model_value(self, value: Any) -> Any:
if value is None:
return None
if self.format is None:
return str(value)
if value == '':
return 0
return self.format % float(value)
def _value_to_event_value(self, value: Any) -> Any:
return float(value) if value else 0