Creating a Design System in React: A Comprehensive Guide

Aayush Agarwal
5 min readFeb 5, 2024

--

Design system for React

Design systems are a crucial part of developing scalable, maintainable, and consistent web applications. They serve as a single source of truth for UI components and styles, ensuring teams can work more efficiently and cohesively. With its component-based architecture, React is an excellent choice for building a design system. This article will guide you through creating a design system in React, covering everything from using Storybook to managing components, making your design system publishable, and other essential concepts.

1. What is a Design System?

A design system is a collection of reusable components, guided by clear standards, that can be assembled to build any number of applications. It typically includes design tokens (colors, typography, spacing), UI components, and even coding guidelines. The goal is to improve UI consistency and quality across a project or an organization.

2. Setting Up Your Environment

Initializing Your Project

Start by setting up a new React project if you haven’t already. This project will house your design system.bashCopy code

npx create-react-app my-design-system
cd my-design-system

Adding Storybook

Storybook is a tool that allows you to develop and design UI components in isolation. It serves as an interactive component library, making it easier for your team to browse and understand the capabilities of your design system.

npx sb init

This command sets up Storybook in your project.

Global Configuration

  1. Modify .storybook/preview.js:
  2. Add a global decorator in your .storybook/preview.js file to wrap all stories with the ThemeProvider. Import your theme and apply the theme globally.
import React from 'react';
import { ThemeProvider } from 'react-jss';
import { theme } from '../src/theme'; // Adjust the import path to your theme file

export const decorators = [
(Story) => (
<ThemeProvider theme={theme}>
<Story />
</ThemeProvider>
),
];

Introduction to React-JSS

React-JSS provides CSS-in-JS styling with a focus on performance and theming. It allows you to define styles in JavaScript, benefiting from the full power of JS and theming capabilities. React-JSS is particularly well-suited for design systems due to its seamless integration with React components and support for dynamic themes.

Installing React-JSS

Install react-jss to add it to your project:

npm install react-jss

Configuring Theming with ThemeProvider

React-JSS’s ThemeProvider allows you to define a theme object that can be accessed by all components within the provider. This is particularly useful for defining global design tokens such as colors, fonts, and spacing.

Defining Your Theme

Create a theme.js file in your project:

// src/theme.js
export const theme = {
colors: {
primary: '#007bff',
secondary: '#6c757d',
},
// Add more theme tokens like fonts, spacing, etc.
};

Using ThemeProvider

Wrap your application or component hierarchy with ThemeProvider and pass your theme object to it:

// src/App.js
import React from 'react';
import { ThemeProvider } from 'react-jss';
import { theme } from './theme';
import { Button } from './components/Button';

function App() {
return (
<ThemeProvider theme={theme}>
<div className="App">
<Button>Click Me</Button>
</div>
</ThemeProvider>
);
}

export default App;

3. Structuring Your Design System

Atomic Design Methodology

Consider using the Atomic Design methodology to structure your components. This involves organizing your UI into atoms, molecules, organisms, templates, and pages, promoting reusability and modularity.

Directory Structure

Create a components directory in your project. Inside, organize your components based on the Atomic Design principles:

src/
└── components/
├── atoms/
├── molecules/
├── organisms/
└── templates/

4. Creating Components

When building components, aim for reusability and configurability. Use props to make components adaptable to different needs. Document each component’s props using JSDoc comments to enhance their discoverability and usability in Storybook.js.

Implementing the Button Component

// src/components/atoms/Button/Button.tsx
import React from 'react';
import { useStyles } from './styles';

interface ButtonProps {
color?: string;
children: React.ReactNode;
onClick?: () => void;
}

const Button: React.FC<ButtonProps> = ({
children,
color,
...props
}) => {
const classes = useStyles({
color
});

return (
<button className={classes.button} {...props}>
{children}
</button>
);
};

export default Button;

Implementing the styles in React-JSS

// src/components/atoms/Button/styles.tsx
import { createUseStyles } from 'react-jss';
import { Theme } from '../../theme/types';

interface Props {
color?:string;
}

export const useStyles = createUseStyles<
string,
Props,
Theme
>((theme) => ({
button: {
background: props => props.color ?? theme.colors.primary,
color: '#fff',
padding: '10px 20px',
border: 'none',
cursor: 'pointer',
'&:hover': {
background: theme.colors.secondary,
},
},
}));

Writing the button story

  1. Define the default export to describe your component and named exports for each story. Here’s an example story for the Button component that allows users to interact with the component’s props.
// src/components/atoms/Button.stories.tsx
import React from 'react';
import { Button } from './Button'; // Adjust the import path to your Button component

// Default export that defines component metadata
export default {
title: 'Components/Button',
component: Button,
};

// Template function to create a Button story
const Template = (args) => <Button {...args} />;

// Story variants
export const Primary = Template.bind({});
Primary.args = {
children: 'Click Me',
};

export const Secondary = Template.bind({});
Secondary.args = {
children: 'Secondary Button',
// Add props to adjust styling for a secondary button, if applicable
};

5. Making Your Design System Publishable

Preparing for Publication

To share your design system across projects, you’ll want to make it publishable as an npm package. Create a package.json file in your design system directory, specifying the name, version, and entry point of your package.

Building Your Design System

Before publishing, compile your components into a distributable format using a bundler like Webpack or Rollup. This ensures that your components can be easily consumed by other projects.

Publishing to npm

Once your package is ready, publish it to npm to make it accessible to your team and projects.

npm login
npm publish

6. Versioning and Updates

Adopt semantic versioning (semver) for your design system package to manage updates and communicate changes to consumers effectively. This includes patch versions for bug fixes, minor versions for backward-compatible feature additions, and major versions for breaking changes.

7. Documentation and Guidelines

Beyond the components themselves, your design system should include comprehensive documentation and usage guidelines. This can cover coding standards, design principles, and component usage instructions. Tools like Docusaurus or GitBook can help you create and maintain this documentation.

8. Engaging the Community

Encourage feedback and contributions from users of your design system. Open channels for communication, such as Slack or GitHub discussions, to gather insights and continually improve your system.

Conclusion

Creating a design system in React is an investment in your team’s efficiency, consistency, and quality of work. By leveraging tools like Storybook and npm, you can build a robust, reusable, and easily maintainable set of components that will serve as the foundation for your projects. Remember, a design system is a living entity that evolves with your project’s needs, so engage your community, solicit feedback, and iterate on your system to keep it relevant and useful.

--

--

Aayush Agarwal
Aayush Agarwal

Written by Aayush Agarwal

SDE at TagMango (YC W20) | Enthusiast in Web & App Development | Driving Innovation in Software Solutions

No responses yet