Description
Describe the problem
It's not very clear how to use decorators/the render function in Svelte correctly.
For example, the return value of a decorator function is a Story Result - but it's not quite clear how to use that in Svelte. In React, it's just some JSX, but in Svelte it's an object with 4 undocumented properties - Component
and props
(self-explaining), but also on
and decorator
(no idea what they do).
It is mentioned that you can choose what component to render and with what props:
return {
Component: SomeComponent,
props: { prop1: 1, prop2: 2 }
}
But nothing more than that. The section about custom render functions linked in a couple of pages is missing (choose Svelte and click on "Custom render function" in the table of contents on the right). I tried returning some component in the decorator
property, which I realised exists because of TypeScript autocomplete, but it doesn't seem to do anything:
return {
Component: SomeComponent,
props: { prop1: 1, prop2: 2 },
decorator: MyWrapper,
}
I would expect this to render this:
<MyWrapper>
<SomeComponent {...props} />
</MyWrapper>
But the wrapper component is never actually instantiated. It also seems like you can actually use React hooks in decorators, even when using Svelte. By doing some more digging, this seems to work:
// preview.ts
import type { Preview } from "@storybook/svelte"
import { global } from "@storybook/global"
import { useEffect } from "@storybook/preview-api"
const preview: Preview = {
decorators: [
(Story) => {
useEffect(() => {
console.log("In effect", global.document.getElementById("storybook-root"))
}, [])
return Story()
}
]
}
But this isn't really documented anywhere, The only time this is mentioned is in the "Writing addons" sections. But even there, global
is not documented. Both are extremely useful for decorators.
Finally, in this PR it is stated you can set Svelte contexts, which is not stated anywhere in the docs. This is incredibly useful, as the only decorator example given in the docs is () => MarginDecorator
, and it doesn't explain how one would set the margin size, for example, based on global arguments. This is actually possible to do (as described in the PR above) like so:
import { setContext } from 'svelte'
const decorator = () => {
setContext('marginDecoratorSize', 12)
return MarginDecorator
}
The MarginDecorator
component can then do getContext('marginDecoratorSize')
.
Additional context
No response
Metadata
Metadata
Assignees
Type
Projects
Status