The Storybook Design Token Addon allows you to generate design token documentation from your stylesheets. Read more about the motivation for this addon here: https://dev.to/psqrrl/managing-design-tokens-using-storybook-5975
Contents:
- Automatically generates design token descriptions from your annotated stylesheets and icon files
- Parses CSS, SCSS/SASS, Less and SVG files
- Provides various presenters to render examples of your design tokens
- Automatically detects aliases of your css, sass or less variables
- Preview changes to tokens directly in your browser (only for CSS variables)
- Displays property values that match token values, so you can easily recognize hard coded values in your stylesheets
First, install the package using your package manager of choice.
npm install --save-dev storybook-design-token
or
yarn add --dev storybook-design-token
Skip to the next section if you are using Storybook 5.2 or older. For Storybook 5.2 and older add the addon to your main.js
config file in the .storybook directory:
// main.js
module.exports = {
stories: ['../**/*.stories.js'],
addons: ['storybook-design-token']
};
Then add neccessary parameters to your preview.js
in your .storybook directory:
import { addParameters } from '@storybook/react';
const cssReq = require.context('!!raw-loader!../src', true, /.\.css$/);
const cssTokenFiles = cssReq
.keys()
.map((filename) => ({ filename, content: cssReq(filename).default }));
const scssReq = require.context('!!raw-loader!../src', true, /.\.scss$/);
const scssTokenFiles = scssReq
.keys()
.map((filename) => ({ filename, content: scssReq(filename).default }));
const lessReq = require.context('!!raw-loader!../src', true, /.\.less$/);
const lessTokenFiles = lessReq
.keys()
.map((filename) => ({ filename, content: lessReq(filename).default }));
const svgIconsReq = require.context('!!raw-loader!../src', true, /.\.svg$/);
const svgIconTokenFiles = svgIconsReq
.keys()
.map((filename) => ({ filename, content: svgIconsReq(filename).default }));
addParameters({
designToken: {
files: {
css: cssTokenFiles,
scss: scssTokenFiles,
less: lessTokenFiles,
svgIcons: svgIconTokenFiles
},
options: {
hideMatchingHardCodedValues: false
}
}
});
Create a file called addons.js
in your .storybook directory and import the plugin:
// addon.js
import 'storybook-design-token/register';
In your config.js
in your .storybook directory: tell the plugin what files to parse. Important: Call addParameters
before making the configure
call.
// config.js
import { configure, addParameters } from '@storybook/react';
// [...]
const cssReq = require.context('!!raw-loader!../src', true, /.\.css$/);
const cssTokenFiles = cssReq
.keys()
.map((filename) => ({ filename, content: cssReq(filename).default }));
const scssReq = require.context('!!raw-loader!../src', true, /.\.scss$/);
const scssTokenFiles = scssReq
.keys()
.map((filename) => ({ filename, content: scssReq(filename).default }));
const lessReq = require.context('!!raw-loader!../src', true, /.\.less$/);
const lessTokenFiles = lessReq
.keys()
.map((filename) => ({ filename, content: lessReq(filename).default }));
const svgIconsReq = require.context('!!raw-loader!../src', true, /.\.svg$/);
const svgIconTokenFiles = svgIconsReq
.keys()
.map((filename) => ({ filename, content: svgIconsReq(filename).default }));
addParameters({
designToken: {
files: {
css: cssTokenFiles,
scss: scssTokenFiles,
less: lessTokenFiles,
svgIcons: svgIconTokenFiles
},
options: {
hideMatchingHardCodedValues: false
}
}
});
// [...]
configure(loadStories, module);
Make sure to specify the right path after !!raw-loader!
. The path has to be relative to your config file location.
The plugin tries to organize your tokens into token groups. You can annotate token groups by adding specific comment blocks to your stylesheets. Only annotated tokens will be listed by the plugin. The specified presenter is used to render examples of your tokens. See Presenters for further information.
CSS example:
/**
* @tokens Colors
* @presenter Color
*/
:root {
--n0: #fff; /* Token Description */
--n100: #fbfbfb;
--n200: #edeeef;
--n300: #e4e5e7;
--primary: var(--n300);
}
/**
* @tokens Border Radius
* @presenter BorderRadius
*/
:root {
--border-radius-m: 4px;
--border-radius-l: 8px;
}
SCSS example:
/**
* @tokens Colors
* @presenter Color
*/
$n0: #fff; /* Token Description */
$n100: #fbfbfb;
$n200: #edeeef;
$n300: #e4e5e7;
$primary: $n300;
/**
* @tokens Border Radius
* @presenter BorderRadius
*/
$border-radius-m: 4px;
$border-radius-l: 8px;
SVG example:
<?xml version="1.0" encoding="UTF-8"?>
<svg
data-token-name="check"
width="24px"
height="24px"
viewBox="0 0 24 24"
version="1.1"
xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink"
>
<path
d="M9.35221912,16.3536125 L19.5004166,5.34255149 C19.9029667,4.90884428 20.5808871,4.88358644 21.0145944,5.28613652 C21.4483016,5.6886866 21.4735594,6.36660707 21.0710093,6.80031428 L10.1375155,18.6574532 C9.71359736,19.1141823 8.99084087,19.1141823 8.56692275,18.6574532 L3.28613652,12.890538 C2.88358644,12.4568308 2.90884428,11.7789103 3.34255149,11.3763602 C3.77625869,10.9738101 4.45417917,10.999068 4.85672925,11.4327752 L9.35221912,16.3536125 Z"
fill="currentColor"
></path>
</svg>
Make sure to specify the data-token-name
attribute.
Presenters are used to render examples for your design tokens. The following presenters are available:
- Animation
- BorderRadius
- Border
- Color
- Easing
- FontFamily
- FontSize
- FontWeight
- Gradient
- LineHeight
- Opacity
- Shadow
- Spacing
- Svg