use dioxus_core::{ScopeId, ScopeState}; use std::{ cell::{Ref, RefCell, RefMut}, collections::HashSet, rc::Rc, sync::Arc, }; type ProvidedState = Rc>>; // Tracks all the subscribers to a shared State pub(crate) struct ProvidedStateInner { value: T, notify_any: Arc, consumers: HashSet, } impl ProvidedStateInner { pub(crate) fn notify_consumers(&mut self) { for consumer in self.consumers.iter() { (self.notify_any)(*consumer); } } } /// This hook provides some relatively light ergonomics around shared state. /// /// It is not a substitute for a proper state management system, but it is capable enough to provide use_state - type /// ergonomics in a pinch, with zero cost. /// /// # Example /// /// ```rust /// # use dioxus::prelude::*; /// # /// # fn app(cx: Scope) -> Element { /// # render! { /// # Parent{} /// # } /// # } /// /// #[derive(Clone, Copy)] /// enum Theme { /// Light, /// Dark, /// } /// /// // Provider /// fn Parent<'a>(cx: Scope<'a>) -> Element<'a> { /// use_shared_state_provider(cx, || Theme::Dark); /// let theme = use_shared_state::(cx).unwrap(); /// /// render! { /// button{ /// onclick: move |_| { /// let current_theme = *theme.read(); /// *theme.write() = match current_theme { /// Theme::Dark => Theme::Light, /// Theme::Light => Theme::Dark, /// }; /// }, /// "Change theme" /// } /// Child{} /// } /// } /// /// // Consumer /// fn Child<'a>(cx: Scope<'a>) -> Element<'a> { /// let theme = use_shared_state::(cx).unwrap(); /// let current_theme = *theme.read(); /// /// render! { /// match &*theme.read() { /// Theme::Dark => { /// "Dark mode" /// } /// Theme::Light => { /// "Light mode" /// } /// } /// } /// } /// ``` /// /// # How it works /// /// Any time a component calls `write`, every consumer of the state will be notified - excluding the provider. /// /// Right now, there is not a distinction between read-only and write-only, so every consumer will be notified. pub fn use_shared_state(cx: &ScopeState) -> Option<&UseSharedState> { let state: &Option> = &*cx.use_hook(move || { let scope_id = cx.scope_id(); let root = cx.consume_context::>()?; root.borrow_mut().consumers.insert(scope_id); let state = UseSharedState { inner: root }; let owner = UseSharedStateOwner { state, scope_id }; Some(owner) }); state.as_ref().map(|s| &s.state) } struct UseSharedStateOwner { state: UseSharedState, scope_id: ScopeId, } impl Drop for UseSharedStateOwner { fn drop(&mut self) { // we need to unsubscribe when our component is unmounted let mut root = self.state.inner.borrow_mut(); root.consumers.remove(&self.scope_id); } } pub struct UseSharedState { pub(crate) inner: Rc>>, } impl UseSharedState { pub fn notify_consumers(&self) { self.inner.borrow_mut().notify_consumers(); } pub fn read(&self) -> Ref<'_, T> { Ref::map(self.inner.borrow(), |inner| &inner.value) } /// Calling "write" will force the component to re-render /// /// /// TODO: We prevent unncessary notifications only in the hook, but we should figure out some more global lock pub fn write(&self) -> RefMut<'_, T> { let mut value = self.inner.borrow_mut(); value.notify_consumers(); RefMut::map(value, |inner| &mut inner.value) } /// Allows the ability to write the value without forcing a re-render pub fn write_silent(&self) -> RefMut<'_, T> { RefMut::map(self.inner.borrow_mut(), |inner| &mut inner.value) } } impl Clone for UseSharedState { fn clone(&self) -> Self { Self { inner: self.inner.clone(), } } } impl PartialEq for UseSharedState { fn eq(&self, other: &Self) -> bool { let first = self.inner.borrow(); let second = other.inner.borrow(); first.value == second.value } } /// Provide some state for components down the hierarchy to consume without having to drill props. See [`use_shared_state`] to consume the state /// /// /// # Example /// /// ```rust /// # use dioxus::prelude::*; /// # /// # fn app(cx: Scope) -> Element { /// # render! { /// # Parent{} /// # } /// # } /// /// #[derive(Clone, Copy)] /// enum Theme { /// Light, /// Dark, /// } /// /// // Provider /// fn Parent<'a>(cx: Scope<'a>) -> Element<'a> { /// use_shared_state_provider(cx, || Theme::Dark); /// let theme = use_shared_state::(cx).unwrap(); /// /// render! { /// button{ /// onclick: move |_| { /// let current_theme = *theme.read(); /// *theme.write() = match current_theme { /// Theme::Dark => Theme::Light, /// Theme::Light => Theme::Dark, /// }; /// }, /// "Change theme" /// } /// // Children components that consume the state... /// } /// } /// ``` pub fn use_shared_state_provider(cx: &ScopeState, f: impl FnOnce() -> T) { cx.use_hook(|| { let state: ProvidedState = Rc::new(RefCell::new(ProvidedStateInner { value: f(), notify_any: cx.schedule_update_any(), consumers: HashSet::new(), })); cx.provide_context(state); }); }