vertigo_forms/select/
select.rs1use std::ops::Not;
2
3use vertigo::{Computed, DomNode, Value, bind, computed_tuple, dom, render::render_list};
4
5pub struct Select<T: Clone + PartialEq + 'static> {
29 pub value: Value<T>,
30 pub options: Computed<Vec<T>>,
31}
32
33impl<T> Select<T>
34where
35 T: Clone + From<String> + PartialEq + ToString + 'static,
36{
37 pub fn into_component(self) -> Self {
38 self
39 }
40
41 pub fn mount(&self) -> DomNode {
42 let Self { value, options } = self;
43 let on_change = bind!(value, |new_value: String| {
44 value.set(new_value.into());
45 });
46
47 let empty = computed_tuple!(value, options).render_value_option(|(value, options)| {
48 options
49 .contains(&value)
50 .not()
51 .then(|| dom! { <option value="" selected="selected" /> })
52 });
53
54 let list = bind!(
55 options,
56 value.render_value(move |value| render_list(
57 &options,
58 |item| item.to_string(),
59 move |item| {
60 let text_item = item.to_string();
61 if item == &value {
62 dom! { <option value={&text_item} selected="selected">{text_item}</option> }
63 } else {
64 dom! { <option value={&text_item}>{text_item}</option> }
65 }
66 }
67 ))
68 );
69
70 dom! {
71 <select {on_change}>
72 {empty}
73 {list}
74 </select>
75 }
76 }
77}