Skip to content

mattpocock/use-fsm-reducer

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

23 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

useFsmReducer

Heavily inspired by David Khourshid's article.

useFsmReducer is a hook which helps to bring state machine code quality to a common React pattern - useReducer.

It helps you bootstrap a reducer, similar to redux-toolkit, but also helps extract your effects and help you scale complexity.

Installation

npm i use-fsm-reducer

Example Code

import useFsmReducer from 'use-fsm-reducer';

/**
 * Our state can be in any of these states.
 * Note that data is only declared when we're in
 * the 'loaded' state.
 */
type State =
  | { type: 'pending' }
  | { type: 'errored' }
  | { type: 'loaded'; data: string };

/**
 * We declare all possible actions that can flow
 * through this reducer
 */
type Action =
  | { type: 'reportDataLoaded'; data: string }
  | { type: 'reportError' }
  | { type: 'retry' };

/**
 * We declare all effects that this reducer can fire off.
 *
 * An effect is anything you would put inside a useEffect - an
 * impure function that happens as a result of piece
 * of state changing, or an action firing.
 */
type Effect =
  | { type: 'loadData' }
  | { type: 'showErrorMessage'; message: string };

/**
 * You get back state and dispatch, the API you're
 * used to from useReducer
 */
const [state, dispatch] = useFsmReducer<State, Action, Effect>({
  /**
   * Here, we declare the initial state. Each state
   * must have a 'type'.
   */
  initialState: {
    type: 'pending',
  },
  /**
   * We can run some initial effects on mount.
   */
  runEffectsOnMount: [{ type: 'loadData' }],
  states: {
    /**
     * Each state can handle actions on its own,
     * and fire off effects.
     */
    pending: {
      on: {
        reportDataLoaded: (state, action) => {
          return {
            type: 'loaded',
            data: action.data,
          };
        },
        reportError: () => {
          return {
            type: 'errored',
            /**
             * Return an array of all the effects
             * you'd like this state change to trigger
             */
            effects: [{ type: 'showErrorMessage', message: 'Oh noooo!' }],
          };
        },
      },
    },
    errored: {
      on: {
        retry: () => {
          return {
            type: 'pending',
            effects: [{ type: 'loadData' }],
          };
        },
      },
    },
  },
  effects: {
    loadData: ({ dispatch }) => {
      /** Fetch some data and fire off more actions */
      fetch('...')
        .then(res => res.json())
        .then(data => {
          dispatch({ type: 'reportDataLoaded', data: JSON.stringify(data) });
        })
        .catch(() => {
          dispatch({ type: 'reportError' });
        });
    },
    showErrorMessage: ({ effect }) => {
      console.error(effect.message);
    },
  },
});

There's another example in examples/form.ts

About

useReducer for state machine enthusiasts

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published