dioxus/packages/hooks/src/use_shared_state.rs

230 lines
6 KiB
Rust
Raw Normal View History

use dioxus_core::{ScopeId, ScopeState};
2021-10-11 19:35:20 +00:00
use std::{
cell::{Ref, RefCell, RefMut},
2021-10-11 19:35:20 +00:00
collections::HashSet,
2022-02-23 19:00:01 +00:00
rc::Rc,
sync::Arc,
2021-10-11 19:35:20 +00:00
};
2022-03-02 22:57:57 +00:00
type ProvidedState<T> = Rc<RefCell<ProvidedStateInner<T>>>;
2021-10-11 19:35:20 +00:00
// Tracks all the subscribers to a shared State
2021-12-10 02:19:31 +00:00
pub struct ProvidedStateInner<T> {
2021-10-11 19:35:20 +00:00
value: Rc<RefCell<T>>,
notify_any: Arc<dyn Fn(ScopeId)>,
2021-10-11 19:35:20 +00:00
consumers: HashSet<ScopeId>,
}
impl<T> ProvidedStateInner<T> {
pub(crate) fn notify_consumers(&mut self) {
for consumer in self.consumers.iter() {
(self.notify_any)(*consumer);
}
}
2021-12-10 02:19:31 +00:00
pub fn write(&self) -> RefMut<T> {
self.value.borrow_mut()
}
pub fn read(&self) -> Ref<T> {
self.value.borrow()
}
2021-10-11 19:35:20 +00:00
}
/// 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.
2021-10-11 19:35:20 +00:00
///
/// # 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::<Theme>(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::<Theme>(cx).unwrap();
/// let current_theme = *theme.read();
///
/// render! {
/// match &*theme.read() {
/// Theme::Dark => {
/// "Dark mode"
/// }
/// Theme::Light => {
/// "Light mode"
/// }
/// }
/// }
/// }
2021-10-11 19:35:20 +00:00
/// ```
///
/// # 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<T: 'static>(cx: &ScopeState) -> Option<&UseSharedState<T>> {
let state: &Option<UseSharedStateOwner<T>> = &*cx.use_hook(move || {
let scope_id = cx.scope_id();
let root = cx.consume_context::<ProvidedState<T>>()?;
root.borrow_mut().consumers.insert(scope_id);
let value = root.borrow().value.clone();
let state = UseSharedState { value, root };
let owner = UseSharedStateOwner { state, scope_id };
Some(owner)
});
state.as_ref().map(|s| &s.state)
2021-10-11 19:35:20 +00:00
}
struct UseSharedStateOwner<T: 'static> {
state: UseSharedState<T>,
2021-10-11 19:35:20 +00:00
scope_id: ScopeId,
}
impl<T> Drop for UseSharedStateOwner<T> {
2021-11-07 03:11:17 +00:00
fn drop(&mut self) {
// we need to unsubscribe when our component is unmounted
let mut root = self.state.root.borrow_mut();
root.consumers.remove(&self.scope_id);
2021-11-07 03:11:17 +00:00
}
}
2021-10-11 19:35:20 +00:00
pub struct UseSharedState<T> {
pub(crate) value: Rc<RefCell<T>>,
pub(crate) root: Rc<RefCell<ProvidedStateInner<T>>>,
2021-10-11 19:35:20 +00:00
}
impl<T> UseSharedState<T> {
2021-10-11 19:35:20 +00:00
pub fn read(&self) -> Ref<'_, T> {
self.value.borrow()
}
pub fn notify_consumers(&self) {
self.root.borrow_mut().notify_consumers();
2021-10-11 19:35:20 +00:00
}
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
2021-11-03 19:13:50 +00:00
pub fn write(&self) -> RefMut<'_, T> {
2021-10-11 19:35:20 +00:00
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()
}
2021-12-10 02:19:31 +00:00
pub fn inner(&self) -> Rc<RefCell<ProvidedStateInner<T>>> {
self.root.clone()
}
2021-10-11 19:35:20 +00:00
}
impl<T> Clone for UseSharedState<T> {
2021-10-11 19:35:20 +00:00
fn clone(&self) -> Self {
Self {
value: self.value.clone(),
root: self.root.clone(),
2021-10-11 19:35:20 +00:00
}
}
}
impl<T: PartialEq> PartialEq for UseSharedState<T> {
fn eq(&self, other: &Self) -> bool {
self.value == other.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::<Theme>(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...
/// }
/// }
/// ```
2022-12-07 23:11:51 +00:00
pub fn use_shared_state_provider<T: 'static>(cx: &ScopeState, f: impl FnOnce() -> T) {
cx.use_hook(|| {
2022-03-02 22:57:57 +00:00
let state: ProvidedState<T> = Rc::new(RefCell::new(ProvidedStateInner {
value: Rc::new(RefCell::new(f())),
notify_any: cx.schedule_update_any(),
consumers: HashSet::new(),
2022-03-02 22:57:57 +00:00
}));
2022-12-03 00:24:49 +00:00
cx.provide_context(state);
});
2021-10-11 19:35:20 +00:00
}