-
-
Notifications
You must be signed in to change notification settings - Fork 9.9k
Expand file tree
/
Copy pathpreview.tsx
More file actions
114 lines (104 loc) · 3.64 KB
/
preview.tsx
File metadata and controls
114 lines (104 loc) · 3.64 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
import type { ComponentType } from 'react';
import { definePreview as definePreviewBase } from 'storybook/internal/csf';
import type { AddonTypes, InferTypes, Meta, Preview, Story } from 'storybook/internal/csf';
import type { PreviewAddon } from 'storybook/internal/csf';
import type {
Args,
ArgsStoryFn,
ComponentAnnotations,
DecoratorFunction,
ProjectAnnotations,
Renderer,
StoryAnnotations,
} from 'storybook/internal/types';
import type { RemoveIndexSignature, SetOptional, Simplify, UnionToIntersection } from 'type-fest';
import * as reactAnnotations from './entry-preview';
import * as reactArgTypesAnnotations from './entry-preview-argtypes';
import * as reactDocsAnnotations from './entry-preview-docs';
import type { AddMocks } from './public-types';
import type { ReactRenderer } from './types';
export function __definePreview<Addons extends PreviewAddon<never>[]>(
input: ProjectAnnotations<ReactRenderer> & { addons: Addons }
): ReactPreview<InferTypes<Addons>> {
const preview = definePreviewBase({
...input,
addons: [
reactAnnotations,
reactArgTypesAnnotations,
reactDocsAnnotations,
...(input.addons ?? []),
],
}) as unknown as ReactPreview<InferTypes<Addons>>;
const defineMeta = preview.meta.bind(preview);
preview.meta = (_input) => {
const meta = defineMeta(_input);
const defineStory = meta.story.bind(meta);
meta.story = (__input: any) => {
const story = defineStory(__input);
story.Component = story.__compose();
return story;
};
return meta;
};
return preview;
}
// @ts-expect-error We cannot implement the meta faithfully here, but that is okay.
export interface ReactPreview<T extends AddonTypes> extends Preview<ReactRenderer & T> {
meta<
TArgs extends Args,
Decorators extends DecoratorFunction<ReactRenderer & T, any>,
// Try to make Exact<Partial<TArgs>, TMetaArgs> work
TMetaArgs extends Partial<TArgs>,
>(
meta: {
render?: ArgsStoryFn<ReactRenderer & T, TArgs>;
component?: ComponentType<TArgs>;
decorators?: Decorators | Decorators[];
args?: TMetaArgs;
} & Omit<
ComponentAnnotations<ReactRenderer & T, TArgs>,
'decorators' | 'component' | 'args' | 'render'
>
): ReactMeta<
ReactRenderer &
T & {
args: Simplify<
TArgs & Simplify<RemoveIndexSignature<DecoratorsArgs<ReactRenderer & T, Decorators>>>
>;
},
{ args: Partial<TArgs> extends TMetaArgs ? {} : TMetaArgs }
>;
}
type DecoratorsArgs<TRenderer extends Renderer, Decorators> = UnionToIntersection<
Decorators extends DecoratorFunction<TRenderer, infer TArgs> ? TArgs : unknown
>;
interface ReactMeta<T extends ReactRenderer, MetaInput extends ComponentAnnotations<T>>
extends Meta<T> {
// Required args don't need to be provided when the user uses an empty render
story<
TInput extends
| (() => ReactRenderer['storyResult'])
| (StoryAnnotations<T, T['args']> & {
render: () => ReactRenderer['storyResult'];
}),
>(
story?: TInput
): ReactStory<T, TInput extends () => ReactRenderer['storyResult'] ? { render: TInput } : TInput>;
story<
TInput extends Simplify<
StoryAnnotations<
T,
// TODO: infer mocks from story itself as well
AddMocks<T['args'], MetaInput['args']>,
SetOptional<T['args'], keyof T['args'] & keyof MetaInput['args']>
>
>,
>(
story?: TInput
// @ts-expect-error hard
): ReactStory<T, TInput>;
}
export interface ReactStory<T extends ReactRenderer, TInput extends StoryAnnotations<T, T['args']>>
extends Story<T, TInput> {
Component: ComponentType<Partial<T['args']>>;
}