added ssr to use_element_bounding

This commit is contained in:
Maccesch 2024-01-03 21:11:28 +00:00
parent d9ff05afea
commit 34c913caae
2 changed files with 100 additions and 77 deletions

View file

@ -1,6 +1,5 @@
use crate::{use_event_listener_with_options, UseEventListenerOptions};
use cfg_if::cfg_if; use cfg_if::cfg_if;
use leptos::{ev::change, *}; use leptos::*;
/// Reactive [`window.devicePixelRatio`](https://developer.mozilla.org/en-US/docs/Web/API/Window/devicePixelRatio) /// Reactive [`window.devicePixelRatio`](https://developer.mozilla.org/en-US/docs/Web/API/Window/devicePixelRatio)
/// ///
@ -33,6 +32,9 @@ pub fn use_device_pixel_ratio() -> Signal<f64> {
cfg_if! { if #[cfg(feature = "ssr")] { cfg_if! { if #[cfg(feature = "ssr")] {
let pixel_ratio = Signal::derive(|| 1.0); let pixel_ratio = Signal::derive(|| 1.0);
} else { } else {
use crate::{use_event_listener_with_options, UseEventListenerOptions};
use leptos::ev::change;
let initial_pixel_ratio = window().device_pixel_ratio(); let initial_pixel_ratio = window().device_pixel_ratio();
let (pixel_ratio, set_pixel_ratio) = create_signal(initial_pixel_ratio); let (pixel_ratio, set_pixel_ratio) = create_signal(initial_pixel_ratio);

View file

@ -1,9 +1,6 @@
use crate::core::ElementMaybeSignal; use crate::core::ElementMaybeSignal;
use crate::{ use cfg_if::cfg_if;
use_event_listener_with_options, use_resize_observer, use_window, UseEventListenerOptions,
};
use default_struct_builder::DefaultBuilder; use default_struct_builder::DefaultBuilder;
use leptos::ev::{resize, scroll};
use leptos::*; use leptos::*;
/// Reactive [bounding box](https://developer.mozilla.org/en-US/docs/Web/API/Element/getBoundingClientRect) of an HTML element /// Reactive [bounding box](https://developer.mozilla.org/en-US/docs/Web/API/Element/getBoundingClientRect) of an HTML element
@ -29,6 +26,9 @@ use leptos::*;
/// view! { <div node_ref=el></div> } /// view! { <div node_ref=el></div> }
/// # } /// # }
/// ``` /// ```
/// ## Server-Side Rendering
///
/// On the server the returned signals always are `0.0` and `update` is a no-op.
pub fn use_element_bounding<El, T>(target: El) -> UseElementBoundingReturn<impl Fn() + Clone> pub fn use_element_bounding<El, T>(target: El) -> UseElementBoundingReturn<impl Fn() + Clone>
where where
El: Into<ElementMaybeSignal<T, web_sys::Element>> + Clone, El: Into<ElementMaybeSignal<T, web_sys::Element>> + Clone,
@ -46,13 +46,6 @@ where
El: Into<ElementMaybeSignal<T, web_sys::Element>> + Clone, El: Into<ElementMaybeSignal<T, web_sys::Element>> + Clone,
T: Into<web_sys::Element> + Clone + 'static, T: Into<web_sys::Element> + Clone + 'static,
{ {
let UseElementBoundingOptions {
reset,
window_resize,
window_scroll,
immediate,
} = options;
let (height, set_height) = create_signal(0.0); let (height, set_height) = create_signal(0.0);
let (width, set_width) = create_signal(0.0); let (width, set_width) = create_signal(0.0);
let (left, set_left) = create_signal(0.0); let (left, set_left) = create_signal(0.0);
@ -62,86 +55,114 @@ where
let (x, set_x) = create_signal(0.0); let (x, set_x) = create_signal(0.0);
let (y, set_y) = create_signal(0.0); let (y, set_y) = create_signal(0.0);
let target = target.into(); cfg_if! { if #[cfg(feature = "ssr")] {
let _ = target;
let _ = options;
let update = { let _ = set_height;
let target = target.clone(); let _ = set_width;
let _ = set_left;
let _ = set_right;
let _ = set_top;
let _ = set_bottom;
let _ = set_x;
let _ = set_y;
move || { let update = move || ();
let el = target.get_untracked(); } else {
use leptos::ev::{resize, scroll};
use crate::{
use_event_listener_with_options, use_resize_observer, use_window, UseEventListenerOptions,
};
if let Some(el) = el { let UseElementBoundingOptions {
let rect = el.into().get_bounding_client_rect(); reset,
window_resize,
window_scroll,
immediate,
} = options;
set_height.set(rect.height()); let target = target.into();
set_width.set(rect.width());
set_left.set(rect.x()); let update = {
set_right.set(rect.x() + rect.width()); let target = target.clone();
set_top.set(rect.y());
set_bottom.set(rect.y() + rect.height()); move || {
set_x.set(rect.x()); let el = target.get_untracked();
set_y.set(rect.y());
} else if reset { if let Some(el) = el {
set_height.set(0.0); let rect = el.into().get_bounding_client_rect();
set_width.set(0.0);
set_left.set(0.0); set_height.set(rect.height());
set_right.set(0.0); set_width.set(rect.width());
set_top.set(0.0); set_left.set(rect.x());
set_bottom.set(0.0); set_right.set(rect.x() + rect.width());
set_x.set(0.0); set_top.set(rect.y());
set_y.set(0.0); set_bottom.set(rect.y() + rect.height());
set_x.set(rect.x());
set_y.set(rect.y());
} else if reset {
set_height.set(0.0);
set_width.set(0.0);
set_left.set(0.0);
set_right.set(0.0);
set_top.set(0.0);
set_bottom.set(0.0);
set_x.set(0.0);
set_y.set(0.0);
}
} }
} };
};
use_resize_observer(target.clone(), { use_resize_observer(target.clone(), {
let update = update.clone();
move |_, _| {
update();
}
});
let _ = watch(
move || target.get(),
{
let update = update.clone(); let update = update.clone();
move |_, _, _| {
move |_, _| {
update(); update();
} }
}, });
false,
);
if window_scroll { let _ = watch(
let _ = use_event_listener_with_options( move || target.get(),
use_window(),
scroll,
{ {
let update = update.clone(); let update = update.clone();
move |_| update() move |_, _, _| {
update();
}
}, },
UseEventListenerOptions::default() false,
.capture(true)
.passive(true),
); );
}
if window_resize { if window_scroll {
let _ = use_event_listener_with_options( let _ = use_event_listener_with_options(
use_window(), use_window(),
resize, scroll,
{ {
let update = update.clone(); let update = update.clone();
move |_| update() move |_| update()
}, },
UseEventListenerOptions::default().passive(true), UseEventListenerOptions::default()
); .capture(true)
} .passive(true),
);
}
if immediate { if window_resize {
update(); let _ = use_event_listener_with_options(
} use_window(),
resize,
{
let update = update.clone();
move |_| update()
},
UseEventListenerOptions::default().passive(true),
);
}
if immediate {
update();
}
}}
UseElementBoundingReturn { UseElementBoundingReturn {
height: height.into(), height: height.into(),