Play 函数
观看视频教程
Play
函数是在 story 渲染后执行的小代码片段。使您能够与组件交互并测试原本需要用户干预的场景。
设置 interactions 插件
我们建议您在开始使用 play
函数编写 story 之前,安装 Storybook 的 addon-interactions
。它是它的完美补充,包括一组方便的 UI 控件,让您可以控制执行流程。在任何时候,您都可以暂停、恢复、倒带和单步执行每个交互。还为您提供了一个易于使用的调试器来解决潜在问题。
运行以下命令来安装插件和所需的依赖项。
npm install @storybook/test @storybook/addon-interactions --save-dev
更新您的 Storybook 配置 (在 .storybook/main.js|ts
中) 以包含 interactions 插件。
// 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)'],
addons: [
// Other Storybook addons
'@storybook/addon-interactions', // 👈 Register the addon
],
};
export default config;
使用 play 函数编写 Stories
Storybook 的 play
函数是在 story 完成渲染后运行的小代码片段。在 addon-interactions
的帮助下,它允许您构建组件交互和测试场景,而这些场景在没有用户干预的情况下是不可能实现的。例如,如果您正在开发一个注册表单并想要验证它,您可以编写以下带有 play
函数的 story
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, within } from '@storybook/test';
import { RegistrationForm } from './RegistrationForm';
const meta: Meta<typeof RegistrationForm> = {
component: RegistrationForm,
};
export default meta;
type Story = StoryObj<typeof RegistrationForm>;
/*
* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const FilledForm: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
const emailInput = canvas.getByLabelText('email', {
selector: 'input',
});
await userEvent.type(emailInput, 'example-email@email.com', {
delay: 100,
});
const passwordInput = canvas.getByLabelText('password', {
selector: 'input',
});
await userEvent.type(passwordInput, 'ExamplePassword', {
delay: 100,
});
// See https://storybook.org.cn/docs/essentials/actions#automatically-matching-args to learn how to setup logging in the Actions panel
const submitButton = canvas.getByRole('button');
await userEvent.click(submitButton);
},
};
有关可用 API 事件的概述,请参阅组件测试文档。
当 Storybook 完成 story 的渲染时,它会执行在 play
函数中定义的步骤,与组件交互并填写表单的信息。所有这些都无需用户干预。如果您查看 Interactions
面板,您将看到逐步流程。
组合 Stories
得益于组件 Story Format,这是一种基于 ES6 模块的文件格式,您还可以组合您的 play
函数,类似于其他现有的 Storybook 功能(例如,args)。例如,如果您想验证组件的特定工作流程,您可以编写以下 stories
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
/*
* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const FirstStory: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
await userEvent.type(canvas.getByTestId('an-element'), 'example-value');
},
};
export const SecondStory: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
await userEvent.type(canvas.getByTestId('other-element'), 'another value');
},
};
export const CombinedStories: Story = {
play: async ({ context, canvasElement }) => {
const canvas = within(canvasElement);
// Runs the FirstStory and Second story play function before running this story's play function
await FirstStory.play(context);
await SecondStory.play(context);
await userEvent.type(canvas.getByTestId('another-element'), 'random value');
},
};
通过组合 stories,您正在重新创建整个组件工作流程,并且可以发现潜在问题,同时减少需要编写的样板代码。
处理事件
大多数现代 UI 的构建都侧重于交互(例如,单击按钮、选择选项、勾选复选框),为最终用户提供丰富的体验。借助 play
函数,您可以将相同级别的交互融入到您的 stories 中。
组件交互的常见类型是按钮单击。如果您需要在 story 中重现它,您可以将 story 的 play
函数定义如下
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { fireEvent, userEvent, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
/* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const ClickExample: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
// See https://storybook.org.cn/docs/essentials/actions#automatically-matching-args to learn how to setup logging in the Actions panel
await userEvent.click(canvas.getByRole('button'));
},
};
export const FireEventExample: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
// See https://storybook.org.cn/docs/essentials/actions#automatically-matching-args to learn how to setup logging in the Actions panel
await fireEvent.click(canvas.getByTestId('data-testid'));
},
};
当 Storybook 加载 story 并执行该函数时,它会与组件交互并触发按钮单击,类似于用户会执行的操作。
除了单击事件之外,您还可以使用 play
函数编写其他事件脚本。例如,如果您的组件包含带有各种选项的选择器,您可以编写以下 story 并测试每个场景
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
// Function to emulate pausing between interactions
function sleep(ms: number) {
return new Promise((resolve) => setTimeout(resolve, ms));
}
/* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const ExampleChangeEvent: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
const select = canvas.getByRole('listbox');
await userEvent.selectOptions(select, ['One Item']);
await sleep(2000);
await userEvent.selectOptions(select, ['Another Item']);
await sleep(2000);
await userEvent.selectOptions(select, ['Yet another item']);
},
};
除了事件之外,您还可以基于其他类型的异步方法,使用 play
函数创建交互。例如,假设您正在处理一个实现了验证逻辑的组件(例如,电子邮件验证、密码强度)。在这种情况下,您可以在 play
函数中引入延迟,以模拟用户交互并断言提供的值是否有效
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
/* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const DelayedStory: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
const exampleElement = canvas.getByLabelText('example-element');
// The delay option sets the amount of milliseconds between characters being typed
await userEvent.type(exampleElement, 'random string', {
delay: 100,
});
const AnotherExampleElement = canvas.getByLabelText('another-example-element');
await userEvent.type(AnotherExampleElement, 'another random string', {
delay: 100,
});
},
};
当 Storybook 加载 story 时,它会与组件交互,填写其输入并触发任何定义的验证逻辑。
您还可以使用 play
函数来验证基于特定交互的元素是否存在。例如,如果您正在开发一个组件并想检查用户引入错误信息时会发生什么。在这种情况下,您可以编写以下 story
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, waitFor, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
/* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const ExampleAsyncStory: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
const Input = canvas.getByLabelText('Username', {
selector: 'input',
});
await userEvent.type(Input, 'WrongInput', {
delay: 100,
});
// See https://storybook.org.cn/docs/essentials/actions#automatically-matching-args to learn how to setup logging in the Actions panel
const Submit = canvas.getByRole('button');
await userEvent.click(Submit);
await waitFor(async () => {
await userEvent.hover(canvas.getByTestId('error'));
});
},
};
查询元素
如果需要,您还可以调整 play
函数,以根据查询(例如,角色、文本内容)查找元素。例如
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
/* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const ExampleWithRole: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
// See https://storybook.org.cn/docs/essentials/actions#automatically-matching-args to learn how to setup logging in the Actions panel
await userEvent.click(canvas.getByRole('button', { name: / button label/i }));
},
};
您可以在 Testing Library 文档中阅读有关查询元素的更多信息。
当 Storybook 加载 story 时,play
函数开始执行并查询 DOM 树,期望在 story 渲染时元素可用。如果您的测试失败,您将能够快速验证其根本原因。
否则,如果组件不是立即可用,例如,由于在 play
函数中定义的先前步骤或某些异步行为,您可以调整您的 story 并等待 DOM 树发生更改,然后再查询元素。例如
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
/* See https://storybook.org.cn/docs/writing-stories/play-function#working-with-the-canvas
* to learn more about using the canvasElement to query the DOM
*/
export const AsyncExample: Story = {
play: async ({ canvasElement }) => {
const canvas = within(canvasElement);
// Other steps
// Waits for the component to be rendered before querying the element
await canvas.findByRole('button', { name: / button label/i });
},
};
使用 Canvas
默认情况下,您在 play
函数中编写的每个交互都将从 Canvas 的顶层元素开始执行。这对于较小的组件(例如,按钮、复选框、文本输入)是可以接受的,但对于复杂的组件(例如,表单、页面)或多个 stories 来说可能效率低下。为了适应这种情况,您可以调整交互以从组件的根目录开始执行。例如
// Replace your-framework with the name of your framework
import type { Meta, StoryObj } from '@storybook/your-framework';
import { userEvent, within } from '@storybook/test';
import { MyComponent } from './MyComponent';
const meta: Meta<typeof MyComponent> = {
component: MyComponent,
};
export default meta;
type Story = StoryObj<typeof MyComponent>;
export const ExampleStory: Story = {
play: async ({ canvasElement }) => {
// Assigns canvas to the component root element
const canvas = within(canvasElement);
// Starts querying from the component's root element
await userEvent.type(canvas.getByTestId('example-element'), 'something');
await userEvent.click(canvas.getByRole('button'));
},
};
将这些更改应用到您的 stories 可以提高性能,并使用 addon-interactions
改进错误处理。