use crate::core::ElementMaybeSignal; use crate::use_event_listener::use_event_listener_with_options; use crate::{use_debounce_fn_with_arg, use_throttle_fn_with_arg_and_options, ThrottleOptions}; use cfg_if::cfg_if; use default_struct_builder::DefaultBuilder; use leptos::ev::scrollend; use leptos::*; use std::rc::Rc; use wasm_bindgen::JsCast; /// Reactive scroll position and state. /// /// ## Demo /// /// [Link to Demo](https://github.com/Synphonyte/leptos-use/tree/main/examples/use_scroll) /// /// ## Usage /// /// ``` /// # use leptos::*; /// # use leptos::ev::resize; /// # use leptos::html::Div; /// # use leptos_use::{use_scroll, UseScrollReturn}; /// # /// # #[component] /// # fn Demo() -> impl IntoView { /// let element = create_node_ref::
(); /// /// let UseScrollReturn { /// x, y, set_x, set_y, is_scrolling, arrived_state, directions, .. /// } = use_scroll(element); /// /// view! { ///
"..."
/// } /// # } /// ``` /// /// ### With Offsets /// /// You can provide offsets when you use [`use_scroll_with_options`]. /// These offsets are thresholds in pixels when a side is considered to have arrived. This is reflected in the return field `arrived_state`. /// /// ``` /// # use leptos::*; /// # use leptos::html::Div; /// # use leptos::ev::resize; /// # use leptos_use::{use_scroll_with_options, UseScrollReturn, UseScrollOptions, ScrollOffset}; /// # /// # #[component] /// # fn Demo() -> impl IntoView { /// # let element = create_node_ref::
(); /// # /// let UseScrollReturn { /// x, /// y, /// set_x, /// set_y, /// is_scrolling, /// arrived_state, /// directions, /// .. /// } = use_scroll_with_options( /// element, /// UseScrollOptions::default().offset(ScrollOffset { /// top: 30.0, /// bottom: 30.0, /// right: 30.0, /// left: 30.0, /// }), /// ); /// # /// # view! { /// #
"..."
/// # } /// # } /// ``` /// /// ### Setting Scroll Position /// /// Set the `x` and `y` values to make the element scroll to that position. /// /// ``` /// # use leptos::*; /// # use leptos::html::Div; /// # use leptos::ev::resize; /// # use leptos_use::{use_scroll, UseScrollReturn}; /// # /// # #[component] /// # fn Demo() -> impl IntoView { /// let element = create_node_ref::
(); /// /// let UseScrollReturn { /// x, y, set_x, set_y, .. /// } = use_scroll(element); /// /// view! { ///
"..."
/// /// /// } /// # } /// ``` /// /// ### Smooth Scrolling /// /// Set `behavior: smooth` to enable smooth scrolling. The `behavior` option defaults to `auto`, /// which means no smooth scrolling. See the `behavior` option on /// [Element.scrollTo](https://developer.mozilla.org/en-US/docs/Web/API/Element/scrollTo) for more information. /// /// ``` /// # use leptos::*; /// # use leptos::ev::resize; /// # use leptos::html::Div; /// # use leptos_use::{use_scroll_with_options, UseScrollReturn, UseScrollOptions, ScrollBehavior}; /// # /// # #[component] /// # fn Demo() -> impl IntoView { /// # let element = create_node_ref::
(); /// # /// let UseScrollReturn { /// x, y, set_x, set_y, .. /// } = use_scroll_with_options( /// element, /// UseScrollOptions::default().behavior(ScrollBehavior::Smooth), /// ); /// # /// # view! { /// #
"..."
/// # } /// # } /// ``` /// /// or as a `Signal`: /// /// ``` /// # use leptos::*; /// # use leptos::ev::resize; /// # use leptos::html::Div; /// # use leptos_use::{use_scroll_with_options, UseScrollReturn, UseScrollOptions, ScrollBehavior}; /// # /// # #[component] /// # fn Demo() -> impl IntoView { /// # let element = create_node_ref::
(); /// # /// let (smooth, set_smooth) = create_signal(false); /// /// let behavior = Signal::derive(move || { /// if smooth.get() { ScrollBehavior::Smooth } else { ScrollBehavior::Auto } /// }); /// /// let UseScrollReturn { /// x, y, set_x, set_y, .. /// } = use_scroll_with_options( /// element, /// UseScrollOptions::default().behavior(behavior), /// ); /// # /// # view! { /// #
"..."
/// # } /// # } /// ``` /// /// ## Server-Side Rendering /// /// On the server this returns signals that don't change and setters that are noops. pub fn use_scroll(element: El) -> UseScrollReturn where El: Clone, El: Into>, T: Into + Clone + 'static, { use_scroll_with_options(element, Default::default()) } /// Version of [`use_scroll`] with options. See [`use_scroll`] for how to use. #[allow(unused_variables)] pub fn use_scroll_with_options(element: El, options: UseScrollOptions) -> UseScrollReturn where El: Clone, El: Into>, T: Into + Clone + 'static, { let (internal_x, set_internal_x) = create_signal(0.0); let (internal_y, set_internal_y) = create_signal(0.0); let (is_scrolling, set_is_scrolling) = create_signal(false); let arrived_state = create_rw_signal(Directions { left: true, right: false, top: true, bottom: false, }); let directions = create_rw_signal(Directions { left: false, right: false, top: false, bottom: false, }); cfg_if! { if #[cfg(feature = "ssr")] { let set_x = Box::new(|_| {}); let set_y = Box::new(|_| {}); let measure = Box::new(|| {}); } else { let signal = element.into(); let behavior = options.behavior; let scroll_to = { let signal = signal.clone(); move |x: Option, y: Option| { let element = signal.get_untracked(); if let Some(element) = element { let element = element.into(); let mut scroll_options = web_sys::ScrollToOptions::new(); scroll_options.behavior(behavior.get_untracked().into()); if let Some(x) = x { scroll_options.left(x); } if let Some(y) = y { scroll_options.top(y); } element.scroll_to_with_scroll_to_options(&scroll_options); } } }; let set_x = { let scroll_to = scroll_to.clone(); Box::new(move |x| scroll_to(Some(x), None)) }; let set_y = Box::new(move |y| scroll_to(None, Some(y))); let on_scroll_end = { let on_stop = Rc::clone(&options.on_stop); move |e| { if !is_scrolling.get_untracked() { return; } set_is_scrolling.set(false); directions.update(|directions| { directions.left = false; directions.right = false; directions.top = false; directions.bottom = false; on_stop.clone()(e); }); } }; let throttle = options.throttle; let on_scroll_end_debounced = use_debounce_fn_with_arg(on_scroll_end.clone(), throttle + options.idle); let offset = options.offset; let set_arrived_state = move |target: web_sys::Element| { let style = window() .get_computed_style(&target) .expect("failed to get computed style"); if let Some(style) = style { let display = style .get_property_value("display") .expect("failed to get display"); let flex_direction = style .get_property_value("flex-direction") .expect("failed to get flex-direction"); let scroll_left = target.scroll_left() as f64; let scroll_left_abs = scroll_left.abs(); directions.update(|directions| { directions.left = scroll_left < internal_x.get_untracked(); directions.right = scroll_left > internal_x.get_untracked(); }); let left = scroll_left_abs <= offset.left; let right = scroll_left_abs + target.client_width() as f64 >= target.scroll_width() as f64 - offset.right - ARRIVED_STATE_THRESHOLD_PIXELS; arrived_state.update(|arrived_state| { if display == "flex" && flex_direction == "row-reverse" { arrived_state.left = right; arrived_state.right = left; } else { arrived_state.left = left; arrived_state.right = right; } }); set_internal_x.set(scroll_left); let mut scroll_top = target.scroll_top() as f64; // patch for mobile compatibility if target == document().unchecked_into::() && scroll_top == 0.0 { scroll_top = document().body().expect("failed to get body").scroll_top() as f64; } let scroll_top_abs = scroll_top.abs(); directions.update(|directions| { directions.top = scroll_top < internal_y.get_untracked(); directions.bottom = scroll_top > internal_y.get_untracked(); }); let top = scroll_top_abs <= offset.top; let bottom = scroll_top_abs + target.client_height() as f64 >= target.scroll_height() as f64 - offset.bottom - ARRIVED_STATE_THRESHOLD_PIXELS; // reverse columns and rows behave exactly the other way around, // bottom is treated as top and top is treated as the negative version of bottom arrived_state.update(|arrived_state| { if display == "flex" && flex_direction == "column-reverse" { arrived_state.top = bottom; arrived_state.bottom = top; } else { arrived_state.top = top; arrived_state.bottom = bottom; } }); set_internal_y.set(scroll_top); } }; let on_scroll_handler = { let on_scroll = Rc::clone(&options.on_scroll); move |e: web_sys::Event| { let target: web_sys::Element = event_target(&e); set_arrived_state(target); set_is_scrolling.set(true); on_scroll_end_debounced.clone()(e.clone()); on_scroll.clone()(e); } }; let target = { let signal = signal.clone(); Signal::derive(move || { let element = signal.get(); element.map(|element| element.into().unchecked_into::()) }) }; if throttle >= 0.0 { let throttled_scroll_handler = use_throttle_fn_with_arg_and_options( on_scroll_handler.clone(), throttle, ThrottleOptions { trailing: true, leading: false, }, ); let handler = move |e: web_sys::Event| { throttled_scroll_handler.clone()(e); }; let _ = use_event_listener_with_options::< _, Signal>, web_sys::EventTarget, _, >( target, ev::scroll, handler, options.event_listener_options.clone().unwrap_or_default(), ); } else { let _ = use_event_listener_with_options::< _, Signal>, web_sys::EventTarget, _, >( target, ev::scroll, on_scroll_handler, options.event_listener_options.clone().unwrap_or_default(), ); } let _ = use_event_listener_with_options::< _, Signal>, web_sys::EventTarget, _, >( target, scrollend, on_scroll_end, options.event_listener_options.unwrap_or_default(), ); let measure = Box::new(move || { let el = signal.get_untracked(); if let Some(el) = el { let el = el.into(); set_arrived_state(el); } }); }} UseScrollReturn { x: internal_x.into(), set_x, y: internal_y.into(), set_y, is_scrolling: is_scrolling.into(), arrived_state: arrived_state.into(), directions: directions.into(), measure, } } /// We have to check if the scroll amount is close enough to some threshold in order to /// more accurately calculate arrivedState. This is because scrollTop/scrollLeft are non-rounded /// numbers, while scrollHeight/scrollWidth and clientHeight/clientWidth are rounded. /// https://developer.mozilla.org/en-US/docs/Web/API/Element/scrollHeight#determine_if_an_element_has_been_totally_scrolled const ARRIVED_STATE_THRESHOLD_PIXELS: f64 = 1.0; /// Options for [`use_scroll`]. #[derive(DefaultBuilder)] /// Options for [`use_scroll_with_options`]. pub struct UseScrollOptions { /// Throttle time in milliseconds for the scroll events. Defaults to 0 (disabled). throttle: f64, /// After scrolling ends we wait idle + throttle milliseconds before we consider scrolling to have stopped. /// Defaults to 200. idle: f64, /// Threshold in pixels when we consider a side to have arrived (`UseScrollReturn::arrived_state`). offset: ScrollOffset, /// Callback when scrolling is happening. on_scroll: Rc, /// Callback when scrolling stops (after `idle` + `throttle` milliseconds have passed). on_stop: Rc, /// Options passed to the `addEventListener("scroll", ...)` call #[builder(into)] event_listener_options: Option, /// When changing the `x` or `y` signals this specifies the scroll behaviour. /// Can be `Auto` (= not smooth) or `Smooth`. Defaults to `Auto`. #[builder(into)] behavior: MaybeSignal, } impl Default for UseScrollOptions { fn default() -> Self { Self { throttle: 0.0, idle: 200.0, offset: ScrollOffset::default(), on_scroll: Rc::new(|_| {}), on_stop: Rc::new(|_| {}), event_listener_options: Default::default(), behavior: Default::default(), } } } /// The scroll behavior. /// Can be `Auto` (= not smooth) or `Smooth`. Defaults to `Auto`. #[derive(Default, Copy, Clone)] pub enum ScrollBehavior { #[default] Auto, Smooth, } impl From for web_sys::ScrollBehavior { fn from(val: ScrollBehavior) -> Self { match val { ScrollBehavior::Auto => web_sys::ScrollBehavior::Auto, ScrollBehavior::Smooth => web_sys::ScrollBehavior::Smooth, } } } /// The return value of [`use_scroll`]. pub struct UseScrollReturn { pub x: Signal, pub set_x: Box, pub y: Signal, pub set_y: Box, pub is_scrolling: Signal, pub arrived_state: Signal, pub directions: Signal, pub measure: Box, } #[derive(Copy, Clone)] pub struct Directions { pub left: bool, pub right: bool, pub top: bool, pub bottom: bool, } #[derive(Default, Copy, Clone)] /// Threshold in pixels when we consider a side to have arrived (`UseScrollReturn::arrived_state`). pub struct ScrollOffset { pub left: f64, pub top: f64, pub right: f64, pub bottom: f64, }