From 1a45bcb9806443a94e0ff3ca563e1bc9f118ee9f Mon Sep 17 00:00:00 2001 From: jonniebigodes Date: Fri, 8 Oct 2021 19:32:55 +0100 Subject: [PATCH] more polishing and snippets adjustments --- ...my-component-play-function-waitfor.mdx.mdx | 39 +++++++++++ .../my-component-play-function-waitfor.ts.mdx | 35 ++++++++++ ...ent-play-function-with-selectevent.mdx.mdx | 16 +++-- ...nent-play-function-with-selectevent.ts.mdx | 15 ++-- ...ister-component-with-play-function.mdx.mdx | 25 +++++-- ...gister-component-with-play-function.ts.mdx | 25 +++++-- .../my-component-play-function-waitfor.js.mdx | 34 +++++++++ ...my-component-play-function-waitfor.mdx.mdx | 35 ++++++++++ ...-component-play-function-with-delay.js.mdx | 11 ++- ...component-play-function-with-delay.mdx.mdx | 9 ++- ...nent-play-function-with-selectevent.js.mdx | 16 +++-- ...ent-play-function-with-selectevent.mdx.mdx | 17 ++--- ...gister-component-with-play-function.js.mdx | 22 +++++- ...ister-component-with-play-function.mdx.mdx | 21 +++++- .../svelte/button-story-using-args.js.mdx | 3 + .../my-component-play-function-waitfor.js.mdx | 38 ++++++++++ ...my-component-play-function-waitfor.mdx.mdx | 36 ++++++++++ ...nent-play-function-with-selectevent.js.mdx | 15 ++-- ...ent-play-function-with-selectevent.mdx.mdx | 24 ++++--- ...gister-component-with-play-function.js.mdx | 23 ++++++- ...ister-component-with-play-function.mdx.mdx | 23 ++++++- .../vue/button-story-using-args.2.js.mdx | 2 + .../vue/button-story-using-args.3.js.mdx | 2 + .../my-component-play-function-waitfor.js.mdx | 39 +++++++++++ ...my-component-play-function-waitfor.mdx.mdx | 36 ++++++++++ ...-component-play-function-with-delay.js.mdx | 9 ++- ...component-play-function-with-delay.mdx.mdx | 9 ++- ...nent-play-function-with-selectevent.js.mdx | 17 +++-- ...ent-play-function-with-selectevent.mdx.mdx | 16 +++-- ...gister-component-with-play-function.js.mdx | 23 ++++++- ...ister-component-with-play-function.mdx.mdx | 22 +++++- docs/writing-stories/args.md | 8 +-- docs/writing-stories/decorators.md | 6 +- docs/writing-stories/introduction.md | 2 +- docs/writing-stories/play-function.md | 69 ++++++++++++------- 35 files changed, 611 insertions(+), 131 deletions(-) create mode 100644 docs/snippets/angular/my-component-play-function-waitfor.mdx.mdx create mode 100644 docs/snippets/angular/my-component-play-function-waitfor.ts.mdx create mode 100644 docs/snippets/react/my-component-play-function-waitfor.js.mdx create mode 100644 docs/snippets/react/my-component-play-function-waitfor.mdx.mdx create mode 100644 docs/snippets/svelte/my-component-play-function-waitfor.js.mdx create mode 100644 docs/snippets/svelte/my-component-play-function-waitfor.mdx.mdx create mode 100644 docs/snippets/vue/my-component-play-function-waitfor.js.mdx create mode 100644 docs/snippets/vue/my-component-play-function-waitfor.mdx.mdx diff --git a/docs/snippets/angular/my-component-play-function-waitfor.mdx.mdx b/docs/snippets/angular/my-component-play-function-waitfor.mdx.mdx new file mode 100644 index 00000000000..e470114f988 --- /dev/null +++ b/docs/snippets/angular/my-component-play-function-waitfor.mdx.mdx @@ -0,0 +1,39 @@ +```md + + +import { Meta, Story } from '@storybook/addon-docs'; + + + +import { screen, fireEvent, waitFor } from '@testing-library/react'; + +import userEvent from '@testing-library/user-event'; + +import { MyComponent } from './MyComponent.component'; + + + + + + + + { + const emailInput = screen.getByLabelText('example-element', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'WrongInput', { + delay: 100, + }); + + const Button = screen.getByRole('button'); + + await fireEvent.click(Button); + + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + }} /> +``` \ No newline at end of file diff --git a/docs/snippets/angular/my-component-play-function-waitfor.ts.mdx b/docs/snippets/angular/my-component-play-function-waitfor.ts.mdx new file mode 100644 index 00000000000..54b4cfd45f4 --- /dev/null +++ b/docs/snippets/angular/my-component-play-function-waitfor.ts.mdx @@ -0,0 +1,35 @@ +```ts +// MyComponent.stories.ts + +// These are placeholders until the addon-interaction is out +import userEvent from '@testing-library/user-event'; +import { screen, fireEvent, waitFor } from '@testing-library/angular'; + +import { MyComponent } from './MyComponent.component'; + +export default { + component: MyComponent, + //๐Ÿ‘‡ Marking the onSubmit with this configuration will log the event in the Actions panel + argTypes: { onSubmit: { action: true } }, +}; + +export const ExampleAsyncStory = { + play: async () => { + const emailInput = screen.getByLabelText('example-element', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'WrongInput', { + delay: 100, + }); + + const Button = screen.getByRole('button'); + + await fireEvent.click(Button); + + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + }, +}; +``` \ No newline at end of file diff --git a/docs/snippets/angular/my-component-play-function-with-selectevent.mdx.mdx b/docs/snippets/angular/my-component-play-function-with-selectevent.mdx.mdx index dece3b9d982..fbf430b22eb 100644 --- a/docs/snippets/angular/my-component-play-function-with-selectevent.mdx.mdx +++ b/docs/snippets/angular/my-component-play-function-with-selectevent.mdx.mdx @@ -18,18 +18,20 @@ export const sleep= (ms) => { return new Promise((resolve) => setTimeout(resolve, ms)); } + + { - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, - }); + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { + + await fireEvent.change(dropdown, { target: { value: 'Yet another item' }, }); }} /> diff --git a/docs/snippets/angular/my-component-play-function-with-selectevent.ts.mdx b/docs/snippets/angular/my-component-play-function-with-selectevent.ts.mdx index 665a910fe50..5a1e8670466 100644 --- a/docs/snippets/angular/my-component-play-function-with-selectevent.ts.mdx +++ b/docs/snippets/angular/my-component-play-function-with-selectevent.ts.mdx @@ -16,17 +16,18 @@ function sleep(ms: any) { return new Promise((resolve) => setTimeout(resolve, ms)); } +// Queries the element by it's role and fires the event export const ExampleChangeEvent = { play: async () => { - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, - }); + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { + + await fireEvent.change(dropdown, { target: { value: 'Yet another item' }, }); }, diff --git a/docs/snippets/angular/register-component-with-play-function.mdx.mdx b/docs/snippets/angular/register-component-with-play-function.mdx.mdx index 3e3e33bb2ae..6a3b3231093 100644 --- a/docs/snippets/angular/register-component-with-play-function.mdx.mdx +++ b/docs/snippets/angular/register-component-with-play-function.mdx.mdx @@ -5,7 +5,7 @@ import { Meta, Story } from '@storybook/addon-docs'; -import { screen } from '@testing-library/angular'; +import { screen, fireEvent } from '@testing-library/angular'; import userEvent from '@testing-library/user-event'; import { RegistrationForm } from './RegistrationForm.component'; @@ -17,9 +17,26 @@ import { RegistrationForm } from './RegistrationForm.component'; { - await userEvent.type(screen.getByTestId('username')); - await userEvent.type(screen.getByTestId('password')); - await userEvent.click(screen.getByTestId('submit')); + + const emailInput = screen.getByLabelText('email', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: 'input', + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const submitButton = screen.getByRole('button'); + + await fireEvent.click(submitButton); }} /> ``` \ No newline at end of file diff --git a/docs/snippets/angular/register-component-with-play-function.ts.mdx b/docs/snippets/angular/register-component-with-play-function.ts.mdx index a3a91fa7b08..f26915516af 100644 --- a/docs/snippets/angular/register-component-with-play-function.ts.mdx +++ b/docs/snippets/angular/register-component-with-play-function.ts.mdx @@ -3,7 +3,7 @@ // These are placeholders until the addon-interaction is out import userEvent from '@testing-library/user-event'; -import { screen } from '@testing-library/angular'; +import { screen, fireEvent } from '@testing-library/angular'; import { RegistrationForm } from './RegistrationForm.component'; @@ -15,9 +15,26 @@ export default { export const FilledForm = { play: async () => { - await userEvent.type(screen.getByTestId('username')); - await userEvent.type(screen.getByTestId('password')); - await userEvent.click(screen.getByTestId('submit')); + + const emailInput = screen.getByLabelText('email', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: 'input', + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const submitButton = screen.getByRole('button'); + + await fireEvent.click(submitButton); }, }; ``` \ No newline at end of file diff --git a/docs/snippets/react/my-component-play-function-waitfor.js.mdx b/docs/snippets/react/my-component-play-function-waitfor.js.mdx new file mode 100644 index 00000000000..cfb8c269bf2 --- /dev/null +++ b/docs/snippets/react/my-component-play-function-waitfor.js.mdx @@ -0,0 +1,34 @@ +```js +// MyComponent.stories.js | MyComponent.stories.jsx | MyComponent.stories.ts | MyComponent.stories.tsx + +// These are placeholders until the addon-interaction is out +import { screen, waitFor } from '@testing-library/react'; + +import userEvent from '@testing-library/user-event'; + +import { MyComponent } from './MyComponent.js'; + +export default { + component: MyComponent, + //๐Ÿ‘‡ Marking the onSubmit with this configuration will log the event in the Actions panel + argTypes: { onSubmit: { action: true } }, +}; + +export const ExampleAsyncStory = { + play: async () => { + const Input = screen.getByLabelText('example-element', { + selector: 'input', + }); + + await userEvent.type(Input, 'WrongInput', { + delay: 100, + }); + const Button = screen.getByRole('button'); + await userEvent.click(Submit); + + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + }, +}; +``` \ No newline at end of file diff --git a/docs/snippets/react/my-component-play-function-waitfor.mdx.mdx b/docs/snippets/react/my-component-play-function-waitfor.mdx.mdx new file mode 100644 index 00000000000..29a18eb58d0 --- /dev/null +++ b/docs/snippets/react/my-component-play-function-waitfor.mdx.mdx @@ -0,0 +1,35 @@ +```md + + +import { Meta, Story } from '@storybook/addon-docs'; + + + +import { screen, waitFor } from '@testing-library/react'; + +import userEvent from '@testing-library/user-event'; + +import { MyComponent } from './MyComponent.js'; + + + + + + + + { + const Input = screen.getByLabelText('example-element'); + + await userEvent.type(Input, 'WrongInput', { + delay: 100, + }); + + const Button = screen.getByRole('button'); + await userEvent.click(Submit); + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + }} /> +``` \ No newline at end of file diff --git a/docs/snippets/react/my-component-play-function-with-delay.js.mdx b/docs/snippets/react/my-component-play-function-with-delay.js.mdx index 1c474ca8d43..dbb3d0ee045 100644 --- a/docs/snippets/react/my-component-play-function-with-delay.js.mdx +++ b/docs/snippets/react/my-component-play-function-with-delay.js.mdx @@ -14,11 +14,16 @@ export default { export const DelayedStory = { play: async () => { - // The delay option sets the amount of milliseconds between characters being typed - await userEvent.type(screen.getByTestId('example-element'), 'random string', { + + const exampleElement= screen.getByLabelText('example-element'); + + // The delay option set the ammount of milliseconds between characters being typed + await userEvent.type(exampleElement, 'random string', { delay: 100, }); - await userEvent.type(screen.getByTestId('another-example-element'), 'another random string', { + + const AnotherExampleElement= screen.getByLabelText('another-example-element'); + await userEvent.type(AnotherExampleElement, 'another random string', { delay: 100, }); }, diff --git a/docs/snippets/react/my-component-play-function-with-delay.mdx.mdx b/docs/snippets/react/my-component-play-function-with-delay.mdx.mdx index 8e9a25775b7..5b6afda7e04 100644 --- a/docs/snippets/react/my-component-play-function-with-delay.mdx.mdx +++ b/docs/snippets/react/my-component-play-function-with-delay.mdx.mdx @@ -18,10 +18,15 @@ import { MyComponent } from './MyComponent.js'; { - await userEvent.type(screen.getByTestId('example-element'), 'random string', { + const exampleElement= screen.getByLabelText('example-element'); + + await userEvent.type(exampleElement, 'random string', { delay: 100, }); - await userEvent.type(screen.getByTestId('another-example-element'), 'another random string', { + + const AnotherExampleElement= screen.getByLabelText('another-example-element'); + + await userEvent.type(AnotherExampleElement, 'another random string', { delay: 100, }); }} /> diff --git a/docs/snippets/react/my-component-play-function-with-selectevent.js.mdx b/docs/snippets/react/my-component-play-function-with-selectevent.js.mdx index f2ba9308811..67f07fcb141 100644 --- a/docs/snippets/react/my-component-play-function-with-selectevent.js.mdx +++ b/docs/snippets/react/my-component-play-function-with-selectevent.js.mdx @@ -15,17 +15,19 @@ function sleep(ms) { return new Promise((resolve) => setTimeout(resolve, ms)); } + // Queries the element by it's role and fires the event + export const ExampleChangeEvent = { play: async () => { - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, - }); + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { + + await fireEvent.change(dropdown, { target: { value: 'Yet another item' }, }); }, diff --git a/docs/snippets/react/my-component-play-function-with-selectevent.mdx.mdx b/docs/snippets/react/my-component-play-function-with-selectevent.mdx.mdx index ef0ef375542..0ea8c7a7e43 100644 --- a/docs/snippets/react/my-component-play-function-with-selectevent.mdx.mdx +++ b/docs/snippets/react/my-component-play-function-with-selectevent.mdx.mdx @@ -15,19 +15,20 @@ export const sleep = (ms) => { return new Promise((resolve) => setTimeout(resolve, ms)); } + { - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, - }); + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Yet another item' }, + + await fireEvent.change(dropdown, { + target: { value: 'Yet another item' }, }); }} /> ``` \ No newline at end of file diff --git a/docs/snippets/react/register-component-with-play-function.js.mdx b/docs/snippets/react/register-component-with-play-function.js.mdx index a80ef0549fe..153e1859c4d 100644 --- a/docs/snippets/react/register-component-with-play-function.js.mdx +++ b/docs/snippets/react/register-component-with-play-function.js.mdx @@ -16,9 +16,25 @@ export default { export const FilledForm = { play: async () => { - await userEvent.type(screen.getByTestId('username')); - await userEvent.type(screen.getByTestId('password')); - await userEvent.click(screen.getByTestId('submit')); + + const emailInput = screen.getByLabelText('email', { + selector: "input", + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: "input", + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const Submit = screen.getByRole('button'); + await userEvent.click(Submit); }, }; ``` \ No newline at end of file diff --git a/docs/snippets/react/register-component-with-play-function.mdx.mdx b/docs/snippets/react/register-component-with-play-function.mdx.mdx index bbec13506d1..a15c81e77e8 100644 --- a/docs/snippets/react/register-component-with-play-function.mdx.mdx +++ b/docs/snippets/react/register-component-with-play-function.mdx.mdx @@ -17,8 +17,23 @@ import { RegistrationForm } from './RegistrationForm.js'; { - await userEvent.type(screen.getByTestId('username')); - await userEvent.type(screen.getByTestId('password')); - await userEvent.click(screen.getByTestId('submit')); + const emailInput = screen.getByLabelText('email', { + selector: "input", + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: "input", + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const Submit = screen.getByRole('button'); + await userEvent.click(Submit); }} /> ``` \ No newline at end of file diff --git a/docs/snippets/svelte/button-story-using-args.js.mdx b/docs/snippets/svelte/button-story-using-args.js.mdx index b3602f01ba1..48a2b48a302 100644 --- a/docs/snippets/svelte/button-story-using-args.js.mdx +++ b/docs/snippets/svelte/button-story-using-args.js.mdx @@ -7,6 +7,9 @@ export default { component: Button, }; + +//๐Ÿ‘‡ The render function is a framework specific construct to tell how the story should render + export const Primary = { args: { backgroundColor: '#ff0', diff --git a/docs/snippets/svelte/my-component-play-function-waitfor.js.mdx b/docs/snippets/svelte/my-component-play-function-waitfor.js.mdx new file mode 100644 index 00000000000..e2005e76e14 --- /dev/null +++ b/docs/snippets/svelte/my-component-play-function-waitfor.js.mdx @@ -0,0 +1,38 @@ +```js +// MyComponent.stories.js + +// These are placeholders until the addon-interaction is out +import userEvent from '@testing-library/user-event'; +import { screen, fireEvent, waitFor } from '@testing-library/svelte'; + +import MyComponent from './MyComponent.svelte'; + +export default { + component: MyComponent, + //๐Ÿ‘‡ Marking the onSubmit with this configuration will log the event in the Actions panel + argTypes: { onSubmit: { action: true } }, +}; + +export const ExampleAsyncStory = { + play: async () => { + const exampleElement = screen.getByLabelText('example-element', { + selector: 'input', + }); + + // The delay option set the ammount of milliseconds between characters being typed + await userEvent.type(exampleElement, 'WrongInput', { + delay: 100, + }); + + const Button = screen.getByRole('button'); + await fireEvent.click(Button); + + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + }, + render: () => ({ + component: MyComponent, + }), +}; +``` \ No newline at end of file diff --git a/docs/snippets/svelte/my-component-play-function-waitfor.mdx.mdx b/docs/snippets/svelte/my-component-play-function-waitfor.mdx.mdx new file mode 100644 index 00000000000..9ffa29b3299 --- /dev/null +++ b/docs/snippets/svelte/my-component-play-function-waitfor.mdx.mdx @@ -0,0 +1,36 @@ +```md + + +import { Meta, Story } from '@storybook/addon-docs'; + + + +import userEvent from '@testing-library/user-event'; +import { screen, fireEvent, waitFor } from "@testing-library/svelte"; + +import MyComponent from './MyComponent.svelte'; + + + + + + { + const Input = screen.getByLabelText('example-element'); + await userEvent.type(Input, 'WrongInput', { + delay: 100, + }); + + const Button = screen.getByRole('button'); + + await fireEvent.click(Button); + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + + }} + render={() => ({ + component: MyComponent, + })} /> +``` \ No newline at end of file diff --git a/docs/snippets/svelte/my-component-play-function-with-selectevent.js.mdx b/docs/snippets/svelte/my-component-play-function-with-selectevent.js.mdx index ef14a920ca9..8577793de92 100644 --- a/docs/snippets/svelte/my-component-play-function-with-selectevent.js.mdx +++ b/docs/snippets/svelte/my-component-play-function-with-selectevent.js.mdx @@ -15,17 +15,18 @@ function sleep(ms) { return new Promise((resolve) => setTimeout(resolve, ms)); } + // Queries the element by it's role and fires the event export const ExampleChangeEvent = { play: async () => { - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, - }); + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { + + await fireEvent.change(dropdown, { target: { value: 'Yet another item' }, }); }, diff --git a/docs/snippets/svelte/my-component-play-function-with-selectevent.mdx.mdx b/docs/snippets/svelte/my-component-play-function-with-selectevent.mdx.mdx index c05d5cab1ac..f1e89c72984 100644 --- a/docs/snippets/svelte/my-component-play-function-with-selectevent.mdx.mdx +++ b/docs/snippets/svelte/my-component-play-function-with-selectevent.mdx.mdx @@ -17,22 +17,24 @@ export const sleep(ms) => { return new Promise((resolve) => setTimeout(resolve, ms)); } + + { - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + play={async () => { + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); + await sleep(2000); + + await fireEvent.change(dropdown, { + target: { value: 'Yet another item' }, }); - await sleep(2000); - await fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Yet another item' }, - }); }} - render={()=>({ + render={() => ({ Component: MyComponent, })} /> ``` \ No newline at end of file diff --git a/docs/snippets/svelte/register-component-with-play-function.js.mdx b/docs/snippets/svelte/register-component-with-play-function.js.mdx index 87aca931ba2..93a39a788cf 100644 --- a/docs/snippets/svelte/register-component-with-play-function.js.mdx +++ b/docs/snippets/svelte/register-component-with-play-function.js.mdx @@ -15,9 +15,26 @@ export default { export const FilledForm = { play: async () => { - await userEvent.type(screen.getByTestId('username'), 'username'); - await userEvent.type(screen.getByTestId('password'), 'password'); - await userEvent.click(screen.getByTestId('submit')); + + const emailInput = screen.getByLabelText('email', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: 'input', + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const submitButton = screen.getByRole('button'); + + await fireEvent.click(submitButton); }, render: () => ({ Component: MyComponent, diff --git a/docs/snippets/svelte/register-component-with-play-function.mdx.mdx b/docs/snippets/svelte/register-component-with-play-function.mdx.mdx index c926cce2905..fd35658a30c 100644 --- a/docs/snippets/svelte/register-component-with-play-function.mdx.mdx +++ b/docs/snippets/svelte/register-component-with-play-function.mdx.mdx @@ -17,9 +17,26 @@ import RegistrationForm from './RegistrationForm.svelte'; { - await userEvent.type(screen.getByTestId('username'), 'username'); - await userEvent.type(screen.getByTestId('password'), 'password'); - await userEvent.click(screen.getByTestId('submit')); + + const emailInput = screen.getByLabelText('email', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: 'input', + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const submitButton = screen.getByRole('button'); + + await fireEvent.click(submitButton); }} render={() => ({ Component: RegistrationForm, diff --git a/docs/snippets/vue/button-story-using-args.2.js.mdx b/docs/snippets/vue/button-story-using-args.2.js.mdx index 0f7890c5f0a..17f2bf622c9 100644 --- a/docs/snippets/vue/button-story-using-args.2.js.mdx +++ b/docs/snippets/vue/button-story-using-args.2.js.mdx @@ -7,6 +7,8 @@ export default { component: Button, }; +//๐Ÿ‘‡ The render function is a framework specific construct to tell how the story should render + export const Primary = { args: { backgroundColor: '#ff0', diff --git a/docs/snippets/vue/button-story-using-args.3.js.mdx b/docs/snippets/vue/button-story-using-args.3.js.mdx index cff83ef8907..20290019044 100644 --- a/docs/snippets/vue/button-story-using-args.3.js.mdx +++ b/docs/snippets/vue/button-story-using-args.3.js.mdx @@ -7,6 +7,8 @@ export default { component: Button, }; +//๐Ÿ‘‡ The render function is a framework specific construct to tell how the story should render + export const Primary = { args: { backgroundColor: '#ff0', diff --git a/docs/snippets/vue/my-component-play-function-waitfor.js.mdx b/docs/snippets/vue/my-component-play-function-waitfor.js.mdx new file mode 100644 index 00000000000..8915e2357c6 --- /dev/null +++ b/docs/snippets/vue/my-component-play-function-waitfor.js.mdx @@ -0,0 +1,39 @@ +```js +// MyComponent.stories.js + +// These are placeholders until the addon-interaction is out +import userEvent from '@testing-library/user-event'; +import { screen, fireEvent, waitFor } from '@testing-library/vue'; + +import MyComponent from './MyComponent.vue'; + +export default { + component: MyComponent, + //๐Ÿ‘‡ Marking the onSubmit with this configuration will log the event in the Actions panel + argTypes: { onSubmit: { action: true } }, +}; + +export const ExampleAsyncStory = { + play: async () => { + const exampleElement = screen.getByLabelText('example-element', { + selector: 'input', + }); + + // The delay option set the ammount of milliseconds between characters being typed + await userEvent.type(exampleElement, 'WrongInput', { + delay: 100, + }); + + const Button = screen.getByRole('button'); + await fireEvent.click(Button); + + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + }, + render: () => ({ + components: { MyComponent }, + template: '', + }), +}; +``` \ No newline at end of file diff --git a/docs/snippets/vue/my-component-play-function-waitfor.mdx.mdx b/docs/snippets/vue/my-component-play-function-waitfor.mdx.mdx new file mode 100644 index 00000000000..967b55a752d --- /dev/null +++ b/docs/snippets/vue/my-component-play-function-waitfor.mdx.mdx @@ -0,0 +1,36 @@ +```md + + +import { Meta, Story } from '@storybook/addon-docs'; + + + +import userEvent from '@testing-library/user-event'; +import { screen, fireEvent, waitFor } from '@testing-library/vue'; + +import MyComponent from './MyComponent.vue'; + + + + + + { + const Input = screen.getByLabelText('example-element'); + await userEvent.type(Input, 'WrongInput', { + delay: 100, + }); + + const Button = screen.getByRole('button'); + + await fireEvent.click(Button); + await waitFor(async () => { + await userEvent.hover(screen.getByTestId('error')); + }); + }} + render={() => ({ + components: { MyComponent }, + template: '', + })} /> +``` \ No newline at end of file diff --git a/docs/snippets/vue/my-component-play-function-with-delay.js.mdx b/docs/snippets/vue/my-component-play-function-with-delay.js.mdx index f5f11f590bf..2fb4c1f284f 100644 --- a/docs/snippets/vue/my-component-play-function-with-delay.js.mdx +++ b/docs/snippets/vue/my-component-play-function-with-delay.js.mdx @@ -13,11 +13,16 @@ export default { export const DelayedStory = { play: async () => { + + const exampleElement= screen.getByLabelText('example-element'); + // The delay option set the ammount of milliseconds between characters being typed - await userEvent.type(screen.getByTestId('example-element'), 'random string', { + await userEvent.type(exampleElement, 'random string', { delay: 100, }); - await userEvent.type(screen.getByTestId('another-example-element'), 'another random string', { + + const AnotherExampleElement= screen.getByLabelText('another-example-element'); + await userEvent.type(AnotherExampleElement, 'another random string', { delay: 100, }); }, diff --git a/docs/snippets/vue/my-component-play-function-with-delay.mdx.mdx b/docs/snippets/vue/my-component-play-function-with-delay.mdx.mdx index 2f4c3b3ae26..d300371e498 100644 --- a/docs/snippets/vue/my-component-play-function-with-delay.mdx.mdx +++ b/docs/snippets/vue/my-component-play-function-with-delay.mdx.mdx @@ -17,10 +17,15 @@ import MyComponent from './MyComponent.vue'; { - await userEvent.type(screen.getByTestId('example-element'), 'random string', { + const exampleElement= screen.getByLabelText('example-element'); + + await userEvent.type(exampleElement, 'random string', { delay: 100, }); - await userEvent.type(screen.getByTestId('another-example-element'), 'another random string', { + + const AnotherExampleElement= screen.getByLabelText('another-example-element'); + + await userEvent.type(AnotherExampleElement, 'another random string', { delay: 100, }); }} diff --git a/docs/snippets/vue/my-component-play-function-with-selectevent.js.mdx b/docs/snippets/vue/my-component-play-function-with-selectevent.js.mdx index 3df9baa38aa..16e2bb4f555 100644 --- a/docs/snippets/vue/my-component-play-function-with-selectevent.js.mdx +++ b/docs/snippets/vue/my-component-play-function-with-selectevent.js.mdx @@ -12,17 +12,20 @@ export default { component: MyComponent, }; + +// Queries the element by it's role and fires the event + export const ExampleChangeEvent = { play: async () => { - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, - }); + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { + + await fireEvent.change(dropdown, { target: { value: 'Yet another item' }, }); }, diff --git a/docs/snippets/vue/my-component-play-function-with-selectevent.mdx.mdx b/docs/snippets/vue/my-component-play-function-with-selectevent.mdx.mdx index 0931a9fae05..53ff3c55578 100644 --- a/docs/snippets/vue/my-component-play-function-with-selectevent.mdx.mdx +++ b/docs/snippets/vue/my-component-play-function-with-selectevent.mdx.mdx @@ -18,18 +18,20 @@ export const sleep= (ms) => { return new Promise((resolve) => setTimeout(resolve, ms)); } + + { - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'One Item' }, - }); + const dropdown = screen.getByRole('listbox'); + + await fireEvent.change(dropdown, { target: { value: 'One Item'} }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { - target: { value: 'Another Item' }, - }); + + await fireEvent.change(dropdown, { target: { value: 'Another Item' } }); await sleep(2000); - fireEvent.change(screen.getByTestId('select'), { + + await fireEvent.change(dropdown, { target: { value: 'Yet another item' }, }); }} diff --git a/docs/snippets/vue/register-component-with-play-function.js.mdx b/docs/snippets/vue/register-component-with-play-function.js.mdx index c42825a1e0d..5b700f93504 100644 --- a/docs/snippets/vue/register-component-with-play-function.js.mdx +++ b/docs/snippets/vue/register-component-with-play-function.js.mdx @@ -15,9 +15,26 @@ export default { export const FilledForm = { play: async () => { - await userEvent.type(screen.getByTestId('username')); - await userEvent.type(screen.getByTestId('password')); - await userEvent.click(screen.getByTestId('submit')); + const emailInput = screen.getByLabelText('email', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: 'input', + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const submitButton = screen.getByRole('button'); + + await fireEvent.click(submitButton); + }, render: () => ({ components: { RegistrationForm }, diff --git a/docs/snippets/vue/register-component-with-play-function.mdx.mdx b/docs/snippets/vue/register-component-with-play-function.mdx.mdx index 9c7af735c45..29abbdd76c5 100644 --- a/docs/snippets/vue/register-component-with-play-function.mdx.mdx +++ b/docs/snippets/vue/register-component-with-play-function.mdx.mdx @@ -17,9 +17,25 @@ import RegistrationForm from './RegistrationForm.vue'; { - await userEvent.type(screen.getByTestId('username')); - await userEvent.type(screen.getByTestId('password')); - await userEvent.click(screen.getByTestId(submit')); + const emailInput = screen.getByLabelText('email', { + selector: 'input', + }); + + await userEvent.type(emailInput, 'example-email@email.com', { + delay: 100, + }); + + const passwordInput = screen.getByLabelText('password', { + selector: 'input', + }); + + await userEvent.type(passwordInput, 'ExamplePassword', { + delay: 100, + }); + + const submitButton = screen.getByRole('button'); + + await fireEvent.click(submitButton); }} render={() => ({ components: { RegistrationForm }, diff --git a/docs/writing-stories/args.md b/docs/writing-stories/args.md index a8bef31b87b..380c3223093 100644 --- a/docs/writing-stories/args.md +++ b/docs/writing-stories/args.md @@ -54,7 +54,7 @@ In the above example, we use the [object spread](https://developer.mozilla.org/e ## Component args -You can also define args at the component level; they will apply to all the component's stories unless you overwrite them. To do so, use the `arg` key on the `default` CSF export: +You can also define args at the component level; they will apply to all the component's stories unless you overwrite them. To do so, use the `args` key on the `default` CSF export: @@ -93,9 +93,7 @@ Here's how you can combine args for multiple stories of the same component.
- -๐Ÿ’กNote: If you find yourself applying this pattern often, you should consider using [component-level args](#component-args). - +๐Ÿ’กNote: If you find yourself re-using the same args for most of a component's stories, you should consider using [component-level args](#component-args).
Args are useful when writing stories for composite components that are assembled from other components. Composite components often pass their arguments unchanged to their child components, and similarly, their stories can be compositions of their child components stories. With args, you can directly compose the arguments: @@ -146,7 +144,7 @@ You can also override the set of initial args for the active story by adding an ?path=/story/avatar--default&args=style:rounded;size:100 ``` -As a safety guard against [XSS](https://owasp.org/www-community/attacks/xss/) attacks, the arg's keys and values provided in the URL are limited to alphanumeric characters, spaces, underscores, and dashes. Any other types will be ignored and removed from the URL, but you can still use them with the Controls addon and code. +As a safeguard against [XSS](https://owasp.org/www-community/attacks/xss/) attacks, the arg's keys and values provided in the URL are limited to alphanumeric characters, spaces, underscores, and dashes. Any other types will be ignored and removed from the URL, but you can still use them with the Controls addon and [within your story](#mapping-to-complex-arg-values). The `args` param is always a set of `key: value` pairs delimited with a semicolon `;`. Values will be coerced (cast) to their respective `argTypes` (which may have been automatically inferred). Objects and arrays are supported. Special values `null` and `undefined` can be set by prefixing with a bang `!`. For example, `args=obj.key:val;arr[0]:one;arr[1]:two;nil:!null` will be interpreted as: diff --git a/docs/writing-stories/decorators.md b/docs/writing-stories/decorators.md index 9dceb3306b8..767ee5163a5 100644 --- a/docs/writing-stories/decorators.md +++ b/docs/writing-stories/decorators.md @@ -38,9 +38,9 @@ Some components require a โ€œharnessโ€ to render in a useful way. For instance, ## โ€œContextโ€ for mocking -Framework-specific libraries (e.g., React's Styled Components, Vue's Fortawesome) may require additional configuration to render correctly in Storybook. +Framework-specific libraries (e.g., [Styled Components](https://styled-components.com/), [Fortawesome](https://github.com/FortAwesome/vue-fontawesome) for Vue) may require additional configuration to render correctly in Storybook. -For example, if you're working with React's Styled components and your components use themes, add a single global decorator to [`.storybook/preview.js`](../configure/overview.md#configure-story-rendering) to enable them. Or with Vue, extend Storybook's application and register your library: +For example, if you're working with Styled Components and your components use a theme, add a single global decorator to [`.storybook/preview.js`](../configure/overview.md#configure-story-rendering) to provide it. Or with Vue, extend Storybook's application and register your library: @@ -59,7 +59,7 @@ For example, if you're working with React's Styled components and your component -In the example above, the values provided are hardcoded. Still, you may want to vary them, either per-story basis (i.e., if the values you're adding are relevant to a specific story) or in a user-controlled way (e.g., provide a theme switcher or a different set of icons). +In the example above, the values provided are hardcoded. Still, you may want to vary them, either on a per-story basis (i.e., if the values you're providing are relevant to a specific story) or in a user-controlled way (e.g., provide a theme switcher or a different set of icons). The second argument to a decorator function is the **story context** which in particular contains the keys: diff --git a/docs/writing-stories/introduction.md b/docs/writing-stories/introduction.md index 899ac8e78a9..36aea9372e2 100644 --- a/docs/writing-stories/introduction.md +++ b/docs/writing-stories/introduction.md @@ -222,7 +222,7 @@ This parameter would instruct the backgrounds addon to reconfigure itself whenev ### Using decorators -Decorators are a mechanism to wrap a component in arbitrary markup when rendering a story. Components are often created with assumptions about โ€˜whereโ€™ they render. Your styles might expect a theme or layout wrapper. Or your UI might expect specific context or data providers. +Decorators are a mechanism to wrap a component in arbitrary markup when rendering a story. Components are often created with assumptions about โ€˜whereโ€™ they render. Your styles might expect a theme or layout wrapper, or your UI might expect specific context or data providers. A simple example is adding padding to a componentโ€™s stories. Accomplish this using a decorator that wraps the stories in a `div` with padding, like so: diff --git a/docs/writing-stories/play-function.md b/docs/writing-stories/play-function.md index 18fce25a257..f2737050f58 100644 --- a/docs/writing-stories/play-function.md +++ b/docs/writing-stories/play-function.md @@ -6,7 +6,7 @@ title: 'Play function' ## Setup the interactions addon -We recommend installing Storybook's `addon-interactions` before you start writing stories with the `play` function. It's the perfect complement for it, including a handy set of UI controls to allow you command over the execution flow. At any time, you can pause, resume, rewind, and step through each interaction. Also providing you with an easy-to-use debugger for potential issues. +We recommend installing Storybook's [`addon-interactions`](/addons/@storybook/addon-interactions/) before you start writing stories with the `play` function. It's the perfect complement for it, including a handy set of UI controls to allow you command over the execution flow. At any time, you can pause, resume, rewind, and step through each interaction. Also providing you with an easy-to-use debugger for potential issues. Run the following command to install the addon. @@ -78,34 +78,11 @@ Thanks to the [Component Story Format](../api/csf.md), an ES6 module based file By combining the stories, you're recreating the entire component workflow and can spot potential issues while reducing the boilerplate code you need to write. -## Delaying interactions - -Assuming that you're working with a component with validation logic implemented (e.g., email validation, password strength). In that case, you can introduce delays within your `play` function to emulate user interaction and assert if the values provided are valid or not. For example: - - - - - - - -When Storybook loads the story, it interacts with the component, filling in its inputs and triggering any validation logic defined. - ## Working with events Most modern UIs are built focusing on interaction (e.g., button clicks, selecting options, ticking checkboxes), providing rich experiences to the end-user. With the `play` function, you can incorporate the same level of interaction into your stories. -The most common scenario for component interaction is a button click. If you need to reproduce it in your story, you can adjust your story's `play` function to the following: +The most common scenario for component interaction is a button click. If you need to reproduce it in your story, you can define your story's `play` function to the following: @@ -145,6 +122,46 @@ Asides from click events, you can also script additional events with the `play` +In addition to events, you can also create interactions with the `play` function based on other types of asynchronous methods. For instance, let's assume that you're working with a component with validation logic implemented (e.g., email validation, password strength). In that case, you can introduce delays within your `play` function to emulate user interaction and assert if the values provided are valid or not: + + + + + + + +When Storybook loads the story, it interacts with the component, filling in its inputs and triggering any validation logic defined. + +You can also use the `play` function to verify the existence of an element based on a specific interaction. For instance, if you're working on a component and want to check what happens if a user introduces the wrong information. In that case, you could write the following story: + + + + + + + ## Querying elements If you need, you can also adjust your `play` function to find elements based on queries (e.g., role, text content). For example: @@ -187,4 +204,4 @@ With each interaction you write inside your `play` function when Storybook execu -Applying these changes into your stories, you'll get a performance boost and improved error handling with the `addon-interactions`. +Applying these changes to your stories can provide a performance boost and improved error handling with [`addon-interactions`](/addons/@storybook/addon-interactions/). \ No newline at end of file