Skip to content

Learning

Before we start

In the examples we will use the @reatom/framework package which is an umbrella package for the most common reatom packages. You can install it with the command

npm install --save @reatom/framework

Basic primitives

The reatom is based on three basic primitives: Atom, Action and Context. Below you can see how they are used together, after which we will look at each line and what happens in it

Context

The first thing we did was create a context. It is required for read, modify and subscribe operations. Later we will see what tricks it allows us to do, but now we will focus on the fact that it is enough to create one context for the whole application

import { createCtx } from '@reatom/framework'
const ctx = createCtx()

Atom

Create Atom

import { atom } from '@reatom/framework'
const aAtom = atom(1, 'aAtom')
const bAtom = atom(2, 'bAtom')

An atom is like a variable, it has a type and a value. However, unlike a variable, we can track changes in its value and react to that changes in some way. To create an atom we use the atom factory function

In line above we create atom with initial value 1 and name aAtom.
The name, though not required, will come in handy during the debug stage

Atoms can also be computable, i.e. use the values of other atoms. This line of code can be read as - “To find out the value of cAtom you need to read the current value of aAtom and bAtom add summarize it”.

const cAtom = atom((ctx) => ctx.spy(aAtom) + ctx.spy(bAtom), 'cAtom')

Computed atoms should be pure functions to archive the correct order of all computations

Read Atom

To read the value of an atom you need a previously created context

import { atom, createCtx } from '@reatom/framework'

const ctx = createCtx()
const aAtom = atom(1, 'aAtom')
const bAtom = atom(2, 'bAtom')
const cAtom = atom((ctx) => ctx.spy(aAtom) + ctx.spy(bAtom), 'cAtom')

ctx.get(aAtom) // 1
ctx.get(cAtom) // 3

It is important to note that the retrieval of the value of an atom will happen only after its reading. In other words, it means that if a computed atom has not been read by anyone, the atom will not run the function passed to it

Update Atom

To change the value in an atom you also need a context, but this time you need to pass it to the atom

import { atom, createCtx } from '@reatom/framework'

const ctx = createCtx()
const aAtom = atom(1, 'aAtom')
aAtom(ctx, 3)

The current value of the atom can also be used in the update operation, by passing function

const greetAtom = atom('Hello', 'greetAtom')
greetAtom(ctx, (greet) => greet + ', atom') // 'Hello, atom'
greetAtom(ctx, (greet) => greet + ', atom') // 'Hello, atom, atom'

The previous state can also be used in computable atoms

import { atom, isDeepEqual } from '@reatom/framework'

const listAtom = atom<number[]>([1, 2, 3, 4, 5], 'listAtom')
const evenListAtom = atom((ctx, state = []) => {
  const newState = ctx.spy(listAtom).filter((n) => n % 2 === 0)
  return isDeepEqual(state, newState) ? state : newState
}, 'evenListAtom')

Subscribe to Atom

Finally you can subscribe to atom changes using context

import { atom, createCtx } from '@reatom/framework'

const ctx = createCtx()
const aAtom = atom(1, 'aAtom')
const bAtom = atom(2, 'bAtom')
const cAtom = atom((ctx) => ctx.spy(aAtom) + ctx.spy(bAtom), 'cAtom')

ctx.subscribe(cAtom, (c) => {
  const a = ctx.get(aAtom)
  const b = ctx.get(bAtom)
  console.log(`${a} + ${b} = ${c}`)
})

aAtom(ctx, 3)
// logs: 3 + 2 = 5

bAtom(ctx, 4)
// logs: 3 + 4 = 7

bAtom(ctx, 4)
// does not log anything, as the state is not changed

Actions

Actions are transactions

Setting atoms manually is good thing but more often we want to do many changes at once. Let’s edit example so that we are able to change a and b simultaneously.

import { createCtx, atom, action } from '@reatom/framework'

const ctx = createCtx()

const aAtom = atom(1, 'aAtom')
const bAtom = atom(2, 'bAtom')
const cAtom = atom((ctx) => ctx.spy(aAtom) + ctx.spy(bAtom), 'cAtom')

const setParams = action((ctx, a: number, b: number) => {
  console.log(`change a=${a}, b=${b}`)
  aAtom(ctx, a)
  bAtom(ctx, b)
}, 'setParams')

ctx.subscribe(cAtom, (c) => {
  const a = ctx.get(aAtom)
  const b = ctx.get(bAtom)
  console.log(`${a} + ${b} = ${c}`)
})

setParams(ctx, 10, 12)
// change a=10, b=12
// 10 + 12 = 22

setParams(ctx, 10, 12)
// change a=10, b=12
// (does not log the result because it has not changed)

As we see here the subscribe callback only called with both a and b values changed. This is called “transactions”. They help to reduce the number of subscriber calls, and avoid the creation of unwanted intermediate states

Async actions

Creating asynchronous actions is also possible, but keep in mind that async operations must be called inside ctx.schedule callback

import { createCtx, atom, action } from '@reatom/framework'

export const dataAtom = atom(null)

export const fetchData = action(async (ctx) => {
  const data = await ctx.schedule(() => {
    const response = await fetch('https://jsonplaceholder.typicode.com/todos/1')
    const payload = await response.json();
    return payload;
  })
  dataAtom(ctx, data)
})

Actions nesting

You can call actions from other actions. And asynchronous actions will return the promise

import { action, atom } from '@reatom/core'

export const todoAtom = atom(null)
export const isLoadingAtom = atom(false);

export const fetchTodo = action(async (ctx) => {
  const response = await ctx.schedule(() => fetch('https://jsonplaceholder.typicode.com/todos/1'))
  return await response.json();
})

export const loadTodo = action(async (ctx) => {
  try {
    isLoadingAtom(ctx, true)
    const data = await ctx.schedule((ctx) => fetchTodo(ctx))
    todoAtom(ctx, data);
  } catch (e) {
    console.error(e)
  } finally {
    isLoadingAtom(ctx, false)
  }
})

Stakblitz

Advanced

Multiple contexts

Contexts are used to glue up atoms and actions, track transactions and many more features. You can use same dependency trees in different contexts:

import { createCtx, atom } from "@reatom/framework"

const ctx1 = createCtx()
const ctx2 = createCtx()

const someAtom = atom(1, 'someAtom')

console.log(ctx1.get(someAtom), ctx2.get(someAtom))
// logs: 1, 1

// change value of an atom only in one context
someAtom(ctx1, 100)

console.log(ctx1.get(someAtom), ctx2.get(someAtom))
// logs: 100, 1

Computed atoms can’t depend on atom values from different contexts and actions can’t change atoms of different context. Context will initiate new item state referring to that atom.

This enables us to easily test things. But beware of function closures because they are not context dependent!

import { createCtx, atom } from "@reatom/framework"

const ctx1 = createCtx()
const ctx2 = createCtx()

// DON'T DO THIS
let someExternalVar = 1
const someAtom = atom(() => someExternalVar, 'someAtom')

// check using ctx1
console.log(ctx1.get(someAtom))
// logs: 1

someExternalVar = 100

// check using ctx1 and ctx2
console.log(ctx1.get(someAtom), ctx2.get(someAtom))
// logs: 1, 100
// because ctx1 cached 1 and ctx2 was only read when value changed