|
@@ -1,5 +1,5 @@
|
|
|
from copy import deepcopy
|
|
|
-from typing import Any, Callable, Dict, List, Optional, Union
|
|
|
+from typing import Any, Callable, Dict, List, Literal, Optional, Union
|
|
|
|
|
|
from ..events import GenericEventArguments
|
|
|
from .choice_element import ChoiceElement
|
|
@@ -14,6 +14,7 @@ class Select(ChoiceElement, DisableableElement, component='select.js'):
|
|
|
value: Any = None,
|
|
|
on_change: Optional[Callable[..., Any]] = None,
|
|
|
with_input: bool = False,
|
|
|
+ new_value_mode: Optional[Literal['add', 'add-unique', 'toggle']] = None,
|
|
|
multiple: bool = False,
|
|
|
clearable: bool = False,
|
|
|
) -> None:
|
|
@@ -24,11 +25,17 @@ class Select(ChoiceElement, DisableableElement, component='select.js'):
|
|
|
The options can be specified as a list of values, or as a dictionary mapping values to labels.
|
|
|
After manipulating the options, call `update()` to update the options in the UI.
|
|
|
|
|
|
+ If `with_input` is True, an input field is shown to filter the options.
|
|
|
+
|
|
|
+ If `new_value_mode` is not None, it implies `with_input=True` and the user can enter new values in the input field.
|
|
|
+ See `Quasar's documentation <https://quasar.dev/vue-components/select#the-new-value-mode-prop>`_ for details.
|
|
|
+
|
|
|
:param options: a list ['value1', ...] or dictionary `{'value1':'label1', ...}` specifying the options
|
|
|
:param label: the label to display above the selection
|
|
|
:param value: the initial value
|
|
|
:param on_change: callback to execute when selection changes
|
|
|
:param with_input: whether to show an input field to filter the options
|
|
|
+ :param new_value_mode: handle new values from user input (default: None, i.e. no new values)
|
|
|
:param multiple: whether to allow multiple selections
|
|
|
:param clearable: whether to add a button to clear the selection
|
|
|
"""
|
|
@@ -41,6 +48,11 @@ class Select(ChoiceElement, DisableableElement, component='select.js'):
|
|
|
super().__init__(options=options, value=value, on_change=on_change)
|
|
|
if label is not None:
|
|
|
self._props['label'] = label
|
|
|
+ if new_value_mode is not None:
|
|
|
+ if isinstance(options, dict) and new_value_mode == 'add':
|
|
|
+ raise ValueError('new_value_mode "add" is not supported for dict options')
|
|
|
+ self._props['new-value-mode'] = new_value_mode
|
|
|
+ with_input = True
|
|
|
if with_input:
|
|
|
self.original_options = deepcopy(options)
|
|
|
self._props['use-input'] = True
|
|
@@ -55,11 +67,21 @@ class Select(ChoiceElement, DisableableElement, component='select.js'):
|
|
|
if self.multiple:
|
|
|
if e.args is None:
|
|
|
return []
|
|
|
- return [self._values[arg['value']] for arg in e.args]
|
|
|
+ else:
|
|
|
+ args = [self._values[arg['value']] if isinstance(arg, dict) else arg for arg in e.args]
|
|
|
+ for arg in e.args:
|
|
|
+ if isinstance(arg, str):
|
|
|
+ self._handle_new_value(arg)
|
|
|
+ return [arg for arg in args if arg in self._values]
|
|
|
else:
|
|
|
if e.args is None:
|
|
|
return None
|
|
|
- return self._values[e.args['value']]
|
|
|
+ else:
|
|
|
+ if isinstance(e.args, str):
|
|
|
+ self._handle_new_value(e.args)
|
|
|
+ return e.args if e.args in self._values else None
|
|
|
+ else:
|
|
|
+ return self._values[e.args['value']]
|
|
|
|
|
|
def _value_to_model_value(self, value: Any) -> Any:
|
|
|
# pylint: disable=no-else-return
|
|
@@ -78,3 +100,28 @@ class Select(ChoiceElement, DisableableElement, component='select.js'):
|
|
|
return {'value': index, 'label': self._labels[index]}
|
|
|
except ValueError:
|
|
|
return None
|
|
|
+
|
|
|
+ def _handle_new_value(self, value: str) -> None:
|
|
|
+ mode = self._props['new-value-mode']
|
|
|
+ if isinstance(self.options, list):
|
|
|
+ if mode == 'add':
|
|
|
+ self.options.append(value)
|
|
|
+ elif mode == 'add-unique':
|
|
|
+ if value not in self.options:
|
|
|
+ self.options.append(value)
|
|
|
+ elif mode == 'toggle':
|
|
|
+ if value in self.options:
|
|
|
+ self.options.remove(value)
|
|
|
+ else:
|
|
|
+ self.options.append(value)
|
|
|
+ # NOTE: self._labels and self._values are updated via self.options since they share the same references
|
|
|
+ else:
|
|
|
+ if mode in 'add-unique':
|
|
|
+ if value not in self.options:
|
|
|
+ self.options[value] = value
|
|
|
+ elif mode == 'toggle':
|
|
|
+ if value in self.options:
|
|
|
+ self.options.pop(value)
|
|
|
+ else:
|
|
|
+ self.options.update({value: value})
|
|
|
+ self._update_values_and_labels()
|