core
Tiny, efficient, featured and extensible core to handle reactivity right. The ultimate state manager. Build anything, from a small widget to a huge application.
included in @reatom/framework
The raw API description is below.
About
Reatom allows you to describe both simple and complex logic using three main components: atoms for data reference, actions for logic processing, and context (ctx
) for system isolation. This core is a perfect solution for building your own high-order library or framework.
Reatom is inspired by the React and Redux architectures. All processed data should be immutable, computations should be pure, and all side effects should be scheduled for a separate effects queue using ctx.schedule(callback)
. Only consistent data transactions should be applied. All prerequisites can be checked in this article: What is a state manager.
Installation
npm i @reatom/core
The build target is
last 4 chrome versions
, if you need to support older environments, you should transpile it by yourself.
Usage
Let’s describe a simple example of a search input with a tip and a list of goods. This code is written in TypeScript, but you could also use JavaScript; a lot of types are inferred automatically. Take your attention to the comments; they will help you to understand the core concepts.
// ~/ctx.ts
import { createCtx } from '@reatom/core'
// create context in the app root and use it to start all computations
// for tests or SSR you will want to create a different context
export const ctx = createCtx()
All atoms and actions accepts ctx
by a first argument, it helps you a lot in many things: testing, debugging, SSR, effects chains management and logging. You could add ctx.subscribe(logs => console.log(logs))
or connect separate logger to see all changes in your app.
// ~/features/search/model.ts
import { createCtx, action, atom } from '@reatom/core'
// define your base mutable data references
// by passing a primitive initial values
const searchAtom = atom('')
const isSearchingAtom = atom(false)
const goodsAtom = atom(new Array<Goods>())
// define computed atoms to infer data
// with smart and optimized caching
const tipAtom = atom((ctx) => {
// read and subscribe by `ctx.spy`
const goodsCount = ctx.spy(goodsAtom).length
if (goodsCount === 0) {
// read atom lazy without subscribing by `get`
return ctx.get(searchAtom) ? 'Nothing found' : 'Try to search something'
}
if (goodsCount === 1) {
return `We found one treasure`
}
return `Found ${goodsCount} goods`
})
// define your actions to handle any effects and work with atoms
const onSearch = action((ctx, event) => {
// mutate base atoms by passing relative ctx and the new state
searchAtom(ctx, event.currentTarget.value)
})
const fetchGoods = action((ctx) => {
const search = ctx.get(searchAtom)
// all sync updates inside action automatically batched
// and dependent computations will call after the action callback return
isSearchingAtom(ctx, true)
// schedule side-effects
// which will be called after successful execution of all computations
const promise = ctx.schedule(async () => {
const response = await fetch(`/api/goods?search=${search}`)
if (!response.ok) throw new Error('Network response was not ok')
const goods = await response.json()
// pass a callback to `get` to batch a few updates inside async resolve
ctx.get(() => {
isSearchingAtom(ctx, false)
goodsAtom(ctx, goods)
})
})
// returned promise could be handled in place of the action call
return promise
})
As you can see, most callbacks passed in Reatom units accept ctx
as their first argument - this is the main convention, which allows you to not use imports and write more cleanly. Some advanced helper packages could extend or redefine the ctx
for additional features and typesafety, so when you need it, there will be no code changes required for the feature.
Here we just described the logic of a module which uses ctx
, but does not import it. This is because we want to use the same module in different contexts, such as view components and tests. It is a good architectural practice in itself.
So, we should connect an IO and our module together somewhere.
// ~/features/search/index.ts
import { ctx } from '~/ctx'
import { searchAtom, tipAtom, onSearch, fetchGoods } from './model'
// subscribe to your atoms
ctx.subscribe(tipAtom, (tip) => {
document.getElementById('goods-tip').innerText = tip
})
// handle user interactions by your actions
document.getElementById('search-input').addEventListener('input', (event) => {
onSearch(ctx, event)
fetchGoods(ctx)
})
Do you want to see next the docs for React adapter?
Action handling (advanced)
It is better to stay atoms stupid and handle all logic inside actions. But sometimes you need to turn the direction of your code coupling and make atoms depend on an action. And you can do it!
An action is an atom with a temporal state, which is an array of all passed payloads. This state is cleared after the transaction ends; if you try to get
or spy
an action which wasn’t called, you will receive an empty array. But if the action was called, the array will contain some elements.
// ~/modules/someFlow
import { newMessage } from '~/modules/ws'
const FLOW_NAME = 'someFlow'
export const someFlowAtom = atom(0)
// you need to subscribe to it to start watching `newMessage`
export const someFlowManagerAtom = atom((ctx) => {
console.log('example log for `ctx.get(newMessage)`', ctx.get(newMessage))
ctx.spy(newMessage).forEach(({ payload }) => {
if (payload.relation === FLOW_NAME) someFlowAtom(ctx, payload)
console.log('example log for `ctx.spy(newMessage)[N]`.payload', payload)
})
})
// socket service:
socket.on(
throttle(150, (msgs) =>
// batch updates
ctx.get(() => {
msgs.forEach((msg) => newMessage(ctx, msg))
}),
),
)
// someFlowManagerAtom reducer:
// example log for `ctx.get(newMessage)` [{ params: [1], payload: 1 }, { params: [2], payload: 2 }]
// example log for `ctx.spy(newMessage)[N]` 1
// example log for `ctx.spy(newMessage)[N]` 2
You need to know one rare tricky thing. If during a transaction you call an action and read its dependent atom a few times step by step,
ctx.get
will return an array of all passed payloads, butctx.spy
will return an array with only new elements that weren’t handled in this reducer during this transaction. To make this rare case correct, you should spy your dependencies in the same way each time, without conditions. In other words, for this case your dependencies list should be static.
API
atom
API
import { atom } from '@reatom/core'
The atom()
function is a factory for an atomic-based reactive primitive. Atoms don’t store their data (state, listeners, dependencies) within themselves; they only provide a key to a cache in ctx (context). You can think of an atom as a prototype for a cache. One of the most powerful features of Reatom is that the cache is immutable, and it is recreated on each relative update. The immutability of the cache helps to process transactions and is extremely useful for debugging. Don’t worry, it is also quite efficient.
As Atom is a key, it should be mapped somewhere to its cache. ctx
has an internal WeakMap caches
, which store your data until there is a link to Atom. When you subscribe (connect) and unsubscribe (disconnect) from Atom, the state isn’t reset or deleted; it is still stored in the cache, which will be cleared by the GC only after the link to the Atom disappears from your closures. This behavior is the most intuitive and works just like any variable storing. So, if you define a global Atom available in a few of your modules, the state will always persist in memory during the application lifetime, whether you subscribed or unsubscribed for the Atom, which is useful. If you need to clear the state on disconnect or do other lifetime transformations, check the hooks package and withreset helper.
If you need to create a base mutable atom, just pass the initial value to atom
. Pass the atom name as a second argument (it is optional but strongly recommended). The resulted atom will be mutable (Mut
) with a callable signature (a function); you can mutate it by passing a context and a new value or a reducer function.
// create
const countAtom = atom(0, 'countAtom')
// countAtom: AtomMut<number>
// mutate by setter
countAtom(ctx, 10)
// updates and return 10
// mutate by reducer
countAtom(ctx, (state) => state + 1)
// updates and return 11
All atom state changes should be immutable.
export const listAtom = atom([], 'listAtom')
// add item
listAtom(ctx, (list) => [...list, newItem])
You could create a computed derived atom by passing a function to atom
. The first argument of the passed reducer is a special kind of ctx
with a spy
function, which allows you to subscribe to the passed atom and receive its fresh state. The second argument is an optional previous state
, which you can initiate by defining a default value.
Note to TypeScript users: It is impossible to describe the reducer type with an optional generic state argument, which is returned from the function. If you use the second
state
argument, you should define its type; do not rely on the return type.
const isCountEvenAtom = atom(
(ctx) => ctx.spy(countAtom) % 2 === 0,
'isCountEven',
)
// isCountEvenAtom: Atom<number>
To store a function in Reatom state, just wrap it in a container, like
atom({ fn })
.
Reatom allows you to use native language features to describe your conditions, with all reactive dependencies reconnecting in real-time.
export const currencyAtom = atom<'us' | 'eu'>('us')
export const rateAtom = atom(1)
export const usCurrencyAtom = atom(0)
export const euCurrencyAtom = atom(0)
export const currencyValueAtom = atom((ctx) => {
const currency = ctx.spy(currencyAtom)
// use `if` or `switch` if you want
const valueAtom = { us: usCurrencyAtom, eu: euCurrencyAtom }[currency]
return ctx.spy(valueAtom)
})
Moreover, you could dynamically create and manage atoms.
const currencyAtom = atom('us')
const currenciesAtom = atom({ us: atom(0) })
export const currencyValueAtom = atom((ctx) => {
const currency = ctx.spy(currencyAtom)
let valueAtom = ctx.spy(currenciesAtom)[currency]
if (!valueAtom) {
valueAtom = atom(0)
currenciesAtom(ctx, (state) => ({
...state,
[currency]: valueAtom,
}))
}
return ctx.spy(valueAtom)
})
// update could look like
ctx.get(currenciesAtom)[ctx.get(currencyAtom)](ctx, newValue)
You could handle each update independently by passing a function to the spy
method. It is useful for action-reaction scenarios or if you need to handle a few concurrent updates.
export const changeCurrency = action<string>('changeCurrency')
export const currencyAtom = atom((ctx, state?: string) => {
ctx.spy(languageAtom, (language) => {
state = getCurrencyByLanguage(language)
})
ctx.spy(changeCurrency, (currency) => {
state = currency
})
return state
}, 'currencyAtom')
atom.pipe
API
Pipe is a general chain helper, it applies an operator to the atom to map it to another thing. Classic operator interface is <T extends Atom>(options?: any) => (anAtom: T) => aNewThing
. The main reason is a readable and type-safe way to apply decorators.
const countAtom = atom(0).pipe(
withInit(() => localStorage.getItem('COUNT') ?? 0),
)
// equals to
const countAtom = withInit(() => localStorage.getItem('COUNT') ?? 0)(atom(0))
withInit
allows you to configure the init state of the atom reading, which is more pridictable and safe for a testing sometimes. It is a part of reatom/hooks package.
Operator with
prefix mean that the target atom will be changed somehow and the returned reference will the same. reatom/async uses operators a lot to configure the behavior of the effect by composition, which is good for tree-shaking. Check naming conventions and more examples in this guild
Btw, actions has pipe
too!
atom.onChange
API
All links and computations between atoms and actions are performed in a separate context. However, there can be many cases when you need to describe some logic between two things statically outside a context, such as an action trigger on a data change, etc. The onChange
hook allows you to define this common logic right in the place of your atoms definition.
const searchAtom = atom('', 'searchAtom')
const fetchSearchSuggestion = action((ctx, search) => {
/* ... */
}, 'fetchSearchSuggestion')
searchAtom.onChange((ctx, state) => fetchSearchSuggestion(ctx, state))
// or just
searchAtom.onChange(fetchSearchSuggestion)
onChange
returns an unsubscribe function which you should use if you are adding a hook dynamically to a global atom.
The important difference of a hook from a subscription is that it does not activate the connections.
const searchAtom = atom('', 'searchAtom')
const fetchSearchSuggestion = action((ctx, search) => {
/* ... */
}, 'fetchSearchSuggestion')
const filteredSearchAtom = atom((ctx, state = '') => {
const search = ctx.spy(searchAtom)
return search.length >= 3 ? search : state
}, 'filteredSearchAtom')
// the hook will not called if the atom have no subscription, as it lazy.
filteredSearchAtom.onChange(fetchSearchSuggestion)
action
API
Actions are atoms with temporal states, which live only during a transaction. The action state is an array of parameters and payloads. The array is needed to handle multiple action calls during a transaction batch. Action callbacks can change atoms or call other actions, but their dependencies will only be notified after the callback ends - that is what a batch means.
Possible usage:
const increment = action()
// increment: Action<[], void>
const increment = action('increment')
// increment: Action<[], void>
const add = action<number>()
// add: Action<[number], number>
const add = action<number>('add')
// add: Action<[number], number>
const add = action((ctx, value: number) => value)
// add: Action<[number], number>
const add = action((ctx, value: number) => value, 'add')
// add: Action<[number], number>
const splice = action((ctx, start: number, deleteCount?: number) => {
listAtom(ctx, (list) => {
const newList = list.slice(0)
newList.splice(start, deleteCount)
return newList
})
})
// splice: Action<[number, number?], number>
Action state is Array<{ params: Array<any>, payload: any }>
, but action call returns the payload:
const submit = action((ctx, name, password) => ({ name, password }))
ctx.get(() => {
submit(ctx, 'Joe', 'Bom')
// { name: 'Joe', password: 'Bom' }
submit(ctx, 'Koe', 'Rog')
// { name: 'Koe', password: 'Rog' }
ctx.get(submit)
// [{ name: 'Joe', password: 'Bom' }, { name: 'Koe', password: 'Rog' }]
})
action.onCall
API
The same as atom.onChange, but with the relative arguments: payload
and params
.
const doSome = action((ctx, a, b) => ({ a, b }), 'doSome')
doSome.onCall((ctx, payload, params) => {
console.log(payload, params)
// `doSome(ctx, 1, 2)` will log "{ a: 1, b: 2 }, [1, 2]"
})
ctx
API
CTX
is the main shell for a state of all atoms, all user and meta data leaves here. Each atom and action produces an immutable version of the context and it should not be mutated!
One more rule, which you probably won’t need, but we should still mention it: don’t run one context inside another, such as ctx1.get(() => ctx2.get(anAtom))
- this will throw an error.
ctx.get
atom API
Get fresh atom state
get<T>(anAtom: Atom<T>): T
ctx.get
batch API
Start transaction and batch all updates, same as in action call
get<T>(cb: () => T): T
ctx.subscribe
atom API
Subscribe to atom new state. Passed callback called immediately and after each atom state change.
subscribe<T>(anAtom: Atom<T>, cb: (newState: T) => void): () => void
ctx.subscribe
log API
Subscribe to transaction end. Useful for logging.
subscribe(cb: (logs: Array<AtomCache>, error?: Error) => void): () => void
ctx.schedule
To achieve atomicity, each update (action call / atom mutation) starts a complex batch operation, which tries to optimize your updates and collect them into a new immutable log of new immutable cache snapshots. If some computation throws an error (like can't use property of undefined
) the whole update will be canceled, otherwise the new caches will be merged into the context internal caches
weak map. To achieve purity of computations and the ability to cancel them, all side-effects should be called separately in a different queue, after all computations. This is where schedule
comes in; it accepts an effect callback and returns a promise which will be resolved after the effect call or rejected if the transaction fails.
const fetchData = action((ctx) => {
loadingAtom(ctx, true)
ctx.schedule(effect).then((data) => {
loadingAtom(ctx, false)
dataAtom(ctx, data)
})
})
The unique feature of Reatom and the schedule specially is ability to define the target queue. The second argument of schedule
is a priority number:
-1
- rollback queue, useful when you need to do a side-effect during pure computations. Check example below.0
- computations queue, schedule pure computation, which will call right after current batch.1
- the default near effect queue, used to schedule regular effects. This effects calling could be redefined (delayed) incallNearEffect
option ofcreateCtx
2
- lates effect queue, used to schedule subscribers. This effects calling could be redefined (delayed) incallLateEffect
option ofcreateCtx
.
Read more in lifecycle guild.
ctx.schedule
rollback API
Sometimes you want to do a side-effect during clean calculations or need to store some artifact of an effect and store it. To make it clean, you should describe a rollback (cleanup) function for the case of an unexpected error by passing -1
as the second argument of ctx.schedule
. Check out this example with a debounced action:
const timeoutIdAtom = atom(-1)
// `timeoutIdAtom` update is in a schedule cause an extra transaction - not handy
export const doSome = action((ctx) => {
const timeoutId = ctx.get(timeoutIdAtom)
ctx.schedule(() => {
clearTimeout(timeoutId)
const newTimeoutId = setTimeout(some)
timeoutIdAtom(ctx, newTimeoutId)
})
})
// `timeoutIdAtom` update is during transaction more obvious
export const doSome = action((ctx) => {
const timeoutId = ctx.get(timeoutIdAtom)
ctx.schedule(() => clearTimeout(timeoutId))
const newTimeoutId = setTimeout(some)
timeoutIdAtom(ctx, newTimeoutId)
ctx.schedule(() => clearTimeout(newTimeoutId), -1)
})