mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 08:01:20 +08:00
306 lines
7.5 KiB
Markdown
306 lines
7.5 KiB
Markdown
---
|
||
title: 'Addons API'
|
||
---
|
||
|
||
## Core Addon API
|
||
|
||
This is the core addon API. This is how to get the addon API:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-imports.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
### addons.getChannel()
|
||
|
||
Get an instance to the channel where you can communicate with the manager and the preview. You can find this in both the addon register code and in your addon’s wrapper component (where used inside a story).
|
||
|
||
It has a NodeJS [EventEmitter](https://nodejs.org/api/events.html) compatible API. So, you can use it to emit events and listen for events.
|
||
|
||
### addons.register()
|
||
|
||
This method allows you to register an addon and get the storybook API. You can do this only in the Manager App.
|
||
See how we can use this:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-register.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
Now you'll get an instance to our StorybookAPI. See the [api docs](#storybook-api) for Storybook API regarding using that.
|
||
|
||
### addons.add()
|
||
|
||
This method allows you to add a panel to Storybook. (Storybook's Action Logger is a panel). You can do this only in the Manager App.
|
||
See how you can use this method:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addon-panel-initial.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
The render function is called with `active` and `key`.
|
||
|
||
When the panel is in focus in the UI, the `active` will be true.
|
||
|
||
As you can see, you can set any React Component as the panel. Currently, it's one line of text. But you can do anything you want. You should specify the panel title. It could be a plain text.
|
||
|
||
## makeDecorator API
|
||
|
||
The `makeDecorator` API can be used to create decorators in the style of the official addons. Use it like so:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-makedecorator.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
The options to `makeDecorator` are:
|
||
|
||
- `name`: The name of the export (e.g. `withFoo`)
|
||
- `parameterName`: The name of the parameter your addon uses. This should be unique.
|
||
- `skipIfNoParametersOrOptions`: Don't run your decorator if the user hasn't set options (via `.addDecorator(withFoo(options)))`) or parameters (`.add('story', () => <Story/>, { foo: 'param' })`, or `.addParameters({ foo: 'param' })`).
|
||
- `allowDeprecatedUsage`: support the deprecated "wrapper" usage (`.add('story', () => withFoo(options)(() => <Story/>))`).
|
||
- `wrapper`: your decorator function. Takes the `storyFn`, `context`, and both the `options` and `parameters` (as defined in `skipIfNoParametersOrOptions` above).
|
||
|
||
<div class="aside">
|
||
|
||
Note if the parameters to a story include `{ foo: { disable: true } }` (where `foo` is the `parameterName` of your addon), your decorator will not be called.
|
||
|
||
</div>
|
||
|
||
---
|
||
|
||
## Storybook hooks
|
||
|
||
Writing addons can be simplified a lot by using these Storybook hooks:
|
||
|
||
### useStorybookState
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-usestorybookstate.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
Allows full access to the entire storybook state.
|
||
Your component will re-render whenever the storybook state changes.
|
||
|
||
If you use this, remember your component wil be re-rendered a lot, and you may need to optimize for that using [`React.memo`](https://reactjs.org/docs/react-api.html#reactmemo) or [`useMemo`](https://reactjs.org/docs/hooks-reference.html#usememo) or [`PureComponent`](https://reactjs.org/docs/react-api.html#reactpurecomponent).
|
||
|
||
### useStorybookApi
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-useapi.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
Allows full access to the storybook API.
|
||
|
||
Detail on the storybook api are further down.
|
||
|
||
### useChannel
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-usechannel.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
Allows for both setting subscriptions to events and getting the emitter for emitting custom event unto the channel.
|
||
|
||
The messages can be listened for on both the iframe and the manager side.
|
||
|
||
### useAddonState
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-useaddonstate.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
Extremely useful for addons that need to persist some state.
|
||
|
||
Storybook may unmount your addon component, and so keeping local state, might not work really well.
|
||
|
||
Also some addons consist of multiple parts, some part being in a panel, some in the toolbar etc.
|
||
|
||
With this hook they can all get access to the same bit of state which is persisted even if the components are unmounted.
|
||
|
||
### useParameter
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-useparameter.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
This hook gets you the current story's parameter.
|
||
|
||
If the parameter isn't set, the default value (second argument) is returned instead.
|
||
|
||
---
|
||
|
||
## Storybook API
|
||
|
||
Storybook API allows you to access different functionalities of Storybook UI. You can move an instance to the Storybook API when you register an addon.
|
||
|
||
Let's have a look at API methods.
|
||
|
||
### api.selectStory()
|
||
|
||
With this method, you can select a story via an API. This method accepts two parameters.
|
||
|
||
1. story kind name
|
||
2. story name (optional)
|
||
|
||
Let's say you've got a story like this:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'react/button-story-with-addon-example.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
This is how you can select the above story:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-selectstory.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
### api.selectInCurrentKind()
|
||
|
||
Same as `selectStory`, but accepts a story inside current kind as the only parameter:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-selectincurrentkind.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
### api.setQueryParams()
|
||
|
||
This method allows you to set query string parameters. You can use that as temporary storage for addons. Here's how you set query params.
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-setqueryparams.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
<div class="aside">
|
||
|
||
If you need to remove a query param, use `null` for that. For an example, let's say we need to remove bbc query param. This is how we do it:
|
||
|
||
</div>
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-disablequeryparams.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
### api.getQueryParam()
|
||
|
||
This method allows you to get a query param set by above API `setQueryParams`. For example, let's say we need to get the bbc query param. Then this how we do it:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-getqueryparam.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
### api.getUrlState(overrideParams)
|
||
|
||
This method allows you to get application url state with some changed params. For example, if you want to get a link to a particular story:
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-geturlstate.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|
||
|
||
### api.on(eventName, fn)
|
||
|
||
This method allows you to register a handler function which will be called whenever the user navigates between stories.
|
||
|
||
<!-- prettier-ignore-start -->
|
||
|
||
<CodeSnippets
|
||
paths={[
|
||
'common/storybook-addons-api-on.js.mdx',
|
||
]}
|
||
/>
|
||
|
||
<!-- prettier-ignore-end -->
|