🚀 Socket Launch Week Day 5:Introducing Repository Access Permissions and Custom Roles.Learn more
Sign In

react-machine

Package Overview
Dependencies
Maintainers
1
Versions
10
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

react-machine

A lightweight state machine for React applications

Source
npmnpm
Version
0.2.0
Version published
Weekly downloads
671
56.05%
Maintainers
1
Weekly downloads
 
Created
Source

react machine logo, a man with an open head with a state chart inside

Finite state machine hook for React featuring context, reducers, actions and effects.


When useState or useReducer is not enough, the useMachine hook can be used to express more complex component state and business logic. Machines are especially useful in handling asynchronouse effects in your components (for example, saving a form). In fact, you can think of useMachine as useReducer with native support for asynchronous logic.

Features include:

  • a single useMachine hook for declaratively describing state machines
  • define states and transitions between states
  • immediate transitions with guards
  • internal transitions for updating context or triggering actions
  • transition hooks - reduce, assign, action, guard
  • state enter and exit hooks - reduce, assign, action, invoke, effect
  • invoke for async promise returning functions
  • effect for custom async logic and long running activities
  • pure stateless machine implementation using useReducer and useEffect to hook into React
  • hierarchical and parallel states (coming in V2 in 2021)
  • semantics guided by the SCXML spec (coming in V2 in 2021)

Example

import React from 'react'
import { useMachine } from 'react-machine'

const isSuccess = (ctx) => ctx.item.status === 'success'
const isError = (ctx) => ctx.item.status === 'error'
const increment = (ctx) => ({ ...ctx, count: ctx.count + 1 })
const retry = (ctx) => ctx.retry()

const machine = ({ state, transition, immediate, internal, enter }) => {
  state(
    'loading',
    internal('assign', { assign: true }),
    immediate('counter', { guard: isSuccess }),
    immediate('error', { guard: isError })
  )

  state('error',
    internal('assign', { assign: true }),
    transition('retry', 'loading', { action: retry })
  )

  state(
    'counter',
    enter({ assign: { count: 0 } }),
    internal('assign', { assign: true }),
    internal('increment', { reduce: increment })
  )
}

export function Component({ item, retry }) {
  const [{ name, context }, send] = useMachine(machine, { item, retry })

  if (name === 'loading') return <div>Loading</div>
  if (name === 'error') return <button onClick={() => send('retry')}>Retry</button>
  return <button onClick={() => send('increment')}>{context.count}</button>
}

Also see:

Motivation

react-machine is very much inspired by XState and Robot - thank you for really great libraries 🙏

Comparison to Robot

Robot is a great Finite State Machine library with integrations into React, Preact, Svelte and more. It has a neat succint API that inspired the API of react-machine and boasts a lightweight implementation. Here are some differences:

  • Robot requires every helper function to be imported individually. react-machine uses a similar DSL, but passes helpers as arguments to the machine creation function or takes them as options. This means you only ever need to import useMachine, which is more akin to how you'd use useState or useReducer for managing state.
  • Robot has some support for nesting machines, but react-machine (in the upcoming V2) will support arbitrarily nested and parallel states, adhering more closely to the SCXML spec.
  • Robot does not support internal transitions, making it difficult to update machine context in the middle of an async function invocation.
  • Robot does not have enter and exit hooks, and does not allow custom effect implementations.

Comparison to XState:

XState is the most powerful modern state chart / state machine implementation for JavaScript. It's rich in features and supports React and Vue out of the box. Here are some differences:

  • react-machine strives to create a smaller surface area, less features, less options, less packages. This could be seen as a good or a bad thing depending on your perspective and your requirements. The goal is to seek simplicy, which can be subjective. For example, you will not find actors, machine inter messaging, delayed events or history states in react-machine.
  • related to the point above, full compatibility / serialisation to SCXML is a non goal for react-machine, SCXML (and it's interpration algorithm in particular) is only used to guide the implementation of react-machine.
  • react-machine uses a more functional machine declaration DSL that is closer to that found in Robot, whereas XState declares machines using a deeply nested object notation, this might well be a personal preference, give both a try, and also XState might gain new optional DSL adapters in the future.
  • XState provides visualisation of it's state charts, a feature that could be added to react-machine in the future.

Conclusion

In summary, react-machine is an experiment in creating a lightweight, simple, flexible solution for component level state management in React. The core machine logic is pure and stateless, which allows for delegating the state storage to useReducer and effect execution to useEffect, done soe with concurrent mode compatibility in mind.

API

Machines are created using the API passed into machine description function, here's an exhaustive example showing all possible types of transitions and hooks:

const [state, send] = useMachine(({ state, transition, immediate, internal, enter, exit }) => {
  state(stateName,
    enter({ reduce, assign, action, invoke, effect }),
    transition(event, target, { guard, reduce, assign, action }),
    immediate(target, { guard, reduce, assign, action }),
    internal(event, { guard, reduce, assign, action }),
    exit({ reduce, assign, action }),
  )
}, context)

Hook

State machine description

Transition hooks

useMachine(description, context, options)

Create and initialise the machine.

  • description - the machine description function invoked with state, transition, immediate, internal, enter, exit as arguments.
  • context - the context to be assigned to the machine's state. Since it's common to pass props and other computed data via context, by default, whenever any of the values of the context change, the hook will send an event of type assign with the context object spread onto the event object, this event can be renamed or disabled in options.
  • options - hook options

Available options:

  • assign (default: "assign") - the name of the event to be sent when context values change. Set this to false to disable sending the event altogether.
  • deps - by default all context values are checked for changes in between hook invocations. Use this option to customize the dependency array.

Returns [state, send, machine]:

  • state - current state of shape { name, context, final }
  • send - send an event, e.g. send('save') or send({ type: 'save', item: 'x' })
  • machine - a stateless machine description that could be used to transition to new states
const myMachine = useCallback(({ state, transition }) => {
  state('a', transition('next', 'b'))
  state('b', transition('next', 'c'))
  state('c')
})
const [state, send, machine] = useMachine(myMachine, { x: 0 })
const { name, context, final } = state

state(name, ...transitions)

Declare a state.

  • name - name of the state
  • transitions - any number of available: transition(), immediate(), internal(), enter(), exit()
state('loading')
state('loading', transition('go', 'ready'))
state('loading', immediate('ready', { guard: (ctx) => ctx.loaded }))

transition(event, target, options)

Declare a transition between states.

  • event - the name of the event that will trigger this transition
  • target - the name of the target state
  • options - in the shape of { reduce, assign, action, guard }
transition('save', 'saving')
transition('reset', 'edit', { reduce: ctx => ({ ...ctx, data: null }) })
transition('close', 'closing', { action: ctx => ctx.onClose() })

immediate(target, options)

A special type of transition that is executed immediately upon entering (or re-entering a state with an internal transition). If no guard option is used, the transition will always immediately be applied and move the machine to a new state. If the guard option is used, the transition will only be applied if the guard condition passes. Note that, when immediate transitions take place, all of the intermediate transition hooks and intermediate state enter/exit hooks are triggered, however the effects (including invoke) are only executed for the final state, not any of the intermediate states.

  • target - the name of the target state
  • options - in the shape of { reduce, assign, action, guard }
immediate('ready')
immediate('ready', { guard: { guard: (ctx) => ctx.loaded } })

internal(event, options)

A special type of transition that does not leave the state and does not trigger any enter/exit hooks. Useful for performing actions or updating context without leaving the state. Note: this transition does re-evaluate all immediate transitions of the state.

  • event - the name of the event that will trigger this transition
  • options - in the shape of { reduce, assign, action, guard }
internal('assign', { assign: true })
internal('reset', { assign: { count: 0 } })

enter(options)

Hooks to run when entering a state.

  • options - in the shape of { reduce, assign, action, invoke, effect }
enter({ action: ctx => ctx.start() })
enter({ invoke: ctx => ctx.fetch('/data') })
enter({ assign: { count: 0 } })

exit(options)

Hooks to run when leaving the state.

  • options - in the shape of { reduce, assign, action }
exit({ action: ctx => ctx.stop() })
exit({ assign: { error: null } })

guard

If the guard condition fails, the transition is skipped when matching against the event and selection proceeds to the next transition. Commonly used with immediate transitions, but works with any type of transition.

{ guard: (context, event) => context.status === 'success' }

reduce

Updated context based on current context and the incoming event.

{ reduce: (context, event) => nextContext }
{ reduce: (context, { type, ...payload }) => nextContext }
{ reduce: [reduce1, reduce2] }

assign

Return a partial context update object, that will be immutably assigned to the current context. A commonly useful shortcut for assigning event paylods to the context.

{ assign: (context, { type, ...payload }) => ({ ...context , ...payload }) }
{ assign: true } // same as above
{ assign: { static: 'value' } }
{ assign: [assign1, assign2] }

action

A fire and forget action executed immediately (synchronously) upon sending an event.

{ action: (context, event) => context.onClose() }
{ action: [action1, action2] }

invoke

A way to invoke async functions as part of entering a state. If the promise is fulfilled, an event of shape { type: 'done', data } is sent, and if the promise rejects, an event of { type: 'error', error } is sent. Note, if the machine exits the state while the promise is pending, the results will be ignored and no event will get sent. Note, internally, invoke is turned into an effect.

state('save',
  enter({ invoke: async (context, event) => context.save() }),
  transition('done', 'show', { assign: (ctx, event) => ({ item: event.data }) }),
  transition('error', 'edit', { assign: (ctx, event) => ({ error: event.error }) }),
)

effect

A way of handling side effects, async effects, subscriptions or activities. Once the state is entered, the effect gets started (in useEffect and only after finalising all of the immediate transitions) and can send any number of events. Note that context will be valid when initially running the effect, but will get stale afterwards, and is best read in subsequent internal transitions. Also note that send will be ignored after the effect is cleaned up, and similarly send can not be used in the cleanup function of the effect.

const addPing = (ctx, event) => ({ pings: ctx.pings.concat(event.ping) })
const listenToPings = (context, event, send) => {
  const cancel = context.ponger.subscribe((ping) => {
    send({ type: 'ping', ping })
  })
  return () => cancel()
}

state('save',
  enter({ assign: { pings: [] }, effect: listenToPings }),
  internal('ping', { assign: addPing })
)

Roadmap

V1

  • add debug logging option
  • write proper TypeScript type definitions
  • review the final state logic, e.g. wrt to internal transitions

V2

  • add hierarchical and parallel states
  • only new effects are run, old ones can stay running
  • change from state.name string to state.value object
  • introduce state.matches() api

V3

  • add compatibility with XState visualiser, serialize into compatible JSON

Keywords

react

FAQs

Package last updated on 22 Dec 2020

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts