Skip to content

Latest commit

 

History

History
374 lines (283 loc) · 15.8 KB

README.md

File metadata and controls

374 lines (283 loc) · 15.8 KB

Redux Patterns

These are some recommended patterns for setting up a React + Redux project. This document assumes basic knowledge of Redux.

Table of Contents

Visualizing Redux

Note: Actions can take one of the two roles in this diagram, not both.

Separation of Concerns

The most important takeaway from this diagram is the delineation of responsibilities between the different parts of your app. When developing each part of your app, you should keep in mind the responsibilities of the particular part you are writing. This can be difficult if you are touching every part represented in the diagram for one feature but it is important for long-term maintenance and avoiding highly-coupled spaghetti code.

When writing code, always write from the perspective of the module you are building (keeping in mind that module's responsibilities) rather than from your personal high-level perspective. Future developers (including yourself in six months) will inherit your code, not what's in your mind as you are writing your code. To facilitate this, be sure to make each piece of code you write modular and reaonably readable without context.

Consider the following example. Suppose you are tasked with adding add-to-cart functionality in a new part of your app. There already exists an API function for POSTing to the server and you come across the function definition as you begin implementation.

// api.js

// BAD
const addToCart = (product, user) => {
  const formattedProduct = formatProduct(product);
  ...
};

// GOOD
const addToCart = ({ productId, color, quantity }, userId) => {
  const formattedProduct = formatProduct(productId, color);
  ...
};

Both the addToCart and the formatProduct functions are examples. In the bad example, you would have to figure out what keys the product and user objects need to be provided. As you dive into the implementation of the function, you come across a function called formatProduct. The product object is threaded into this function as well so you now have to search for and open the file in which this function is defined. Of course, the formatProduct function definition calls another utility function that also receives the entire product object. You can see where this going. This is, unfortunately, extremely common. The developer, in this case, wrote their code from their own high-level perspective and did not provide a clear, descriptive API for how to use this function.

In the good example, the functions are written from the perspective of the functions, themselves. These functions don't know what else exists in the app. They can only accept arguments, do something with those arguments, and possibly return something. Thus, the developer provided a clear contract that communicates how to interact with these functions and no unnecessary data is passed into the function.

Responsibilities

The React component tree's responsibility is the visual UI. It is the presentational layer of the app and it should live completely independent of Redux. The component tree should have little to no knowledge of Redux.

The Redux store is responsible for storing the state of your app and the reducers are responsible for translating actions into state mutations. This is where much of your app's business logic should live (aside from any async middleware layer). A reducer should have knowledge of the store state slice that it governs and of actions.

Containers act as the intermediary between the presentational layer and the business logic. It is the containers' responsibility to apply transformed state data and actions into component props. Containers should have knowledge of the wrapped component's props, available actions, and available selectors.

Selectors act as the translation layer between the Redux store's state and the presentational layer. This layer is used to extract and transform state data into a format that can be consumed by the presentational layer. Selectors should have knowledge of the state tree and component props.

Actions act as the communication channel between your containers and reducers. Depending on the pattern you go with, actions will have knowledge of the containers or the reducers, not both.

Reducers and State

Reducer Types

It is recommended to categorize your reducers into two types: entity reducers and domain-specific reducers.

Entity Reducers

Entity reducers are reducers that deal with data entities, typically retrieved from a server endpoint. There are many ways to construct the state tree to house these entities, but typically, you will want some kind of key-value dictionary with ids pointing to objects. A very simple example of an entity slice of state might look something like the following:

{
  19367: { sku: 19367, name: 'Jeans', ... },
  49373: { sku: 49373, name: 'T-Shirt', ... },
  30958: { sku: 30958, name: 'Dress', ... },
  ...
}

In this example, we have a slice of state that stores products. The product's sku is known to be a unique identifier so it is used as the key. You will likely need to transform the data from the server into this format. This work can be done in the reducer when it receives the payload from the server.

A key-value dictionary allows for instant lookup and should be used by other parts of your app to retrieve the data it needs. If another reducer needs to store a reference to an entity or a list of entities, store the ids in the reducer and use selectors to retrieve the actual data.

These reducers should only append, remove, or edit the entities.

Domain-Specific Reducers

These reducers are like your typical Redux reducers. These should be grouped in a way that makes sense for your app. For instance, they can be grouped by page, view, feature, etc. As mentioned previously, if one of these reducers needs to store a reference to an entity, store the entity's id to prevent data duplication. Preventing data duplication will save memory and prevent bugs that result from inconsistent data entries.

Selectors

Selectors act as the layer between the Redux store and the containers. Selectors need to know how the state tree is shaped as well as how the containers expect to receive data. In general, you will probably want two types of selectors: general access selectors and container-specific selectors.

General access selectors are more basic and help you access particular parts of the state tree. These selectors are meant to be shared and are the building blocks used to compose more complex selectors.

Container-specific selectors are composed of general access selectors and custom logic to transform the data into the correct format for the container consuming it. Use reselect for a performance boost when writing selectors that require an operation greater than O(1).

Suppose you are tasked with displaying the cart page for a user in an ecommerce app. You have an entity reducer for all of the products that have been fetched in the app so far. You also have a domain-specific cart view reducer. In the following code sample, we have general access selectors for the product entities and the products in cart as well as a container-specific selector for transforming our product data so that it is suitable for the product list component within the cart view.

// selectors/entities/products.js

const selectProducts = state => state.entities.products;

// selectors/views/cart.js

import { createSelector } from 'reselect';

const selectProductIdsInCart = state => state.cart.productIdsInCart;

const selectProductsForCart = createSelector(
  selectProducts,
  selectProductIdsInCart,
  (products, productIdsInCart) => productIdsInCart.map(productId => {
    const product = products[productId]

    return {
      imageSrc: product.imageUrl,
      productName: `${product.color} ${product.name}`,
      productDescription: product.description
    };
  })
);

Notice that the names of the fields being passed to the container are different from the names of the product entities' fields. Furthermore, there is one field that is composed of multiple pieces of data. This will be the case if the underlying React components are built independent of Redux, as React component props should be named in a way that makes sense for the component. Doing so makes the component more easily understandable to other developers and increases its reuse value. This will be discussed in more detail in the containers and components section.

Additionally, the product entity likely has many more fields than four but in this particular case, the underlying React component only needs three fields. Blindly passing data from the store to components will inevitably lead to spaghetti code and data duplication down the line. It is recommended to format the data before it is passed into the React layer of the app and to only pass the data that is required in the component. By using selectors as a translation layer in the data flow, you will be able to maintain the integrity of the underlying data while also maintaining the modularity of the presentational React layer of the app. In other words, this promotes separation of concerns.

Containers and Components

Forthcoming

Actions

You can conceptualize actions in one of two ways: actions as commands and actions as events. You will need to agree to a pattern that makes sense for your team and try to stick with it.

Actions as Commands

This is, by and far, the most common way people think of actions. In this pattern, actions are typically coupled to reducers. Reducers essentially provide actions as an API to interact with the state that a particular reducer governs. Thus, actions serve as commands that are issued which are then fulfilled by the reducers. Action types should be named ${VERB}_${NOUN}.

Consider the following example. Your app needs to display modals at a global level and will only display one modal at a time. You are tasked with showing a confirmation modal when the user clicks the add to cart button on a product. You also need to add the item to the user's cart.

You've decided to have a modal reducer and a cart reducer. The modal reducer will accept two action types: OPEN_MODAL and CLOSE_MODAL. OPEN_MODAL comes with a payload, which contains a string that represents the modal that needs to be displayed. In one of your top-level components, you will use the string to access a modal from a dictionary of possible modals to display.

Below is sample code of how this could be accomplished using actions as commands. Note that in a real app, you'd likely split the code into domain-specific modules. For simplicity, the sample code puts everything into one place.

To illustrate the modal mechanism, additional modals will also be shown in the sample code.

// actions.js

const addProductToCart = productId => ({ type: ADD_PRODUCT_TO_CART, payload: productId });
const openModal = modalName => ({ type: OPEN_MODAL, payload: modalName });
const closeModal = () => ({ type: CLOSE_MODAL });

// reducers.js

const cartInitialState = { productIds: [] };

const cartReducer = (state = cartInitialState, { type, payload }) => {
  switch (type) {
    case ADD_PRODUCT_TO_CART:
      return { ...state, productIds: [...state.productsIds, payload ] };
    default:
      return state;
  }
};

const modalInitialState = { currentModal: null };

const modalReducer = (state = modalInitialState, { type, payload }) => {
  switch (type) {
    case OPEN_MODAL:
      return { ...state, currentModal: payload };
    case CLOSE_MODAL:
      return { ...state, currentModal: null };
    default:
      return state;
  }
};

// product/container.js

const mapDispatchToProps = dispatch => ({
  onAddToCartClick: productId => {
    dispatch(addProductToCart(productId));
    dispatch(openModal(ADD_TO_CART_CONFIRMATION_MODAL));
  }
});

// sign-up/container.js

const mapDispatchToProps = dispatch => ({
  onTermsAndConditionsClick: () => dispatch(openModal(TERMS_AND_CONDITIONS_MODAL))
});

// home/container.js

const mapDispatchToProps = dispatch => ({
  onSignInClick: () => dispatch(openModal(SIGN_IN_MODAL))
});

Other containers that need to launch a modal would also dispatch the openModal action and pass a modal string. In this scenario, each container needs to know what modal it's supposed to launch, thus housing business logic closer in proximity to the React component tree than to the Redux store.

This pattern lends itself to redux-thunk as thunks are essentially async commands. This pattern can still work with redux-saga if sagas are treated similarly to reducers, in that they provide commands as an API for interacting with them.

Actions as Events

In this pattern, actions represent events that occured outside of the store. This includes user interaction, server responses, etc. As a result, actions are coupled more to the containers and underlying component tree and represent what happened in the past. Containers will fire off these actions simply to inform your app that something happened. After that, it's up to the consumers of these action events to decide how to react. Reducers, in this case, act as reactive watchers. Actions should be labeled in the past tense.

We will use the same example from above to illustrate what this might look like in code. Instead of three specific command actions, we will only need one action, ADD_TO_CART_CLICKED, to signify the fact that the add to cart button was clicked. The logic for this same feature thus lives within the reducers rather than spread across several disparate actions.

Again, additional modals are shown in the sample code and in a real app, this code would be split into modules. The following replicates the functionality of the above example within the actions-as-events pattern.

// actions.js

const addToCartClicked = productId => ({ type: ADD_TO_CART_CLICKED, payload: productId });
const termsAndConditionsClicked = () => ({ type: TERMS_AND_CONDITIONS_CLICKED });
const signInClicked = () => ({ type: SIGN_IN_CLICKED });

// product/container.js

const mapDispatchToProps = dispatch => ({
  onAddToCartClick: productId => dispatch(addToCartClicked(productId))
});

// sign-up/container.js

const mapDispatchToProps = dispatch => ({
  onTermsAndConditionsClicked: () => dispatch(termsAndConditionsClicked())
});

// home/container.js

const mapDispatchToProps = dispatch => ({
  onSignInClicked: () => dispatch(signInClicked())
});

// reducers.js

const cartInitialState = { productIds: [] };

const cartReducer = (state = cartInitialState, { type, payload }) => {
  switch (type) {
    case ADD_TO_CART_CLICKED:
      return { ...state, productsInCart: [...state.productsInCart, payload] };
    default:
      return state;
  }
};

const modalInitialState = { currentModal: null };

const modalReducer = (state = modalInitialState, { type, payload }) => {
  switch (type) {
    case ADD_TO_CART_CLICKED:
      return { ...state, currentModal: ADD_TO_CART_CONFIRMATION_MODAL };
    case TERMS_AND_CONDITIONS_CLICKED:
      return { ...state, currentModal: TERMS_AND_CONDITIONS_MODAL };
    case SIGN_IN_CLICKED:
      return { ...state, currentModal: SIGN_IN_MODAL };
    default:
      return state;
  }
};

In this pattern, the responsibility of documenting the modal logic lives within the reducers instead of the containers. Reducers listen for events and determine state mutations on their own, thus positioning the business logic in closer proximity to the Redux store than to the React component tree.

This pattern lends itself to redux-saga as sagas are watchers and listen for actions as if they were events. This pattern can still work with redux-thunk but will require an extra convention and enforcement.

Pros and Cons

Forthcoming