190 lines
5.3 KiB
Rust
190 lines
5.3 KiB
Rust
#[cfg(feature = "effects")]
|
|
pub mod imports {
|
|
pub use any_spawner::Executor;
|
|
pub use reactive_graph::{
|
|
effect::{Effect, RenderEffect},
|
|
prelude::*,
|
|
signal::RwSignal,
|
|
};
|
|
pub use std::{
|
|
mem,
|
|
sync::{Arc, RwLock},
|
|
};
|
|
pub use tokio::task;
|
|
}
|
|
|
|
#[cfg(feature = "effects")]
|
|
#[tokio::test]
|
|
async fn render_effect_runs() {
|
|
use imports::*;
|
|
|
|
_ = Executor::init_tokio();
|
|
task::LocalSet::new()
|
|
.run_until(async {
|
|
let a = RwSignal::new(-1);
|
|
|
|
// simulate an arbitrary side effect
|
|
let b = Arc::new(RwLock::new(String::new()));
|
|
|
|
// we forget it so it continues running
|
|
// if it's dropped, it will stop listening
|
|
mem::forget(RenderEffect::new({
|
|
let b = b.clone();
|
|
move |_| {
|
|
let formatted = format!("Value is {}", a.get());
|
|
*b.write().unwrap() = formatted;
|
|
}
|
|
}));
|
|
|
|
Executor::tick().await;
|
|
assert_eq!(b.read().unwrap().as_str(), "Value is -1");
|
|
|
|
println!("setting to 1");
|
|
a.set(1);
|
|
|
|
Executor::tick().await;
|
|
assert_eq!(b.read().unwrap().as_str(), "Value is 1");
|
|
})
|
|
.await;
|
|
}
|
|
|
|
#[cfg(feature = "effects")]
|
|
#[tokio::test]
|
|
async fn effect_runs() {
|
|
use imports::*;
|
|
|
|
_ = Executor::init_tokio();
|
|
|
|
task::LocalSet::new()
|
|
.run_until(async {
|
|
let a = RwSignal::new(-1);
|
|
|
|
// simulate an arbitrary side effect
|
|
let b = Arc::new(RwLock::new(String::new()));
|
|
|
|
Effect::new({
|
|
let b = b.clone();
|
|
move || {
|
|
let formatted = format!("Value is {}", a.get());
|
|
*b.write().unwrap() = formatted;
|
|
}
|
|
});
|
|
|
|
Executor::tick().await;
|
|
assert_eq!(b.read().unwrap().as_str(), "Value is -1");
|
|
|
|
println!("setting to 1");
|
|
a.set(1);
|
|
|
|
Executor::tick().await;
|
|
assert_eq!(b.read().unwrap().as_str(), "Value is 1");
|
|
})
|
|
.await
|
|
}
|
|
|
|
#[cfg(feature = "effects")]
|
|
#[tokio::test]
|
|
async fn dynamic_dependencies() {
|
|
use imports::*;
|
|
|
|
_ = Executor::init_tokio();
|
|
|
|
task::LocalSet::new()
|
|
.run_until(async {
|
|
let first = RwSignal::new("Greg");
|
|
let last = RwSignal::new("Johnston");
|
|
let use_last = RwSignal::new(true);
|
|
|
|
let combined_count = Arc::new(RwLock::new(0));
|
|
|
|
mem::forget(RenderEffect::new({
|
|
let combined_count = Arc::clone(&combined_count);
|
|
move |_| {
|
|
*combined_count.write().unwrap() += 1;
|
|
if use_last.get() {
|
|
println!("{} {}", first.get(), last.get());
|
|
} else {
|
|
println!("{}", first.get());
|
|
}
|
|
}
|
|
}));
|
|
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 1);
|
|
|
|
println!("\nsetting `first` to Bob");
|
|
first.set("Bob");
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 2);
|
|
|
|
println!("\nsetting `last` to Bob");
|
|
last.set("Thompson");
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 3);
|
|
|
|
println!("\nsetting `use_last` to false");
|
|
use_last.set(false);
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 4);
|
|
|
|
println!("\nsetting `last` to Jones");
|
|
last.set("Jones");
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 4);
|
|
|
|
println!("\nsetting `last` to Jones");
|
|
last.set("Smith");
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 4);
|
|
|
|
println!("\nsetting `last` to Stevens");
|
|
last.set("Stevens");
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 4);
|
|
|
|
println!("\nsetting `use_last` to true");
|
|
use_last.set(true);
|
|
Executor::tick().await;
|
|
assert_eq!(*combined_count.read().unwrap(), 5);
|
|
})
|
|
.await
|
|
}
|
|
|
|
#[cfg(feature = "effects")]
|
|
#[tokio::test]
|
|
async fn recursive_effect_runs_recursively() {
|
|
use imports::*;
|
|
|
|
_ = Executor::init_tokio();
|
|
task::LocalSet::new()
|
|
.run_until(async {
|
|
let s = RwSignal::new(0);
|
|
|
|
let logged_values = Arc::new(RwLock::new(Vec::new()));
|
|
|
|
mem::forget(RenderEffect::new({
|
|
let logged_values = Arc::clone(&logged_values);
|
|
move |_| {
|
|
let a = s.get();
|
|
println!("a = {a}");
|
|
logged_values.write().unwrap().push(a);
|
|
if a == 0 {
|
|
return;
|
|
}
|
|
s.set(0);
|
|
}
|
|
}));
|
|
|
|
s.set(1);
|
|
Executor::tick().await;
|
|
s.set(2);
|
|
Executor::tick().await;
|
|
s.set(3);
|
|
Executor::tick().await;
|
|
|
|
assert_eq!(0, s.get_untracked());
|
|
assert_eq!(&*logged_values.read().unwrap(), &[0, 1, 0, 2, 0, 3, 0]);
|
|
})
|
|
.await;
|
|
}
|