diff --git a/reactive/Cargo.toml b/reactive/Cargo.toml index 9bbf2bd78..ac8a3d176 100644 --- a/reactive/Cargo.toml +++ b/reactive/Cargo.toml @@ -5,3 +5,4 @@ edition = "2021" license.workspace = true [dependencies] +smallvec = "1.10.0" diff --git a/reactive/src/effect.rs b/reactive/src/effect.rs index 2d92b8ff5..4557e9fc8 100644 --- a/reactive/src/effect.rs +++ b/reactive/src/effect.rs @@ -69,6 +69,19 @@ pub fn untrack(f: impl FnOnce() -> T) -> T { result } +pub fn batch(f: impl FnOnce() -> T) -> T { + RUNTIME.with(|runtime| { + runtime.batching.set(true); + }); + let result = f(); + RUNTIME.with(|runtime| { + runtime.batching.set(false); + runtime.run_pending_effects(); + }); + + result +} + pub(crate) fn run_effect(effect: Rc) { let effect_id = effect.id(); effect_id.dispose(); diff --git a/reactive/src/lib.rs b/reactive/src/lib.rs index 7b715bd66..6f961fc35 100644 --- a/reactive/src/lib.rs +++ b/reactive/src/lib.rs @@ -8,7 +8,7 @@ mod signal; mod trigger; pub use context::{provide_context, use_context}; -pub use effect::{create_effect, untrack}; +pub use effect::{batch, create_effect, untrack}; pub use memo::{create_memo, Memo}; pub use scope::{as_child_of_current_scope, with_scope, Scope}; pub use signal::{create_rw_signal, create_signal, ReadSignal, RwSignal, WriteSignal}; diff --git a/reactive/src/runtime.rs b/reactive/src/runtime.rs index 2e03ed2cf..e9cebe5e5 100644 --- a/reactive/src/runtime.rs +++ b/reactive/src/runtime.rs @@ -1,11 +1,17 @@ use std::{ any::{Any, TypeId}, - cell::RefCell, + cell::{Cell, RefCell}, collections::{HashMap, HashSet}, rc::Rc, }; -use crate::{effect::EffectTrait, id::Id, signal::Signal}; +use smallvec::SmallVec; + +use crate::{ + effect::{run_effect, EffectTrait}, + id::Id, + signal::Signal, +}; thread_local! { pub(crate) static RUNTIME: Runtime = Runtime::new(); @@ -19,6 +25,8 @@ pub(crate) struct Runtime { pub(crate) children: RefCell>>, pub(crate) signals: RefCell>, pub(crate) contexts: RefCell>>, + pub(crate) batching: Cell, + pub(crate) pending_effects: RefCell; 10]>>, } impl Default for Runtime { @@ -35,6 +43,26 @@ impl Runtime { children: RefCell::new(HashMap::new()), signals: Default::default(), contexts: Default::default(), + batching: Cell::new(false), + pending_effects: RefCell::new(SmallVec::new()), + } + } + + pub(crate) fn add_pending_effect(&self, effect: Rc) { + let has_effect = self + .pending_effects + .borrow() + .iter() + .any(|e| e.id() == effect.id()); + if !has_effect { + self.pending_effects.borrow_mut().push(effect); + } + } + + pub(crate) fn run_pending_effects(&self) { + let pending_effects = self.pending_effects.take(); + for effect in pending_effects { + run_effect(effect); } } } diff --git a/reactive/src/signal.rs b/reactive/src/signal.rs index 3f85da690..1eeca89d8 100644 --- a/reactive/src/signal.rs +++ b/reactive/src/signal.rs @@ -347,6 +347,16 @@ impl Signal { } pub(crate) fn run_effects(&self) { + // If we are batching then add it as a pending effect + if RUNTIME.with(|r| r.batching.get()) { + RUNTIME.with(|r| { + for (_, subscriber) in self.subscribers() { + r.add_pending_effect(subscriber); + } + }); + return; + } + for (_, subscriber) in self.subscribers() { run_effect(subscriber); }