Overview
We implement a standard set of React libraries and tools in order to:
- Reduce onboarding time for new developers.
- Reduce complexity & time required for context switching between projects.
- Ensure everyone is contributing to a common knowledge base.
Other Choices?
This guide is more of a strong recommendation than a strict mandate. We recognize that every project is different and some offer particularly unique requirements and technical challenges. That said, deviating from our best practices needs to be justified by a solution below not meeting our needs, so please discuss your idea with your project lead developer or a senior JS developer before implementation. There may be existing solutions that most of our team will already know how to implement and be able to build, debug and help with.
Setting up your workstation
Node + NVM (Node Version Manager)
You should install node with nvm, not with Homebrew or the nodejs.org installer. If you’ve already installed node, please uninstall it before installing nvm. nvm makes it easy to run multiple versions of node side by side on the same machine, allowing you to switch from project to project without running into version issues.
.nvmrc
Each React project should include an .nvmrc
with the version of node the project expects.
Package management
All third-party dependencies should be installed with npm
. Avoid copying & pasting third-party code directly into the project and avoid using other front-ends to the npm registry like yarn.
All projects should include a package-lock.json
file to ensure builds are reliable from machine to machine. Please check this file into version control, and avoid deleting & regenerating it. If you run into an issue that you can only solve by regenerating package-lock.json
, please consult a senior JS developer before doing so to make sure it’s absolutely necessary.
Project Setup
Use create-react-app
, React’s official setup tool, to bootstrap your new application. CRA provides a bare-bones project structure and automatically configures Webpack, Babel, and the various Webpack loaders necessary to import CSS, SVG, and other assets into your components.
Get started by running the following command, making sure that the name you give your React app matches the name of its git repository.
npm init react-app $GIT_REPO_NAME --use-npm
Ejecting
Depending on project requirements (i.e. supporting more than development, test & production environment configs) you might need to eject from CRA and manually configure your project. Consult a senior JS developer if you think you need to do this so we can make sure that ejecting is necessary and your ejected configuration is well documented.
Project structure
We use CRA as a starting point for our structure, but we get a bit more specific about component organization. All of your app code will live inside src/
as CRA will only compile code in that directory with webpack.
my-app/
README.md
node_modules/
package.json
.env
public/
index.html
favicon.ico
src/
assets/
icon.svg
containers/
App/
__tests__/
index.test.js
SubComponent.test.js
index.js
SubComponent.js
style.module.css
components/
form/
form-partials/
layout/
ui/
view-partials/
index.css
index.js
setupTests.js
Assets
Your assets folder should contain any images or SVGs imported by Webpack into your components (CRA’s Webpack config only watches src/
, so you can’t import files from public/
.) Try to put all assets here instead of co-locating them with specific components to reduce possible duplication of assets across multiple components.
Importing SVGs
Webpack allows importing of SVGs as images and as React components. If you need to style an SVG, make use of this feature instead of manually inlining SVG markup into a new component:
import { ReactComponent as Icon } from 'assets/icon.svg';
Containers
Container components differ in that they’re directly connected to a data store (whether that’s Redux or some other tool) and serve to map data and actions down to child components. An example of a container component would be a <UserListPage>
component that connects to your store and receives state and dispatch actions as props. For a simple application, a single container may be all you need, while a more complex app may want a container per route. An even more complex application could have multiple containers per page. For example, you might have a complex modal form that’s available on multiple pages. It’s simpler to make this modal a container and connect it directly to the store than it is to have repetitive prop passing across multiple pages.
For more details on the difference between container & presentational components, refer to this article by Dan Abramov.
Components
While it’s tempting to use React as an HTML template engine, it offers more powerful ways for us to develop a set of re-usable & composable components that aren’t tied to a specific view or use case. Remember, DRY (Don’t Repeat Yourself) applies to markup and styling as well. This is your opportunity to define a domain-specific markup language for your project, so be descriptive in naming. If you’re thorough about building components for all the basic building blocks of your application, you’ll find yourself rarely writing raw HTML in your page components. This can feel counter-intuitive at first, but the end result is code that’s easier to understand at a glance as you’re looking at components that describe what they do instead of large blocks of HTML.
If you’re using an open source UI library, this pattern is still very useful. Instead of importing their components throughout your application, you’ll import them to your components and export your customized version(s). Even if you’re not customizing them much now, having a single source of truth for your app’s implementation of (for example) a modal means that applying future customizations won’t require combing through all your components to make sure you didn’t miss anything.
Your components folder should include a few sub-folders to help with organization. We split our generic components up by their use case:
form
Form components include dropdowns, input fields, date pickers, and other basic building blocks for forms.
Instead of writing repetitive form code
<input
type="test"
className={styles.inputText}
value={user.name}
/>
Refactor your form building blocks (with their default styles) to form components, each with their own CSS module:
import Input from 'components/form/Input';
<Input
value={user.name}
/>
form-partials
Form partials are re-usable collections of form elements. For example, if you’re building an app that captures addresses for both individual people and businesses, you could create an <Address />
partial that’s used in both your person edit and business edit forms.
If you’re copying and pasting something like this into more than 1 form:
import Input from 'components/form/Input';
import PostalCode from 'components/form/PostalCode';
import Dropdown from 'components/form/Dropdown';
import styles from 'globalStylesYouImportInEveryPage.module.css';
<div className={styles.line1}>
<Input
value={user.address.streetNumber}
/>
<Input
value={user.address.streetName}
/>
</div>
<div className={styles.line2}>
<Input
value={user.address.unit}
/>
</div>
<div className={styles.cityProvincePostal}>
<Input
value={user.address.city}
/>
<Dropdown
value={user.address.province}
/>
<PostalCode
value={user.address.postalCode}
/>
</div>
Refactor it into a form partial with it’s own CSS module:
import Address from 'components/form-partials/Address';
<Address
values={user.address}
/>
layout
Layout components provide visual structure and organization for your application. These components include basic page structure, re-usable grids, page sections, headings, basic lists, and sidebars. These components should mostly serve as wrappers for CSS styles, provide little to no interactivity, and offer style configuration through props.
If you’re copying and pasting something like this into more than 1 view:
import styles from 'globalStylesYouImportInEveryPage.module.css';
<div className={styles.container}>
<article className={styles.main}>
Your main content
</article>
<aside className={styles.sidebar}>
Some sidebar content
</aside>
</div>
Refactor it into a set of layout components, each with their own CSS module:
import Layout from 'components/layout/Layout';
<Layout>
<Layout.Main>
Your main content
</Layout.Main>
<Layout.Sidebar>
Some sidebar content
</Layout.Sidebar>
</Layout>
ui
UI components are non-form interactive elements like buttons, menus, modals, sliders, cards, icons, tooltips, and progress bars. If it doesn’t fit in form or layout, it probably belongs here.
If you’re copying and pasting something like this into more than 1 view:
import { Modal, Button } from 'some-react-ui-framework';
import styles from 'globalStylesYouImportInEveryPage.module.css';
<Modal
isOpen={this.isOpen}
>
<h2 className={styles.modalTitle}>
Modal title
</h2>
<div className={styles.modalBody}>
Modal content
</div>
<footer className={styles.modalFooter}>
<Button
onClick={this.doSomething}
className={styles.modalButton}
>
Do A Thing
</Button>
</footer>
</Modal>
Refactor it to a UI component with its own CSS module:
import Modal from 'components/ui/Modal';
<Modal
isOpen={this.isOpen}
title="Modal title"
actions={(
<Button onClick={this.doSomething}>Do A Thing</Button>
)}
>
Modal content
</Modal>
Note how our Modal’s actions
prop makes it easy for implementers to customize footer actions without adding boilerplate markup. Even if you’re using a UI framework, you should still look for ways to reduce repetition and make it easier for other developers to implement your components.
view-partials
View partials are the non-interactive equivalent of form partials. This is where you put small re-usable templates for formatting & displaying data.
If you’re copying & pasting something like this into multiple pages:
import styles from 'globalStylesYouImportInEveryPage.module.css';
<address>
<div className={styles.line1}>
{user.address.streetNumber} {user.address.streetName}
</div>
{user.address.unit && (
<div className={styles.line2}>
Unit {user.address.unit}
</div>
)}
<div className={styles.cityProvincePostal}>
{user.address.city}, {user.address.province} {user.address.postalCode}
</div>
</address>
Refactor it to a view partial with its own CSS module:
import Address from 'components/view-partials/Address';
<Address
address={user.address}
/>
Other files
index.js
This serves as the entry point for your application. You might also mount your router and page containers here (for more details, see Routing). CRA provides this file for you, and you’ll find that it will grow as your application does.
index.css
This is a CSS file that’s not imported as a module. This is a great place to include basic tag selector styles, CSS resets, web font imports, and other CSS that isn’t tied to a specific class or component. Don’t go overboard here. Global styles are brittle and hard to change without visual regression testing of large parts of your application.
setupTests.js
Contains Jest configuration. If you’re adding Jest plugins or configuring Enzyme, this is where that code goes. CRA doesn’t include this file by default but will load it if it exists. You don’t need to eject from CRA just to customize Jest.
State Management
Our preferred solution for state management is the popular Redux library. Redux provides a means of describing the nouns and verbs that make up our state that promotes re-use and testability. That said, this is React, so there’s no one “right” way to use Redux. We need to tweak our approach depending on the scope of our application and the complexity of its business logic. Therefore we’re going to describe a couple recommended approaches to organizing your Redux state: simple & complex. Whether or not your app is simple or complex is up to you (or the lead developer on your project.)
Conventions
Regardless of your approach to state management, there are a few naming and formatting conventions we should always follow:
Namespace actions
Actions should always be prefixed with a namespace for clarity and to avoid accidental action name collision. They should always be declared as SCREAMING_SNAKE_CASE
constants and exported for use in other modules.
Bad:
export const MY_ACTION = 'MY_ACTION';
Good:
export const MY_ACTION = 'myApp/MY_ACTION';
Use action creators
It’s tempting to just dispatch
objects to your store but because we don’t like repeating ourselves and we do like writing code that’s easy to refactor, we should encapsulate actions in functions.
Bad:
import { MY_ACTION } from 'state/actions';
dispatch({
type: MY_ACTION,
myValue,
});
Good:
// actions.js
export const MY_ACTION = 'myApp/MY_ACTION';
export const myAction = myValue => ({
type: MY_ACTION,
myValue,
});
// MyComponent/index.js
import { myAction } from 'state/actions';
dispatch(myAction(myValue));
Use a consistent action object format
{ type, payload, meta }
Sticking to a standard action format leads to code that’s easier to learn and less surprising. This format is loosely adapted from Flux Standard Action but opts to keep errors in meta
to preserve the expectation that payload is an object of data.
type
is always a string action name.payload
is optional, but when present is data that’s meant to be persisted to the store.meta
is any additional information, like errors, HTTP status, etc. This can be standardized further on a per-application basis.
Bad:
It’s not clear from this action what data is persisted to the store and what’s just extra metadata.
// actions.js
export const MY_ACTION = 'myApp/MY_ACTION';
export const myAction = myValue => ({
type: MY_ACTION,
myValue,
extraDataNotInStore: 'foo',
});
Good:
Every action uses the same format so other developers can have clear expectations on what’s dispatched to reducers. Setting some standards around error formatting, i.e adapting a standard meta.errors
property, can make it easier to write code like a reducer for global error state handling.
// actions.js
export const MY_ACTION = 'myApp/MY_ACTION';
export const myAction = myValue => ({
type: MY_ACTION,
payload: {
myValue,
},
meta: {
extraDataNotInStore: 'foo',
}
});
Reducers
Reducers should:
- Make use of ES6 features like default parameters, destructuring & object spread to reduce boilerplate code.
- Use
switch
statements to model action branches, notif/else
object key/value access, or other clever abstractions. While there’s a bit more typing involved withswitch
es, it’s obvious what they’re doing at first glance. Remember, you’ll spend more time reading code than writing it.
import { MY_ACTION } from 'state/actions';
const initialState = {
myValue: null,
anotherValue: null,
};
export const myReducer = (state = initialState, { type, payload, meta }) => {
switch (type) {
case MY_ACTION: {
return {
// Remember to always destructure the current state first ...
...state,
// before updating properties
myValue: payload.myValue,
// Declare initialState in a variable to allow easy resets of individual properties
anotherValue: initialState.anotherValue,
};
}
case default: {
return state;
}
}
}
Simple State
A simple state use case would be something like a multi-step registration & purchase flow, or a React app embedded inside a traditional web app that’s only responsible for a small subset of functionality.
File structure
If your app’s state needs are relatively simple, you can group state code by what type of functionality it is. All actions in one file, all reducers in one file, etc.
state/
actions.js
reducers.js
selectors.js
Complex State
Complex state use cases include fully featured web applications, advanced content workflows, and other applications where a significant amount of state needs to be loaded, manipulated, and saved. Data objects in these scenarios may be deeply nested or interrelated.
File structure
In this situation, you’ll want to break up your state functionality around specific business objects. Think of these state modules similarly to a model in an MVC framework. If you’ve worked with Redux before, you may be familiar with this way of organizing state as the ducks pattern.
Each module:
- Includes a reducer
- Includes actions & action creators
- May include async actions and selectors
- May export action names as
const
s so other reducers can listen for them
Collaborating with your project’s API team when designing a complex state model is essential, and you can often follow their lead on how to organize things (i.e. if you have a set of CRUD project
endpoints, you probably need a projects
state module.) You won’t always want to match your API’s data model 1:1, but it’s not a bad place to start organizing your state.
As an example, these could be a few basic state modules for a project task management system:
state/
projects.js
todos.js
users.js
These are all kept separate in the filesystem, they’re eventually merged into a single reducer, so state modules can optionally import actions from other state modules for use in their reducers.
The following is an example of a users
duck that can try to load data from an API and either add users to the store on success, or log an error on failure. You can use this as a starting point for your ducks, but a more robust error handling solution is probably necessary.
Take note of how we model the act of getting users from the server as three distinct actions: REQUEST
, SUCCESS
, and ERROR
. We’ve found this convention useful for modeling asynchronous logic as it allows us to update state both when an async action completes as well as when it starts.
import { createSelector } from 'reselect'; // see Additional Libraries below
// Actions
const LOAD_REQUEST = 'myApp/users/LOAD_REQUEST';
const LOAD_SUCCESS = 'myApp/users/LOAD_SUCCESS';
const LOAD_ERROR = 'myApp/users/LOAD_ERROR';
// Action creators
export const usersLoadRequest = () => ({
type: LOAD_REQUEST,
});
export const usersLoadSuccess = (users) => ({
type: LOAD_SUCCESS,
payload: {
users,
},
});
export const usersLoadError = (error) => ({
type: LOAD_ERROR,
meta: {
error,
},
});
// Async action creators, or thunks (see Async section below)
export const getUsers = () => async (dispatch) => {
dispatch(usersLoadRequest());
try {
const response = await fetch('/users');
const users = await response.json();
dispatch(usersLoadSuccess(users));
} catch (error) {
dispatch(usersLoadError(error));
}
};
// Selectors
export const selectAllUsers = state => state.users;
export const selectAllActiveUsers = createSelector(
selectAllUsers(),
users => users.filter(user => user.active),
);
// Reducer
const initialState = {
users: [],
isLoading: false,
error: null,
};
export default (state = initialState, { type, payload, meta }) => {
switch (type) {
case LOAD_REQUEST: {
return {
...state,
isLoading: true,
};
}
case LOAD_SUCCESS: {
return {
...state,
isLoading: initialState.isLoading,
users: payload.users,
error: initialState.error,
};
}
case LOAD_ERROR: {
return {
...state,
isLoading: initialState.isLoading,
error: meta.error,
};
}
default: {
return state;
}
}
};
Each of these state modules will be merged together using Redux’s combineReducers
function, ensuring they’re all part of a single store.
import { combineReducers, createStore } from 'redux';
import { Provider } from 'react-redux';
import projects from 'state/projects';
import todos from 'state/todos';
import users from 'state/users';
import App from './App'; // or wherever your "root" component is
const rootReducer = combineReducers({
projects,
todos,
users,
});
const store = createStore(rootReducer);
ReactDOM.render(
<Provider store={store}>
<App />
</Provider>,
document.getElementById('root'),
);
Redux + Async
As Redux is synchronous, we need to add a library for handling asynchronous actions like requesting data from an API and placing it in the store.
For simplicity’s sake, most applications will be fine with redux-thunk and async
/await
. If you understand promises you can work with thunks as they are just actions that return a function that’ll eventually resolve.
If you need more fine grained control over async actions than promises allow, redux-saga is your answer. In addition to handling async logic like thunks, sagas give developers the ability to pause, cancel, and resume async actions. Consult with your project’s dev lead before jumping into redux-saga
as its added power comes with some complexity. As redux-saga
uses ES6 generator functions instead of async
/await
, knowledge of how they work is required to fully take advantage of the flexibility sagas provide.
Additional libraries
normalizr
normalizr takes deeply nested data and extracts it into flat stores based on a schema definition. Most apps won’t need this, but if you’re working with an API that provides deeply nested data (especially if it provides it in a way where there could be multiple sources of truth for a given object) normalizr can make it more manageable.
reselect
reselect makes it easy to create performant, testable, and reusable views into your store. We often need to compute derived data from what’s in Redux, and reselect handles all the hard stuff around performance optimization so you don’t have to.
Freactal
If you’re working on an older project, you may encounter state managed with Formidable Labs’ Freactal. Freactal is conceptually similar to Redux in that you have state, actions that transform state, and functions that can derive data from state, but differs quite a bit in implementation.
Freactal is quite a bit simpler to pick up than Redux as it’s a bit less indirect (i.e. instead of defining actions and reducers separately, Freactal bundles them together as effects
) but as it’s somewhat buggy & no longer actively maintained, we do not recommend using it for new work.
Forms & Validation
Nearly every React application has to deal with forms & form state management at some point. Building forms with React’s built-in component state works great for many simple use cases, but we’ve found that delivering robust forms with client side validation can quickly outgrow the state management tools.
Formik
Formik provides a set of components for form state management, rendering, and validation, and will work with any application state management library. If you’re building anything more complex than a basic one field search form, add Formik to your project with a quick npm install formik
. Their documentation is pretty robust so we won’t duplicate it here, but we do have some general recommendations on how to work with Formik.
Prefer creating separate form components & passing them into the Formik
component instead of inlining them. This makes it easier to use the same form in multiple contexts (i.e. building a single form for both your create and edit pages.)
import { Formik } from 'formik';
import YourFormComponent from './YourFormComponent';
<Formik
component={YourFormComponent}
/>
Pass your components/form
and components/form-partials
components into Formik’s <Field />
component instead of inlining them or using 'input', 'select', etc
. All of our custom form element components should be compatible with <Field />
without additional implementation work.
import { Field } from 'formik';
import Input from 'components/form/Input';
<Field
name="firstName"
component={Input}
/>
Formik: Documentation
Yup
Yup.js is an object schema validator that integrates seamlessly with Formik. With Yup, you build a schema describing the expected shape of your form’s data and Formik will take care of running validation on change and propagating errors to your fields. Yup provides a plethora of built-in validators and allows the addition of custom validations using yup.addMethod
.
Here’s a quick example of a custom validator and a signup form schema.
utils.yup.js
// in a utils/yup.js file
import * as yup from 'yup';
// Set a custom default required message
yup.setLocale({
mixed: {
required: ({ path: label }) => `Please enter a valid ${label}`,
}
})
yup.addMethod(yup.string, 'password', function validatePassword(length, message) {
return this.test('password', message, function testPassword(value) {
const { path, createError } = this;
const valid = value.match(yourPasswordRegexHere);
return new Promise((resolve, reject) => {
if (valid) {
resolve(true);
} else {
reject(createError({
path,
message: message || `${path} must have ${length} at least one number, one letter, and one symbol`,
}));
}
});
});
});
export default yup;
FormComponent.js
import { PHONE_NUMBER } from 'utils/regex';
import yup from 'utils/yup';
const SignupSchema = yup.object({
first_name: yup.string()
.label('first name')
.min(2)
.required(),
email: yup.string()
.email()
.required(),
password: yup.string()
.min(8)
.max(128)
.password(),
password_confirm: yup.string()
.label('password')
.required(({ path: label }) => `Those two ${label}s do not match. Please try again.`)
.oneOf(
[yup.ref('password'), null],
({ path: label }) => `Those two ${label}s do not match. Please try again.`
),
phone_number: yup.string()
.label('phone number')
.matches(
PHONE_NUMBER,
)
.required(),
});
Yup: Best Practices
- Make liberal use of
yup.addMethod
& descriptive validator names to avoid repeating validation code. ie.birthDate.string().isAdult()
is always easier to understand at a glance than a series of min/max validations full of date parsing code. Additionally, custom validators are easily unit testable, while one-off custom validation rules in a schema are not. - Any regexes used with
matches
should be named constants, even if you think it’s obvious what the regex does. - Make use of
label()
to give your fields human-friendly names in error messages, if necessary. - Use
yup.setLocale
to customize default messages instead of passing the same message to multiple validators.
Yup: Documenation
Routing
We employ react-router for page routing and conditional rendering of components.
Basic Pattern
While React Router supports a variety of routing schemes for familiarity and ease of use we employ a pattern where each route corresponds to a page container component. React Router’s <Switch />
component makes this easy by only rendering the first matching <Route />
it finds.
Here’s a basic example of a Home page and a set of user CRUD routes. Note that we put the most specific routes first. This is because <Switch />
matches from top to bottom. We can take advantage of this behaviour, notice how our final route lacks a path
param. This means this route will only render when nothing else matches, giving us an easy way to implement a generic 404 page.
// App/index.js
import {
BrowserRouter as Router,
Route,
Switch,
} from 'react-router-dom';
export const App = () => (
<Router>
<Switch>
<Route
path="/users/:id/edit"
component={UserEdit}
/>
<Route
path="/users/:id"
component={UserView}
/>
<Route
path="/users/new"
component={UserCreate}
/>
<Route
path="/users"
component={UserList}
/>
<Route
path="/"
component={Home}
/>
<Route
component={NotFound}
>
</Switch>
</Router>
);
Use route constants & generatePath
That works pretty well, but we want to DRY this up to avoid easy to write & hard to notice bugs like route typos, as well as make future route refactoring easier.
// utils/routes.js
export default {
users: {
edit: '/users/:id/edit',
view: '/users/:id',
create: '/users/new',
list: '/users',
},
}
// App/index.js
import {
BrowserRouter as Router,
Route,
Switch,
} from 'react-router-dom';
import {
users,
} from 'utils/routes';
export const App = () => (
<Router>
<Switch>
<Route
path={users.edit}
component={UserEdit}
/>
<Route
path={users.view}
component={UserView}
/>
<Route
path={users.create}
component={UserCreate}
/>
<Route
path={users.list}
component={UserList}
/>
<Route
path="/"
component={Home}
/>
<Route
component={NotFound}
>
</Switch>
</Router>
);
// SomeComponent.js
import {
Link
} from 'react-router-dom';
import {
users,
} from 'utils/routes';
export const SomeComponent = () => (
<Link
to={users.list}
>
Users
</Link>
);
Now we’ve only defined our routes in a single place & made it easy to import into other components. That said, how do we link to routes with dynamic parameters when they’re stored in a string constant in a different module? React Router helpfully provides a generatePath
function that uses our route definition and a parameter object to generate a URL:
// SomeOtherComponent.js
import {
Link,
} from 'react-router-dom';
import {
generatePath,
} from 'react-router';
import {
users,
} from 'utils/routes';
export const SomeOtherComponent = ({ user }) => (
<Link
to={generatePath(users.edit, { id: user.id })}
>
Edit {user.name}
</Link>
);
We can also use this function when navigating programmatically:
history.push(generatePath(users.edit, { id: user.id }));
Use Link & NavLink components for internal links
Remember to take advantage of React Router’s helper components when implementing navigation. <Link />
provides some helpful props for inline links, and <NavLink />
is aware of the current route and can take care of applying active styling to your navigation items.
Using routes for conditional display
Your application may have global UI elements like a header or footer that need to display different content based on the current route. You can also employ React Router to handle these checks instead of building them yourself.
For example, let’s say you want to build a component that will display a different heading depending on which resource the user is currently accessing.
// PageTitle.js
import {
Route,
Switch,
} from 'react-router-dom';
import {
users,
tasks,
} from 'utils/routes';
export const PageTitle = () => (
<Switch>
<Route
path={users.list}
>
<h1>Users</h1>
</Route>
<Route
path={tasks.list}
>
<h1>Tasks</h1>
</Route>
</Switch>
);
Notice we’re matching on each resource’s list
route. This makes sure each route matches any route that starts with /users
or /tasks
, respectively.
Testing with MemoryRouter
While writing component tests, you may run into React Router warnings and errors when rendering any components that include a Route component or are otherwise aware of their current location.
React Router provides a <MemoryRouter />
component to help us out. All we need to do is wrap it around the component we’re rendering for testing.
// components/PageTitle/test/index.test.js
import {
MemoryRouter,
} from 'react-router';
import { shallow, mount } from 'enzyme';
import PageTitle from '..';
describe('PageTitle', () => {
it('should render', () => {
const wrapper = mount(
<MemoryRouter>
<PageTitle />
</MemoryRouter>
);
expect(wrapper).toMatchSnapshot();
});
});
Now your test errors are suppressed, but the component is probably rendering an empty div. By default, <MemoryRouter />
is set to the base /
route, so this component won’t render at all. Thankfully, we can pass in a history stack to simulate different routes and ensure our components are responding to them correctly.
// components/PageTitle/test/index.test.js
import {
MemoryRouter,
} from 'react-router';
import { shallow, mount } from 'enzyme';
import {
users,
} from 'utils/routes';
import PageTitle from '..';
describe('PageTitle', () => {
it('should render', () => {
const wrapper = mount(
<MemoryRouter>
<PageTitle />
</MemoryRouter>
);
expect(wrapper).toMatchSnapshot();
});
it('should render "Users" on user pages', () => {
const wrapper = mount(
<MemoryRouter
initialEntries={[users.list]}
>
<PageTitle />
</MemoryRouter>
);
expect(wrapper.find('h1').text()).toBe('Users');
});
});
Styling
We recognize that a one-size-fits-all approach to CSS isn’t really possible and that overcomplicating styling for smaller apps can lead to unnecessary complexity. Knowing that, we make use of two approaches for styling: vanilla CSS, and CSS Modules.
Vanilla CSS
CRA compiles .css
files into a single style bundle using Webpack, and automatically runs your code through Autoprefixer. Please refer to our general CSS style guide for information on structuring and naming your selectors.
If you find yourself writing a lot of CSS or getting heavily into complex BEM selectors, you might want to move along to CSS Modules.
CSS Modules
While a traditional CSS approach can work well with simple React apps, we prefer making use of CSS Modules for styling more complex applications. CSS Modules support is built into CRA, all you have to do is use the .module.css
extension instead of .css
, and co-locate it with your React component. This tells Webpack a couple things:
- Affix some additional text to your selectors post-compilation to make sure they’re unique across your entire project. i.e.
.container
instyles.modules.css
is transformed to something like.styles__container_v23fn2
. - Allow importing of this stylesheet as a JS object. This is how you associate the randomly generated unique class name to your component without knowing it in advance. In practice, this looks like this:
components/ui/YourComponent/styles.module.css
.yourComponent {
color: #fff;
}
components/ui/YourComponent/index.js
import styles from './styles.module.css';
const ComponentName = ({ children }) => (
<div className={styles.yourComponent}>
{children}
</div>
);
CSS Modules Guidelines
Use camelCase instead of kebab-case for class & variable names
You’ll be importing your styles to JS, so use JS-friendly naming conventions. Writing & reading styles.myClass
is easier than styles['my-class']
.
Class names should be simple
BEM and other naming conventions designed to avoid specificity bugs aren’t necessary with CSS Modules. Your class names are guaranteed to be unique so you can name classes based on what they mean in the context of your module. You can have 50 components, each with their own .container
class, and they’ll never conflict even if some of those components are nested in others.
Writing classes this way also makes Sass nesting a lot less necssary as you don’t have to repeat the same prefixes over and over again.
Avoid this:
.socialLinks {
}
.socialLinks__item {
}
.socialLinks__item--facebook {
}
Do this instead:
.socialLinks {
}
.item {
}
.facebook {
}
Overriding Child Components
If you need to override styles in a child component from a parent component, expose props for passing CSS class names to the element you need to override instead of relying on child element selectors. These selectors lead to hard to debug CSS, as it’s not always obvious which parent component is overriding child styles and those parent styles may wind up applying to more components than you intended.
Avoid this:
.list {
background: black;
}
.list li {
color: white;
}
import styles from './styles.module.css';
import ListItem from 'ListItem';
const MyComponent = ({ items }) => (
<ul className={styles.list}>
{items.map(item => (
<ListItem>
{item}
</ListItem>
))}
</ul>
);
Do this instead:
/* MyComponent/styles.module.css */
.list {
background: black;
}
.item {
color: white;
}
/* MyComponent/index.js */
import styles from './styles.module.css';
import ListItem from 'components/layout/ListItem';
const MyComponent = ({
items,
}) => (
<ul className={styles.list}>
{items.map(item => (
<ListItem
className={styles.item}
>
{item}
</ListItem>
))}
</ul>
);
/* ListItem/index.js */
import styles from './styles.module.css';
const ListItem = ({
children,
className = null,
}) => (
<li
className={[
styles.item,
className,
]}
>
{children}
</Link>
);
Variables
If you need compile-time variables, don’t jump straight to including Sass. CSS Modules allows declaration of variables that you can import into CSS & and JS modules using its @value
keyword. This is especially handy for sharing things like common media query breakpoints between your styles and conditional rendering tools like react-media
(which the below example illustrates) or the DOM API’s window.mediaQueryListener
.
/* variables.module.css */
@value colorWhite: #fff;
@value screenSmall: screen and (min-width: 768px);
/* MyComponent/styles.module.css */
@value (
colorWhite,
screenSmall
) from 'variables.css';
.container {
width: 100%;
color: colorWhite;
}
@media screenSmall {
.container {
width: 80%;
}
}
/* MyComponent/index.js */
import { screenSmall } from 'variables.module.css';
import SmallScreenComponent from 'MobileFriendlyComponent';
import LargeScreenComponent from 'MobileFriendlyComponent';
import styles from './styles.module.css';
const MyComponent = ({
children,
}) => (
<div className={styles.container}>
<Media query={screenSmall}>
{matches => (
<LargeScreenComponent>
) : (
<SmallScreenComponent>
)}
</Media>
</div>
);
Conditional classes
Install the classnames
package if you need to apply classes based on JS data as conditional string concatenation can get unwieldy with two or more optional classes. classnames
’s array syntax makes this a lot easier to write & read.
import classNames from 'classnames';
import styles from './styles.module.css';
const MyComponent = ({
children,
conditionA,
conditionB,
conditionC,
}) => (
<div
className={classNames([
styles.myComponent,
conditionA && styles.conditionalClassA,
conditionB && styles.conditionalClassB,
conditionC && styles.conditionalClassC,
])}
>
{children}
</div>
);
Global selectors
Only use :global
selectors to override another library’s CSS if there are no other ways of doing so. We should try to make use of third-party UI libraries that allow passing classNames to components for style customization. That said, some otherwise very good tools may not support this or we may need to integrate with non-React JS libraries. Avoid using :global
otherwise as it opts you out of all the local scoping benefits provided by CSS Modules.
A good example is Semantic UI. While Semantic provides className
properties for its components, its heavily specific style of CSS means that built-in styles will often win over yours, particularly if you’re trying to style sub-elements like a dropdown menu item. Furthermore, you can’t directly target their classes (i.e. ui.dropdown .menu > .item
) as CSS Modules will modulify those class names.
To overcome this, you can write a selector like below which will tell CSS Modules to modulify yourDropdown
(and allow you to access it in JS like normal) but print .ui.dropdown .menu > .item
as is.
⚠️ You should try to always prefix your
:global
rules with a local class, otherwise they’ll escape the scope of your module and potentially override styles in other components.
/* MyDropdown/styles.module.css */
.myDropdown:global(.ui.dropdown .menu > .item) {
background-color: black;
}
/* MyDropdown/index.js */
import Dropdown from 'semantic-ui-react';
import styles from './styles.module.css';
const MyDropdown = ({
items
}) => (
<Dropdown
className={styles.myDropdown}
>
{/* etc... */}
</Dropdown>
);
Prefer composable components to the composes
keyword
CSS Modules provides the composes
keyword as a means of adding style rules from one class into another, similar to mixins in Sass. While this is powerful functionality that can make it easier to share styles across semantically different components, we should avoid using it when creating composable components is possible.
For example, you should avoid creating a .list
class and composing it with various components that implement a <ul>
. Instead:
- Create
<List />
and<ListItem />
components that have their own style modules and accept aclassName
for further customization - Replace
<ul>
s &<li>
s elsewhere in your app with your new base components. - If you find you’re passing similar customizations to various
<List>
instances, move that customization to<List>
’s CSS module and apply it conditionally using a conditional className like the above example.
You now have a single, explicit defintion of what your app considers a list, and you can update its markup or styles across the entire site in one place.
Utilize computed keys to apply modifier classes using props
If you have a component that has multiple variants, use dynamic property access to make applying these classes easier. You can even organize these variants in a separate CSS file to avoid the need to prefix each variant class.
/* MyComponent/variants.module.css */
@value colorBrand, colorDanger from 'variables.module.css';
.primary {
background-color: colorBrand;
}
.danger {
background-color: colorDanger;
}
/* MyComponent/index.js */
import classNames from 'classnames';
import styles from './styles.module.css';
import variants from './variants.module.css';
const MyComponent = ({
children,
variant = 'primary',
}) => (
<div
className={classNames([
styles.myComponent,
variants[variant]
])}
>
{children}
</div>
);
MyComponent.propTypes = {
variant: PropTypes.oneOf([
'primary',
'danger',
]),
};
Sass + CSS Modules
If you find yourself missing Sass features and you’re double super sure you can’t address your needs using CSS Modules’ built-in fuctionality, refer to CRA’s documentation on adding Sass support. Remember that you’ll lose the ability to import variables into JS files if you convert them to Sass variables, so you’ll need to re-engineer components accordingly.
UI Libraries + Frameworks
There are an assortment of popular React UI libraries/kits/frameworks that aim to speed up development by providing basic UI components so you don’t have to build everything from scratch. We believe these tools have a place in our development workflow, but they should only be chosen if they actually save us more time across the span of the project. These tools should only be chosen after a discussion with your team lead or a senior front-end developer.
When is a UI library appropriate?
- Your project is a back-end/line of business system that doesn’t need to be heavily customized to match the client’s brand.
- You need a large variety of components quickly, and your project team has agreed ahead of time that a UI library’s general look and feel is adequate.
- The client is already using a UI library as a basis for their web presence and it’s already React friendly.
- The library you’ve chosen provides good accessibility features (it’s screen reader friendly, allows users to control all components without a pointing device, etc.)
When is a UI library inappropriate?
- Your project is a customer-facing or otherwise heavily branded experience, requiring significant customization to a UI library’s CSS or behaviour. This is a lot of our projects, meaning you should probably build custom components in most cases.
- You only need a couple components from the library. Remember, it’s simpler in the long term to build & maintain some basic form components than to integrate a large third party dependency.
- Your project team lacks a front-end developer and this is the quickest way to get going. This is addressing a short-term resourcing issue with a long-term engineering solution we have to live with for the entire lifespan of the application, and a conversation with your team should happen before making this call.
Aside on Semantic UI
We use this on a lot of older projects but have had a lot of issues with customizing their styles and functionality. It’s good to be familiar with this library if you’re maintaining older projcets, but do not use Semantic UI on new work.
Other helpful libraries
Just because you’ve made the call to build your application’s components from scratch it doesn’t mean you have to do everything yourself. Through usage on previous projects we’ve found these tools to be extremely helpful for building common but complex UI components:
Dropdowns & autocompletes
- downshift provides everything you need to build custom dropdown & autocomplete components.
Datepickers
- react-dates AirBnB’s datepicker component as a library. Supports single dates & ranges.
- react-datepicker provides basic date & time pickers. Range pickers aren’t included, but examples are provided in their documentation.
Tooltips/popovers
- tippy.js-react provides primatives for building tooltips, popovers, and other components that overlay inline content.
Scroll position control & animation
General UI animation
- react-transition-group basic hooks for component enter/leave transitions. Bring your own CSS animations.
- react-motion provides physics based animation primatives for React components. Reach for this when
react-transition-group
isn’t enough.
Refer to this dev wiki’s Libraries page for non-React specific JS tools.
Testing
Unit Testing
CRA comes with Jest pre-configured and ready for you to write your first test. Please reference the CRA documentation for the latest info on how to run tests and some basic information on setting them up. Beyond this, we have a few best practices for test writing and organization.
Place .test.js files next to the component they test
Jest automatically detects files with a .test.js
extension and incorporates them into the test suite. The best place for these files is right next to the file they’re testing. If your component has any sub components, their test files should be named similarly.
MyComponent/
index.js
index.test.js
MySubComponent.js
MySubComponent.test.js
Organize for better readability
Jest lets you use describe
blocks to group tests by functionality or another measure that makes sense for your project. This makes your test suites easier to read and gives Jest metadata to use when displaying test failure feedback in your terminal.
Assuming you’re testing a module that contains a single function:
/* myModule.test.js */
import {
functionA,
} from './myModule';
describe('myModule', () => { // the name of the module being tested
describe('functionA', () => { // the name of the function being tested
it('should return A', () => { // a specific test case
expect(functionA()).toBe('A');
});
it('should not return G', () => { // a different, equally contrived test case
expect(functionA()).not.toBe('G');
});
});
});
Use Enzyme to simulate component rendering
We need to bring in another package to allow us to simulate the mounting and rendering of React components without a browser. Follow the instructions on the CRA site to add Enzyme to your project and make use of shallow
and render
whenever you need to see whether or not a React component behaves the way you expect.
Use snapshot tests to detect potential UI regressions
Snapshot testing makes preventing UI regressions much easier. On first run, Jest takes a “snapshot” (a plain text rendering of your component’s output, not an image) and stores it in a snapshots folder alongside your test which is then added to your next git commit.
On subsequent runs, Jest renders the component and compares the to your stored snapshot. If they don’t match, the tests fail. This doesn’t necessarily mean your app is broken, it just means that some of your components are now rendering differently. Jest will show you a diff for each failing component. If you see differences you expect, you can update the snapshot using the Jest CLI, otherwise it’s time to manually review those components for a regression.
Use mocking to test functionality that relies on code or factors you can’t control
Mocking is the process of faking a third-party dependency or a built-in API to create the conditions necessary for your test to run. It can be anything from testing that your app handles API issues by simulating a 500 error from a network request, simulating screen size by overriding mediaQueryListener
’s behaviour, to mocking Date
so you don’t have to update tests that calculate an age difference every time the current year rolls over. Let’s use that last one as an example.
Assuming this test was written in 2019, it will start failing next year even though the tested logic is working correctly:
describe('getAge', () => {
it("should calculate a user's age", () => {
const birthDate = '1990-01-01';
expect(getAge(birthDate)).toBe(29);
});
});
This will always work as we’re mocking the current date, ensuring new Date()
always returns January 1, 2019 regardless of what time it currently is:
describe('getAge', () => {
beforeEach(() => {
clock.set('2019-01-01T00:00:00.000Z'); // provided by jest-plugin-clock
});
it("should calculate a user's age", () => {
const birthDate = '1990-01-01';
expect(getAge(birthDate)).toBe(29);
});
});
Turn your bugs into test cases
Practicing test driven development (TDD) with UI can be challenging as it’s hard to write tests for a potentially complex UI, especially when much of your business logic is happening in an API somewhere. That said, bug stories offer a great opportunity to practice TDD in a React project.
Once you get a handle on the problematic behaviour in question and verify that it’s real & replicable, write a test case that simulates it. Then, update your code until the tests pass.
Separate complex logic into smaller, easily testable modules whenever possible
Testing complex logic that relies on React state can be confusing and time consuming. Instead, we can take a more modular approach by refactoring code that isn’t directly responsible for presentation and testing that directly.
An example would be a UI element that takes the start and end dates of a user’s vacation and displays how long they’ll be away, in days. You could embed this date calculation into a React component, possibly as a function that accesses state:
const MyComponent = () => {
const [{ startDate, endDate }, updateDates] = useState({ startDate: null, endDate: null });
const calculateDays = () => {
return endDate - startDate; // for brevity, let's assume date diff calculation in JS is sensible & easy
};
return (
<div>
{calculateDays()}
</div>
);
};
To test this you’d need to fully mount the component, simulate state changes to set startDate
and endDate
, then check the rendered output for the number of days you expect. To make this simpler, we can refactor our date logic into a separate function.
/* MyComponent/index.js */
import { calculateDays } from 'utils/date';
const MyComponent = () => {
const [{ startDate, endDate }, updateDates] = useState({ startDate: null, endDate: null });
return (
<div>
{calculateDays(startDate, endDate)}
</div>
);
};
/* utils/date.js */
export calculateDays(startDate, endDate) {
return endDate - startDate; // for brevity, let's assume date diff calculation in JS is sensible & easy
}
/* utils/date.test.js */
describe('date', () => {
describe('calculateDays', () => {
it('should calculate the number of days between two dates', () => {
expect(calculateDays(new Date('2019-01-01'), new Date('2019-01-07'))).toBe(6);
});
});
});
Now your test is a lot simpler and as a happy side effect you can easily re-use this logic in other components if necessary.
Helpful Plugins
- jest-plugin-clock: Makes mocking JS
Date
s super easy. Essential if you’re testing anything time-sensitive. - jest-fetch-mock: Mock
fetch
responses and rejections. Enables the testing of API dependent code without hitting a real API or even needing an internet connection. - jest-localstorage-mock: Essential for testing code that touches localStorage or sessionStorage.
Integration/Functional Testing
Linting & Coding Style
We’ve adopted a standard coding style in order to make our projects easier to onboard and cut down on worries about tabs vs. spaces, casing styles, etc. This coding standard will be documented in brief here, but is fully captured in our eslint-config-domain7
NPM package. It’s highly recommended to configure your text editor to automatically lint your React code as you work, as our React projects often have pre-commit & pre-push checks that will fail if your code doesn’t pass. It’s much easier to learn and adapt a standard when you get feedback in small chunks instead of all at once.
Our standard extends eslint-config-airbnb. Please review their documentation (and the React sub-page.)
Our customizations
- We automatically include
jest
as a global and bundle thereact
,jsx-a11y
, andfilenames
plugins. - We allow non-destructuring props assignment in case destructuring would cause name conflicts.
- We allow dangling underscores (i.e.
const variable__
) - We enforce dangling commas in multi-line array, object, and function arguments lists. This makes it easier to re-order lines without causing syntax errors. This is not enforced on single-line lists.
// Good: const foo = { a: 1, b: 2, }; // Bad: const foo = { a: 1, b: 2 }
- We enforce a two space indent.
- We validate
aria-*
attributes. - We require
for
attributes on<label>
components. - We require corresponding key events if mouse events are present
- We disable Airbnb’s max line length.
- We allow multiple
inline.chained().method().calls()
. console
anddebugger
statements are warnings, not errors, as we need these to not crash CRA’s dev server. Your project will often treat these as errors when running pre-commit/pre-push checks, however, so you do have to clean them up eventually.${Template} literals
are preferred to'string' + 'concatentation'
.- We allow
PropTypes.object
even though it’s best to define aPropTypes.shape
when you know the full list of properties your component expects. - We ban the
.jsx
extension. Please use.js
for React components. - We warn on deprected functions but don’t throw an error because sometimes refactoring isn’t feasible at the moment. Try to fix this stuff when you find it though!
- We allow
this.setState
incomponentDidUpdate
. While this isn’t optimal code, it’s sometimes necessary. - We disable
import/no-extraneous-dependencies
in Jest tests when importingdevDependencies
. - We disable
react/prefer-stateless-function
because refactoring functional components to class components when you need to add state can be unnecessarily time consuming. If your project is primarily using class components, feel free to use one when a stateless function would work. While React is moving away from classes, we still support them in older projects.
Ignore comments & per-project customization
Lint rules are a guideline to provide some consistency, but are always up for discussion if they’re not meeting our needs or aren’t providing value.
You may need to occasionally override eslint on a case-by-case basis, and this is OK! That said, please check with your lead dev and/or expect PR feedback on whether or not they’re necessary.
We sometimes will use a project as an opportunity to try something new, (i.e.,avoid using class components and this.state
to embrace Hooks and the direction React is going in the future) so individual projects may minimally override or customize our standard. Be sure to check their eslint configuration if something doesn’t totally line up with your expectations or the documentation here.
Remember, our linting guidelines are a living document and are always up for discussion with your team lead and other team members. Our standard should be an ongoing coversation on what we feel is best practice, not an excuse to stick to what we know.
Code Editor Config
We believe in letting developers choose the best tools for themselves. When working on React projects, we recommend a modern text editor with good ES6+ syntax support. A few suggestions are:
While we’re not prescriptive about text editors, there are a few essential editor plugins that you’ll need to install to best work on our codebases. (Since we support many editors, we’re not linking specific plugins here. You’ll need to search for the plugin for your editor yourself.)
- editorconfig provides a standard way to share text editor configuration settings like which line endings to use and how many spaces are in an indent. Our projects all have an
.editorconfig
and installing this plugin will make your editor automatically adapt those settings. - eslint will automatically run your changes against our ESLint standard and provide helpful inline feedback. Many ESLint plugins support automatically fixing lint errors on save, which we recommend enabling.
- stylelint provides syntax checking and linting for CSS. You may additionally need sass-lint if your project is using SCSS.
Browser developer tools
- React Developer Tools - inspect react component tree (state, props), debug rerenderings
- Redux DevTools - track dispatched actions, inspect state at any action, track action payload
- Immutable.js Object Formatter - inspect immutable objects in console or in react props/state