Source code for arcade.gui.widgets.dropdown

from copy import deepcopy

from pyglet.event import EVENT_HANDLED

import arcade
from arcade import uicolor
from arcade.gui import UIEvent, UIMousePressEvent
from arcade.gui.events import UIControllerButtonPressEvent, UIOnChangeEvent, UIOnClickEvent
from arcade.gui.experimental.focus import UIFocusMixin
from arcade.gui.ui_manager import UIManager
from arcade.gui.widgets import UILayout, UIWidget
from arcade.gui.widgets.buttons import UIFlatButton
from arcade.gui.widgets.layout import UIBoxLayout


class _UIDropdownOverlay(UIFocusMixin, UIBoxLayout):
    """Represents the dropdown options overlay.

    Currently only handles closing the overlay when clicked outside of the options.
    """

    # TODO move also options logic to this class

    def show(self, manager: UIManager):
        manager.add(self, layer=UIManager.OVERLAY_LAYER)

    def hide(self):
        """Hide the overlay."""
        self.set_focus(None)
        if self.parent:
            self.parent.remove(self)

    def on_event(self, event: UIEvent) -> bool | None:
        if isinstance(event, UIMousePressEvent):
            # Click outside of dropdown options
            if not self.rect.point_in_rect((event.x, event.y)):
                self.hide()
                return EVENT_HANDLED

        if isinstance(event, UIControllerButtonPressEvent):
            # TODO find a better and more generic way to handle controller events for this
            if event.button == "b":
                self.hide()
                return EVENT_HANDLED

        return super().on_event(event)


[docs] class UIDropdown(UILayout): """A dropdown layout. When clicked displays a list of options provided. Triggers an event when an option is clicked, the event can be read by .. code:: py dropdown = Dropdown() @dropdown.event() def on_change(event: UIOnChangeEvent): print(event.old_value, event.new_value) Args: x: x coordinate of bottom left y: y coordinate of bottom left width: Width of each of the option. height: Height of each of the option. default: The default value shown. options: The options displayed when the layout is clicked. primary_style: The style of the primary button. dropdown_style: The style of the buttons in the dropdown. active_style: The style of the dropdown button, which represents the active option. """ DIVIDER = None DEFAULT_BUTTON_STYLE = { "normal": UIFlatButton.UIStyle( font_color=uicolor.GREEN_NEPHRITIS, ), "hover": UIFlatButton.UIStyle( font_color=uicolor.WHITE, bg=uicolor.DARK_BLUE_WET_ASPHALT, border=uicolor.GRAY_CONCRETE, ), "press": UIFlatButton.UIStyle( font_color=uicolor.DARK_BLUE_MIDNIGHT_BLUE, bg=uicolor.WHITE_CLOUDS, border=uicolor.GRAY_CONCRETE, ), "disabled": UIFlatButton.UIStyle( font_color=uicolor.WHITE_SILVER, bg=uicolor.GRAY_ASBESTOS, ), } DEFAULT_DROPDOWN_STYLE = { "normal": UIFlatButton.UIStyle(), "hover": UIFlatButton.UIStyle( font_color=uicolor.WHITE, bg=uicolor.DARK_BLUE_WET_ASPHALT, border=uicolor.GRAY_CONCRETE, ), "press": UIFlatButton.UIStyle( font_color=uicolor.DARK_BLUE_MIDNIGHT_BLUE, bg=uicolor.WHITE_CLOUDS, border=uicolor.GRAY_CONCRETE, ), "disabled": UIFlatButton.UIStyle( font_color=uicolor.WHITE_SILVER, bg=uicolor.GRAY_ASBESTOS, ), } def __init__( self, *, x: float = 0, y: float = 0, width: float = 150, height: float = 30, default: str | None = None, options: list[str | None] | None = None, primary_style=None, dropdown_style=None, active_style=None, **kwargs, ): if primary_style is None: primary_style = self.DEFAULT_BUTTON_STYLE if dropdown_style is None: dropdown_style = self.DEFAULT_DROPDOWN_STYLE if active_style is None: active_style = self.DEFAULT_BUTTON_STYLE # TODO handle if default value not in options or options empty if options is None: options = [] self._options = options self._value = default super().__init__(x=x, y=y, width=width, height=height, **kwargs) self._default_style = deepcopy(primary_style) self._dropdown_style = deepcopy(dropdown_style) self._active_style = deepcopy(active_style) # Setup button showing value self._default_button = UIFlatButton( text=self._value or "", width=self.width, height=self.height, style=self._default_style ) self._default_button.on_click = self._on_button_click # type: ignore self._overlay = _UIDropdownOverlay() self._update_options() # add children after super class setup self.add(self._default_button) self.register_event_type("on_change") @property def value(self) -> str | None: """Current selected option.""" return self._value @value.setter def value(self, value: str | None): """Change the current selected option to a new option.""" old_value = self._value self._value = value self._default_button.text = self._value or "" self._update_options() self.dispatch_event("on_change", UIOnChangeEvent(self, old_value, value)) self.trigger_render() def _update_options(self): # generate options self._overlay.clear() for option in self._options: if option is None: # None = UIDropdown.DIVIDER, required by pyright self._overlay.add( UIWidget(width=self.width, height=2).with_background(color=arcade.color.GRAY) ) continue else: button = self._overlay.add( UIFlatButton( text=option, width=self.width, height=self.height, style=self._active_style if self.value == option else self._dropdown_style, ) ) button.on_click = self._on_option_click self._overlay.detect_focusable_widgets() def _show_overlay(self): manager = self.get_ui_manager() if manager is None: raise Exception("UIDropdown could not find UIManager in its parents.") self._overlay.show(manager) def _on_button_click(self, _: UIOnClickEvent): self._show_overlay() def _on_option_click(self, event: UIOnClickEvent): source: UIFlatButton = event.source self._overlay.hide() self.value = source.text
[docs] def do_layout(self): """Position the overlay, this is not a common thing to do in do_layout, but is required for the dropdown.""" self._default_button.rect = self.rect # resize layout to contain widgets overlay = self._overlay rect = overlay.rect if overlay.size_hint_min is not None: rect = rect.resize(*overlay.size_hint_min) self._overlay.rect = rect.align_top(self.bottom - 2).align_left(self._default_button.left)
[docs] def on_change(self, event: UIOnChangeEvent): """To be implemented by the user, triggered when the current selected value is changed to a different option. """ pass