From 894a8571ee1a3bbbdc16bb6a64192819779ba7b3 Mon Sep 17 00:00:00 2001 From: Fabian Dill Date: Fri, 10 Jan 2025 20:21:02 +0100 Subject: [PATCH] kvui: add autocompleting new hint text input (#3535) Co-authored-by: black-sliver <59490463+black-sliver@users.noreply.github.com> Co-authored-by: Silvris <58583688+Silvris@users.noreply.github.com> --- data/client.kv | 5 ++++ kvui.py | 65 +++++++++++++++++++++++++++++++++++++++++++++++--- 2 files changed, 67 insertions(+), 3 deletions(-) diff --git a/data/client.kv b/data/client.kv index 3455f2a23657..f0f31769e411 100644 --- a/data/client.kv +++ b/data/client.kv @@ -147,3 +147,8 @@ rectangle: self.x-2, self.y-2, self.width+4, self.height+4 : pos_hint: {'center_y': 0.5, 'center_x': 0.5} + + size_hint_y: None + height: dp(30) + multiline: False + write_tab: False diff --git a/kvui.py b/kvui.py index b2ab004e274a..f47e45b93c07 100644 --- a/kvui.py +++ b/kvui.py @@ -40,7 +40,7 @@ from kivy.base import ExceptionHandler, ExceptionManager from kivy.clock import Clock from kivy.factory import Factory -from kivy.properties import BooleanProperty, ObjectProperty +from kivy.properties import BooleanProperty, ObjectProperty, NumericProperty from kivy.metrics import dp from kivy.effects.scroll import ScrollEffect from kivy.uix.widget import Widget @@ -64,6 +64,7 @@ from kivy.uix.recycleview.layout import LayoutSelectionBehavior from kivy.animation import Animation from kivy.uix.popup import Popup +from kivy.uix.dropdown import DropDown from kivy.uix.image import AsyncImage fade_in_animation = Animation(opacity=0, duration=0) + Animation(opacity=1, duration=0.25) @@ -305,6 +306,50 @@ def apply_selection(self, rv, index, is_selected): """ Respond to the selection of items in the view. """ self.selected = is_selected + +class AutocompleteHintInput(TextInput): + min_chars = NumericProperty(3) + + def __init__(self, **kwargs): + super().__init__(**kwargs) + + self.dropdown = DropDown() + self.dropdown.bind(on_select=lambda instance, x: setattr(self, 'text', x)) + self.bind(on_text_validate=self.on_message) + + def on_message(self, instance): + App.get_running_app().commandprocessor("!hint "+instance.text) + + def on_text(self, instance, value): + if len(value) >= self.min_chars: + self.dropdown.clear_widgets() + ctx: context_type = App.get_running_app().ctx + if not ctx.game: + return + item_names = ctx.item_names._game_store[ctx.game].values() + + def on_press(button: Button): + split_text = MarkupLabel(text=button.text).markup + return self.dropdown.select("".join(text_frag for text_frag in split_text + if not text_frag.startswith("["))) + lowered = value.lower() + for item_name in item_names: + try: + index = item_name.lower().index(lowered) + except ValueError: + pass # substring not found + else: + text = escape_markup(item_name) + text = text[:index] + "[b]" + text[index:index+len(value)]+"[/b]"+text[index+len(value):] + btn = Button(text=text, size_hint_y=None, height=dp(30), markup=True) + btn.bind(on_release=on_press) + self.dropdown.add_widget(btn) + if not self.dropdown.attach_to: + self.dropdown.open(self) + else: + self.dropdown.dismiss() + + class HintLabel(RecycleDataViewBehavior, BoxLayout): selected = BooleanProperty(False) striped = BooleanProperty(False) @@ -570,8 +615,10 @@ def connect_bar_validate(sender): # show Archipelago tab if other logging is present self.tabs.add_widget(panel) - hint_panel = self.add_client_tab("Hints", HintLog(self.json_to_kivy_parser)) + hint_panel = self.add_client_tab("Hints", HintLayout()) + self.hint_log = HintLog(self.json_to_kivy_parser) self.log_panels["Hints"] = hint_panel.content + hint_panel.content.add_widget(self.hint_log) if len(self.logging_pairs) == 1: self.tabs.default_tab_text = "Archipelago" @@ -698,7 +745,7 @@ def set_new_energy_link_value(self): def update_hints(self): hints = self.ctx.stored_data.get(f"_read_hints_{self.ctx.team}_{self.ctx.slot}", []) - self.log_panels["Hints"].refresh_hints(hints) + self.hint_log.refresh_hints(hints) # default F1 keybind, opens a settings menu, that seems to break the layout engine once closed def open_settings(self, *largs): @@ -753,6 +800,17 @@ def fix_heights(self): element.height = element.texture_size[1] +class HintLayout(BoxLayout): + orientation = "vertical" + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + boxlayout = BoxLayout(orientation="horizontal", size_hint_y=None, height=dp(30)) + boxlayout.add_widget(Label(text="New Hint:", size_hint_x=None, size_hint_y=None, height=dp(30))) + boxlayout.add_widget(AutocompleteHintInput()) + self.add_widget(boxlayout) + + status_names: typing.Dict[HintStatus, str] = { HintStatus.HINT_FOUND: "Found", HintStatus.HINT_UNSPECIFIED: "Unspecified", @@ -769,6 +827,7 @@ def fix_heights(self): } + class HintLog(RecycleView): header = { "receiving": {"text": "[u]Receiving Player[/u]"},