leptos_use/use_scroll.rs
1use crate::core::{Direction, Directions, IntoElementMaybeSignal};
2use crate::UseEventListenerOptions;
3use cfg_if::cfg_if;
4use default_struct_builder::DefaultBuilder;
5use leptos::prelude::*;
6use leptos::reactive::wrappers::read::Signal;
7use std::rc::Rc;
8
9cfg_if! { if #[cfg(not(feature = "ssr"))] {
10use crate::use_event_listener::use_event_listener_with_options;
11use crate::{
12 sendwrap_fn, use_debounce_fn_with_arg, use_throttle_fn_with_arg_and_options, ThrottleOptions,
13};
14use leptos::ev;
15use leptos::ev::scrollend;
16use wasm_bindgen::JsCast;
17
18/// We have to check if the scroll amount is close enough to some threshold in order to
19/// more accurately calculate arrivedState. This is because scrollTop/scrollLeft are non-rounded
20/// numbers, while scrollHeight/scrollWidth and clientHeight/clientWidth are rounded.
21/// https://developer.mozilla.org/en-US/docs/Web/API/Element/scrollHeight#determine_if_an_element_has_been_totally_scrolled
22const ARRIVED_STATE_THRESHOLD_PIXELS: f64 = 1.0;
23}}
24
25/// Reactive scroll position and state.
26///
27/// ## Demo
28///
29/// [Link to Demo](https://github.com/Synphonyte/leptos-use/tree/main/examples/use_scroll)
30///
31/// ## Usage
32///
33/// ```
34/// # use leptos::prelude::*;
35/// # use leptos::ev::resize;
36/// # use leptos::html::Div;
37/// # use leptos_use::{use_scroll, UseScrollReturn};
38/// #
39/// # #[component]
40/// # fn Demo() -> impl IntoView {
41/// let element = NodeRef::<Div>::new();
42///
43/// let UseScrollReturn {
44/// x, y, set_x, set_y, is_scrolling, arrived_state, directions, ..
45/// } = use_scroll(element);
46///
47/// view! {
48/// <div node_ref=element>"..."</div>
49/// }
50/// # }
51/// ```
52///
53/// ### With Offsets
54///
55/// You can provide offsets when you use [`use_scroll_with_options`].
56/// These offsets are thresholds in pixels when a side is considered to have arrived. This is reflected in the return field `arrived_state`.
57///
58/// ```
59/// # use leptos::prelude::*;
60/// # use leptos::html::Div;
61/// # use leptos::ev::resize;
62/// # use leptos_use::{use_scroll_with_options, UseScrollReturn, UseScrollOptions, ScrollOffset};
63/// #
64/// # #[component]
65/// # fn Demo() -> impl IntoView {
66/// # let element = NodeRef::<Div>::new();
67/// #
68/// let UseScrollReturn {
69/// x,
70/// y,
71/// set_x,
72/// set_y,
73/// is_scrolling,
74/// arrived_state,
75/// directions,
76/// ..
77/// } = use_scroll_with_options(
78/// element,
79/// UseScrollOptions::default().offset(ScrollOffset {
80/// top: 30.0,
81/// bottom: 30.0,
82/// right: 30.0,
83/// left: 30.0,
84/// }),
85/// );
86/// #
87/// # view! { /// # <div node_ref=element>"..."</div>
88/// # }
89/// # }
90/// ```
91///
92/// ### Setting Scroll Position
93///
94/// Set the `x` and `y` values to make the element scroll to that position.
95///
96/// ```
97/// # use leptos::prelude::*;
98/// # use leptos::html::Div;
99/// # use leptos::ev::resize;
100/// # use leptos_use::{use_scroll, UseScrollReturn};
101/// #
102/// # #[component]
103/// # fn Demo() -> impl IntoView {
104/// let element = NodeRef::<Div>::new();
105///
106/// let UseScrollReturn {
107/// x, y, set_x, set_y, ..
108/// } = use_scroll(element);
109///
110/// view! {
111/// <div node_ref=element>"..."</div>
112/// <button on:click=move |_| set_x(x.get_untracked() + 10.0)>"Scroll right 10px"</button>
113/// <button on:click=move |_| set_y(y.get_untracked() + 10.0)>"Scroll down 10px"</button>
114/// }
115/// # }
116/// ```
117///
118/// ### Smooth Scrolling
119///
120/// Set `behavior: smooth` to enable smooth scrolling. The `behavior` option defaults to `auto`,
121/// which means no smooth scrolling. See the `behavior` option on
122/// [Element.scrollTo](https://developer.mozilla.org/en-US/docs/Web/API/Element/scrollTo) for more information.
123///
124/// ```
125/// # use leptos::prelude::*;
126/// # use leptos::ev::resize;
127/// # use leptos::html::Div;
128/// # use leptos_use::{use_scroll_with_options, UseScrollReturn, UseScrollOptions, ScrollBehavior};
129/// #
130/// # #[component]
131/// # fn Demo() -> impl IntoView {
132/// # let element = NodeRef::<Div>::new();
133/// #
134/// let UseScrollReturn {
135/// x, y, set_x, set_y, ..
136/// } = use_scroll_with_options(
137/// element,
138/// UseScrollOptions::default().behavior(ScrollBehavior::Smooth),
139/// );
140/// #
141/// # view! { /// # <div node_ref=element>"..."</div>
142/// # }
143/// # }
144/// ```
145///
146/// or as a `Signal`:
147///
148/// ```
149/// # use leptos::prelude::*;
150/// # use leptos::ev::resize;
151/// # use leptos::html::Div;
152/// # use leptos_use::{use_scroll_with_options, UseScrollReturn, UseScrollOptions, ScrollBehavior};
153/// #
154/// # #[component]
155/// # fn Demo() -> impl IntoView {
156/// # let element = NodeRef::<Div>::new();
157/// #
158/// let (smooth, set_smooth) = signal(false);
159///
160/// let behavior = Signal::derive(move || {
161/// if smooth.get() { ScrollBehavior::Smooth } else { ScrollBehavior::Auto }
162/// });
163///
164/// let UseScrollReturn {
165/// x, y, set_x, set_y, ..
166/// } = use_scroll_with_options(
167/// element,
168/// UseScrollOptions::default().behavior(behavior),
169/// );
170/// #
171/// # view! { /// # <div node_ref=element>"..."</div>
172/// # }
173/// # }
174/// ```
175///
176/// ## SendWrapped Return
177///
178/// The returned closures `set_x`, `set_y` and `measure` are sendwrapped functions. They can
179/// only be called from the same thread that called `use_scroll`.
180///
181/// ## Server-Side Rendering
182///
183/// On the server this returns signals that don't change and setters that are noops.
184pub fn use_scroll<El, M>(
185 element: El,
186) -> UseScrollReturn<
187 impl Fn(f64) + Clone + Send + Sync,
188 impl Fn(f64) + Clone + Send + Sync,
189 impl Fn() + Clone + Send + Sync,
190>
191where
192 El: IntoElementMaybeSignal<web_sys::Element, M>,
193{
194 use_scroll_with_options(element, Default::default())
195}
196
197/// Version of [`use_scroll`] with options. See [`use_scroll`] for how to use.
198#[cfg_attr(feature = "ssr", allow(unused_variables))]
199pub fn use_scroll_with_options<El, M>(
200 element: El,
201 options: UseScrollOptions,
202) -> UseScrollReturn<
203 impl Fn(f64) + Clone + Send + Sync,
204 impl Fn(f64) + Clone + Send + Sync,
205 impl Fn() + Clone + Send + Sync,
206>
207where
208 El: IntoElementMaybeSignal<web_sys::Element, M>,
209{
210 let (internal_x, set_internal_x) = signal(0.0);
211 let (internal_y, set_internal_y) = signal(0.0);
212
213 let (is_scrolling, set_is_scrolling) = signal(false);
214
215 let arrived_state = RwSignal::new(Directions {
216 left: true,
217 right: false,
218 top: true,
219 bottom: false,
220 });
221 let directions = RwSignal::new(Directions {
222 left: false,
223 right: false,
224 top: false,
225 bottom: false,
226 });
227
228 let set_x;
229 let set_y;
230 let measure;
231
232 #[cfg(feature = "ssr")]
233 {
234 set_x = |_| {};
235 set_y = |_| {};
236 measure = || {};
237 }
238
239 #[cfg(not(feature = "ssr"))]
240 {
241 let signal = element.into_element_maybe_signal();
242 let behavior = options.behavior;
243
244 let scroll_to = move |x: Option<f64>, y: Option<f64>| {
245 let element = signal.get_untracked();
246
247 if let Some(element) = element {
248 let scroll_options = web_sys::ScrollToOptions::new();
249 scroll_options.set_behavior(behavior.get_untracked().into());
250
251 if let Some(x) = x {
252 scroll_options.set_left(x);
253 }
254 if let Some(y) = y {
255 scroll_options.set_top(y);
256 }
257
258 element.scroll_to_with_scroll_to_options(&scroll_options);
259 }
260 };
261
262 set_x = sendwrap_fn!(move |x| scroll_to(Some(x), None));
263
264 set_y = sendwrap_fn!(move |y| scroll_to(None, Some(y)));
265
266 let on_scroll_end = {
267 let on_stop = Rc::clone(&options.on_stop);
268
269 move |e| {
270 if !is_scrolling.try_get_untracked().unwrap_or_default() {
271 return;
272 }
273
274 set_is_scrolling.set(false);
275 directions.update(|directions| {
276 directions.left = false;
277 directions.right = false;
278 directions.top = false;
279 directions.bottom = false;
280 on_stop.clone()(e);
281 });
282 }
283 };
284
285 let throttle = options.throttle;
286
287 let on_scroll_end_debounced =
288 use_debounce_fn_with_arg(on_scroll_end.clone(), throttle + options.idle);
289
290 let offset = options.offset;
291
292 let set_arrived_state = move |target: web_sys::Element| {
293 let style = window()
294 .get_computed_style(&target)
295 .expect("failed to get computed style");
296
297 if let Some(style) = style {
298 let display = style
299 .get_property_value("display")
300 .expect("failed to get display");
301 let flex_direction = style
302 .get_property_value("flex-direction")
303 .expect("failed to get flex-direction");
304
305 let scroll_left = target.scroll_left() as f64;
306 let scroll_left_abs = scroll_left.abs();
307
308 directions.update(|directions| {
309 directions.left = scroll_left < internal_x.get_untracked();
310 directions.right = scroll_left > internal_x.get_untracked();
311 });
312
313 let left = scroll_left_abs <= offset.left;
314 let right = scroll_left_abs + target.client_width() as f64
315 >= target.scroll_width() as f64 - offset.right - ARRIVED_STATE_THRESHOLD_PIXELS;
316
317 arrived_state.update(|arrived_state| {
318 if display == "flex" && flex_direction == "row-reverse" {
319 arrived_state.left = right;
320 arrived_state.right = left;
321 } else {
322 arrived_state.left = left;
323 arrived_state.right = right;
324 }
325 });
326 set_internal_x.set(scroll_left);
327
328 let mut scroll_top = target.scroll_top() as f64;
329
330 // patch for mobile compatibility
331 if target == document().unchecked_into::<web_sys::Element>() && scroll_top == 0.0 {
332 scroll_top = document().body().expect("failed to get body").scroll_top() as f64;
333 }
334
335 let scroll_top_abs = scroll_top.abs();
336
337 directions.update(|directions| {
338 directions.top = scroll_top < internal_y.get_untracked();
339 directions.bottom = scroll_top > internal_y.get_untracked();
340 });
341
342 let top = scroll_top_abs <= offset.top;
343 let bottom = scroll_top_abs + target.client_height() as f64
344 >= target.scroll_height() as f64
345 - offset.bottom
346 - ARRIVED_STATE_THRESHOLD_PIXELS;
347
348 // reverse columns and rows behave exactly the other way around,
349 // bottom is treated as top and top is treated as the negative version of bottom
350 arrived_state.update(|arrived_state| {
351 if display == "flex" && flex_direction == "column-reverse" {
352 arrived_state.top = bottom;
353 arrived_state.bottom = top;
354 } else {
355 arrived_state.top = top;
356 arrived_state.bottom = bottom;
357 }
358 });
359
360 set_internal_y.set(scroll_top);
361 }
362 };
363
364 let on_scroll_handler = {
365 let on_scroll = Rc::clone(&options.on_scroll);
366
367 move |e: web_sys::Event| {
368 let target: web_sys::Element = event_target(&e);
369
370 set_arrived_state(target);
371 set_is_scrolling.set(true);
372
373 on_scroll_end_debounced.clone()(e.clone());
374 on_scroll.clone()(e);
375 }
376 };
377
378 let target = Signal::derive_local(move || {
379 let element = signal.get();
380 element.map(|element| element.unchecked_into::<web_sys::EventTarget>())
381 });
382
383 if throttle >= 0.0 {
384 let throttled_scroll_handler = use_throttle_fn_with_arg_and_options(
385 on_scroll_handler.clone(),
386 throttle,
387 ThrottleOptions {
388 trailing: true,
389 leading: false,
390 },
391 );
392
393 let handler = move |e: web_sys::Event| {
394 throttled_scroll_handler.clone()(e);
395 };
396
397 let _ = use_event_listener_with_options::<
398 _,
399 Signal<Option<web_sys::EventTarget>, LocalStorage>,
400 _,
401 _,
402 >(target, ev::scroll, handler, options.event_listener_options);
403 } else {
404 let _ = use_event_listener_with_options::<
405 _,
406 Signal<Option<web_sys::EventTarget>, LocalStorage>,
407 _,
408 _,
409 >(
410 target,
411 ev::scroll,
412 on_scroll_handler,
413 options.event_listener_options,
414 );
415 }
416
417 let _ = use_event_listener_with_options::<
418 _,
419 Signal<Option<web_sys::EventTarget>, LocalStorage>,
420 _,
421 _,
422 >(
423 target,
424 scrollend,
425 on_scroll_end,
426 options.event_listener_options,
427 );
428
429 measure = sendwrap_fn!(move || {
430 if let Some(el) = signal.try_get_untracked().flatten() {
431 set_arrived_state(el);
432 }
433 });
434 }
435
436 UseScrollReturn {
437 x: internal_x.into(),
438 set_x,
439 y: internal_y.into(),
440 set_y,
441 is_scrolling: is_scrolling.into(),
442 arrived_state: arrived_state.into(),
443 directions: directions.into(),
444 measure,
445 }
446}
447
448/// Options for [`use_scroll`].
449#[derive(DefaultBuilder)]
450/// Options for [`use_scroll_with_options`].
451#[cfg_attr(feature = "ssr", allow(dead_code))]
452pub struct UseScrollOptions {
453 /// Throttle time in milliseconds for the scroll events. Defaults to 0 (disabled).
454 throttle: f64,
455
456 /// After scrolling ends we wait idle + throttle milliseconds before we consider scrolling to have stopped.
457 /// Defaults to 200.
458 idle: f64,
459
460 /// Threshold in pixels when we consider a side to have arrived (`UseScrollReturn::arrived_state`).
461 offset: ScrollOffset,
462
463 /// Callback when scrolling is happening.
464 on_scroll: Rc<dyn Fn(web_sys::Event)>,
465
466 /// Callback when scrolling stops (after `idle` + `throttle` milliseconds have passed).
467 on_stop: Rc<dyn Fn(web_sys::Event)>,
468
469 /// Options passed to the `addEventListener("scroll", ...)` call
470 event_listener_options: UseEventListenerOptions,
471
472 /// When changing the `x` or `y` signals this specifies the scroll behaviour.
473 /// Can be `Auto` (= not smooth) or `Smooth`. Defaults to `Auto`.
474 #[builder(into)]
475 behavior: Signal<ScrollBehavior>,
476}
477
478impl Default for UseScrollOptions {
479 fn default() -> Self {
480 Self {
481 throttle: 0.0,
482 idle: 200.0,
483 offset: ScrollOffset::default(),
484 on_scroll: Rc::new(|_| {}),
485 on_stop: Rc::new(|_| {}),
486 event_listener_options: Default::default(),
487 behavior: Default::default(),
488 }
489 }
490}
491
492/// The scroll behavior.
493/// Can be `Auto` (= not smooth) or `Smooth`. Defaults to `Auto`.
494#[derive(Default, Copy, Clone)]
495pub enum ScrollBehavior {
496 #[default]
497 Auto,
498 Smooth,
499}
500
501impl From<ScrollBehavior> for web_sys::ScrollBehavior {
502 fn from(val: ScrollBehavior) -> Self {
503 match val {
504 ScrollBehavior::Auto => web_sys::ScrollBehavior::Auto,
505 ScrollBehavior::Smooth => web_sys::ScrollBehavior::Smooth,
506 }
507 }
508}
509
510/// The return value of [`use_scroll`].
511pub struct UseScrollReturn<SetXFn, SetYFn, MFn>
512where
513 SetXFn: Fn(f64) + Clone + Send + Sync,
514 SetYFn: Fn(f64) + Clone + Send + Sync,
515 MFn: Fn() + Clone + Send + Sync,
516{
517 /// X coordinate of scroll position
518 pub x: Signal<f64>,
519
520 /// Sets the value of `x`. This does also scroll the element.
521 pub set_x: SetXFn,
522
523 /// Y coordinate of scroll position
524 pub y: Signal<f64>,
525
526 /// Sets the value of `y`. This does also scroll the element.
527 pub set_y: SetYFn,
528
529 /// Is true while the element is being scrolled.
530 pub is_scrolling: Signal<bool>,
531
532 /// Sets the field that represents a direction to true if the
533 /// element is scrolled all the way to that side.
534 pub arrived_state: Signal<Directions>,
535
536 /// The directions in which the element is being scrolled are set to true.
537 pub directions: Signal<Directions>,
538
539 /// Re-evaluates the `arrived_state`.
540 pub measure: MFn,
541}
542
543#[derive(Default, Copy, Clone, Debug)]
544/// Threshold in pixels when we consider a side to have arrived (`UseScrollReturn::arrived_state`).
545pub struct ScrollOffset {
546 pub left: f64,
547 pub top: f64,
548 pub right: f64,
549 pub bottom: f64,
550}
551
552impl ScrollOffset {
553 /// Sets the value of the provided direction
554 pub fn set_direction(mut self, direction: Direction, value: f64) -> Self {
555 match direction {
556 Direction::Top => self.top = value,
557 Direction::Bottom => self.bottom = value,
558 Direction::Left => self.left = value,
559 Direction::Right => self.right = value,
560 }
561
562 self
563 }
564}