use dioxus_core::{ScopeId, ScopeState}; use std::{ cell::{Cell, Ref, RefCell, RefMut}, collections::HashSet, rc::Rc, sync::Arc, }; type ProvidedState = Rc>>; // Tracks all the subscribers to a shared State pub struct ProvidedStateInner { value: Rc>, notify_any: Arc, consumers: HashSet, } impl ProvidedStateInner { pub(crate) fn notify_consumers(&mut self) { for consumer in self.consumers.iter() { (self.notify_any)(*consumer); } } pub fn write(&self) -> RefMut { self.value.borrow_mut() } pub fn read(&self) -> Ref { self.value.borrow() } } /// 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 /// /// ## Provider /// /// ```rust, ignore /// /// /// ``` /// /// ## Consumer /// /// ```rust, ignore /// /// /// ``` /// /// # 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> { let state = cx.use_hook(|| { let scope_id = cx.scope_id(); let root = cx.consume_context::>(); if let Some(root) = root.as_ref() { root.borrow_mut().consumers.insert(scope_id); } let value = root.as_ref().map(|f| f.borrow().value.clone()); SharedStateInner { root, value, scope_id, needs_notification: Cell::new(false), } }); state.needs_notification.set(false); match (&state.value, &state.root) { (Some(value), Some(root)) => Some(UseSharedState { cx, value, root, needs_notification: &state.needs_notification, }), _ => None, } } struct SharedStateInner { root: Option>, value: Option>>, scope_id: ScopeId, needs_notification: Cell, } impl Drop for SharedStateInner { fn drop(&mut self) { // we need to unsubscribe when our component is unounted if let Some(root) = &self.root { let mut root = root.borrow_mut(); root.consumers.remove(&self.scope_id); } } } pub struct UseSharedState<'a, T: 'static> { pub(crate) cx: &'a ScopeState, pub(crate) value: &'a Rc>, pub(crate) root: &'a Rc>>, pub(crate) needs_notification: &'a Cell, } impl<'a, T: 'static> UseSharedState<'a, T> { pub fn read(&self) -> Ref<'_, T> { self.value.borrow() } pub fn notify_consumers(self) { if !self.needs_notification.get() { self.root.borrow_mut().notify_consumers(); self.needs_notification.set(true); } } pub fn read_write(&self) -> (Ref<'_, T>, &Self) { (self.read(), self) } /// 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> { self.cx.needs_update(); self.notify_consumers(); self.value.borrow_mut() } /// Allows the ability to write the value without forcing a re-render pub fn write_silent(&self) -> RefMut<'_, T> { self.value.borrow_mut() } pub fn inner(&self) -> Rc>> { self.root.clone() } } impl Copy for UseSharedState<'_, T> {} impl<'a, T> Clone for UseSharedState<'a, T> where T: 'static, { fn clone(&self) -> Self { UseSharedState { cx: self.cx, value: self.value, root: self.root, needs_notification: self.needs_notification, } } } /// Provide some state for components down the hierarchy to consume without having to drill props. pub fn use_shared_state_provider(cx: &ScopeState, f: impl FnOnce() -> T) { cx.use_hook(|| { let state: ProvidedState = Rc::new(RefCell::new(ProvidedStateInner { value: Rc::new(RefCell::new(f())), notify_any: cx.schedule_update_any(), consumers: HashSet::new(), })); cx.provide_context(state); }); }