mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 13:01:07 +08:00
295 lines
8.0 KiB
TypeScript
295 lines
8.0 KiB
TypeScript
import global from 'global';
|
|
import React, { Fragment, useEffect } from 'react';
|
|
import isChromatic from 'chromatic/isChromatic';
|
|
import {
|
|
Global,
|
|
ThemeProvider,
|
|
themes,
|
|
createReset,
|
|
convert,
|
|
styled,
|
|
useTheme,
|
|
} from '@storybook/theming';
|
|
import { Symbols } from '@storybook/components';
|
|
import type { PreviewWeb } from '@storybook/preview-web';
|
|
import { DocsContext } from '@storybook/preview-web';
|
|
import type { ReactFramework } from '@storybook/react';
|
|
import type { Channel } from '@storybook/channels';
|
|
|
|
import { DocsContainer } from '../blocks/src/blocks/DocsContainer';
|
|
|
|
const { document } = global;
|
|
|
|
const ThemeBlock = styled.div(
|
|
{
|
|
position: 'absolute',
|
|
top: 0,
|
|
left: 0,
|
|
right: '50vw',
|
|
width: '50vw',
|
|
height: '100vh',
|
|
bottom: 0,
|
|
overflow: 'auto',
|
|
padding: 10,
|
|
},
|
|
({ theme }) => ({
|
|
background: theme.background.content,
|
|
color: theme.color.defaultText,
|
|
}),
|
|
({ side }) =>
|
|
side === 'left'
|
|
? {
|
|
left: 0,
|
|
right: '50vw',
|
|
}
|
|
: {
|
|
right: 0,
|
|
left: '50vw',
|
|
}
|
|
);
|
|
|
|
const ThemeStack = styled.div(
|
|
{
|
|
position: 'relative',
|
|
minHeight: 'calc(50vh - 15px)',
|
|
},
|
|
({ theme }) => ({
|
|
background: theme.background.content,
|
|
color: theme.color.defaultText,
|
|
})
|
|
);
|
|
|
|
const PlayFnNotice = styled.div(
|
|
{
|
|
position: 'absolute',
|
|
bottom: '1rem',
|
|
right: '1rem',
|
|
border: '1px solid #ccc',
|
|
borderRadius: '5px',
|
|
padding: '1rem',
|
|
fontSize: '12px',
|
|
'> *': {
|
|
display: 'block',
|
|
},
|
|
},
|
|
({ theme }) => ({
|
|
background: theme.background.content,
|
|
color: theme.color.defaultText,
|
|
})
|
|
);
|
|
|
|
const ThemedSetRoot = () => {
|
|
const theme = useTheme();
|
|
|
|
useEffect(() => {
|
|
document.body.style.background = theme.background.content;
|
|
document.body.style.color = theme.color.defaultText;
|
|
return () => {
|
|
//
|
|
};
|
|
});
|
|
|
|
return null;
|
|
};
|
|
|
|
// eslint-disable-next-line no-underscore-dangle
|
|
const preview = (window as any).__STORYBOOK_PREVIEW__ as PreviewWeb<ReactFramework>;
|
|
const channel = (window as any).__STORYBOOK_ADDONS_CHANNEL__ as Channel;
|
|
export const loaders = [
|
|
async () => ({ globalValue: 1 }),
|
|
|
|
async ({ parameters: { relativeCsfPaths } }) => {
|
|
if (!relativeCsfPaths) return {};
|
|
|
|
const csfFiles = await Promise.all(
|
|
(relativeCsfPaths as string[]).map(async (relativePath) => {
|
|
const webpackPath = `./ui/blocks/src/${relativePath.replace(/^..\//, '')}.tsx`;
|
|
const entry = preview.storyStore.storyIndex!.importPathToEntry(webpackPath);
|
|
|
|
if (!entry) {
|
|
throw new Error(`Couldn't find story file at ${webpackPath} (passed as ${relativePath})`);
|
|
}
|
|
|
|
return preview.storyStore.loadCSFFileByStoryId(entry.id);
|
|
})
|
|
);
|
|
|
|
return {
|
|
docsContext: new DocsContext(
|
|
channel,
|
|
preview.storyStore,
|
|
preview.renderStoryToElement.bind(preview),
|
|
csfFiles,
|
|
false
|
|
),
|
|
};
|
|
},
|
|
];
|
|
|
|
export const decorators = [
|
|
(Story, { loaded: { docsContext } }) =>
|
|
docsContext ? (
|
|
<DocsContainer context={docsContext}>
|
|
<Story />
|
|
</DocsContainer>
|
|
) : (
|
|
<Story />
|
|
),
|
|
(StoryFn, { globals, parameters, playFunction }) => {
|
|
const defaultTheme = isChromatic() && !playFunction ? 'stacked' : 'light';
|
|
const theme = globals.theme || parameters.theme || defaultTheme;
|
|
|
|
switch (theme) {
|
|
case 'side-by-side': {
|
|
return (
|
|
<Fragment>
|
|
<Symbols icons={['folder', 'component', 'document', 'bookmarkhollow']} />
|
|
<ThemeProvider theme={convert(themes.light)}>
|
|
<Global styles={createReset} />
|
|
</ThemeProvider>
|
|
<ThemeProvider theme={convert(themes.light)}>
|
|
<ThemeBlock side="left" data-side="left">
|
|
<StoryFn />
|
|
</ThemeBlock>
|
|
</ThemeProvider>
|
|
<ThemeProvider theme={convert(themes.dark)}>
|
|
<ThemeBlock side="right" data-side="right">
|
|
<StoryFn />
|
|
</ThemeBlock>
|
|
</ThemeProvider>
|
|
</Fragment>
|
|
);
|
|
}
|
|
case 'stacked': {
|
|
return (
|
|
<Fragment>
|
|
<Symbols icons={['folder', 'component', 'document', 'bookmarkhollow']} />
|
|
<ThemeProvider theme={convert(themes.light)}>
|
|
<Global styles={createReset} />
|
|
</ThemeProvider>
|
|
<ThemeProvider theme={convert(themes.light)}>
|
|
<ThemeStack side="left" data-side="left">
|
|
<StoryFn />
|
|
</ThemeStack>
|
|
</ThemeProvider>
|
|
<ThemeProvider theme={convert(themes.dark)}>
|
|
<ThemeStack side="right" data-side="right">
|
|
<StoryFn />
|
|
</ThemeStack>
|
|
</ThemeProvider>
|
|
</Fragment>
|
|
);
|
|
}
|
|
default: {
|
|
return (
|
|
<ThemeProvider theme={convert(themes[theme])}>
|
|
<Symbols icons={['folder', 'component', 'document', 'bookmarkhollow']} />
|
|
<Global styles={createReset} />
|
|
<ThemedSetRoot />
|
|
{!parameters.theme && isChromatic() && playFunction && (
|
|
<PlayFnNotice>
|
|
<span>Detected play function.</span>
|
|
<span>Rendering in a single theme</span>
|
|
</PlayFnNotice>
|
|
)}
|
|
<StoryFn />
|
|
</ThemeProvider>
|
|
);
|
|
}
|
|
}
|
|
},
|
|
];
|
|
|
|
export const parameters = {
|
|
exportedParameter: 'exportedParameter',
|
|
actions: { argTypesRegex: '^on.*' },
|
|
options: {
|
|
storySort: (a, b) =>
|
|
a.title === b.title ? 0 : a.id.localeCompare(b.id, undefined, { numeric: true }),
|
|
},
|
|
docs: {
|
|
theme: themes.light,
|
|
},
|
|
controls: {
|
|
presetColors: [
|
|
{ color: '#ff4785', title: 'Coral' },
|
|
{ color: '#1EA7FD', title: 'Ocean' },
|
|
{ color: 'rgb(252, 82, 31)', title: 'Orange' },
|
|
{ color: 'RGBA(255, 174, 0, 0.5)', title: 'Gold' },
|
|
{ color: 'hsl(101, 52%, 49%)', title: 'Green' },
|
|
{ color: 'HSLA(179,65%,53%,0.5)', title: 'Seafoam' },
|
|
{ color: '#6F2CAC', title: 'Purple' },
|
|
{ color: '#2A0481', title: 'Ultraviolet' },
|
|
{ color: 'black' },
|
|
{ color: '#333', title: 'Darkest' },
|
|
{ color: '#444', title: 'Darker' },
|
|
{ color: '#666', title: 'Dark' },
|
|
{ color: '#999', title: 'Mediumdark' },
|
|
{ color: '#ddd', title: 'Medium' },
|
|
{ color: '#EEE', title: 'Mediumlight' },
|
|
{ color: '#F3F3F3', title: 'Light' },
|
|
{ color: '#F8F8F8', title: 'Lighter' },
|
|
{ color: '#FFFFFF', title: 'Lightest' },
|
|
'#fe4a49',
|
|
'#FED766',
|
|
'rgba(0, 159, 183, 1)',
|
|
'HSLA(240,11%,91%,0.5)',
|
|
'slategray',
|
|
],
|
|
},
|
|
};
|
|
|
|
export const globals = {
|
|
foo: 'fooValue',
|
|
};
|
|
|
|
export const globalTypes = {
|
|
foo: { defaultValue: 'fooDefaultValue' },
|
|
bar: { defaultValue: 'barDefaultValue' },
|
|
theme: {
|
|
name: 'Theme',
|
|
description: 'Global theme for components',
|
|
toolbar: {
|
|
icon: 'circlehollow',
|
|
title: 'Theme',
|
|
items: [
|
|
{ value: 'light', icon: 'circlehollow', title: 'light' },
|
|
{ value: 'dark', icon: 'circle', title: 'dark' },
|
|
{ value: 'side-by-side', icon: 'sidebar', title: 'side by side' },
|
|
{ value: 'stacked', icon: 'bottombar', title: 'stacked' },
|
|
],
|
|
},
|
|
},
|
|
locale: {
|
|
name: 'Locale',
|
|
description: 'Internationalization locale',
|
|
toolbar: {
|
|
icon: 'globe',
|
|
shortcuts: {
|
|
next: {
|
|
label: 'Go to next language',
|
|
keys: ['L'],
|
|
},
|
|
previous: {
|
|
label: 'Go to previous language',
|
|
keys: ['K'],
|
|
},
|
|
reset: {
|
|
label: 'Reset language',
|
|
keys: ['meta', 'shift', 'L'],
|
|
},
|
|
},
|
|
items: [
|
|
{ title: 'Reset locale', type: 'reset' },
|
|
{ value: 'en', right: '🇺🇸', title: 'English' },
|
|
{ value: 'es', right: '🇪🇸', title: 'Español' },
|
|
{ value: 'zh', right: '🇨🇳', title: '中文' },
|
|
{ value: 'kr', right: '🇰🇷', title: '한국어' },
|
|
],
|
|
},
|
|
},
|
|
};
|
|
|
|
export const argTypes = { color: { control: 'color' } };
|
|
export const args = { color: 'red' };
|