I enjoy the concept of reducer composition in Redux, but have run into a scenario where I would like to split a reducer up, but the children reducers would then rely on state-slices from the others to make their changes.
For Example
In my state, I need to keep track of the following:
[ 2, 3, 4, 5, 6 ])[ 1 .. (selectedRank - 1) ]
Initially, I had a larger reducer which encapsulated all of these aspects:
function rankAndTraining(state = {
  selectedRank            : 4,
  availableRanks          : [ 2, 3, 4, 5, 6 ],
  availableTrainingLevels : [ 1, 2, 3 ],
  selectedTrainingLevel   : 2,
}, action) {
  .
  .
  .
    case SELECT_RANK: 
      let newRank = action.rank;
      if(!availableRanks.contains(newRank)) {
        // If new rank is not an allowed value, do nothing.
        return state;
      }
      // Update selectedRank with newRank...
      // Update availableTrainingLevels array based on newRank...
      // [ 1 ... (selectedRank - 1) ]
      // Update selectedTrainingLevel if it is now out of range 
      // of availableTrainingLevel (i.e. set to highest value of new range)
      return state;
  .
  .
  .
}
I wanted to split this reducer up, as I felt that the ranks and the training levels could be maintained in separate reducers.
However, there are dependencies across the state pieces that would still need to be handled if the reducer was split, such as:
If I were to split the training level bits into another reducer, then they would have no way to know the result of this "rank check", since all they would "see" in their state-slice is the training level pieces.
In this scenario, is the above reducer really the "smallest slice" of state that I can get, given the dependencies between them? Or is there a better way to split it that I may not be seeing?
If you use the Redux thunk middleware you can inspect the entire state first before you actually dispatch your action at all instead of dispatching your action and then conditionally update your state in the reducer:
function selectRankIfAllowed() {
  return (dispatch, getState) => {
    const { availableRanks } = getState(); 
    if(!availableRanks.contains(newRank)) {
      // If new rank is not an allowed value, do nothing.
      return state;
    }
    dispatch(selectRank());
  };
}
https://github.com/gaearon/redux-thunk
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With