leptos_use/watch_throttled.rs
1use crate::{utils::ThrottleOptions, watch_with_options, WatchOptions};
2use default_struct_builder::DefaultBuilder;
3
4/// A throttled version of `leptos::watch`.
5///
6/// ## Demo
7///
8/// [Link to Demo](https://github.com/Synphonyte/leptos-use/tree/main/examples/watch_throttled)
9///
10/// ## Usage
11///
12/// ```
13/// # use leptos::prelude::*;
14/// # use leptos::logging::log;
15/// # use leptos_use::watch_throttled;
16/// #
17/// # pub fn Demo() -> impl IntoView {
18/// # let (source, set_source) = signal(0);
19/// #
20/// watch_throttled(
21/// move || source.get(),
22/// move |_, _, _| {
23/// log!("changed!");
24/// },
25/// 500.0,
26/// );
27///
28/// # view! { }
29/// # }
30/// ```
31///
32/// This really is only shorthand shorthand for `watch_with_options(deps, callback, WatchOptions::default().throttle(ms))`.
33///
34/// Please note that if the current component is cleaned up before the throttled callback is called, the throttled callback will not be called.
35///
36/// There's also `watch_throttled_with_options` where you can specify the other watch options (except `filter`).
37///
38/// ```
39/// # use leptos::prelude::*;
40/// # use leptos::logging::log;
41/// # use leptos_use::{watch_throttled_with_options, WatchThrottledOptions};
42/// #
43/// # pub fn Demo() -> impl IntoView {
44/// # let (source, set_source) = signal(0);
45/// #
46/// watch_throttled_with_options(
47/// move || source.get(),
48/// move |_, _, _| {
49/// log!("changed!");
50/// },
51/// 500.0,
52/// WatchThrottledOptions::default().leading(true).trailing(false),
53/// );
54///
55/// # view! { }
56/// # }
57/// ```
58///
59/// ## Recommended Reading
60///
61/// - [**Debounce vs Throttle**: Definitive Visual Guide](https://redd.one/blog/debounce-vs-throttle)
62/// - [Debouncing and Throttling Explained Through Examples](https://css-tricks.com/debouncing-throttling-explained-examples/)
63///
64/// ## Server-Side Rendering
65///
66/// On the server the callback
67/// will never be called except if you set `immediate` to `true` in which case the callback will be
68/// called exactly once.
69///
70/// ## See also
71///
72/// * `leptos::watch`
73/// * [`fn@crate::watch_debounced`]
74pub fn watch_throttled<W, T, DFn, CFn>(
75 deps: DFn,
76 callback: CFn,
77 ms: f64,
78) -> impl Fn() + Clone + Send + Sync
79where
80 DFn: Fn() -> W + 'static,
81 CFn: Fn(&W, Option<&W>, Option<T>) -> T + Clone + 'static,
82 W: Clone + 'static,
83 T: Clone + 'static,
84{
85 watch_with_options(deps, callback, WatchOptions::default().throttle(ms))
86}
87
88/// Version of [`fn@watch_throttled`] that accepts `WatchThrottledOptions`. See [`watch_throttled`] for how to use.
89pub fn watch_throttled_with_options<W, T, DFn, CFn>(
90 deps: DFn,
91 callback: CFn,
92 ms: f64,
93 options: WatchThrottledOptions,
94) -> impl Fn() + Clone + Send + Sync
95where
96 DFn: Fn() -> W + 'static,
97 CFn: Fn(&W, Option<&W>, Option<T>) -> T + Clone + 'static,
98 W: Clone + 'static,
99 T: Clone + 'static,
100{
101 watch_with_options(
102 deps,
103 callback,
104 WatchOptions::default()
105 .throttle_with_options(
106 ms,
107 ThrottleOptions::default()
108 .leading(options.leading)
109 .trailing(options.trailing),
110 )
111 .immediate(options.immediate),
112 )
113}
114
115/// Options for [`watch_throttled_with_options`].
116#[derive(DefaultBuilder)]
117pub struct WatchThrottledOptions {
118 /// If `immediate` is false, the `callback` will not run immediately but only after
119 /// the first change is detected of any signal that is accessed in `deps`.
120 /// Defaults to `true`.
121 immediate: bool,
122
123 /// Invoke on the trailing edge of the timeout. Defaults to `true`.
124 pub trailing: bool,
125 /// Invoke on the leading edge of the timeout. Defaults to `true`.
126 pub leading: bool,
127}
128
129impl Default for WatchThrottledOptions {
130 fn default() -> Self {
131 Self {
132 immediate: false,
133 trailing: true,
134 leading: true,
135 }
136 }
137}