2 releases
0.0.2 | Jul 21, 2024 |
---|---|
0.0.1 | May 30, 2024 |
#454 in Rust patterns
240KB
7K
SLoC
sigmut
sigmut
is a state management framework designed to be used as a foundation for UI frameworks.
[!WARNING] Warning: This crate is still in the very early stages of development. APIs will change. Documentation is sparse.
Features
- Signals-based API
- Separation of "state changes" and "state calculations"
- Easy-to-use single-threaded model
- Support for asynchronous operations using
async
/await
- Glitch-free (no unnecessary calculations based on outdated states)
- Capable of implementing more efficient reactive primitives
Signals-based API
In sigmut
, state management is conducted using the following reactive primitives:
State<T>
: Similar toRc<RefCell<T>>
, but with added functionality to observe changes.Signal<T>
: Similar toRc<dyn Fn() -> &T>
, but with added functionality to observe changes in the result.effect
: A function that is called again when there are changes to the dependent state.
use sigmut::{Signal, State};
let mut rt = sigmut::core::Runtime::new();
let a = State::new(0);
let b = State::new(1);
let c = Signal::new({
let a = a.clone();
let b = b.clone();
move |sc| a.get(sc) + b.get(sc)
});
let _e = c.effect(|x| println!("{x}"));
rt.update(); // prints "1"
a.set(2, rt.ac());
rt.update(); // prints "3"
a.set(3, rt.ac());
b.set(5, rt.ac());
rt.update(); // prints "8"
Dependencies between states are automatically tracked, and recalculations are automatically triggered when changes occur.
This mechanism is a recent trend and is also adopted by other state management libraries, such as the following:
Separation of "state changes" and "state calculations"
Many state management libraries simplify programs by separating state changes from state calculations.
In Elm, the Model-View-Update architecture separates state changes (Update) from state calculations (View).
In React, the rule to Components and Hooks must be pure
prohibits state changes during state calculations. In React's StrictMode, state calculations are called an extra time to ensure this rule is followed.
In SolidJS, state changes made during state calculations are deferred until the state calculation is complete.
In sigmut
, state changes and state calculations are separated using SignalContext
and ActionContext
.
ActionContext
: Used for state changesSignalContext
: Used for state calculations
By requiring functions that perform state changes or state calculations to use the corresponding context, the distinction between state changes and state calculations is made clear, and the compiler can enforce this separation.
The "separation of state changes and state calculations" simplifies the program by treating state as immutable during state calculations, which is similar to Rust's ownership concept. Internally, sigmut
uses RefCell
, but this similarity helps avoid BorrowError
during state calculations. If you are using many Rc<RefCell<T>>
, switching to sigmut
can result in a more robust program with fewer BorrowError
occurrences.
Easy-to-use single-threaded model
sigmut
adopts a single-threaded model for the following reasons:
- Simple and easy to handle
- No risk of deadlocks
- No need for synchronization, allowing for instant retrieval of the current value
- Interoperability with
async/await
, enabling the benefits of multithreading - Capable of being glitch-free (no unnecessary calculations based on outdated states)
Support for asynchronous operations using async
/await
sigmut
integrates with async/await
, allowing asynchronous operations to be treated as synchronous Poll<T>
state. This enables interoperability with asynchronous runtimes like tokio
.
For more details, refer to functions and types with names that include async
, future
, or stream
.
Glitch-free (no unnecessary calculations based on outdated states)
Some state management libraries use outdated caches during state calculations, which can lead to unexpected results. While these unexpected results are quickly recalculated and the unintended calculation outcomes are discarded, this can still cause issues, including potential panics. Therefore, the problem is not fully resolved simply because recalculation occurs.
In sigmut
, caches are managed by categorizing them into three types: clean
, dirty
, and maybe dirty
. By consistently and accurately checking the validity of these caches, sigmut
avoids the issues associated with using outdated caches during state calculations.
Capable of implementing more efficient reactive primitives
sigmut
includes a low-level module, sigmut::core
, that handles only state change notifications. By using this module, you can implement more efficient reactive primitives under specific conditions.
An implementation example of this is SignalVec<T>
. SignalVec<T>
is similar to Signal<Vec<T>>
, but it allows you to obtain the change history since the last access, enabling more efficient processing.
License
This project is dual licensed under Apache-2.0/MIT. See the two LICENSE-* files for details.
Contribution
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.
Dependencies
~3.5–5.5MB
~103K SLoC