# Storybook for Next.js ## Table of Contents - [Supported Features](#supported-features) - [Requirements](#requirements) - [Getting Started](#getting-started) - [In a project without Storybook](#in-a-project-without-storybook) - [In a project with Storybook](#in-a-project-with-storybook) - [Documentation](#documentation) - [Options](#options) - [Next.js's Image Component](#nextjss-image-component) - [Local Images](#local-images) - [Remote Images](#remote-images) - [Optimization](#optimization) - [AVIF](#avif) - [Next.js Routing](#nextjs-routing) - [Overriding defaults](#overriding-defaults) - [Global Defaults](#global-defaults) - [Default Router](#default-router) - [Actions Integration Caveats](#actions-integration-caveats) - [Sass/Scss](#sassscss) - [Css/Sass/Scss Modules](#csssassscss-modules) - [Styled JSX](#styled-jsx) - [Postcss](#postcss) - [Absolute Imports](#absolute-imports) - [Runtime Config](#runtime-config) - [Custom Webpack Config](#custom-webpack-config) - [Typescript](#typescript) - [Notes for Yarn v2 and v3 users](#notes-for-yarn-v2-and-v3-users) - [FAQ](#faq) - [Stories for pages](#stories-for-pages) - [Statically imported images won't load](#statically-imported-images-wont-load) - [Module not found: Error: Can't resolve [package name]](#module-not-found-error-cant-resolve-package-name) - [Acknowledgements](#acknowledgements) ## Supported Features 👉 [Next.js's Image Component](#nextjss-image-component) 👉 [Next.js Routing](#nextjs-routing) 👉 [Sass/Scss](#sassscss) 👉 [Css/Sass/Scss Modules](#csssassscss-modules) 👉 [Styled JSX](#styled-jsx) 👉 [Postcss](#postcss) 👉 [Absolute Imports](#absolute-imports) 👉 [Runtime Config](#runtime-config) 👉 [Custom Webpack Config](#custom-webpack-config) 👉 [Typescript](#typescript) (already supported out of the box by Storybook) ## Requirements - [Next.js](https://nextjs.org/) >= 9.x - [Storybook](https://storybook.js.org/) >= 7.x ## Getting Started ### In a project without Storybook Follow the prompts after running this command in your Next.js project's root directory: ```bash npx storybook init ``` [More on getting started with Storybook](https://storybook.js.org/docs/react/get-started/introduction) ### In a project with Storybook Update your `main.js` to look something like this: Install the framework: ```bash yarn install @storybook/nextjs ``` ```js // .storybook/main.js module.exports = { framework: { name: '@storybook/nextjs', options: {}; } } ``` ## Documentation ### Options You can be pass an options object for addional configuration if needed. For example: ```js // .storybook/main.js const path = require('path'); module.exports = { // other config ommited for brevity framework: { name: '@storybook/nextjs', options: { nextConfigPath: path.resolve(__dirname, '../next.config.js'), }, }, // ... }; ``` - `nextConfigPath`: The absolute path to the `next.config.js` ### Next.js's Image Component [next/image](https://nextjs.org/docs/api-reference/next/image) is [notoriously difficult](https://github.com/vercel/next.js/issues/18393) to get working with Storybook. This framework allows you to use Next.js's `Image` component with no configuration! #### Local Images [Local images](https://nextjs.org/docs/basic-features/image-optimization#local-images) work just fine! Keep in mind that this feature was [only added in Next.js v11](https://nextjs.org/blog/next-11#automatic-size-detection-local-images). ```js import Image from 'next/image'; import profilePic from '../public/me.png'; function Home() { return ( <>

My Homepage

Picture of the author

Welcome to my homepage!

); } ``` #### Remote Images [Remote images](https://nextjs.org/docs/basic-features/image-optimization#remote-images) also work just fine! ```js import Image from 'next/image'; export default function Home() { return ( <>

My Homepage

Picture of the author

Welcome to my homepage!

); } ``` #### Optimization All Next.js `Image`s are automatically [unoptimized](https://nextjs.org/docs/api-reference/next/image#unoptimized) for you. If [placeholder="blur"](https://nextjs.org/docs/api-reference/next/image#placeholder) is used, the [blurDataURL](https://nextjs.org/docs/api-reference/next/image#blurdataurl) used is the [src](https://nextjs.org/docs/api-reference/next/image#src) of the image (thus effectively disabling the placeholder). See [this issue](https://github.com/vercel/next.js/issues/18393) for more discussion on how Next.js `Image`s are handled for Storybook. #### AVIF This format is not supported by this framework yet. Feel free to [open up an issue](https://github.com/storybookjs/storybook/issues) if this is something you want to see. ### Next.js Routing [Next.js's router](https://nextjs.org/docs/routing/introduction) is automatically stubbed for you so that when the router is interacted with, all of its interactions are automatically logged to the [Storybook actions tab](https://storybook.js.org/docs/react/essentials/actions) if you have the actions addon. #### Overriding defaults Per-story overrides can be done by adding a `nextRouter` property onto the story [parameters](https://storybook.js.org/docs/react/writing-stories/parameters). The framework will shallowly merge whatever you put here into the router. ```js // SomeComponentThatUsesTheRouter.stories.js import SomeComponentThatUsesTheRouter from './SomeComponentThatUsesTheRouter'; export default { component: SomeComponentThatUsesTheRouter, }; // if you have the actions addon // you can click the links and see the route change events there export const Example = { parameters: { nextRouter: { path: '/profile/[id]', asPath: '/profile/ryanclementshax', query: { id: 'ryanclementshax', }, }, }, }; ``` #### Global Defaults Global defaults can be set in [preview.js](https://storybook.js.org/docs/react/configure/overview#configure-story-rendering) and will be shallowly merged with the default router. ```js // .storybook/main.js export const parameters = { nextRouter: { path: '/some-default-path', asPath: '/some-default-path', query: {}, }, }; ``` #### Default Router The default values on the stubbed router are as follows (see [globals](https://storybook.js.org/docs/react/essentials/toolbars-and-globals#globals) for more details on how globals work) ```ts const defaultRouter = { locale: context?.globals?.locale, route: '/', pathname: '/', query: {}, asPath: '/', push(...args: unknown[]) { action('nextRouter.push')(...args); return Promise.resolve(true); }, replace(...args: unknown[]) { action('nextRouter.replace')(...args); return Promise.resolve(true); }, reload(...args: unknown[]) { action('nextRouter.reload')(...args); }, back(...args: unknown[]) { action('nextRouter.back')(...args); }, prefetch(...args: unknown[]) { action('nextRouter.prefetch')(...args); return Promise.resolve(); }, beforePopState(...args: unknown[]) { action('nextRouter.beforePopState')(...args); }, events: { on(...args: unknown[]) { action('nextRouter.events.on')(...args); }, off(...args: unknown[]) { action('nextRouter.events.off')(...args); }, emit(...args: unknown[]) { action('nextRouter.events.emit')(...args); }, }, isFallback: false, }; ``` #### Actions Integration Caveats If you override a function, you lose the automatic action tab integration and have to build it out yourself. ```js // .storybook/main.js export const parameters = { nextRouter: { push() { // The default implementation that logs the action into the action tab is lost }, }, }; ``` Doing this yourself looks something like this (make sure you install the `@storybook/addon-actions` package): ```js // .storybook/main.js import { action } from '@storybook/addon-actions'; export const parameters = { nextRouter: { push(...args) { // custom logic can go here // this logs to the actions tab action('nextRouter.push')(...args); // return whatever you want here return Promise.resolve(true); }, }, }; ``` ### Sass/Scss [Global sass/scss stylesheets](https://nextjs.org/docs/basic-features/built-in-css-support#sass-support) are supported without any additional configuration as well. Just import them into [preview.js](https://storybook.js.org/docs/react/configure/overview#configure-story-rendering) ```js import '../styles/globals.scss'; ``` This will automatically include any of your [custom sass configurations](https://nextjs.org/docs/basic-features/built-in-css-support#customizing-sass-options) in your `next.config.js` file. ```js // next.config.js const path = require('path'); module.exports = { // any options here are included in sass compilation for your stories sassOptions: { includePaths: [path.join(__dirname, 'styles')], }, }; ``` ### Css/Sass/Scss Modules [css modules](https://nextjs.org/docs/basic-features/built-in-css-support#adding-component-level-css) work as expected. ```js // this import works just fine in Storybook now import styles from './Button.module.css'; // sass/scss is also supported // import styles from './Button.module.scss' // import styles from './Button.module.sass' export function Button() { return ( ); } ``` ### Styled JSX The built in CSS-in-JS solution for Next.js is [styled-jsx](https://nextjs.org/docs/basic-features/built-in-css-support#css-in-js), and this framework supports that out of the box too, zero config. ```js // This works just fine in Storybook now function HelloWorld() { return (
Hello world

scoped!

); } export default HelloWorld; ``` You can use your own babel config too. This is an example of how you can customize styled-jsx. ```json // .babelrc or whatever config file you use { "presets": [ [ "next/babel", { "styled-jsx": { "plugins": ["@styled-jsx/plugin-sass"] } } ] ] } ``` If you use a monorepo, you may need to add the babel config yourself to your storybook project. Just add a babel config to your storybook project with the following contents to get started. ```json { "presets": ["next/babel"] } ``` ### Postcss Next.js lets you [customize postcss config](https://nextjs.org/docs/advanced-features/customizing-postcss-config#default-behavior). Thus this framework will automatically handle your postcss config for you. This allows for cool things like zero config tailwindcss! (See [Next.js' example](https://github.com/vercel/next.js/tree/canary/examples/with-tailwindcss)) ### Absolute Imports Goodbye `../`! Absolute imports from the root directory work just fine. ```js // All good! import Button from 'components/button'; // Also good! import styles from 'styles/HomePage.module.css'; export default function HomePage() { return ( <>

Hello World