thaw/thaw/src/tab_list/tab.rs
2024-07-11 20:44:19 +08:00

77 lines
2.7 KiB
Rust

use super::{TabListInjection, TabRegisterData};
use leptos::{html, prelude::*};
use std::ops::Deref;
use thaw_utils::{class_list, mount_style};
#[component]
pub fn Tab(
#[prop(optional, into)] class: MaybeProp<String>,
#[prop(into)] value: String,
children: Children,
) -> impl IntoView {
mount_style("tab", include_str!("./tab.css"));
let tab_ref = NodeRef::<html::Button>::new();
let tab_list = TabListInjection::use_();
let value = StoredValue::new(value);
tab_list.register(TabRegisterData {
value: value.get_value(),
tab_ref,
});
on_cleanup(move || {
value.with_value(|v| tab_list.unregister(v));
});
let selected = Memo::new(move |_| {
tab_list
.selected_value
.with(|selected_value| value.with_value(|value| value == selected_value))
});
let on_select = move |_| {
tab_list.registered_tabs.with_untracked(|registered_tabs| {
if let Some(previous_selected_tab) = tab_list
.selected_value
.with_untracked(|selected_value| registered_tabs.get(selected_value))
{
let tab_el = tab_ref.get_untracked().unwrap();
let selected_tab_rect = tab_el.get_bounding_client_rect();
let previous_selected_tab_rect = previous_selected_tab
.tab_ref
.get_untracked()
.unwrap()
.get_bounding_client_rect();
let offset = previous_selected_tab_rect.x() - selected_tab_rect.x();
let scale = previous_selected_tab_rect.width() / selected_tab_rect.width();
let style = tab_el.deref().style();
let _ =
style.set_property("--thaw-tab__indicator--offset", &format!("{offset:.0}px"));
let _ = style.set_property("--thaw-tab__indicator--scale", &format!("{scale:.2}"));
request_animation_frame(move || {
let _ = style.set_property("--thaw-tab__indicator--offset", "0px");
let _ = style.set_property("--thaw-tab__indicator--scale", "1");
});
}
});
tab_list.on_select(value.get_value());
};
view! {
<button
class=class_list![
"thaw-tab", ("thaw-tab--selected", move || selected.get()), class
]
role="tab"
aria-selected=move || if selected.get() { "true" } else { "false" }
node_ref=tab_ref
on:click=on_select
>
<span class="thaw-tab__content">
{children()}
</span>
</button>
}
}