leptos_use/use_throttle_fn.rs
1use crate::utils::{create_filter_wrapper, create_filter_wrapper_with_arg, throttle_filter};
2use leptos::prelude::Signal;
3use std::sync::{Arc, Mutex};
4
5pub use crate::utils::ThrottleOptions;
6
7/// Throttle execution of a function.
8/// Especially useful for rate limiting execution of handlers on events like resize and scroll.
9///
10/// > Throttle is a spring that throws balls: After a ball flies out it needs some time to shrink back, so it cannot throw any more balls until it's ready.
11///
12/// ## Demo
13///
14/// [Link to Demo](https://github.com/Synphonyte/leptos-use/tree/main/examples/use_throttle_fn)
15///
16/// ## Usage
17///
18/// ```
19/// # use leptos::prelude::*;
20/// # use leptos_use::use_throttle_fn;
21/// #
22/// # #[component]
23/// # fn Demo() -> impl IntoView {
24/// let mut throttled_fn = use_throttle_fn(
25/// || {
26/// // do something, it will be called at most 1 time per second
27/// },
28/// 1000.0,
29/// );
30/// view! {
31/// <button on:click=move |_| { throttled_fn(); }>
32/// "Smash me!"
33/// </button>
34/// }
35/// # }
36/// ```
37///
38/// Please note that if the current component is cleaned up before the throttled callback is called, the throttled callback will not be called.
39///
40/// You can provide options when you use [`use_throttle_fn_with_options`].
41///
42/// ```
43/// # use leptos::prelude::*;
44/// # use leptos_use::{ThrottleOptions, use_throttle_fn_with_options};
45/// # #[component]
46/// # fn Demo() -> impl IntoView {
47/// let throttled_fn = use_throttle_fn_with_options(
48/// || {
49/// // do something, it will be called at most 1 time per second
50/// },
51/// 1000.0,
52/// ThrottleOptions::default()
53/// .leading(true)
54/// .trailing(true),
55/// );
56/// # view! { }
57/// # }
58/// ```
59///
60/// If you want to throttle a function that takes an argument there are also the versions
61/// [`use_throttle_fn_with_arg`] and [`use_throttle_fn_with_arg_and_options`].
62///
63/// ## SendWrapped Return
64///
65/// The returned closure is a sendwrapped function. It can
66/// only be called from the same thread that called `use_throttle_...`.
67///
68/// ## Recommended Reading
69///
70/// - [**Debounce vs Throttle**: Definitive Visual Guide](https://redd.one/blog/debounce-vs-throttle)
71/// - [Debouncing and Throttling Explained Through Examples](https://css-tricks.com/debouncing-throttling-explained-examples/)
72///
73/// ## Server-Side Rendering
74///
75/// Internally this uses `setTimeout` which is not supported on the server. So usually calling
76/// a throttled function on the server will simply be ignored.
77pub fn use_throttle_fn<F, R>(
78 func: F,
79 ms: impl Into<Signal<f64>> + 'static,
80) -> impl Fn() -> Arc<Mutex<Option<R>>> + Clone + Send + Sync
81where
82 F: Fn() -> R + Clone + 'static,
83 R: 'static,
84{
85 use_throttle_fn_with_options(func, ms, Default::default())
86}
87
88/// Version of [`use_throttle_fn`] with throttle options. See the docs for [`use_throttle_fn`] for how to use.
89pub fn use_throttle_fn_with_options<F, R>(
90 func: F,
91 ms: impl Into<Signal<f64>> + 'static,
92 options: ThrottleOptions,
93) -> impl Fn() -> Arc<Mutex<Option<R>>> + Clone + Send + Sync
94where
95 F: Fn() -> R + Clone + 'static,
96 R: 'static,
97{
98 create_filter_wrapper(Arc::new(throttle_filter(ms, options)), func)
99}
100
101/// Version of [`use_throttle_fn`] with an argument for the throttled function. See the docs for [`use_throttle_fn`] for how to use.
102pub fn use_throttle_fn_with_arg<F, Arg, R>(
103 func: F,
104 ms: impl Into<Signal<f64>> + 'static,
105) -> impl Fn(Arg) -> Arc<Mutex<Option<R>>> + Clone + Send + Sync
106where
107 F: Fn(Arg) -> R + Clone + 'static,
108 Arg: Clone + 'static,
109 R: 'static,
110{
111 use_throttle_fn_with_arg_and_options(func, ms, Default::default())
112}
113
114/// Version of [`use_throttle_fn_with_arg`] with throttle options. See the docs for [`use_throttle_fn`] for how to use.
115pub fn use_throttle_fn_with_arg_and_options<F, Arg, R>(
116 func: F,
117 ms: impl Into<Signal<f64>> + 'static,
118 options: ThrottleOptions,
119) -> impl Fn(Arg) -> Arc<Mutex<Option<R>>> + Clone + Send + Sync
120where
121 F: Fn(Arg) -> R + Clone + 'static,
122 Arg: Clone + 'static,
123 R: 'static,
124{
125 create_filter_wrapper_with_arg(Arc::new(throttle_filter(ms, options)), func)
126}