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`] pub fn use_media_query(cx: Scope, query: impl Into>) -> Signal { let query = query.into(); let (matches, set_matches) = create_signal(cx, false); let media_query: Rc>> = Rc::new(RefCell::new(None)); let remove_listener: RemoveListener = Rc::new(RefCell::new(None)); let listener: Rc>>> = Rc::new(OnceCell::new()); let cleanup = { let remove_listener = Rc::clone(&remove_listener); move || { if let Some(remove_listener) = remove_listener.take().as_ref() { remove_listener(); } } }; 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(media_query.matches()); 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(false); } } }; { let update = update.clone(); listener .set(Box::new(move |_| update()) as Box>) .expect("cell is empty"); } create_effect(cx, move |_| update()); on_cleanup(cx, cleanup); matches.into() } type RemoveListener = Rc>>>;