From 0f989bed738c280832f5540bce09ca565c57bb4b Mon Sep 17 00:00:00 2001 From: luoxiaozero <48741584+luoxiaozero@users.noreply.github.com> Date: Wed, 28 Aug 2024 00:05:52 +0800 Subject: [PATCH] Feat/combobox aria (#244) * pref: Combobox blur * feat: Combobox input navigation --- .../use_active_descendant.rs | 7 ++ .../active_descendant/use_option_walker.rs | 20 ++++ thaw/src/combobox/combobox.rs | 97 +++++++++---------- thaw/src/combobox/docs/mod.md | 2 +- 4 files changed, 75 insertions(+), 51 deletions(-) diff --git a/thaw/src/_aria/active_descendant/use_active_descendant.rs b/thaw/src/_aria/active_descendant/use_active_descendant.rs index df8c325..8ac308d 100644 --- a/thaw/src/_aria/active_descendant/use_active_descendant.rs +++ b/thaw/src/_aria/active_descendant/use_active_descendant.rs @@ -96,4 +96,11 @@ impl ActiveDescendantController { None } } + + pub fn find(&self, predicate: impl Fn(String) -> bool) -> Option { + let target = self.option_walker.find(predicate)?; + let id = target.id(); + self.focus_active_descendant(target); + Some(id) + } } diff --git a/thaw/src/_aria/active_descendant/use_option_walker.rs b/thaw/src/_aria/active_descendant/use_option_walker.rs index 7ea1123..48de274 100644 --- a/thaw/src/_aria/active_descendant/use_option_walker.rs +++ b/thaw/src/_aria/active_descendant/use_option_walker.rs @@ -87,4 +87,24 @@ impl OptionWalker { tree_walker.previous_node().unwrap_throw()?.dyn_into().ok() }) } + + pub fn find(&self, predicate: impl Fn(String) -> bool) -> Option { + self.0.with_value(|tree_walker| { + let Some((tree_walker, _)) = tree_walker.as_ref() else { + return None; + }; + tree_walker.set_current_node(&tree_walker.root()); + let mut current: Option = + tree_walker.first_child().unwrap_throw()?.dyn_into().ok(); + + while let Some(cur) = current.as_ref() { + if predicate(cur.id()) { + break; + } + current = tree_walker.next_node().unwrap_throw()?.dyn_into().ok(); + } + + current + }) + } } diff --git a/thaw/src/combobox/combobox.rs b/thaw/src/combobox/combobox.rs index 3d174b0..9015667 100644 --- a/thaw/src/combobox/combobox.rs +++ b/thaw/src/combobox/combobox.rs @@ -68,56 +68,6 @@ pub fn Combobox( }); } - #[cfg(any(feature = "csr", feature = "hydrate"))] - { - let handle = window_event_listener(ev::click, move |ev| { - use leptos::wasm_bindgen::__rt::IntoJsResult; - if !is_show_listbox.get_untracked() { - return; - } - let el = ev.target(); - let mut el: Option = - el.into_js_result().map_or(None, |el| Some(el.into())); - let body = document().body().unwrap(); - if let Some(listbox_el) = listbox_ref.get_untracked() { - if let Some(trigger_el) = trigger_ref.get_untracked() { - while let Some(current_el) = el { - if current_el == *body { - break; - }; - if current_el == **listbox_el { - return; - } - if current_el == **trigger_el { - return; - } - el = current_el.parent_element(); - } - } - } - is_show_listbox.set(false); - }); - on_cleanup(move || handle.remove()); - } - - let on_input = move |ev| { - let input_value = event_target_value(&ev); - if selected_options.with_untracked(|options| match options { - VecModelWithValue::T(v) => v != &input_value, - VecModelWithValue::Option(v) => { - if let Some(v) = v.as_ref() { - v != &input_value - } else { - false - } - } - VecModelWithValue::Vec(_) => false, - }) { - selected_options.set(vec![]); - } - value.set(input_value); - }; - let multiselect = selected_options.is_vec(); let combobox_injection = ComboboxInjection { value, @@ -130,6 +80,51 @@ pub fn Combobox( let (set_listbox, active_descendant_controller) = use_active_descendant(move |el| el.class_list().contains("thaw-combobox-option")); + let on_input = { + let active_descendant_controller = active_descendant_controller.clone(); + move |ev| { + let input_value = event_target_value(&ev); + if selected_options.with_untracked(|options| match options { + VecModelWithValue::T(v) => v != &input_value, + VecModelWithValue::Option(v) => { + if let Some(v) = v.as_ref() { + v != &input_value + } else { + false + } + } + VecModelWithValue::Vec(_) => false, + }) { + selected_options.set(vec![]); + } + value.set(input_value); + let Some(value) = value.with_untracked(|value| { + let value = value.trim().to_ascii_lowercase(); + if value.is_empty() { + None + } else { + Some(value) + } + }) else { + active_descendant_controller.blur(); + return; + }; + if active_descendant_controller + .find(|id| { + options.with_value(|options| { + let Some((_, text, _)) = options.get(&id) else { + return false; + }; + text.to_ascii_lowercase().contains(&value) + }) + }) + .is_none() + { + active_descendant_controller.blur(); + } + } + }; + let on_blur = { let active_descendant_controller = active_descendant_controller.clone(); move |_| { @@ -148,6 +143,7 @@ pub fn Combobox( }); active_descendant_controller.blur(); validate.run(Some(ComboboxRuleTrigger::Blur)); + is_show_listbox.set(false); } }; @@ -209,6 +205,7 @@ pub fn Combobox( .unwrap_or_default() } node_ref=clear_icon_ref + on:mousedown=|e| e.prevent_default() > - + }