|
@@ -0,0 +1,244 @@
|
|
|
+use std::{
|
|
|
+ cell::{Ref, RefCell, RefMut},
|
|
|
+ ops::{Deref, DerefMut},
|
|
|
+ rc::Rc,
|
|
|
+ sync::Arc,
|
|
|
+};
|
|
|
+
|
|
|
+use dioxus_core::{
|
|
|
+ prelude::{current_scope_id, has_context, provide_context, schedule_update_any},
|
|
|
+ ScopeId, ScopeState,
|
|
|
+};
|
|
|
+
|
|
|
+use crate::{CopyValue, Effect};
|
|
|
+
|
|
|
+pub fn use_signal<T: 'static>(cx: &ScopeState, f: impl FnOnce() -> T) -> Signal<T> {
|
|
|
+ *cx.use_hook(|| Signal::new(f()))
|
|
|
+}
|
|
|
+
|
|
|
+#[derive(Clone)]
|
|
|
+struct Unsubscriber {
|
|
|
+ scope: ScopeId,
|
|
|
+ subscribers: Rc<RefCell<Vec<Rc<RefCell<Vec<ScopeId>>>>>>,
|
|
|
+}
|
|
|
+
|
|
|
+impl Drop for Unsubscriber {
|
|
|
+ fn drop(&mut self) {
|
|
|
+ for subscribers in self.subscribers.borrow().iter() {
|
|
|
+ subscribers.borrow_mut().retain(|s| *s != self.scope);
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+fn current_unsubscriber() -> Unsubscriber {
|
|
|
+ match has_context() {
|
|
|
+ Some(rt) => rt,
|
|
|
+ None => {
|
|
|
+ let owner = Unsubscriber {
|
|
|
+ scope: current_scope_id().expect("in a virtual dom"),
|
|
|
+ subscribers: Default::default(),
|
|
|
+ };
|
|
|
+ provide_context(owner).expect("in a virtual dom")
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+pub(crate) struct SignalData<T> {
|
|
|
+ pub(crate) subscribers: Rc<RefCell<Vec<ScopeId>>>,
|
|
|
+ pub(crate) effect_subscribers: Rc<RefCell<Vec<Effect>>>,
|
|
|
+ pub(crate) update_any: Arc<dyn Fn(ScopeId)>,
|
|
|
+ pub(crate) value: T,
|
|
|
+}
|
|
|
+
|
|
|
+pub struct Signal<T: 'static> {
|
|
|
+ pub(crate) inner: CopyValue<SignalData<T>>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<T: 'static> Signal<T> {
|
|
|
+ pub fn new(value: T) -> Self {
|
|
|
+ Self {
|
|
|
+ inner: CopyValue::new(SignalData {
|
|
|
+ subscribers: Default::default(),
|
|
|
+ effect_subscribers: Default::default(),
|
|
|
+ update_any: schedule_update_any().expect("in a virtual dom"),
|
|
|
+ value,
|
|
|
+ }),
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn origin_scope(&self) -> ScopeId {
|
|
|
+ self.inner.origin_scope()
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn read(&self) -> Ref<T> {
|
|
|
+ let inner = self.inner.read();
|
|
|
+ if let Some(effect) = Effect::current() {
|
|
|
+ let mut effect_subscribers = inner.effect_subscribers.borrow_mut();
|
|
|
+ if !effect_subscribers.contains(&effect) {
|
|
|
+ effect_subscribers.push(effect);
|
|
|
+ }
|
|
|
+ } else if let Some(current_scope_id) = current_scope_id() {
|
|
|
+ log::trace!(
|
|
|
+ "{:?} subscribed to {:?}",
|
|
|
+ self.inner.value,
|
|
|
+ current_scope_id
|
|
|
+ );
|
|
|
+ let mut subscribers = inner.subscribers.borrow_mut();
|
|
|
+ if !subscribers.contains(¤t_scope_id) {
|
|
|
+ subscribers.push(current_scope_id);
|
|
|
+ drop(subscribers);
|
|
|
+ let unsubscriber = current_unsubscriber();
|
|
|
+ inner.subscribers.borrow_mut().push(unsubscriber.scope);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ Ref::map(inner, |v| &v.value)
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn write(&self) -> Write<'_, T> {
|
|
|
+ let inner = self.inner.write();
|
|
|
+ let borrow = RefMut::map(inner, |v| &mut v.value);
|
|
|
+ Write {
|
|
|
+ write: borrow,
|
|
|
+ signal: SignalSubscriberDrop { signal: *self },
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn update_subscribers(&self) {
|
|
|
+ {
|
|
|
+ let inner = self.inner.read();
|
|
|
+ for &scope_id in &*inner.subscribers.borrow() {
|
|
|
+ log::trace!(
|
|
|
+ "Write on {:?} triggered update on {:?}",
|
|
|
+ self.inner.value,
|
|
|
+ scope_id
|
|
|
+ );
|
|
|
+ (inner.update_any)(scope_id);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ let subscribers = {
|
|
|
+ let self_read = self.inner.read();
|
|
|
+ let mut effects = self_read.effect_subscribers.borrow_mut();
|
|
|
+ std::mem::take(&mut *effects)
|
|
|
+ };
|
|
|
+ for effect in subscribers {
|
|
|
+ log::trace!(
|
|
|
+ "Write on {:?} triggered effect {:?}",
|
|
|
+ self.inner.value,
|
|
|
+ effect
|
|
|
+ );
|
|
|
+ effect.try_run();
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn set(&self, value: T) {
|
|
|
+ *self.write() = value;
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn with<O>(&self, f: impl FnOnce(&T) -> O) -> O {
|
|
|
+ let write = self.read();
|
|
|
+ f(&*write)
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn with_mut<O>(&self, f: impl FnOnce(&mut T) -> O) -> O {
|
|
|
+ let mut write = self.write();
|
|
|
+ f(&mut *write)
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<T: Clone + 'static> Signal<T> {
|
|
|
+ pub fn value(&self) -> T {
|
|
|
+ self.read().clone()
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<T: 'static> PartialEq for Signal<T> {
|
|
|
+ fn eq(&self, other: &Self) -> bool {
|
|
|
+ self.inner == other.inner
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+struct SignalSubscriberDrop<T: 'static> {
|
|
|
+ signal: Signal<T>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<T: 'static> Drop for SignalSubscriberDrop<T> {
|
|
|
+ fn drop(&mut self) {
|
|
|
+ self.signal.update_subscribers();
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+pub struct Write<'a, T: 'static, I: 'static = T> {
|
|
|
+ write: RefMut<'a, T>,
|
|
|
+ signal: SignalSubscriberDrop<I>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, T: 'static, I: 'static> Write<'a, T, I> {
|
|
|
+ pub fn map<O>(myself: Self, f: impl FnOnce(&mut T) -> &mut O) -> Write<'a, O, I> {
|
|
|
+ let Self { write, signal } = myself;
|
|
|
+ Write {
|
|
|
+ write: RefMut::map(write, f),
|
|
|
+ signal,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn filter_map<O>(
|
|
|
+ myself: Self,
|
|
|
+ f: impl FnOnce(&mut T) -> Option<&mut O>,
|
|
|
+ ) -> Option<Write<'a, O, I>> {
|
|
|
+ let Self { write, signal } = myself;
|
|
|
+ let write = RefMut::filter_map(write, f).ok();
|
|
|
+ write.map(|write| Write {
|
|
|
+ write,
|
|
|
+ signal: signal,
|
|
|
+ })
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<'a, T: 'static> Deref for Write<'a, T> {
|
|
|
+ type Target = T;
|
|
|
+
|
|
|
+ fn deref(&self) -> &Self::Target {
|
|
|
+ &*self.write
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<T> DerefMut for Write<'_, T> {
|
|
|
+ fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
+ &mut *self.write
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+pub struct ReadOnlySignal<T: 'static> {
|
|
|
+ inner: Signal<T>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<T: 'static> ReadOnlySignal<T> {
|
|
|
+ pub fn new(signal: Signal<T>) -> Self {
|
|
|
+ Self { inner: signal }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn origin_scope(&self) -> ScopeId {
|
|
|
+ self.inner.origin_scope()
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn read(&self) -> Ref<T> {
|
|
|
+ self.inner.read()
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn with<O>(&self, f: impl FnOnce(&T) -> O) -> O {
|
|
|
+ self.inner.with(f)
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<T: Clone + 'static> ReadOnlySignal<T> {
|
|
|
+ pub fn value(&self) -> T {
|
|
|
+ self.read().clone()
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<T: 'static> PartialEq for ReadOnlySignal<T> {
|
|
|
+ fn eq(&self, other: &Self) -> bool {
|
|
|
+ self.inner == other.inner
|
|
|
+ }
|
|
|
+}
|