Application state management

Although React_useReducer and React_useContext have many sound use cases by themselves, when they are combined they offer a way to acheive a system of global state management, without utilising third-party libraries like Redux.

Requirements

The Context API combined with a reducer addresses the following needs:

  • be able to consume global state from anywhere in the component hierarchy without prop-drilling
  • update global state from within child components from anywhere in the component hierarchy
  • access, maintain and update complex structured data (arrays/objects) not just single values, as global state

Implementation

In essence the approach is as follows:

  • Create a context provider that houses the global state object
  • Attach this state to a reducer function that operates on the state object through dispatch methods

First off, our initial state and overall state object:

const initialState = {
  count: 0,
};

Next, we create our context. This is what we will invoke when we want to consume our state.

export const CounterContext = React.createContext({});

Now we need a reducer that will handle the state updates. We will just use the same setup as we used in the example of React_useReducer:

function reducer(state, action) {
  switch (action.type) {
    case "increase":
      return { ...state, counter: state.counter + 1 };
      break;
    case "decrease":
      return { ...state, counter: state.counter - 1 };
      break;
    case "increase_by_payload":
      return { ...state, counter: state.counter + action.payload };
    default:
      throw new Error();
  }
  return newState;
}

Finally, we need to make a provider as a top-level component that receives the reducer’s state and dispatch methods as props:

export const CounterProvider = ({children}) => {
    // We pass our reducer function and initial state to useReducer as params
    const [state, dispatch] = React.useReducer(reducer, initialState)

    // We create our provider and pass the reducer state and update method as props. This is the provider to the CounterContext consumer
    <CounterContext.Provider value={{state, dispatch}}>
        {children}
    </CounterContext.Provider>
}

Now whenever we want to consume or update our global state, we can invoke the context within a lower-level component, for example:

const {state, dispatch} = useContext(CounterContext);

dispatch({
    type 'increase_by_payload',
    payload: 4
})

console.log(state.counter) // 4