Skip to content

[Documentation]: Svelte - Better documentation for decorators/render function #26896

Open
@rChaoz

Description

@rChaoz

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

No type

Projects

Status

No status

Milestone

No milestone

Relationships

None yet

Development

No branches or pull requests

Issue actions