123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120 |
- use crate::dependency::Dependency;
- use crate::{use_callback, use_signal};
- use dioxus_core::prelude::*;
- use dioxus_signals::Memo;
- use dioxus_signals::{ReactiveContext, ReadOnlySignal, Readable, Signal, SignalData};
- use dioxus_signals::{Storage, Writable};
- use futures_util::StreamExt;
- /// Creates a new unsync Selector. The selector will be run immediately and whenever any signal it reads changes.
- ///
- /// Selectors can be used to efficiently compute derived data from signals.
- ///
- /// ```rust
- /// use dioxus::prelude::*;
- /// use dioxus_signals::*;
- ///
- /// fn App() -> Element {
- /// let mut count = use_signal(|| 0);
- /// let double = use_memo(move || count * 2);
- /// count += 1;
- /// assert_eq!(double(), count * 2);
- ///
- /// rsx! { "{double}" }
- /// }
- /// ```
- #[track_caller]
- pub fn use_memo<R: PartialEq>(f: impl FnMut() -> R + 'static) -> Memo<R> {
- let callback = use_callback(f);
- #[allow(clippy::redundant_closure)]
- use_hook(|| Signal::memo(move || callback()))
- }
- /// Creates a new unsync Selector with some local dependencies. The selector will be run immediately and whenever any signal it reads or any dependencies it tracks changes
- ///
- /// Selectors can be used to efficiently compute derived data from signals.
- ///
- /// ```rust
- /// use dioxus::prelude::*;
- ///
- /// fn App() -> Element {
- /// let mut local_state = use_signal(|| 0);
- /// let double = use_memo_with_dependencies((&local_state(),), move |(local_state,)| local_state * 2);
- /// local_state.set(1);
- ///
- /// rsx! { "{double}" }
- /// }
- /// ```
- #[track_caller]
- pub fn use_memo_with_dependencies<R: PartialEq, D: Dependency>(
- dependencies: D,
- f: impl FnMut(D::Out) -> R + 'static,
- ) -> ReadOnlySignal<R>
- where
- D::Out: 'static,
- {
- use_maybe_sync_memo_with_dependencies(dependencies, f)
- }
- /// Creates a new Selector that may be sync with some local dependencies. The selector will be run immediately and whenever any signal it reads or any dependencies it tracks changes
- ///
- /// Selectors can be used to efficiently compute derived data from signals.
- ///
- /// ```rust
- /// use dioxus::prelude::*;
- /// use dioxus_signals::*;
- ///
- /// fn App() -> Element {
- /// let mut local_state = use_signal(|| 0i32);
- /// let double: ReadOnlySignal<i32, SyncStorage> = use_maybe_sync_memo_with_dependencies((&local_state(),), move |(local_state,)| local_state * 2);
- /// local_state.set(1);
- ///
- /// rsx! { "{double}" }
- /// }
- /// ```
- #[track_caller]
- pub fn use_maybe_sync_memo_with_dependencies<
- R: PartialEq,
- D: Dependency,
- S: Storage<SignalData<R>>,
- >(
- dependencies: D,
- mut f: impl FnMut(D::Out) -> R + 'static,
- ) -> ReadOnlySignal<R, S>
- where
- D::Out: 'static,
- {
- let mut dependencies_signal = use_signal(|| dependencies.out());
- let selector = use_hook(|| {
- // Get the current reactive context
- let (rc, mut changed) = ReactiveContext::new();
- // Create a new signal in that context, wiring up its dependencies and subscribers
- let mut state: Signal<R, S> =
- rc.run_in(|| Signal::new_maybe_sync(f(dependencies_signal.read().clone())));
- spawn(async move {
- loop {
- // Wait for context to change
- let _ = changed.next().await;
- let new = rc.run_in(|| f(dependencies_signal.read().clone()));
- if new != *state.peek() {
- *state.write() = new;
- }
- }
- });
- // And just return the readonly variant of that signal
- ReadOnlySignal::new_maybe_sync(state)
- });
- // This will cause a re-run of the selector if the dependencies change
- let changed = { dependencies.changed(&*dependencies_signal.read()) };
- if changed {
- dependencies_signal.set(dependencies.out());
- }
- selector
- }
|