2023-07-16 01:07:52 +01:00
use crate ::utils ::Pausable ;
use default_struct_builder ::DefaultBuilder ;
use leptos ::* ;
2023-08-15 12:31:25 +01:00
use std ::cell ::{ Cell , RefCell } ;
2023-07-16 01:07:52 +01:00
use std ::rc ::Rc ;
use wasm_bindgen ::closure ::Closure ;
use wasm_bindgen ::JsCast ;
/// Call function on every requestAnimationFrame.
/// With controls of pausing and resuming.
///
/// ## Demo
///
/// [Link to Demo](https://github.com/Synphonyte/leptos-use/tree/main/examples/use_raf_fn)
///
/// ## Usage
///
/// ```
/// # use leptos::*;
/// # use leptos_use::use_raf_fn;
/// use leptos_use::utils::Pausable;
/// #
/// # #[component]
2023-07-27 18:06:36 +01:00
/// # fn Demo() -> impl IntoView {
/// let (count, set_count) = create_signal(0);
2023-07-16 01:07:52 +01:00
///
2023-07-27 18:06:36 +01:00
/// let Pausable { pause, resume, is_active } = use_raf_fn(move |_| {
2023-07-16 01:07:52 +01:00
/// set_count.update(|count| *count += 1);
/// });
///
2023-07-27 18:06:36 +01:00
/// view! { <div>Count: { count }</div> }
2023-07-16 01:07:52 +01:00
/// }
/// ```
///
/// You can use `use_raf_fn_with_options` and set `immediate` to `false`. In that case
/// you have to call `resume()` before the `callback` is executed.
pub fn use_raf_fn (
2023-08-04 13:19:34 +01:00
callback : impl Fn ( UseRafFnCallbackArgs ) + 'static ,
2023-07-16 01:07:52 +01:00
) -> Pausable < impl Fn ( ) + Clone , impl Fn ( ) + Clone > {
2023-07-27 18:06:36 +01:00
use_raf_fn_with_options ( callback , UseRafFnOptions ::default ( ) )
2023-07-16 01:07:52 +01:00
}
/// Version of [`use_raf_fn`] that takes a `UseRafFnOptions`. See [`use_raf_fn`] for how to use.
pub fn use_raf_fn_with_options (
2023-08-04 13:19:34 +01:00
callback : impl Fn ( UseRafFnCallbackArgs ) + 'static ,
2023-07-16 01:07:52 +01:00
options : UseRafFnOptions ,
) -> Pausable < impl Fn ( ) + Clone , impl Fn ( ) + Clone > {
let UseRafFnOptions { immediate } = options ;
2023-08-15 12:31:25 +01:00
let raf_handle = Rc ::new ( Cell ::new ( None ::< i32 > ) ) ;
2023-07-16 01:07:52 +01:00
2023-07-27 18:06:36 +01:00
let ( is_active , set_active ) = create_signal ( false ) ;
2023-07-16 01:07:52 +01:00
let loop_ref = Rc ::new ( RefCell ::new ( Box ::new ( | _ : f64 | { } ) as Box < dyn Fn ( f64 ) > ) ) ;
let request_next_frame = {
let loop_ref = Rc ::clone ( & loop_ref ) ;
2023-08-15 12:31:25 +01:00
let raf_handle = Rc ::clone ( & raf_handle ) ;
2023-07-16 01:07:52 +01:00
move | | {
let loop_ref = Rc ::clone ( & loop_ref ) ;
2023-08-15 12:31:25 +01:00
raf_handle . set (
2023-07-16 01:07:52 +01:00
window ( )
. request_animation_frame (
Closure ::once_into_js ( move | timestamp : f64 | {
loop_ref . borrow ( ) ( timestamp ) ;
} )
. as_ref ( )
. unchecked_ref ( ) ,
)
. ok ( ) ,
) ;
}
} ;
let loop_fn = {
let request_next_frame = request_next_frame . clone ( ) ;
2023-08-16 16:23:29 +01:00
let previous_frame_timestamp = Cell ::new ( 0.0_ f64 ) ;
2023-07-16 01:07:52 +01:00
move | timestamp : f64 | {
if ! is_active . get ( ) {
return ;
}
2023-08-15 12:31:25 +01:00
let prev_timestamp = previous_frame_timestamp . get ( ) ;
2023-07-16 01:07:52 +01:00
let delta = if prev_timestamp > 0.0 {
timestamp - prev_timestamp
} else {
0.0
} ;
callback ( UseRafFnCallbackArgs { delta , timestamp } ) ;
2023-08-15 12:31:25 +01:00
previous_frame_timestamp . set ( timestamp ) ;
2023-07-16 01:07:52 +01:00
request_next_frame ( ) ;
}
} ;
let _ = loop_ref . replace ( Box ::new ( loop_fn ) ) ;
let resume = move | | {
if ! is_active . get ( ) {
set_active . set ( true ) ;
request_next_frame ( ) ;
}
} ;
let pause = move | | {
set_active . set ( false ) ;
2023-08-15 12:31:25 +01:00
let handle = raf_handle . get ( ) ;
2023-07-16 01:07:52 +01:00
if let Some ( handle ) = handle {
let _ = window ( ) . cancel_animation_frame ( handle ) ;
}
2023-08-15 12:31:25 +01:00
raf_handle . set ( None ) ;
2023-07-16 01:07:52 +01:00
} ;
if immediate {
resume ( ) ;
}
2023-08-15 12:31:25 +01:00
on_cleanup ( pause . clone ( ) ) ;
2023-07-16 01:07:52 +01:00
Pausable {
resume ,
pause ,
is_active : is_active . into ( ) ,
}
}
/// Options for [`use_raf_fn_with_options`].
#[ derive(DefaultBuilder) ]
pub struct UseRafFnOptions {
/// Start the requestAnimationFrame loop immediately on creation. Defaults to `true`.
/// If false the loop will only start when you call `resume()`.
immediate : bool ,
}
impl Default for UseRafFnOptions {
fn default ( ) -> Self {
Self { immediate : true }
}
}
/// Type of the argument for the callback of [`use_raf_fn`].
pub struct UseRafFnCallbackArgs {
/// Time elapsed between this and the last frame.
pub delta : f64 ,
/// Time elapsed since the creation of the web page. See [MDN Docs](https://developer.mozilla.org/en-US/docs/Web/API/DOMHighResTimeStamp#the_time_origin Time origin).
pub timestamp : f64 ,
}