2023-07-11 23:38:50 +00:00
|
|
|
use self::error::{UseSharedStateError, UseSharedStateResult};
|
2021-12-14 07:27:59 +00:00
|
|
|
use dioxus_core::{ScopeId, ScopeState};
|
2021-10-11 19:35:20 +00:00
|
|
|
use std::{
|
2023-03-02 17:42:02 +00:00
|
|
|
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
|
|
|
};
|
|
|
|
|
2023-07-12 00:53:48 +00:00
|
|
|
#[cfg(not(debug_assertions))]
|
|
|
|
type Location = ();
|
|
|
|
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
type Location = &'static std::panic::Location<'static>;
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! debug_location {
|
|
|
|
() => {{
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
{
|
|
|
|
std::panic::Location::caller()
|
|
|
|
}
|
|
|
|
#[cfg(not(debug_assertions))]
|
|
|
|
{
|
|
|
|
()
|
|
|
|
}
|
|
|
|
}};
|
|
|
|
}
|
|
|
|
|
|
|
|
pub mod diagnostics {
|
2023-07-11 23:38:50 +00:00
|
|
|
use std::panic::Location;
|
|
|
|
|
2023-07-12 00:53:48 +00:00
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
|
|
pub enum BorrowKind {
|
|
|
|
Mutable,
|
|
|
|
Immutable,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
|
|
pub struct PreviousBorrow {
|
|
|
|
pub location: Location<'static>,
|
|
|
|
pub kind: BorrowKind,
|
|
|
|
}
|
|
|
|
|
2023-07-12 14:36:17 +00:00
|
|
|
impl PreviousBorrow {
|
|
|
|
pub fn display_opt(value: &Option<Self>) -> String {
|
|
|
|
value
|
|
|
|
.as_ref()
|
|
|
|
.map(|value| value.to_string())
|
|
|
|
.unwrap_or_default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl std::fmt::Display for PreviousBorrow {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
let Self { location, kind } = self;
|
|
|
|
write!(f, "{location} ({kind:?})")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-12 00:53:48 +00:00
|
|
|
impl<T> super::UseSharedState<T> {
|
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2023-07-12 00:53:48 +00:00
|
|
|
#[allow(unused_must_use)]
|
|
|
|
pub(super) fn debug_track_borrow(&self) {
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
self.previous_borrow
|
|
|
|
.borrow_mut()
|
|
|
|
.insert(PreviousBorrow::borrowed());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2023-07-12 00:53:48 +00:00
|
|
|
#[allow(unused_must_use)]
|
|
|
|
pub(super) fn debug_track_borrow_mut(&self) {
|
2023-07-11 23:38:50 +00:00
|
|
|
#[cfg(debug_assertions)]
|
2023-07-12 00:53:48 +00:00
|
|
|
self.previous_borrow
|
|
|
|
.borrow_mut()
|
|
|
|
.insert(PreviousBorrow::borrowed_mut());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl PreviousBorrow {
|
|
|
|
#[track_caller]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[inline(never)]
|
2023-07-12 00:53:48 +00:00
|
|
|
pub fn borrowed() -> Self {
|
|
|
|
Self {
|
|
|
|
location: *Location::caller(),
|
|
|
|
kind: BorrowKind::Immutable,
|
2023-07-11 23:38:50 +00:00
|
|
|
}
|
2023-07-12 00:53:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[track_caller]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[inline(never)]
|
2023-07-12 00:53:48 +00:00
|
|
|
pub fn borrowed_mut() -> Self {
|
|
|
|
Self {
|
|
|
|
location: *Location::caller(),
|
|
|
|
kind: BorrowKind::Mutable,
|
2023-07-11 23:38:50 +00:00
|
|
|
}
|
2023-07-12 00:53:48 +00:00
|
|
|
}
|
2023-07-11 23:38:50 +00:00
|
|
|
}
|
2023-07-12 00:53:48 +00:00
|
|
|
}
|
2023-07-11 23:38:50 +00:00
|
|
|
|
2023-07-12 00:53:48 +00:00
|
|
|
pub mod error {
|
2023-07-11 23:38:50 +00:00
|
|
|
#[derive(thiserror::Error, Debug)]
|
|
|
|
pub enum UseSharedStateError {
|
2023-07-12 00:53:48 +00:00
|
|
|
#[cfg_attr(
|
|
|
|
debug_assertions,
|
|
|
|
error(
|
2023-07-12 14:36:17 +00:00
|
|
|
"[{location}] {type_name} is already borrowed at [{previous_borrow}], so it cannot be borrowed mutably.",
|
|
|
|
previous_borrow = super::diagnostics::PreviousBorrow::display_opt(.previous_borrow)
|
2023-07-12 00:53:48 +00:00
|
|
|
)
|
|
|
|
)]
|
|
|
|
#[cfg_attr(
|
|
|
|
not(debug_assertions),
|
|
|
|
error("{type_name} is already borrowed, so it cannot be borrowed mutably.")
|
|
|
|
)]
|
2023-07-11 23:38:50 +00:00
|
|
|
AlreadyBorrowed {
|
|
|
|
source: core::cell::BorrowMutError,
|
|
|
|
type_name: &'static str,
|
|
|
|
/// Only available in debug mode
|
2023-07-12 00:53:48 +00:00
|
|
|
location: super::Location,
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
previous_borrow: Option<super::diagnostics::PreviousBorrow>,
|
2023-07-11 23:38:50 +00:00
|
|
|
},
|
2023-07-12 00:53:48 +00:00
|
|
|
#[cfg_attr(
|
|
|
|
debug_assertions,
|
|
|
|
error(
|
2023-07-12 14:36:17 +00:00
|
|
|
"[{location}] {type_name} is already borrowed mutably at [{previous_borrow}], so it cannot be borrowed anymore.",
|
|
|
|
previous_borrow = super::diagnostics::PreviousBorrow::display_opt(.previous_borrow)
|
2023-07-12 00:53:48 +00:00
|
|
|
)
|
|
|
|
)]
|
|
|
|
#[cfg_attr(
|
|
|
|
not(debug_assertions),
|
|
|
|
error("{type_name} is already borrowed mutably, so it cannot be borrowed anymore.")
|
|
|
|
)]
|
2023-07-11 23:38:50 +00:00
|
|
|
AlreadyBorrowedMutably {
|
|
|
|
source: core::cell::BorrowError,
|
|
|
|
type_name: &'static str,
|
|
|
|
/// Only available in debug mode
|
2023-07-12 00:53:48 +00:00
|
|
|
location: super::Location,
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
previous_borrow: Option<super::diagnostics::PreviousBorrow>,
|
2023-07-11 23:38:50 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
pub type UseSharedStateResult<T> = Result<T, UseSharedStateError>;
|
|
|
|
}
|
|
|
|
|
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
|
2023-03-02 17:54:21 +00:00
|
|
|
pub(crate) struct ProvidedStateInner<T> {
|
|
|
|
value: T,
|
2022-02-23 18:47:17 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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
|
2022-01-29 16:43:10 +00:00
|
|
|
/// ergonomics in a pinch, with zero cost.
|
2021-10-11 19:35:20 +00:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2023-03-02 17:42:02 +00:00
|
|
|
/// ```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! {
|
2023-06-20 17:13:34 +00:00
|
|
|
/// match current_theme {
|
2023-03-02 17:42:02 +00:00
|
|
|
/// 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.
|
2023-03-02 17:42:02 +00:00
|
|
|
pub fn use_shared_state<T: 'static>(cx: &ScopeState) -> Option<&UseSharedState<T>> {
|
|
|
|
let state: &Option<UseSharedStateOwner<T>> = &*cx.use_hook(move || {
|
2022-01-02 07:15:04 +00:00
|
|
|
let scope_id = cx.scope_id();
|
2023-03-02 17:42:02 +00:00
|
|
|
let root = cx.consume_context::<ProvidedState<T>>()?;
|
2022-01-02 07:15:04 +00:00
|
|
|
|
2023-03-02 17:42:02 +00:00
|
|
|
root.borrow_mut().consumers.insert(scope_id);
|
2022-01-02 07:15:04 +00:00
|
|
|
|
2023-07-12 00:53:48 +00:00
|
|
|
let state = UseSharedState::new(root);
|
2023-03-02 17:42:02 +00:00
|
|
|
let owner = UseSharedStateOwner { state, scope_id };
|
|
|
|
Some(owner)
|
2022-01-02 07:15:04 +00:00
|
|
|
});
|
2023-03-02 17:42:02 +00:00
|
|
|
state.as_ref().map(|s| &s.state)
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|
|
|
|
|
2023-03-02 18:02:12 +00:00
|
|
|
/// This wrapper detects when the hook is dropped and will unsubscribe when the component is unmounted
|
2023-03-02 17:59:25 +00:00
|
|
|
struct UseSharedStateOwner<T> {
|
2023-03-02 17:42:02 +00:00
|
|
|
state: UseSharedState<T>,
|
2021-10-11 19:35:20 +00:00
|
|
|
scope_id: ScopeId,
|
|
|
|
}
|
2023-03-02 17:42:02 +00:00
|
|
|
|
|
|
|
impl<T> Drop for UseSharedStateOwner<T> {
|
2021-11-07 03:11:17 +00:00
|
|
|
fn drop(&mut self) {
|
2023-03-02 17:42:02 +00:00
|
|
|
// we need to unsubscribe when our component is unmounted
|
2023-03-02 17:54:21 +00:00
|
|
|
let mut root = self.state.inner.borrow_mut();
|
2023-03-02 17:42:02 +00:00
|
|
|
root.consumers.remove(&self.scope_id);
|
2021-11-07 03:11:17 +00:00
|
|
|
}
|
|
|
|
}
|
2021-10-11 19:35:20 +00:00
|
|
|
|
2023-03-02 18:02:12 +00:00
|
|
|
/// State that is shared between components through the context system
|
2023-03-02 17:42:02 +00:00
|
|
|
pub struct UseSharedState<T> {
|
2023-03-02 17:54:21 +00:00
|
|
|
pub(crate) inner: Rc<RefCell<ProvidedStateInner<T>>>,
|
2023-07-12 00:53:48 +00:00
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
previous_borrow: Rc<RefCell<Option<diagnostics::PreviousBorrow>>>,
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|
|
|
|
|
2023-03-02 17:42:02 +00:00
|
|
|
impl<T> UseSharedState<T> {
|
2023-07-12 00:53:48 +00:00
|
|
|
fn new(inner: Rc<RefCell<ProvidedStateInner<T>>>) -> Self {
|
|
|
|
Self {
|
|
|
|
inner,
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
previous_borrow: Default::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-02 18:02:12 +00:00
|
|
|
/// Notify all consumers of the state that it has changed. (This is called automatically when you call "write")
|
2023-03-02 17:42:02 +00:00
|
|
|
pub fn notify_consumers(&self) {
|
2023-03-02 17:54:21 +00:00
|
|
|
self.inner.borrow_mut().notify_consumers();
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|
|
|
|
|
2023-07-11 19:41:52 +00:00
|
|
|
/// Try reading the shared state
|
2023-07-11 23:53:00 +00:00
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2023-07-11 19:41:52 +00:00
|
|
|
pub fn try_read(&self) -> UseSharedStateResult<Ref<'_, T>> {
|
2023-07-12 14:20:19 +00:00
|
|
|
match self.inner.try_borrow() {
|
|
|
|
Ok(value) => {
|
|
|
|
self.debug_track_borrow();
|
|
|
|
Ok(Ref::map(value, |inner| &inner.value))
|
|
|
|
}
|
|
|
|
Err(source) => Err(UseSharedStateError::AlreadyBorrowedMutably {
|
2023-07-11 19:41:52 +00:00
|
|
|
source,
|
|
|
|
type_name: std::any::type_name::<Self>(),
|
2023-07-12 00:53:48 +00:00
|
|
|
location: debug_location!(),
|
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
previous_borrow: *self.previous_borrow.borrow(),
|
2023-07-12 14:20:19 +00:00
|
|
|
}),
|
|
|
|
}
|
2023-07-11 19:41:52 +00:00
|
|
|
}
|
2023-07-11 20:54:24 +00:00
|
|
|
|
2023-03-02 18:02:12 +00:00
|
|
|
/// Read the shared value
|
2023-07-11 23:53:00 +00:00
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2023-03-02 17:54:21 +00:00
|
|
|
pub fn read(&self) -> Ref<'_, T> {
|
2023-07-11 19:41:52 +00:00
|
|
|
match self.try_read() {
|
|
|
|
Ok(value) => value,
|
|
|
|
Err(message) => panic!(
|
|
|
|
"Reading the shared state failed: {}\n({:?})",
|
|
|
|
message, message
|
|
|
|
),
|
|
|
|
}
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|
|
|
|
|
2023-07-11 19:41:52 +00:00
|
|
|
/// Try writing the shared state
|
2023-07-11 23:53:00 +00:00
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2023-07-11 19:41:52 +00:00
|
|
|
pub fn try_write(&self) -> UseSharedStateResult<RefMut<'_, T>> {
|
2023-07-12 14:20:19 +00:00
|
|
|
match self.inner.try_borrow_mut() {
|
|
|
|
Ok(mut value) => {
|
|
|
|
self.debug_track_borrow_mut();
|
|
|
|
value.notify_consumers();
|
|
|
|
Ok(RefMut::map(value, |inner| &mut inner.value))
|
|
|
|
}
|
|
|
|
Err(source) => Err(UseSharedStateError::AlreadyBorrowed {
|
2023-07-11 19:41:52 +00:00
|
|
|
source,
|
|
|
|
type_name: std::any::type_name::<Self>(),
|
2023-07-11 23:38:50 +00:00
|
|
|
location: crate::debug_location!(),
|
2023-07-12 00:53:48 +00:00
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
previous_borrow: *self.previous_borrow.borrow(),
|
2023-07-12 14:20:19 +00:00
|
|
|
}),
|
|
|
|
}
|
2023-07-11 19:41:52 +00:00
|
|
|
}
|
2023-07-11 20:54:24 +00:00
|
|
|
|
2021-10-11 19:35:20 +00:00
|
|
|
/// Calling "write" will force the component to re-render
|
|
|
|
///
|
|
|
|
///
|
2023-03-02 18:02:12 +00:00
|
|
|
// TODO: We prevent unncessary notifications only in the hook, but we should figure out some more global lock
|
2023-07-11 23:53:00 +00:00
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2021-11-03 19:13:50 +00:00
|
|
|
pub fn write(&self) -> RefMut<'_, T> {
|
2023-07-11 19:41:52 +00:00
|
|
|
match self.try_write() {
|
|
|
|
Ok(value) => value,
|
|
|
|
Err(message) => panic!(
|
|
|
|
"Writing to shared state failed: {}\n({:?})",
|
|
|
|
message, message
|
|
|
|
),
|
|
|
|
}
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|
|
|
|
|
2023-07-11 19:41:52 +00:00
|
|
|
/// Tries writing the value without forcing a re-render
|
2023-07-11 23:53:00 +00:00
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2023-07-11 19:41:52 +00:00
|
|
|
pub fn try_write_silent(&self) -> UseSharedStateResult<RefMut<'_, T>> {
|
2023-07-12 14:20:19 +00:00
|
|
|
match self.inner.try_borrow_mut() {
|
|
|
|
Ok(value) => {
|
|
|
|
self.debug_track_borrow_mut();
|
|
|
|
Ok(RefMut::map(value, |inner| &mut inner.value))
|
|
|
|
}
|
|
|
|
Err(source) => Err(UseSharedStateError::AlreadyBorrowed {
|
2023-07-11 19:41:52 +00:00
|
|
|
source,
|
|
|
|
type_name: std::any::type_name::<Self>(),
|
2023-07-11 23:38:50 +00:00
|
|
|
location: crate::debug_location!(),
|
2023-07-12 00:53:48 +00:00
|
|
|
#[cfg(debug_assertions)]
|
|
|
|
previous_borrow: *self.previous_borrow.borrow(),
|
2023-07-12 14:20:19 +00:00
|
|
|
}),
|
|
|
|
}
|
2023-07-11 19:41:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Writes the value without forcing a re-render
|
2023-07-11 23:53:00 +00:00
|
|
|
#[cfg_attr(debug_assertions, track_caller)]
|
2023-07-12 10:27:17 +00:00
|
|
|
#[cfg_attr(debug_assertions, inline(never))]
|
2021-10-11 19:35:20 +00:00
|
|
|
pub fn write_silent(&self) -> RefMut<'_, T> {
|
2023-07-11 19:41:52 +00:00
|
|
|
match self.try_write_silent() {
|
|
|
|
Ok(value) => value,
|
|
|
|
Err(message) => panic!(
|
|
|
|
"Writing to shared state silently failed: {}\n({:?})",
|
|
|
|
message, message
|
|
|
|
),
|
|
|
|
}
|
2021-12-10 02:19:31 +00:00
|
|
|
}
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|
|
|
|
|
2023-03-02 17:42:02 +00:00
|
|
|
impl<T> Clone for UseSharedState<T> {
|
2021-10-11 19:35:20 +00:00
|
|
|
fn clone(&self) -> Self {
|
2023-03-02 17:42:02 +00:00
|
|
|
Self {
|
2023-03-02 17:54:21 +00:00
|
|
|
inner: self.inner.clone(),
|
2023-07-12 00:53:48 +00:00
|
|
|
#[cfg(debug_assertions)]
|
2023-07-12 14:24:39 +00:00
|
|
|
previous_borrow: self.previous_borrow.clone(),
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-02 17:42:02 +00:00
|
|
|
impl<T: PartialEq> PartialEq for UseSharedState<T> {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
2023-03-02 17:54:21 +00:00
|
|
|
let first = self.inner.borrow();
|
|
|
|
let second = other.inner.borrow();
|
|
|
|
first.value == second.value
|
2023-03-02 17:42:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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) {
|
2022-07-11 19:50:56 +00:00
|
|
|
cx.use_hook(|| {
|
2022-03-02 22:57:57 +00:00
|
|
|
let state: ProvidedState<T> = Rc::new(RefCell::new(ProvidedStateInner {
|
2023-03-02 17:54:21 +00:00
|
|
|
value: f(),
|
2022-01-02 07:15:04 +00:00
|
|
|
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);
|
2022-01-02 07:15:04 +00:00
|
|
|
});
|
2021-10-11 19:35:20 +00:00
|
|
|
}
|