A tiny state manager for big applications
- Redux inspired
- No boilerplate
- Zero dependencies, 1k gzipped, ~120 SLOC
- Expressive syntax for describing transactions, ImmerJS inspired, but using plain JS objects.
- Automatic individual patches recording, again ImmerJS inspired, and without Proxy voodoo magic :)
- Middleware: logging, time travel, etc. out of the box
- Extremely simple and non-opinionated
- Directly usable in SvelteJS applications: follows Svelte's store API
- Plugin available for VueJS (so you can use it instead of, say, VueX)
The Redux's concept of a single immutable store is very solid and allows for lots of powerful paradigms implemented in a simple way thanks to referential equality. Unfortunately it has been traditionally associated with lots of boilerplate and noise. Actions, action creators, reducers, reducer combiners, you name it...
It does not have to be this way!
Redux tightly couples concepts of actions and reducers (mutations). Mutations may only happen as a result of actions. In the MobX / VueX world this coupling is softened, and mutations only need to happen within actions in the so-called "strict mode". The focus is still though on treating the actions as semantic units driving the state change, while the mutations are happening under the hood, in a semi-magical, hard to trace reactive fashion.
tinyX attempts to have the best of both worlds without over-engineering things
import { writable } from 'svelte/store';
import { tx } from 'tinyx';
import logger from 'tinyx/middleware/logger';
const store =
logger(
tx(
writable({ todos: [] })
));
// Transaction
function ADD_TODO(task) {
return ({ update }) => update('todos', todos => [...todos, { task }])
}
store.commit(ADD_TODO, 'Start using tinyX');
// prints
// [ADD_TODO]: Start using tinyX [ { path: [ 'todos' ], oldValue: [], newValue: [ [Object] ] } ]
tinyX builds upon the concept of a store as a simple contract, as it is popularized in SvelteJS, but it does not have any direct dependencies on Svelte.
From Svelte API docs:
A store is simply an object with a
subscribe
method that allows interested parties to be notified whenever the store value changes. A writable store hasset
andupdate
methods in addition tosubscribe
.
Thus, in Svelte applications it would be natural to use svelte/store/writable
as the base for the single store root.
import { writable } from 'svelte/store';
import { tx } from 'tinyx';
const store = tx(writable({ todos: [] }));
You can create your own stores without relying on svelte/store
, by implementing the writable store contract:
store = { subscribe: (subscription: (value: any) => void) => () => void, set: (value: any) => void }
Like in Redux, the store represents the root of your unique state tree. Again, like in Redux, the whole tree is immutably replaced upon every change by a reducer.
tinyX just provides an expressive and intuitive way to build the reducers from elementary operations i.e. { set, update, remove }
You can put anything into your store: functions, promises, whatever you please, it is your store ! It is never augmented with proxies or tampered with in any way.
- A mutation is a function applying elementary operations to a temporary sub-state, e.g.
({ set, update }) => {
set('important', true);
update('text', value => value.toUpperCase());
set('style', 'color', 'red');
}
A mutation can use 5 elementary operations, each following a keyPath
from the root of the state tree across plain objects, arrays and ES6 maps
get(...keyPath)
: returns a value atkeyPath
set(...keyPath, value)
: sets a value atkeyPath
update(...keyPath, updater)
: updates a value atkeyPath
remove(...keyPath)
: removes a value atkeyPath
apply(...keyPath, mutation)
: applies another mutation atkeyPath
Each operation replaces the whole temporary sub-state (a la ImmutableJS), but unlike ImmutableJS they support ES6 Maps.
tinyX automatically wraps every mutation in an ImmerJS-like produce
helper to make the atomic reducer:
reducer = produce(mutation)
which is just a function state => newState
.
- A transaction is a function
payload => mutation
, e.g.
function ADD_TODO(task) {
return ({ update }) => update('todos', todos => [...todos, { task }])
}
Transactions may be committed to the tinyX store at an arbitrary location, always resulting in a synchronous atomic update of the whole tree:
store.commit(ADD_TODO, 'Start using tinyX', ...keyPath);
store.commit
returns an array of individual changes in the format [{ path, oldValue, newValue }]
Note how the transactions are described in a concise semantic fashion with zero dependencies.
It is a good practice to create logical modules containing transactions only and nothing else, and import them as needed.
This could serve as a reference to everything that can happen to your application state, in one place.
It is also a good practice to declare transactions with the function
keyword, so that middleware (e.g. a logger) could have access to the function name.
With tinyX there is no middleware in the classical sense: you don't need a special semantic concept for this. The store is a simple contract with only three methods in the interface: Store = { subscribe, get, commit }
, so if you want your transactions to travel through extra layers of processing (i.e. logging) you simply implement a transformer, e.g. logger = (store: Store) -> Store
and commit your transactions into transformed store.
import { tx } from 'tinyx';
import logger from 'tinyx/middleware/logger';
const store = logger(tx(writable(initialState)));
store.commit(ADD_TODO, task); // will get logged to console
You can import applyMiddleware from 'tinyx/middleware'
to add syntactic sugar for this:
const store = applyMiddleware(tx(writable(initialState)), [...middleware]);
Note Check out a logger and a generic undo/redo examples in the middleware
folder.
Out of the box tinyx
is compliant to the Svelte Readable
store contract, and thus can be directly referenced as $store
in Svelte components.
You can also augment it with Svelte Writable
contract traits set
and update
using tinyx/middleware/writable-traits
which work by committing predefined mutations SET(value)
and UPDATE(updater)
using the store's commit
function.
Unlike Redux or VueX, tinyX is totally non-opinionated in the way it treats actions (whether user-initiated actions or asynchronous events). They become just plain Javascript functions, totally decoupled from any components. An action would usually take the store it operates on as its first argument, though it is not a strict requirement.
For instance, the undo_redo
middleware exports two such generic actions, namely, undo
and redo
, as well as a generic wrapper undoable
, which, applied to any action, makes it, well, undoable :)
Example:
import { writable } from 'svelte/store';
import { tx } from 'tinyx';
import applyMiddleware from 'tinyx/middleware';
import { undo, redo, undoable, enableUndoRedo } from 'tinyx/middleware/undo_redo';
import logger from 'tinyx/middleware/logger';
const store = applyMiddleware(tx(writable({ todos: [] })), [enableUndoRedo, logger]);
// Export a new action
export const addTodo = undoable((store, task) => store.commit(ADD_TODO, task));
addTodo(store, 'Run 10 miles in the morning !');
undo(store);
redo(store);
tinyX exports a couple of helpers select
and derived
sharing a similar signature (store, selector) => subStore
.
For select
selector = state => keyPath
, e.g.
import { tx, select } from 'tinyx';
const store = tx(writable({
documents: new Map(),
activeDocumentId: null
}));
const activeDocument = select(store, ({ activeDocumentId }) => ['documents', activeDocumentId]);
activeDocument
will have the same API as the root store, you can commit transactions to it, and they will travel through all the middleware attached to the root store.
You can also wrap selected sub-stores in extra middleware, so that only transactions committed directly into it would go through the extra layer. This functional approach gives incredible flexibility to mix and match middleware and sub-trees. In a sense, select
itself may be called a middleware: it is just a store transformer preserving the Store
contract.
For derived
selector = state => derivedState
, e.g.
import { tx, derived } from 'tinyx';
const store = tx(writable({
documents: new Map(),
activeDocumentId: null
}));
const activeDocument = derived(store, ({ documents, activeDocumentId }) => documents.get(activeDocumentId));
You cannot commit transactions to derived stores (they are Readable
only), but the selector
does not have to represent a sub-tree: it can be any derived state.
Svelte developers would enjoy the same $activeDocument
syntactic sugar: reading from it creates an auto-subscription that only notifies subscribers if the corresponding state is affected.
With tinyX you can now easily (and you should !!!) turn on the immutable: true
Svelte compiler option and enjoy improved performance and strictly predictable re-renders.
There is a VueJS plugin in the plugins
folder, that injects the tinyX store into all Vue components created and hooks its .subscribe
method to the Vue reactivity system, so the components can react to transactions committed to the store.
Usage:
import VueStore from 'tinyx/plugins/vue'
import Vue from 'vue'
Vue.use(VueStore, { store, name: 'tx' });
All components will then have the tx()
method returning the latest state root.
More than happy to accept a PR with an equivalent of react-redux
to use tinyX with React.