leptos-use/src/use_media_query.rs

99 lines
2.7 KiB
Rust
Raw Normal View History

2023-06-10 03:19:00 +01:00
use crate::use_event_listener;
use crate::utils::CloneableFnMutWithArg;
use leptos::ev::change;
use leptos::*;
use std::cell::{OnceCell, RefCell};
use std::rc::Rc;
/// Reactive [Media Query](https://developer.mozilla.org/en-US/docs/Web/CSS/Media_Queries/Testing_media_queries).
///
/// ## Demo
///
/// [Link to Demo](https://github.com/Synphonyte/leptos-use/tree/main/examples/use_media_query)
///
/// ## Usage
///
/// ```
/// # use leptos::*;
/// # use leptos_use::use_media_query;
/// #
/// # #[component]
/// # fn Demo(cx: Scope) -> impl IntoView {
/// #
/// let is_large_screen = use_media_query(cx, "(min-width: 1024px)");
///
/// let is_dark_preferred = use_media_query(cx, "(prefers-color-scheme: dark)");
/// #
/// # view! { cx, }
/// # }
/// ```
///
/// ## See also
///
/// * [`use_preferred_dark`]
/// * [`use_preferred_contrast`]
2023-06-10 03:19:00 +01:00
pub fn use_media_query(cx: Scope, query: impl Into<MaybeSignal<String>>) -> Signal<bool> {
let query = query.into();
let (matches, set_matches) = create_signal(cx, false);
let media_query: Rc<RefCell<Option<web_sys::MediaQueryList>>> = Rc::new(RefCell::new(None));
2023-06-10 19:43:51 +01:00
let remove_listener: RemoveListener = Rc::new(RefCell::new(None));
2023-06-10 03:19:00 +01:00
let listener: Rc<OnceCell<Box<dyn CloneableFnMutWithArg<web_sys::Event>>>> =
Rc::new(OnceCell::new());
2023-06-13 17:48:32 +01:00
let cleanup = {
let remove_listener = Rc::clone(&remove_listener);
move || {
if let Some(remove_listener) = remove_listener.take().as_ref() {
remove_listener();
}
2023-06-10 03:19:00 +01:00
}
};
2023-06-13 17:48:32 +01:00
let update = {
let cleanup = cleanup.clone();
let listener = Rc::clone(&listener);
move || {
cleanup();
let mut media_query = media_query.borrow_mut();
*media_query = window().match_media(&query.get()).unwrap_or(None);
if let Some(media_query) = media_query.as_ref() {
set_matches.set(media_query.matches());
2023-06-13 17:48:32 +01:00
remove_listener.replace(Some(Box::new(use_event_listener(
cx,
media_query.clone(),
change,
listener
.get()
.expect("cell should be initialized by now")
.clone(),
))));
} else {
set_matches.set(false);
2023-06-13 17:48:32 +01:00
}
2023-06-10 03:19:00 +01:00
}
};
2023-06-13 17:48:32 +01:00
{
let update = update.clone();
listener
.set(Box::new(move |_| update()) as Box<dyn CloneableFnMutWithArg<web_sys::Event>>)
.expect("cell is empty");
}
2023-06-10 03:19:00 +01:00
create_effect(cx, move |_| update());
on_cleanup(cx, cleanup);
matches.into()
}
2023-06-10 19:43:51 +01:00
type RemoveListener = Rc<RefCell<Option<Box<dyn Fn()>>>>;