Next.js 版 Storybook
Next.js 版 Storybook 是一个框架,可以轻松地为 Next.js 应用程序隔离开发和测试 UI 组件。它包括
- 🔀 路由
- 🖼 图像优化
- ⤵️ 绝对导入
- 🎨 样式
- 🎛 Webpack 和 Babel 配置
- 💫 以及更多!
要求
- Next.js ≥ 13.5
- Storybook ≥ 7.0
开始使用
在没有 Storybook 的项目中
在您的 Next.js 项目根目录中运行此命令后,按照提示操作
npm create storybook@latest
在已有 Storybook 的项目中
此框架旨在与 Storybook 7+ 版本一起使用。如果您尚未使用 v7,请使用此命令升级
npx storybook@latest upgrade
自动迁移
运行上面的 upgrade
命令时,您应该会收到提示,询问您是否迁移到 @storybook/nextjs
,它应该为您处理一切。如果自动迁移不适用于您的项目,请参考下面的手动迁移。
手动迁移
首先,安装框架
npm install --save-dev @storybook/nextjs
然后,更新您的 .storybook/main.js|ts
以更改 framework 属性
import { StorybookConfig } from '@storybook/nextjs';
const config: StorybookConfig = {
// ...
// framework: '@storybook/react-webpack5', 👈 Remove this
framework: '@storybook/nextjs', // 👈 Add this
};
export default config;
最后,如果您之前使用 Storybook 插件与 Next.js 集成,则使用此框架时不再需要这些插件,可以将其删除
import { StorybookConfig } from '@storybook/nextjs';
const config: StorybookConfig = {
// ...
addons: [
// ...
// 👇 These can both be removed
// 'storybook-addon-next',
// 'storybook-addon-next-router',
],
};
export default config;
使用 Vite
(⚠️ 实验性)
您可以使用我们新鲜出炉的实验性 @storybook/experimental-nextjs-vite
框架,它基于 Vite,无需 Webpack 和 Babel。它支持此处记录的所有功能。
将 Next.js 框架与 Vite 一起使用需要 Next.js 14.1.0 或更高版本。
npm install --save-dev @storybook/experimental-nextjs-vite
然后,更新您的 .storybook/main.js|ts
以更改 framework 属性
import { StorybookConfig } from '@storybook/experimental-nextjs-vite';
const config: StorybookConfig = {
// ...
// framework: '@storybook/react-webpack5', 👈 Remove this
framework: '@storybook/experimental-nextjs-vite', // 👈 Add this
};
export default config;
如果您的 Storybook 配置在 webpackFinal
中包含自定义 Webpack 操作,则您可能需要在 viteFinal
中创建等效项。
有关更多信息,请参阅 Vite 构建器文档。
最后,如果您之前使用 Storybook 插件与 Next.js 集成,则使用此框架时不再需要这些插件,可以将其删除
import { StorybookConfig } from '@storybook/experimental-nextjs-vite';
const config: StorybookConfig = {
// ...
addons: [
// ...
// 👇 These can both be removed
// 'storybook-addon-next',
// 'storybook-addon-next-router',
],
};
export default config;
运行设置向导
如果一切顺利,您应该会看到一个设置向导,它将帮助您开始使用 Storybook,向您介绍主要概念和功能,包括 UI 的组织方式、如何编写您的第一个 story,以及如何利用 controls 测试组件对各种输入的响应。
如果您跳过了向导,您可以随时通过将 ?path=/onboarding
查询参数添加到您的 Storybook 实例的 URL 中再次运行它,前提是示例 stories 仍然可用。
Next.js 的 Image 组件
此框架允许您使用 Next.js 的 next/image,无需任何配置。
本地图像
支持本地图像。
// index.jsx
import Image from 'next/image';
import profilePic from '../public/me.png';
function Home() {
return (
<>
<h1>My Homepage</h1>
<Image
src={profilePic}
alt="Picture of the author"
// width={500} automatically provided
// height={500} automatically provided
// blurDataURL="../public/me.png" set to equal the image itself (for this framework)
// placeholder="blur" // Optional blur-up while loading
/>
<p>Welcome to my homepage!</p>
</>
);
}
远程图像
也支持远程图像。
// index.jsx
import Image from 'next/image';
export default function Home() {
return (
<>
<h1>My Homepage</h1>
<Image src="/me.png" alt="Picture of the author" width={500} height={500} />
<p>Welcome to my homepage!</p>
</>
);
}
Next.js 字体优化
next/font 在 Storybook 中部分受支持。next/font/google
和 next/font/local
包受支持。
next/font/google
您无需执行任何操作。next/font/google
开箱即用。
next/font/local
对于本地字体,您必须定义 src 属性。路径相对于调用字体加载器函数的目录。
如果以下组件像这样定义您的 localFont
// src/components/MyComponent.js
import localFont from 'next/font/local';
const localRubikStorm = localFont({ src: './fonts/RubikStorm-Regular.ttf' });
staticDir
映射
如果您使用 @storybook/experimental-nextjs-vite
而不是 @storybook/nextjs
,您可以安全地跳过本节。基于 Vite 的框架会自动处理映射。
您必须通过 staticDirs
配置告诉 Storybook fonts
目录的位置。from
值相对于 .storybook
目录。to
值相对于 Storybook 的执行上下文。很可能它是您项目的根目录。
import { StorybookConfig } from '@storybook/nextjs';
const config: StorybookConfig = {
// ...
staticDirs: [
{
from: '../src/components/fonts',
to: 'src/components/fonts',
},
],
};
export default config;
next/font
不支持的功能
以下功能尚不支持(暂不支持)。未来可能会计划支持这些功能
- 在 next.config.js 中支持字体加载器配置
- fallback 选项
- adjustFontFallback 选项
- preload 选项被忽略。Storybook 以自己的方式处理字体加载。
- display 选项被忽略。所有字体都以 display 设置为 "block" 加载,以使 Storybook 正确加载字体。
在测试期间模拟字体
有时,从 Google 获取字体可能会在您的 Storybook 构建步骤中失败。强烈建议模拟这些请求,因为这些故障也可能导致您的管道失败。Next.js 支持通过 JavaScript 模块模拟字体,该模块位于 env var NEXT_FONT_GOOGLE_MOCKED_RESPONSES
引用的位置。
例如,使用 GitHub Actions
# .github/workflows/ci.yml
- uses: chromaui/action@v1
env:
#👇 the location of mocked fonts to use
NEXT_FONT_GOOGLE_MOCKED_RESPONSES: ${{ github.workspace }}/mocked-google-fonts.js
with:
projectToken: ${{ secrets.CHROMATIC_PROJECT_TOKEN }}
token: ${{ secrets.GITHUB_TOKEN }}
您模拟的字体将如下所示
// mocked-google-fonts.js
//👇 Mocked responses of google fonts with the URL as the key
module.exports = {
'https://fonts.googleapis.com/css?family=Inter:wght@400;500;600;800&display=block': `
/* cyrillic-ext */
@font-face {
font-family: 'Inter';
font-style: normal;
font-weight: 400;
font-display: block;
src: url(https://fonts.gstatic.com/s/inter/v12/UcCO3FwrK3iLTeHuS_fvQtMwCp50KnMw2boKoduKmMEVuLyfAZJhiJ-Ek-_EeAmM.woff2) format('woff2');
unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F;
}
/* more font declarations go here */
/* latin */
@font-face {
font-family: 'Inter';
font-style: normal;
font-weight: 400;
font-display: block;
src: url(https://fonts.gstatic.com/s/inter/v12/UcCO3FwrK3iLTeHuS_fvQtMwCp50KnMw2boKoduKmMEVuLyfAZ9hiJ-Ek-_EeA.woff2) format('woff2');
unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0304, U+0308, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
}`,
};
Next.js 路由
Next.js 的路由器会自动为您进行桩化,以便在与路由器交互时,如果您有 Storybook actions 插件,则其所有交互都会自动记录到 Actions 面板。
您应该仅在 pages
目录中使用 next/router
。在 app
目录中,必须使用 next/navigation
。
覆盖默认值
可以通过在 story parameters 上添加 nextjs.router
属性来完成每个 story 的覆盖。框架会将您在此处放置的任何内容浅合并到路由器中。
import { Meta, StoryObj } from '@storybook/react';
import RouterBasedComponent from './RouterBasedComponent';
const meta: Meta<typeof RouterBasedComponent> = {
component: RouterBasedComponent,
};
export default meta;
type Story = StoryObj<typeof RouterBasedComponent>;
// If you have the actions addon,
// you can interact with the links and see the route change events there
export const Example: Story = {
parameters: {
nextjs: {
router: {
pathname: '/profile/[id]',
asPath: '/profile/1',
query: {
id: '1',
},
},
},
},
};
这些覆盖也可以应用于组件的所有 stories 或 项目中的所有 stories。标准的 参数继承规则适用。
默认路由器
桩化路由器上的默认值如下(有关全局变量如何工作的更多详细信息,请参阅 全局变量)。
// Default router
const defaultRouter = {
// The locale should be configured globally: https://storybook.org.cn/docs/essentials/toolbars-and-globals#globals
locale: globals?.locale,
asPath: '/',
basePath: '/',
isFallback: false,
isLocaleDomain: false,
isReady: true,
isPreview: false,
route: '/',
pathname: '/',
query: {},
};
此外,router
对象包含所有原始方法(例如 push()
、replace()
等),作为可以使用 常规模拟 API 操作和断言的模拟函数。
要覆盖这些默认值,您可以使用 参数和 beforeEach
// .storybook/preview.ts
import { Preview } from '@storybook/react';
// 👇 Must include the `.mock` portion of filename to have mocks typed correctly
import { getRouter } from '@storybook/nextjs/router.mock';
const preview: Preview = {
parameters: {
nextjs: {
// 👇 Override the default router properties
router: {
basePath: '/app/',
},
},
},
async beforeEach() {
// 👇 Manipulate the default router method mocks
getRouter().push.mockImplementation(() => {
/* ... */
});
},
};
Next.js 导航
请注意,next/navigation
只能在 app
目录中的组件/页面中使用。
将 nextjs.appDirectory
设置为 true
如果您的 story 导入的组件使用了 next/navigation
,您需要在该组件的 stories 中将参数 nextjs.appDirectory
设置为 true
import { Meta, StoryObj } from '@storybook/react';
import NavigationBasedComponent from './NavigationBasedComponent';
const meta: Meta<typeof NavigationBasedComponent> = {
component: NavigationBasedComponent,
parameters: {
nextjs: {
appDirectory: true, // 👈 Set this
},
},
};
export default meta;
如果您的 Next.js 项目对每个页面都使用 app
目录(换句话说,它没有 pages
目录),您可以在 .storybook/preview.js|ts
文件中将参数 nextjs.appDirectory
设置为 true
,以将其应用于所有 stories。
import { Preview } from '@storybook/react';
const preview: Preview = {
// ...
parameters: {
// ...
nextjs: {
appDirectory: true,
},
},
};
export default preview;
覆盖默认值
可以通过在 story parameters 上添加 nextjs.navigation
属性来完成每个 story 的覆盖。框架会将您在此处放置的任何内容浅合并到路由器中。
import { Meta, StoryObj } from '@storybook/react';
import NavigationBasedComponent from './NavigationBasedComponent';
const meta: Meta<typeof NavigationBasedComponent> = {
component: NavigationBasedComponent,
parameters: {
nextjs: {
appDirectory: true,
},
},
};
export default meta;
type Story = StoryObj<typeof NavigationBasedComponent>;
// If you have the actions addon,
// you can interact with the links and see the route change events there
export const Example: Story = {
parameters: {
nextjs: {
navigation: {
pathname: '/profile',
query: {
user: '1',
},
},
},
},
};
这些覆盖也可以应用于组件的所有 stories 或 项目中的所有 stories。标准的 参数继承规则适用。
useSelectedLayoutSegment
、useSelectedLayoutSegments
和 useParams
钩子
Storybook 支持 useSelectedLayoutSegment
、useSelectedLayoutSegments
和 useParams
钩子。您必须设置 nextjs.navigation.segments
参数以返回您要使用的 segments 或 params。
import { Meta, StoryObj } from '@storybook/react';
import NavigationBasedComponent from './NavigationBasedComponent';
const meta: Meta<typeof NavigationBasedComponent> = {
component: NavigationBasedComponent,
parameters: {
nextjs: {
appDirectory: true,
navigation: {
segments: ['dashboard', 'analytics'],
},
},
},
};
export default meta;
使用上述配置,stories 中渲染的组件将从钩子接收以下值
// NavigationBasedComponent.js
import { useSelectedLayoutSegment, useSelectedLayoutSegments, useParams } from 'next/navigation';
export default function NavigationBasedComponent() {
const segment = useSelectedLayoutSegment(); // dashboard
const segments = useSelectedLayoutSegments(); // ["dashboard", "analytics"]
const params = useParams(); // {}
// ...
}
要使用 useParams
,您必须使用 segments 数组,其中每个元素都是一个包含两个字符串的数组。第一个字符串是 param 键,第二个字符串是 param 值。
import { Meta, StoryObj } from '@storybook/react';
import NavigationBasedComponent from './NavigationBasedComponent';
const meta: Meta<typeof NavigationBasedComponent> = {
component: NavigationBasedComponent,
parameters: {
nextjs: {
appDirectory: true,
navigation: {
segments: [
['slug', 'hello'],
['framework', 'nextjs'],
],
},
},
},
};
export default meta;
使用上述配置,stories 中渲染的组件将从钩子接收以下值
// ParamsBasedComponent.js
import { useSelectedLayoutSegment, useSelectedLayoutSegments, useParams } from 'next/navigation';
export default function ParamsBasedComponent() {
const segment = useSelectedLayoutSegment(); // hello
const segments = useSelectedLayoutSegments(); // ["hello", "nextjs"]
const params = useParams(); // { slug: "hello", framework: "nextjs" }
...
}
这些覆盖也可以应用于单个 story 或 项目中的所有 stories。标准的 参数继承规则适用。
如果未设置,nextjs.navigation.segments
的默认值为 []
。
默认导航上下文
桩化导航上下文上的默认值如下
// Default navigation context
const defaultNavigationContext = {
pathname: '/',
query: {},
};
此外,router
对象包含所有原始方法(例如 push()
、replace()
等),作为可以使用 常规模拟 API 操作和断言的模拟函数。
要覆盖这些默认值,您可以使用 参数和 beforeEach
// .storybook/preview.ts
import { Preview } from '@storybook/react';
// 👇 Must include the `.mock` portion of filename to have mocks typed correctly
import { getRouter } from '@storybook/nextjs/navigation.mock';
const preview: Preview = {
parameters: {
nextjs: {
// 👇 Override the default navigation properties
navigation: {
pathname: '/app/',
},
},
},
async beforeEach() {
// 👇 Manipulate the default navigation method mocks
getRouter().push.mockImplementation(() => {
/* ... */
});
},
};
Next.js Head
next/head
开箱即用。您可以像在 Next.js 应用程序中一样在 stories 中使用它。请记住,Head children
会被放置到 Storybook 用于渲染 stories 的 iframe 的 head 元素中。
Sass/Scss
也支持全局 Sass/Scss 样式表,无需任何额外配置。只需将它们导入到 .storybook/preview.js|ts
中即可
// .storybook/preview.js|ts
import '../styles/globals.scss';
这将自动包含您的 next.config.js
文件中任何自定义 Sass 配置。
// next.config.js
import * as path from 'path';
export default {
// Any options here are included in Sass compilation for your stories
sassOptions: {
includePaths: [path.join(__dirname, 'styles')],
},
};
CSS/Sass/Scss 模块
CSS 模块按预期工作。
// src/components/Button.jsx
// This import will work in Storybook
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 (
<button type="button" className={styles.error}>
Destroy
</button>
);
}
Styled JSX
Next.js 的内置 CSS-in-JS 解决方案是 styled-jsx,并且此框架也开箱即用地支持它,零配置。
// src/components/HelloWorld.jsx
// This will work in Storybook
function HelloWorld() {
return (
<div>
Hello world
<p>scoped!</p>
<style jsx>{`
p {
color: blue;
}
div {
background: red;
}
@media (max-width: 600px) {
div {
background: blue;
}
}
`}</style>
<style global jsx>{`
body {
background: black;
}
`}</style>
</div>
);
}
export default HelloWorld;
你也可以使用你自己的 babel 配置。这是一个关于如何自定义 styled-jsx 的示例。
// .babelrc (or whatever config file you use)
{
"presets": [
[
"next/babel",
{
"styled-jsx": {
"plugins": ["@styled-jsx/plugin-sass"]
}
}
]
]
}
PostCSS
Next.js 允许你自定义 PostCSS 配置。因此,此框架将自动为你处理 PostCSS 配置。
这允许实现很酷的功能,例如零配置 Tailwind! (请参阅 Next.js 的示例)
绝对路径导入
支持从根目录的绝对路径导入。
// index.jsx
// All good!
import Button from 'components/button';
// Also good!
import styles from 'styles/HomePage.module.css';
export default function HomePage() {
return (
<>
<h1 className={styles.title}>Hello World</h1>
<Button />
</>
);
}
也适用于 .storybook/preview.js|ts
中的全局样式!
// .storybook/preview.js|ts
import 'styles/globals.scss';
// ...
绝对路径导入在 stories/tests 中无法被 mock。有关更多信息,请参阅Mocking modules部分。
模块别名
也支持模块别名。
// index.jsx
// All good!
import Button from '@/components/button';
// Also good!
import styles from '@/styles/HomePage.module.css';
export default function HomePage() {
return (
<>
<h1 className={styles.title}>Hello World</h1>
<Button />
</>
);
}
子路径导入
作为 模块别名 的替代方案,你可以使用子路径导入来导入模块。这遵循 Node 包标准,并且在mock 模块时具有优势。
要配置子路径导入,你需要在项目的 package.json
文件中定义 imports
属性。此属性将子路径映射到实际文件路径。下面的示例为项目中的所有模块配置子路径导入
// package.json
{
"imports": {
"#*": ["./*", "./*.ts", "./*.tsx"]
}
}
由于子路径导入取代了模块别名,你可以从 TypeScript 配置中删除路径别名。
然后可以像这样使用
// index.jsx
import Button from '#components/button';
import styles from '#styles/HomePage.module.css';
export default function HomePage() {
return (
<>
<h1 className={styles.title}>Hello World</h1>
<Button />
</>
);
}
Mock 模块
组件通常依赖于导入到组件文件中的模块。这些模块可能来自外部包或项目内部。在 Storybook 中渲染这些组件或对其进行测试时,你可能希望 mock 这些模块 以控制和断言其行为。
内置 mock 模块
此框架为许多 Next.js 的内部模块提供了 mock
@storybook/nextjs/cache.mock
@storybook/nextjs/headers.mock
@storybook/nextjs/navigation.mock
@storybook/nextjs/router.mock
Mock 其他模块
在 Storybook 中 mock 其他模块的方式取决于你如何将模块导入到组件中。
无论采用哪种方法,第一步都是创建 mock 文件。这是一个名为 session
的模块的 mock 文件示例
import { fn } from '@storybook/test';
import * as actual from './session';
export * from './session';
export const getUserFromSession = fn(actual.getUserFromSession).mockName('getUserFromSession');
使用子路径导入
如果你正在使用子路径导入,你可以调整配置以应用条件,以便在 Storybook 内部使用 mock 模块。下面的示例为四个内部模块配置子路径导入,然后在 Storybook 中 mock 这些模块
{
"imports": {
"#api": {
// storybook condition applies to Storybook
"storybook": "./api.mock.ts",
"default": "./api.ts",
},
"#app/actions": {
"storybook": "./app/actions.mock.ts",
"default": "./app/actions.ts",
},
"#lib/session": {
"storybook": "./lib/session.mock.ts",
"default": "./lib/session.ts",
},
"#lib/db": {
// test condition applies to test environments *and* Storybook
"test": "./lib/db.mock.ts",
"default": "./lib/db.ts",
},
"#*": ["./*", "./*.ts", "./*.tsx"],
},
}
每个子路径必须以 #
开头,以将其与常规模块路径区分开。#*
条目是一个 catch-all,它将所有子路径映射到根目录。
使用模块别名
如果你正在使用模块别名,你可以将 Webpack 别名添加到你的 Storybook 配置中,以指向 mock 文件。
// Replace your-framework with the framework you are using (e.g., react-webpack5, vue3-vite)
import type { StorybookConfig } from '@storybook/your-framework';
const config: StorybookConfig = {
framework: '@storybook/your-framework',
stories: ['../src/**/*.mdx', '../src/**/*.stories.@(js|jsx|mjs|ts|tsx)'],
viteFinal: async (config) => {
if (config.resolve) {
config.resolve.alias = {
...config.resolve?.alias,
// 👇 External module
lodash: require.resolve('./lodash.mock'),
// 👇 Internal modules
'@/api': path.resolve(__dirname, './api.mock.ts'),
'@/app/actions': path.resolve(__dirname, './app/actions.mock.ts'),
'@/lib/session': path.resolve(__dirname, './lib/session.mock.ts'),
'@/lib/db': path.resolve(__dirname, './lib/db.mock.ts'),
};
}
return config;
},
};
export default config;
运行时配置
Next.js 允许运行时配置,它允许你导入一个方便的 getConfig
函数,以获取在 next.config.js
文件中定义的某些配置。
在此框架的 Storybook 上下文中,你可以期望 Next.js 的运行时配置功能可以正常工作。
请注意,由于 Storybook 不会服务器渲染你的组件,因此你的组件只会看到他们在客户端通常看到的内容(即,他们看不到 serverRuntimeConfig
,但会看到 publicRuntimeConfig
)。
例如,考虑以下 Next.js 配置
// next.config.js
module.exports = {
serverRuntimeConfig: {
mySecret: 'secret',
secondSecret: process.env.SECOND_SECRET, // Pass through env variables
},
publicRuntimeConfig: {
staticFolder: '/static',
},
};
在 Storybook 中调用时,调用 getConfig
将返回以下对象
// Runtime config
{
"serverRuntimeConfig": {},
"publicRuntimeConfig": {
"staticFolder": "/static"
}
}
自定义 Webpack 配置
如果你正在使用 @storybook/experimental-nextjs-vite
而不是 @storybook/nextjs
,则可以安全地跳过此部分。基于 Vite 的 Next.js 框架不支持 Webpack 设置。
Next.js 开箱即用地提供了许多功能,例如 Sass 支持,但有时你会向 Next.js 添加自定义 Webpack 配置修改。此框架负责处理你想要添加的大部分 Webpack 修改。如果 Next.js 开箱即用地支持某项功能,则该功能将在 Storybook 中开箱即用。如果 Next.js 不开箱即用地支持某些功能,但使其易于配置,则此框架将为 Storybook 执行相同的操作。
任何希望用于 Storybook 的 Webpack 修改都应在 .storybook/main.js|ts
中进行。
注意:并非所有 Webpack 修改都可以在 next.config.js
和 .storybook/main.js|ts
之间复制/粘贴。建议你研究如何正确地对 Storybook 的 Webpack 配置进行修改,以及 Webpack 的工作原理。
以下是如何使用此框架向 Storybook 添加 SVGR 支持的示例。
import { StorybookConfig } from '@storybook/nextjs';
const config: StorybookConfig = {
// ...
webpackFinal: async (config) => {
config.module = config.module || {};
config.module.rules = config.module.rules || [];
// This modifies the existing image rule to exclude .svg files
// since you want to handle those files with @svgr/webpack
const imageRule = config.module.rules.find((rule) => rule?.['test']?.test('.svg'));
if (imageRule) {
imageRule['exclude'] = /\.svg$/;
}
// Configure .svg files to be loaded with @svgr/webpack
config.module.rules.push({
test: /\.svg$/,
use: ['@svgr/webpack'],
});
return config;
},
};
export default config;
Typescript
Storybook 处理大多数 Typescript 配置,但此框架为 Next.js 对 绝对路径导入和模块路径别名 的支持添加了额外的支持。简而言之,它考虑了你的 tsconfig.json
的 baseUrl 和 paths。因此,如下所示的 tsconfig.json
将开箱即用。
// tsconfig.json
{
"compilerOptions": {
"baseUrl": ".",
"paths": {
"@/components/*": ["components/*"]
}
}
}
React 服务端组件 (RSC)
(⚠️ 实验性)
如果你的应用程序使用React 服务端组件 (RSC),Storybook 可以在浏览器中的 stories 中渲染它们。
要启用此功能,请在你的 .storybook/main.js|ts
配置中设置 experimentalRSC
功能标志
import { StorybookConfig } from '@storybook/nextjs';
const config: StorybookConfig = {
// ...
features: {
experimentalRSC: true,
},
};
export default config;
设置此标志会自动将你的 story 包装在 Suspense 包装器中,该包装器能够在 NextJS 版本的 React 中渲染异步组件。
如果此包装器在你的任何现有 stories 中引起问题,你可以在全局/组件/story 级别使用 react.rsc
参数 选择性地禁用它
import { Meta, StoryObj } from '@storybook/react';
import MyServerComponent from './MyServerComponent';
const meta: Meta<typeof MyServerComponent> = {
component: MyServerComponent,
parameters: {
react: { rsc: false },
},
};
export default meta;
请注意,如果你的服务端组件访问服务端资源(如文件系统或特定于 Node 的库),则将服务端组件包装在 Suspense 中无济于事。要解决此问题,你需要使用 Webpack 别名 或 storybook-addon-module-mock 之类的插件来 mock 你的数据访问层。
如果你的服务端组件通过网络访问数据,我们建议使用 MSW Storybook Addon 来 mock 网络请求。
将来,我们将在 Storybook 中提供更好的 mock 支持,并支持服务端 Actions。
Yarn v2 和 v3 用户注意事项
如果你正在使用 Yarn v2 或 v3,你可能会遇到 Storybook 无法解析 style-loader
或 css-loader
的问题。例如,你可能会收到如下错误
Module not found: Error: Can't resolve 'css-loader'
Module not found: Error: Can't resolve 'style-loader'
这是因为这些版本的 Yarn 具有与 Yarn v1.x 不同的包解析规则。如果你的情况是这样,请直接安装该包。
FAQ
用于获取数据的页面/组件的 Stories
Next.js 页面可以直接在 app
目录中的服务端组件内获取数据,这通常包括仅在 node 环境中运行的模块导入。这(目前)在 Storybook 中不起作用,因为如果你在 stories 中从包含这些 node 模块导入的 Next.js 页面文件导入,你的 Storybook 的 Webpack 将崩溃,因为这些模块将无法在浏览器中运行。要解决此问题,你可以将页面文件中的组件提取到单独的文件中,并在你的 stories 中导入该纯组件。或者,如果由于某些原因不可行,你可以在 Storybook 的 webpackFinal
配置中 polyfill 这些模块。
之前
// app/my-page/index.jsx
async function getData() {
const res = await fetch(...);
// ...
}
// Using this component in your stories will break the Storybook build
export default async function Page() {
const data = await getData();
return // ...
}
之后
// app/my-page/index.jsx
// Use this component in your stories
import MyPage from './components/MyPage';
async function getData() {
const res = await fetch(...);
// ...
}
export default async function Page() {
const data = await getData();
return <MyPage {...data} />;
}
静态导入的图像无法加载
请确保你以与在正常开发中使用 next/image
时相同的方式处理图像导入。
在使用此框架之前,图像导入将导入图像的原始路径(例如 'static/media/stories/assets/logo.svg'
)。现在,图像导入以“Next.js 方式”工作,这意味着你在导入图像时现在会得到一个对象。例如
// Image import object
{
"src": "static/media/stories/assets/logo.svg",
"height": 48,
"width": 48,
"blurDataURL": "static/media/stories/assets/logo.svg"
}
因此,如果 Storybook 中的某些内容未正确显示图像,请确保你期望从导入返回对象,而不仅仅是资源路径。
有关 Next.js 如何处理静态图像导入的更多详细信息,请参阅本地图像。
Module not found: Error: Can't resolve package name
如果你正在使用 Yarn v2 或 v3,你可能会遇到这种情况。有关更多详细信息,请参阅Yarn v2 和 v3 用户注意事项。
如果我正在使用 Vite builder 怎么办?
我们引入了实验性的 Vite builder 支持。只需安装实验性框架包 @storybook/experimental-nextjs-vite
,并将所有 @storybook/nextjs
实例替换为 @storybook/experimental-nextjs-vite
。
Error: You are importing avif images, but you don't have sharp installed. You have to install sharp in order to use image optimization features in Next.js.
sharp
是 Next.js 图像优化功能的依赖项。如果看到此错误,则需要在项目中安装 sharp
。
npm install sharp
yarn add sharp
pnpm add sharp
你可以参考 Next.js 文档中的安装 sharp
以使用内置图像优化以获取更多信息。
API
模块
@storybook/nextjs
包导出几个模块,使你能够mock Next.js 的内部行为。
@storybook/nextjs/export-mocks
类型:{ getPackageAliases: ({ useESM?: boolean }) => void }
getPackageAliases
是一个辅助函数,用于生成设置可移植 stories所需的别名。
// jest.config.ts
import type { Config } from 'jest';
import nextJest from 'next/jest.js';
// 👇 Import the utility function
import { getPackageAliases } from '@storybook/nextjs/export-mocks';
const createJestConfig = nextJest({
// Provide the path to your Next.js app to load next.config.js and .env files in your test environment
dir: './',
});
const config: Config = {
testEnvironment: 'jsdom',
// ... rest of Jest config
moduleNameMapper: {
...getPackageAliases(), // 👈 Add the utility as mapped module names
},
};
export default createJestConfig(config);
@storybook/nextjs/cache.mock
类型:typeof import('next/cache')
此模块导出 next/cache
模块导出的 mock 实现。你可以使用它来创建你自己的 mock 实现,或者在 story 的 play 函数 中断言 mock 调用。
import type { Meta, StoryObj } from '@storybook/react';
import { expect, userEvent, within } from '@storybook/test';
// 👇 Must include the `.mock` portion of filename to have mocks typed correctly
import { revalidatePath } from '@storybook/nextjs/cache.mock';
import MyForm from './my-form';
const meta: Meta<typeof MyForm> = {
component: MyForm,
};
export default meta;
type Story = StoryObj<typeof MyForm>;
export const Submitted: Story = {
async play({ canvasElement }) {
const canvas = within(canvasElement);
const submitButton = canvas.getByRole('button', { name: /submit/i });
await userEvent.click(saveButton);
// 👇 Use any mock assertions on the function
await expect(revalidatePath).toHaveBeenCalledWith('/');
},
};
@storybook/nextjs/headers.mock
类型:来自 Next.js 的 cookies
、headers
和 draftMode
此模块导出 next/headers
模块导出的可写 mock 实现。你可以使用它来设置在你的 story 中读取的 cookies 或 headers,并在之后断言它们已被调用。
Next.js 的默认 headers()
导出是只读的,但此模块公开了允许你写入 headers 的方法
headers().append(name: string, value: string)
:如果 header 已经存在,则将值附加到 header。headers().delete(name: string)
:删除 headerheaders().set(name: string, value: string)
:将 header 设置为提供的值。
对于 cookies,你可以使用现有的 API 来写入它们。例如,cookies().set('firstName', 'Jane')
。
由于 headers()
、cookies()
及其子函数都是 mock,因此你可以在你的 stories 中使用任何 mock 实用程序,例如 headers().getAll.mock.calls
。
import type { Meta, StoryObj } from '@storybook/react';
import { expect, fireEvent, userEvent, within } from '@storybook/test';
// 👇 Must include the `.mock` portion of filename to have mocks typed correctly
import { cookies, headers } from '@storybook/nextjs/headers.mock';
import MyForm from './my-form';
const meta: Meta<typeof MyForm> = {
component: MyForm,
};
export default meta;
type Story = StoryObj<typeof MyForm>;
export const LoggedInEurope: Story = {
async beforeEach() {
// 👇 Set mock cookies and headers ahead of rendering
cookies().set('username', 'Sol');
headers().set('timezone', 'Central European Summer Time');
},
async play() {
// 👇 Assert that your component called the mocks
await expect(cookies().get).toHaveBeenCalledOnce();
await expect(cookies().get).toHaveBeenCalledWith('username');
await expect(headers().get).toHaveBeenCalledOnce();
await expect(cookies().get).toHaveBeenCalledWith('timezone');
},
};
@storybook/nextjs/navigation.mock
类型:typeof import('next/navigation') & getRouter: () => ReturnType<typeof import('next/navigation')['useRouter']>
此模块导出 next/navigation
模块导出的 mock 实现。它还导出一个 getRouter
函数,该函数返回 Next.js 的 router
对象(来自 useRouter
)的 mock 版本,允许操作和断言属性。你可以在 story 的 play 函数 中使用它 mock 实现或断言 mock 调用。
import type { Meta, StoryObj } from '@storybook/react';
import { expect, fireEvent, userEvent, within } from '@storybook/test';
// 👇 Must include the `.mock` portion of filename to have mocks typed correctly
import { redirect, getRouter } from '@storybook/nextjs/navigation.mock';
import MyForm from './my-form';
const meta: Meta<typeof MyForm> = {
component: MyForm,
parameters: {
nextjs: {
// 👇 As in the Next.js application, next/navigation only works using App Router
appDirectory: true,
},
},
};
export default meta;
type Story = StoryObj<typeof MyForm>;
export const Unauthenticated: Story = {
async play() {
// 👇 Assert that your component called redirect()
await expect(redirect).toHaveBeenCalledWith('/login', 'replace');
},
};
export const GoBack: Story = {
async play({ canvasElement }) {
const canvas = within(canvasElement);
const backBtn = await canvas.findByText('Go back');
await userEvent.click(backBtn);
// 👇 Assert that your component called back()
await expect(getRouter().back).toHaveBeenCalled();
},
};
@storybook/nextjs/router.mock
类型: typeof import('next/router') & getRouter: () => ReturnType<typeof import('next/router')['useRouter']>
此模块导出了 next/router
模块导出的模拟实现。它还导出了一个 getRouter
函数,该函数返回 Next.js 的 router
对象(来自 useRouter
)的模拟版本,允许操作和断言属性。您可以使用它来模拟实现或在 story 的 play 函数中断言模拟调用。
import type { Meta, StoryObj } from '@storybook/react';
import { expect, fireEvent, userEvent, within } from '@storybook/test';
// 👇 Must include the `.mock` portion of filename to have mocks typed correctly
import { getRouter } from '@storybook/nextjs/router.mock';
import MyForm from './my-form';
const meta: Meta<typeof MyForm> = {
component: MyForm,
};
export default meta;
type Story = StoryObj<typeof MyForm>;
export const GoBack: Story = {
async play({ canvasElement }) {
const canvas = within(canvasElement);
const backBtn = await canvas.findByText('Go back');
await userEvent.click(backBtn);
// 👇 Assert that your component called back()
await expect(getRouter().back).toHaveBeenCalled();
},
};
选项
您可以根据需要传递选项对象以进行其他配置
// .storybook/main.js
import * as path from 'path';
export default {
// ...
framework: {
name: '@storybook/nextjs',
options: {
image: {
loading: 'eager',
},
nextConfigPath: path.resolve(__dirname, '../next.config.js'),
},
},
};
可用选项如下
builder
类型: Record<string, any>
配置 框架构建器的选项。对于 Next.js,可在 Webpack 构建器文档中找到可用选项。
image
类型: object
传递给 next/image
的每个实例的 Props。有关更多详细信息,请参阅 next/image 文档。
nextConfigPath
类型: string
next.config.js
文件的绝对路径。如果您有自定义的 next.config.js
文件,且该文件不在项目的根目录中,则这是必需的。
参数
此框架在 nextjs
命名空间下,为 Storybook 贡献了以下参数
appDirectory
类型: boolean
默认值: false
如果您的 story 导入了使用 next/navigation
的组件,则需要将参数 nextjs.appDirectory
设置为 true
。由于这是一个参数,您可以将其应用于单个 story,组件的所有 story,或Storybook 中的每个 story。有关更多详细信息,请参阅 Next.js Navigation。
navigation
类型
{
asPath?: string;
pathname?: string;
query?: Record<string, string>;
segments?: (string | [string, string])[];
}
默认值
{
segments: [];
}
传递给 next/navigation
上下文的 router 对象。有关更多详细信息,请参阅 Next.js 的 navigation 文档。
router
类型
{
asPath?: string;
pathname?: string;
query?: Record<string, string>;
}
传递给 next/router
上下文的 router 对象。有关更多详细信息,请参阅 Next.js 的 router 文档。