This change switches back to the original `ctx<props>` syntax for commponents. This lets lifetime elision to remove the need to match exactly which lifetime (props or ctx) gets carried to the output. As such, `Props` is currently required to be static. It *is* possible to loosen this restriction, and will be done in the future, though only through adding metadata about the props through the Props derive macro. Implementing the IS_STATIC trait is unsafe, so the derive macro will do it through some heuristics. For now, this unlocks sharing vnodes from parents to children, enabling pass-thru components, fragments, portals, etc.
7.2 KiB
Recoil.rs - Official global state management solution for Dioxus Apps
Recoil.rs provides a global state management API for Dioxus apps built on the concept of "atomic state." Instead of grouping state together into a single bundle ALA Redux, Recoil provides individual building blocks of state called Atoms. These atoms can be set/get anywhere in the app and combined to craft complex state. Recoil should be easier to learn and more efficient than Redux. Recoil.rs is modeled after the Recoil.JS project.
Recoil.rs is officially supported by the Dioxus team. By doing so, are are "planting our flag in the stand" for atomic state management instead of bundled (Redux-style) state management. Atomic state management fits well with the internals of Dioxus, meaning Recoil.rs state management will be faster, more efficient, and less sensitive to data races than Redux-style apps.
Internally, Dioxus uses batching to speed up linear-style operations. Recoil.rs integrates with this batching optimization, making app-wide changes extremely fast. This way, Recoil.rs can be pushed significantly harder than Redux without the need to enable/disable debug flags to prevent performance slowdowns.
Guide
Atoms
A simple atom of state is defined globally as a const:
const Light: Atom<&str> = |_| "Green";
This atom of state is initialized with a value of "Green"
. The atom that is returned does not actually contain any values. Instead, the atom's key - which is automatically generated in this instance - is used in the context of a Recoil App.
This is then later used in components like so:
fn App(ctx: Context<()>) -> VNode {
// The recoil root must be initialized at the top of the application before any use_recoil hooks
recoil::init_recoil_root(&ctx, |_| {});
let color = recoil::use_read(ctx, Light);
ctx.render(rsx!{
h1 {"Color of light: {color}"}
})
}
Atoms are considered "Writable" objects since any consumer may also set the Atom's value with their own:
fn App(ctx: Context<()>) -> VNode {
let color = recoil::use_read(ctx, Light);
let set_color = recoil::use_write(ctx, Light);
rsx!{in ctx,
div {
h1{"Color: {color}"}
button {onclick: move |_| set_color("red"), "red"}
button {onclick: move |_| set_color("yellow"), "yellow"}
button {onclick: move |_| set_color("green"), "green"}
}
}
}
"Reading" a value with use_read subscribes that component to any changes to that value while "Writing" a value does not. It's a good idea to use write-only
whenever it makes sense to prevent unnecessary evaluations. Both read
and write
may be combined together to provide a use_state
style hook.
fn App(ctx: Context<()>) -> VNode {
let (color, set_color) = recoil::use_read_write(ctx, Light);
rsx!{in ctx,
div {
h1{"Color: {color}"}
button {onclick: move |_| set_color("red"), "red"}
}
}
}
Selectors
Selectors are a concept popular in the JS world as a way of narrowing down a selection of state outside the VDOM lifecycle. Selectors have two functions: 1) summarize/narrow down some complex state and 2) memoize calculations.
Selectors are only readable
- they cannot be set. This differs from RecoilJS where selectors are writable
. Selectors, as you might've guessed, "select" some data from atoms and other selectors.
Selectors provide a SelectorApi
which essentially exposes a read-only RecoilApi
. They have the get
method which allows any readable valued to be obtained for the purpose of generating a new value. A Selector
may only return 'static
data, however SelectorBorrowed
may return borrowed data.
returning static data:
const Light: Atom<&'static str> = |_| "Green";
const NumChars: Selector<usize> = |api| api.get(&Light).len();
Selectors will update their selection and notify subscribed components whenever their dependencies also update. The get
calls in a selector will subscribe that selector to whatever Readable
is being get
-ted. Unlike hooks, you may use get
in conditionals; an internal algorithm decides when a selector needs to be updated based on what it get
-ted last time it was ran.
Selectors may also returned borrowed data:
const Light: Atom<&'static str> = |_| "Green";
const ThreeChars: SelectorBorrowed<str> = |api| api.get(&Light).chars().take(3).unwrap();
Unfortunately, Rust tries to coerce T
(the type in the Selector generics) to the 'static
lifetime because we defined it as a static. SelectorBorrowed
defines a type for a function that returns &T
and provides the right lifetime for T
. If you don't like having to use a dedicated Borrowed
type, regular functions work too:
fn Light(api: &mut AtomBuilder) -> &'static str {
"Green"
}
fn ThreeChars(api: &mut SelectorApi) -> &'static str {
api.get(&Light).chars().take(3).unwrap()
}
Returning borrowed data is generally frowned upon, but may be useful when used wisely.
- If a selected value equals its previous selection (via PartialEq), the old value must be kept around to avoid evaluating subscribed components.
- It's unlikely that a change in a dependency's data will not change the selector's output.
In general, borrowed selectors introduce a slight memory overhead as they need to retain previous state to safely memoize downstream subscribers. The amount of state kept around scales with the amount of gets
in a selector - though as the number of dependencies increase, the less likely the selector actually stays memoized. Recoil tries to optimize this behavior the best it can to balance component evaluations with memory overhead.
Families
You might notice that collections will not be performant with just sets/gets. We don't want to clone our entire HashMap every time we want to insert or remove an entry! That's where Families
come in. Families are memoized collections (HashMap and OrdMap) that wrap the immutable datastructures library im-rc
. Families are defined by a function that takes the FamilyApi and returns a function that provides a default value given a key. In this example, we insert a value into the collection when initialized, and then return a function that takes a key and returns a default value.
const CloudRatings: AtomFamily<&str, i32> = |api| {
api.insert("Oracle", -1);
|key| match key {
"AWS" => 1,
"Azure" => 2,
"GCP" => 3,
_ => 0
}
}
Whenever you select
on a Family
, the ID of the entry is tracked. Other subscribers will only be updated if they too select the same family with the same key and that value is updated. Otherwise, these subscribers will never re-render on an "insert", "remove", or "update" of the collection. You could easily implement this yourself with Atoms, immutable datastructures, and selectors, but our implementation is more efficient and first-class.
fn App(ctx: Context<()>) -> VNode {
let (rating, set_rating) = recoil::use_read_write(ctx, CloudRatings.select("AWS"));
rsx!{in ctx,
div {
h1{ "AWS rating: {rating}" }
button { onclick: move |_| set_rating((rating + 1) % 5), "incr" }
}
}
}