|
@@ -8,25 +8,26 @@ use std::{
|
|
};
|
|
};
|
|
|
|
|
|
/// Create a new tracked state.
|
|
/// Create a new tracked state.
|
|
-/// Tracked state is state that can drive computed state
|
|
|
|
|
|
+/// Tracked state is state that can drive Selector state
|
|
///
|
|
///
|
|
-/// It state will efficiently update any Computed state that is reading from it, but it is not readable on it's own.
|
|
|
|
|
|
+/// It state will efficiently update any Selector state that is reading from it, but it is not readable on it's own.
|
|
///
|
|
///
|
|
/// ```rust
|
|
/// ```rust
|
|
/// use dioxus::prelude::*;
|
|
/// use dioxus::prelude::*;
|
|
///
|
|
///
|
|
/// fn Parent(cx: Scope) -> Element {
|
|
/// fn Parent(cx: Scope) -> Element {
|
|
/// let count = use_tracked_state(cx, || 0);
|
|
/// let count = use_tracked_state(cx, || 0);
|
|
|
|
+///
|
|
/// render! {
|
|
/// render! {
|
|
/// Child {
|
|
/// Child {
|
|
-/// less_than_five: count.compute(|count| *count < 5),
|
|
|
|
|
|
+/// count: count.clone(),
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// }
|
|
///
|
|
///
|
|
/// #[inline_props]
|
|
/// #[inline_props]
|
|
-/// fn Child(cx: Scope, less_than_five: Computed<bool, usize>) -> Element {
|
|
|
|
-/// let less_than_five = less_than_five.use_state(cx);
|
|
|
|
|
|
+/// fn Child(cx: Scope, count: Tracked<usize>) -> Element {
|
|
|
|
+/// let less_than_five = use_selector(cx, count, |count| *count < 5);
|
|
///
|
|
///
|
|
/// render! {
|
|
/// render! {
|
|
/// "{less_than_five}"
|
|
/// "{less_than_five}"
|
|
@@ -40,14 +41,14 @@ pub fn use_tracked_state<T: 'static>(cx: &ScopeState, init: impl FnOnce() -> T)
|
|
})
|
|
})
|
|
}
|
|
}
|
|
|
|
|
|
-/// Tracked state is state that can drive computed state
|
|
|
|
|
|
+/// Tracked state is state that can drive Selector state
|
|
///
|
|
///
|
|
-/// Tracked state will efficiently update any Computed state that is reading from it, but it is not readable on it's own.
|
|
|
|
|
|
+/// Tracked state will efficiently update any Selector state that is reading from it, but it is not readable on it's own.
|
|
#[derive(Clone)]
|
|
#[derive(Clone)]
|
|
pub struct Tracked<I> {
|
|
pub struct Tracked<I> {
|
|
state: Rc<RefCell<I>>,
|
|
state: Rc<RefCell<I>>,
|
|
update_any: std::sync::Arc<dyn Fn(ScopeId)>,
|
|
update_any: std::sync::Arc<dyn Fn(ScopeId)>,
|
|
- subscribers: std::rc::Rc<std::cell::RefCell<Slab<Box<dyn FnMut(&I) + 'static>>>>,
|
|
|
|
|
|
+ subscribers: SubscribedCallbacks<I>,
|
|
}
|
|
}
|
|
|
|
|
|
impl<I: PartialEq> PartialEq for Tracked<I> {
|
|
impl<I: PartialEq> PartialEq for Tracked<I> {
|
|
@@ -67,16 +68,16 @@ impl<I> Tracked<I> {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- /// Create a new computed state from this tracked state
|
|
|
|
|
|
+ /// Create a new Selector state from this tracked state
|
|
pub fn compute<O: PartialEq + 'static>(
|
|
pub fn compute<O: PartialEq + 'static>(
|
|
&self,
|
|
&self,
|
|
mut compute: impl FnMut(&I) -> O + 'static,
|
|
mut compute: impl FnMut(&I) -> O + 'static,
|
|
- ) -> Computed<O, I> {
|
|
|
|
|
|
+ ) -> Selector<O, I> {
|
|
let subscribers = Rc::new(RefCell::new(HashSet::new()));
|
|
let subscribers = Rc::new(RefCell::new(HashSet::new()));
|
|
let state = Rc::new(RefCell::new(compute(&self.state.borrow())));
|
|
let state = Rc::new(RefCell::new(compute(&self.state.borrow())));
|
|
let update_any = self.update_any.clone();
|
|
let update_any = self.update_any.clone();
|
|
|
|
|
|
- Computed {
|
|
|
|
|
|
+ Selector {
|
|
value: state.clone(),
|
|
value: state.clone(),
|
|
subscribers: subscribers.clone(),
|
|
subscribers: subscribers.clone(),
|
|
_tracker: Rc::new(self.track(move |input_state| {
|
|
_tracker: Rc::new(self.track(move |input_state| {
|
|
@@ -88,7 +89,7 @@ impl<I> Tracked<I> {
|
|
if different {
|
|
if different {
|
|
let mut state = state.borrow_mut();
|
|
let mut state = state.borrow_mut();
|
|
*state = new;
|
|
*state = new;
|
|
- for id in subscribers.borrow_mut().drain() {
|
|
|
|
|
|
+ for id in subscribers.borrow().iter().copied() {
|
|
(update_any)(id);
|
|
(update_any)(id);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
@@ -117,7 +118,7 @@ impl<I> Tracked<I> {
|
|
/// A mutable reference to tracked state
|
|
/// A mutable reference to tracked state
|
|
pub struct TrackedMut<'a, I> {
|
|
pub struct TrackedMut<'a, I> {
|
|
state: RefMut<'a, I>,
|
|
state: RefMut<'a, I>,
|
|
- subscribers: std::rc::Rc<std::cell::RefCell<Slab<Box<dyn FnMut(&I) + 'static>>>>,
|
|
|
|
|
|
+ subscribers: SubscribedCallbacks<I>,
|
|
}
|
|
}
|
|
|
|
|
|
impl<'a, I> Deref for TrackedMut<'a, I> {
|
|
impl<'a, I> Deref for TrackedMut<'a, I> {
|
|
@@ -143,8 +144,10 @@ impl<'a, I> Drop for TrackedMut<'a, I> {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+type SubscribedCallbacks<I> = std::rc::Rc<std::cell::RefCell<Slab<Box<dyn FnMut(&I) + 'static>>>>;
|
|
|
|
+
|
|
pub(crate) struct Tracker<I> {
|
|
pub(crate) struct Tracker<I> {
|
|
- subscribers: std::rc::Rc<std::cell::RefCell<Slab<Box<dyn FnMut(&I) + 'static>>>>,
|
|
|
|
|
|
+ subscribers: SubscribedCallbacks<I>,
|
|
id: usize,
|
|
id: usize,
|
|
}
|
|
}
|
|
|
|
|
|
@@ -154,24 +157,33 @@ impl<I> Drop for Tracker<I> {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-/// Computed state is state that is derived from tracked state
|
|
|
|
|
|
+pub fn use_selector<I: 'static, O: Clone + PartialEq + 'static>(
|
|
|
|
+ cx: &ScopeState,
|
|
|
|
+ tracked: &Tracked<I>,
|
|
|
|
+ init: impl FnMut(&I) -> O + 'static,
|
|
|
|
+) -> O {
|
|
|
|
+ let selector = cx.use_hook(|| tracked.compute(init));
|
|
|
|
+ selector.use_state(cx)
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/// Selector state is state that is derived from tracked state
|
|
///
|
|
///
|
|
-/// Whenever the tracked state changes, the computed state will be updated and any components reading from it will be rerun
|
|
|
|
|
|
+/// Whenever the tracked state changes, the Selector state will be updated and any components reading from it will be rerun
|
|
#[derive(Clone)]
|
|
#[derive(Clone)]
|
|
-pub struct Computed<T, I> {
|
|
|
|
|
|
+pub struct Selector<T, I> {
|
|
_tracker: Rc<Tracker<I>>,
|
|
_tracker: Rc<Tracker<I>>,
|
|
value: Rc<RefCell<T>>,
|
|
value: Rc<RefCell<T>>,
|
|
- subscribers: std::rc::Rc<std::cell::RefCell<std::collections::HashSet<ScopeId>>>,
|
|
|
|
|
|
+ subscribers: Rc<RefCell<HashSet<ScopeId>>>,
|
|
}
|
|
}
|
|
|
|
|
|
-impl<T, I> PartialEq for Computed<T, I> {
|
|
|
|
|
|
+impl<T, I> PartialEq for Selector<T, I> {
|
|
fn eq(&self, other: &Self) -> bool {
|
|
fn eq(&self, other: &Self) -> bool {
|
|
std::rc::Rc::ptr_eq(&self.value, &other.value)
|
|
std::rc::Rc::ptr_eq(&self.value, &other.value)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-impl<T: Clone + PartialEq, I> Computed<T, I> {
|
|
|
|
- /// Read the computed state and subscribe to updates
|
|
|
|
|
|
+impl<T: Clone + PartialEq, I> Selector<T, I> {
|
|
|
|
+ /// Read the Selector state and subscribe to updates
|
|
pub fn use_state(&self, cx: &ScopeState) -> T {
|
|
pub fn use_state(&self, cx: &ScopeState) -> T {
|
|
cx.use_hook(|| {
|
|
cx.use_hook(|| {
|
|
let id = cx.scope_id();
|
|
let id = cx.scope_id();
|