Setting up TypeScript with awesome-typescript-loader
We have had the best experience using , but other tutorials may use ts-loader
, just configure accordingly. You can even use babel-loader
with a ts-loader
configuration.
Setting up TypeScript to work with Storybook
We first have to use the custom Webpack config in full control mode, extending default configs by creating a webpack.config.js
file in our Storybook configuration directory (by default, it’s .storybook
):
module.exports = ({ config }) => {
config.module.rules.push({
test: /\.(ts|tsx)$/,
use: [
{
loader: require.resolve('awesome-typescript-loader'),
},
// Optional
{
loader: require.resolve('react-docgen-typescript-loader'),
},
],
});
config.resolve.extensions.push('.ts', '.tsx');
return config;
};
The above example shows a working Webpack config with the integrated. This plugin is not necessary to use Storybook and the section marked // optional
can be safely removed if the features of TSDocgen are not required.
{
"compilerOptions": {
"outDir": "build/lib",
"module": "commonjs",
"target": "es5",
"lib": ["es5", "es6", "es7", "es2017", "dom"],
"sourceMap": true,
"allowJs": false,
"jsx": "react",
"moduleResolution": "node",
"rootDirs": ["src", "stories"],
"baseUrl": "src",
"forceConsistentCasingInFileNames": true,
"noImplicitReturns": true,
"noImplicitThis": true,
"noImplicitAny": true,
"strictNullChecks": true,
"suppressImplicitAnyIndexErrors": true,
"noUnusedLocals": true,
"declaration": true,
"allowSyntheticDefaultImports": true,
"experimentalDecorators": true,
"emitDecoratorMetadata": true
"include": ["src/**/*"],
"exclude": ["node_modules", "build", "scripts"]
}
This is for the default configuration where /stories
is a peer of src
. If you have them all in just src
you may wish to replace "rootDirs": ["src", "stories"]
above with "rootDir": "src",
.
Setting up TypeScript with babel-loader
When using latest create-react-app (CRA 2.0), Babel 7 has native TypeScript support. Setup becomes easier.For a full working demo (that also uses react-docgen-typescript-loader) you can check out this .
Dependencies you may need
config.module.rules.push({
test: /\.(ts|tsx)$/,
loader: require.resolve('babel-loader'),
options: {
presets: [['react-app', { flow: false, typescript: true }]],
},
});
config.resolve.extensions.push('.ts', '.tsx');
return config;
};
tsconfig.json
The default tsconfig.json
that comes with CRA works great. If your stories are outside the src
folder, for example the stories
folder in root, then "rootDirs": ["src", "stories"]
needs to be added to be added to compilerOptions
so it knows what folders to compile. Make sure jsx
is set to preserve. Should be unchanged.
The default storybook index file is stories/index.stories.js
— you’ll want to rename this to stories/index.stories.tsx
.
Import tsx stories
Change config.ts
inside the Storybook config directory (by default, it’s .storybook
) to import stories made with TypeScript:
import { configure } from '@storybook/react';
// automatically import all files ending in *.stories.tsx
const req = require.context('../stories', true, /\.stories\.tsx$/);
function loadStories() {
req.keys().forEach(req);
}
configure(loadStories, module);
Using TypeScript with the TSDocgen addon
The very handy Storybook Info addon autogenerates prop tables documentation for each component, however it doesn’t work with Typescript types. The current solution is to use to preprocess the TypeScript files to give the Info addon what it needs. The webpack config above does this, and so for the rest of your stories you use it as per normal:
Please refer to the react-docgen-typescript-loader docs for writing prop descriptions and other annotations to your Typescript interfaces.
import { addDecorator } from '@storybook/react';
import { withInfo } from '@storybook/addon-info';
// Globally in your .storybook/config.js, or alternatively, per-chapter
addDecorator(
withInfo({
styles: {
header: {
h1: {
marginRight: '20px',
fontSize: '25px',
display: 'inline',
},
body: {
paddingTop: 0,
paddingBottom: 0,
},
h2: {
display: 'inline',
color: '#999',
},
},
infoBody: {
backgroundColor: '#eee',
padding: '0px 5px',
lineHeight: '2',
},
inline: true,
source: false,
})
This can be used like so:
import * as React from 'react';
import { storiesOf } from '@storybook/react';
import { PrimaryButton } from './Button';
import { text, select, boolean } from '@storybook/addon-knobs/react';
storiesOf('Components/Button', module).addWithJSX(
'basic PrimaryButton',
() => (
<PrimaryButton
label={text('label', 'Enroll')}
disabled={boolean('disabled', false)}
onClick={() => alert('hello there')}
/>
),
{
info: {
text: `
### Notes
light button seen on <https://zpl.io/aM49ZBd>
### Usage
~~~js
<PrimaryButton
label={text('label', 'Enroll')}
disabled={boolean('disabled',false)}
onClick={() => alert('hello there')}
/>
~~~
`,
},
}
);
And this is how it looks:
Note: Component docgen information can not be generated for components that are only exported as default. You can work around the issue by exporting the component using a named export.
Setting up Jest tests
The ts-jest README explains pretty clearly how to get Jest to recognize TypeScript code.
Building your TypeScript Storybook
You will need to set up some scripts - these may help:
"scripts": {
"start": "react-scripts-ts start",
"build": "npm run lint && npm run build-lib && build-storybook",
"build-lib-watch": "tsc -w",
"build-lib": "tsc && npm run copy-css-to-lib && npm run copy-svg-to-lib && npm run copy-png-to-lib && npm run copy-woff2-to-lib",
"test": "react-scripts-ts test --env=jsdom",
"test:coverage": "npm test -- --coverage",
"eject": "react-scripts-ts eject",
"storybook": "start-storybook -p 6006",
"build-storybook": "build-storybook",
"copy-css-to-lib": "cpx \"./src/**/*.css\" ./build/lib",
"copy-woff2-to-lib": "cpx \"./src/**/*.woff2\" ./build/lib",
"copy-svg-to-lib": "cpx \"./src/**/*.svg\" ./build/lib",
"copy-png-to-lib": "cpx \"./src/**/*.png\" ./build/lib",
"lint": "tslint -c tslint.json 'src/**/*.{ts,tsx}'"