Merge branch 'next' into deprecate/add-decorator-add-api

This commit is contained in:
Norbert de Langen 2022-10-14 16:16:16 +02:00
commit 414020c0ea
No known key found for this signature in database
GPG Key ID: FD0E78AF9A837762
122 changed files with 480 additions and 141 deletions

View File

@ -122,6 +122,7 @@ jobs:
- code/addons - code/addons
- code/frameworks - code/frameworks
- code/lib - code/lib
- code/ui
- code/renderers - code/renderers
- code/presets - code/presets
chromatic: chromatic:

View File

@ -1,3 +1,4 @@
{ {
"deepscan.enable": true "deepscan.enable": true,
"typescript.tsdk": "./code/node_modules/typescript/lib"
} }

View File

@ -26,13 +26,19 @@ module.exports = {
files: [ files: [
'**/lib/theming/**/*', '**/lib/theming/**/*',
'**/lib/router/**/*', '**/lib/router/**/*',
'**/lib/ui/**/*', '**/ui/manager/**/*',
'**/lib/components/**/*', '**/lib/components/**/*',
], ],
rules: { rules: {
'import/no-extraneous-dependencies': ['error', { bundledDependencies: false }], 'import/no-extraneous-dependencies': ['error', { bundledDependencies: false }],
}, },
}, },
{
files: ['**/ui/*', '**/ui/.storybook/*'],
rules: {
'import/no-extraneous-dependencies': ['error', { packageDir: __dirname }],
},
},
{ {
files: [ files: [
'**/__tests__/**', '**/__tests__/**',

View File

@ -1,7 +1,7 @@
const craTemplates = { const craTemplates = {
'cra/default-js': { 'cra/default-js': {
name: 'Create React App (Javascript)', name: 'Create React App (Javascript)',
script: 'npx create-react-app {{beforeDir}}', script: 'npx create-react-app .',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/cra', framework: '@storybook/cra',
@ -11,7 +11,7 @@ const craTemplates = {
}, },
'cra/default-ts': { 'cra/default-ts': {
name: 'Create React App (Typescript)', name: 'Create React App (Typescript)',
script: 'npx create-react-app {{beforeDir}} --template typescript', script: 'npx create-react-app . --template typescript',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
// Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed. // Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed.
skipTasks: ['smoke-test'], skipTasks: ['smoke-test'],
@ -26,7 +26,7 @@ const craTemplates = {
const reactViteTemplates = { const reactViteTemplates = {
'react-vite/default-js': { 'react-vite/default-js': {
name: 'React Vite (JS)', name: 'React Vite (JS)',
script: 'yarn create vite {{beforeDir}} --template react', script: 'yarn create vite . --template react',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/react-vite', framework: '@storybook/react-vite',
@ -36,7 +36,7 @@ const reactViteTemplates = {
}, },
'react-vite/default-ts': { 'react-vite/default-ts': {
name: 'React Vite (TS)', name: 'React Vite (TS)',
script: 'yarn create vite {{beforeDir}} --template react-ts', script: 'yarn create vite . --template react-ts',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/react-vite', framework: '@storybook/react-vite',
@ -49,7 +49,7 @@ const reactViteTemplates = {
const reactWebpackTemplates = { const reactWebpackTemplates = {
'react-webpack/18-ts': { 'react-webpack/18-ts': {
name: 'React Webpack5 (TS)', name: 'React Webpack5 (TS)',
script: 'yarn create webpack5-react {{beforeDir}}', script: 'yarn create webpack5-react .',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/react-webpack5', framework: '@storybook/react-webpack5',
@ -59,8 +59,7 @@ const reactWebpackTemplates = {
}, },
'react-webpack/17-ts': { 'react-webpack/17-ts': {
name: 'React Webpack5 (TS)', name: 'React Webpack5 (TS)',
script: script: 'yarn create webpack5-react . --version-react="17" --version-react-dom="17"',
'yarn create webpack5-react {{beforeDir}} --version-react="17" --version-react-dom="17"',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/react-webpack5', framework: '@storybook/react-webpack5',
@ -73,7 +72,7 @@ const reactWebpackTemplates = {
const vue3ViteTemplates = { const vue3ViteTemplates = {
'vue3-vite/default-js': { 'vue3-vite/default-js': {
name: 'Vue3 Vite (JS)', name: 'Vue3 Vite (JS)',
script: 'yarn create vite {{beforeDir}} --template vue', script: 'yarn create vite . --template vue',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/vue3-vite', framework: '@storybook/vue3-vite',
@ -83,7 +82,7 @@ const vue3ViteTemplates = {
}, },
'vue3-vite/default-ts': { 'vue3-vite/default-ts': {
name: 'Vue3 Vite (TS)', name: 'Vue3 Vite (TS)',
script: 'yarn create vite {{beforeDir}} --template vue-ts', script: 'yarn create vite . --template vue-ts',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/vue3-vite', framework: '@storybook/vue3-vite',
@ -100,7 +99,7 @@ const vue2ViteTemplates = {
// We don't really want to maintain weird custom scripts like this, // We don't really want to maintain weird custom scripts like this,
// preferring community bootstrap scripts / generators instead. // preferring community bootstrap scripts / generators instead.
script: script:
'yarn create vite {{beforeDir}} --template vanilla && yarn add --dev @vitejs/plugin-vue2 vue-template-compiler vue@2 && echo "import vue2 from \'@vitejs/plugin-vue2\';\n\nexport default {\n\tplugins: [vue2()]\n};" > vite.config.js', 'yarn create vite . --template vanilla && yarn add --dev @vitejs/plugin-vue2 vue-template-compiler vue@2 && echo "import vue2 from \'@vitejs/plugin-vue2\';\n\nexport default {\n\tplugins: [vue2()]\n};" > vite.config.js',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
// Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed. // Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed.
skipTasks: ['smoke-test'], skipTasks: ['smoke-test'],
@ -115,7 +114,7 @@ const vue2ViteTemplates = {
const htmlWebpackTemplates = { const htmlWebpackTemplates = {
'html-webpack/default': { 'html-webpack/default': {
name: 'HTML Webpack5', name: 'HTML Webpack5',
script: 'yarn create webpack5-html {{beforeDir}}', script: 'yarn create webpack5-html .',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/html-webpack5', framework: '@storybook/html-webpack5',
@ -128,7 +127,7 @@ const htmlWebpackTemplates = {
const svelteViteTemplates = { const svelteViteTemplates = {
'svelte-vite/default-js': { 'svelte-vite/default-js': {
name: 'Svelte Vite (JS)', name: 'Svelte Vite (JS)',
script: 'yarn create vite {{beforeDir}} --template svelte', script: 'yarn create vite . --template svelte',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
expected: { expected: {
framework: '@storybook/svelte-vite', framework: '@storybook/svelte-vite',
@ -138,7 +137,7 @@ const svelteViteTemplates = {
}, },
'svelte-vite/default-ts': { 'svelte-vite/default-ts': {
name: 'Svelte Vite (TS)', name: 'Svelte Vite (TS)',
script: 'yarn create vite {{beforeDir}} --template svelte-ts', script: 'yarn create vite . --template svelte-ts',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
// Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed. // Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed.
skipTasks: ['smoke-test'], skipTasks: ['smoke-test'],
@ -205,7 +204,7 @@ const angularCliTemplates = {
const litViteTemplates = { const litViteTemplates = {
'lit-vite/default-js': { 'lit-vite/default-js': {
name: 'Lit Vite (JS)', name: 'Lit Vite (JS)',
script: 'yarn create vite {{beforeDir}} --template lit', script: 'yarn create vite . --template lit',
cadence: ['ci', 'daily', 'weekly'] as any, cadence: ['ci', 'daily', 'weekly'] as any,
// Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed. // Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed.
skipTasks: ['smoke-test'], skipTasks: ['smoke-test'],
@ -217,7 +216,7 @@ const litViteTemplates = {
}, },
'lit-vite/default-ts': { 'lit-vite/default-ts': {
name: 'Lit Vite (TS)', name: 'Lit Vite (TS)',
script: 'yarn create vite {{beforeDir}} --template lit-ts', script: 'yarn create vite . --template lit-ts',
cadence: ['ci', 'daily', 'weekly'] as any, cadence: ['ci', 'daily', 'weekly'] as any,
// Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed. // Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed.
skipTasks: ['smoke-test'], skipTasks: ['smoke-test'],
@ -232,8 +231,7 @@ const litViteTemplates = {
const vueCliTemplates = { const vueCliTemplates = {
'vue-cli/default-js': { 'vue-cli/default-js': {
name: 'Vue-CLI (Default JS)', name: 'Vue-CLI (Default JS)',
script: script: 'npx -p @vue/cli vue create . --default --packageManager=yarn --force --merge',
'npx -p @vue/cli vue create {{beforeDir}} --default --packageManager=yarn --force --merge',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
skipTasks: [ skipTasks: [
// Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed. // Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed.
@ -250,7 +248,7 @@ const vueCliTemplates = {
'vue-cli/vue2-default-js': { 'vue-cli/vue2-default-js': {
name: 'Vue-CLI (Vue2 JS)', name: 'Vue-CLI (Vue2 JS)',
script: script:
'npx -p @vue/cli vue create {{beforeDir}} --default --packageManager=yarn --force --merge --preset=Default\\ (Vue\\ 2)', 'npx -p @vue/cli vue create . --default --packageManager=yarn --force --merge --preset="Default (Vue 2)"',
cadence: ['ci', 'daily', 'weekly'], cadence: ['ci', 'daily', 'weekly'],
skipTasks: [ skipTasks: [
// Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed. // Re-enable once https://github.com/storybookjs/storybook/issues/19351 is fixed.

View File

@ -1 +0,0 @@
export * from './dist/html';

View File

@ -1 +0,0 @@
export * from './dist/html';

View File

@ -82,8 +82,7 @@
}, },
"bundler": { "bundler": {
"entries": [ "entries": [
"./src/index.ts", "./src/index.ts"
"./src/html.tsx"
], ],
"platform": "neutral" "platform": "neutral"
}, },

View File

@ -1,15 +0,0 @@
import { dedent } from 'ts-dedent';
import deprecate from 'util-deprecate';
const deprecatedHtmlEndpoint = deprecate(
() => {},
dedent`
The entry point '@storybook/components/html' is deprecated. Please use '@storybook/components' directly instead.
See https://github.com/storybookjs/storybook/blob/next/MIGRATION.md#deprecated-storybook-components-html-entry-point
`
);
deprecatedHtmlEndpoint();
export * from './typography/DocumentFormatting';
export { components, resetComponents } from './index';

View File

@ -26,7 +26,12 @@ export const getAutoRefs = async (options: Options): Promise<Record<string, Ref>
if (storybook?.url) { if (storybook?.url) {
return { id: name, ...storybook, version }; return { id: name, ...storybook, version };
} }
} catch { } catch (error) {
if ((error as any).code === 'ERR_PACKAGE_PATH_NOT_EXPORTED') {
// silent warning because user can't do anything about it
// "package.json" is not part of the package's "exports" field in its package.json
return undefined;
}
logger.warn(`unable to find package.json for ${d}`); logger.warn(`unable to find package.json for ${d}`);
return undefined; return undefined;
} }

View File

@ -5,7 +5,7 @@ import type { StorybookConfig } from '@storybook/react-webpack5';
const config: StorybookConfig = { const config: StorybookConfig = {
stories: [ stories: [
'../../lib/ui/src/**/*.stories.@(ts|tsx|js|jsx|mdx)', '../../ui/manager/src/**/*.stories.@(ts|tsx|js|jsx|mdx)',
'../../lib/components/src/**/*.stories.@(ts|tsx|js|jsx|mdx)', '../../lib/components/src/**/*.stories.@(ts|tsx|js|jsx|mdx)',
'./../../addons/docs/**/*.stories.@(ts|tsx|js|jsx|mdx)', './../../addons/docs/**/*.stories.@(ts|tsx|js|jsx|mdx)',
'./../../addons/interactions/**/*.stories.@(ts|tsx|js|jsx|mdx)', './../../addons/interactions/**/*.stories.@(ts|tsx|js|jsx|mdx)',

View File

@ -1,13 +0,0 @@
import React from 'react';
export default {
title: 'UI/Addon Panel',
};
export const AllAddons = () => <div>By default all addon panels are rendered</div>;
export const FilteredAddons = () => <div>By default all addon panels are rendered</div>;
FilteredAddons.parameters = {
a11y: { disable: true },
actions: { disable: true },
};

View File

@ -41,13 +41,14 @@
"packages": [ "packages": [
"addons/*", "addons/*",
"addons/storyshots/*", "addons/storyshots/*",
"frameworks/*",
"renderers/*",
"presets/*",
"examples-native/*",
"examples/*", "examples/*",
"examples-native/*",
"frameworks/*",
"lib/*", "lib/*",
"lib/cli/test/run/*" "lib/cli/test/run/*",
"ui/*",
"presets/*",
"renderers/*"
] ]
}, },
"scripts": { "scripts": {
@ -80,6 +81,7 @@
"sandbox": "ts-node ../scripts/sandbox.ts", "sandbox": "ts-node ../scripts/sandbox.ts",
"serve-storybooks": "http-server ./built-storybooks -p 8001", "serve-storybooks": "http-server ./built-storybooks -p 8001",
"smoketest-storybooks": "cross-env STORYBOOK_DISPLAY_WARNING=true DISPLAY_WARNING=true node -r esm ../scripts/smoketest-storybooks.js", "smoketest-storybooks": "cross-env STORYBOOK_DISPLAY_WARNING=true DISPLAY_WARNING=true node -r esm ../scripts/smoketest-storybooks.js",
"storybook:ui": "./lib/cli/bin/index.js dev --port 6006 --config-dir ./ui/.storybook --no-manager-cache",
"test": "NODE_OPTIONS=--max_old_space_size=4096 jest --config ./jest.config.js", "test": "NODE_OPTIONS=--max_old_space_size=4096 jest --config ./jest.config.js",
"test:cli": "npm --prefix lib/cli run test", "test:cli": "npm --prefix lib/cli run test",
"test:e2e-examples-playwright": "playwright test", "test:e2e-examples-playwright": "playwright test",
@ -253,6 +255,7 @@
"@typescript-eslint/eslint-plugin": "^5.15.0", "@typescript-eslint/eslint-plugin": "^5.15.0",
"@typescript-eslint/experimental-utils": "^5.20.0", "@typescript-eslint/experimental-utils": "^5.20.0",
"@typescript-eslint/parser": "^5.15.0", "@typescript-eslint/parser": "^5.15.0",
"@vitejs/plugin-react": "^2.1.0",
"babel-core": "^7.0.0-bridge.0", "babel-core": "^7.0.0-bridge.0",
"babel-eslint": "^10.1.0", "babel-eslint": "^10.1.0",
"babel-jest": "^26.6.3", "babel-jest": "^26.6.3",
@ -284,6 +287,7 @@
"fs-extra": "^9.0.1", "fs-extra": "^9.0.1",
"github-release-from-changelog": "^2.1.1", "github-release-from-changelog": "^2.1.1",
"glob": "^7.1.6", "glob": "^7.1.6",
"global": "^4.4.0",
"http-server": "^0.12.3", "http-server": "^0.12.3",
"husky": "^4.3.7", "husky": "^4.3.7",
"jest": "^26.6.3", "jest": "^26.6.3",
@ -314,6 +318,7 @@
"process": "^0.11.10", "process": "^0.11.10",
"prompts": "^2.4.0", "prompts": "^2.4.0",
"raf": "^3.4.1", "raf": "^3.4.1",
"react": "^16.8.0",
"read-pkg-up": "^7.0.1", "read-pkg-up": "^7.0.1",
"regenerator-runtime": "^0.13.7", "regenerator-runtime": "^0.13.7",
"remark": "^13.0.0", "remark": "^13.0.0",
@ -333,6 +338,7 @@
"tsup": "^6.2.2", "tsup": "^6.2.2",
"typescript": "4.7.4", "typescript": "4.7.4",
"util": "^0.12.4", "util": "^0.12.4",
"vite": "^3.1.7",
"wait-on": "^5.2.1", "wait-on": "^5.2.1",
"web-component-analyzer": "^1.1.6", "web-component-analyzer": "^1.1.6",
"webpack": "5", "webpack": "5",

View File

@ -0,0 +1,58 @@
import vitePluginReact from '@vitejs/plugin-react';
import type { PluginOption } from 'vite';
import type { StorybookConfig } from '../../frameworks/react-vite/dist';
const config: StorybookConfig = {
stories: [
'../manager/src/**/*.stories.@(ts|tsx|js|jsx|mdx)',
// '../../lib/components/src/**/*.stories.@(js|jsx|ts|tsx|mdx)',
// '../../../addons/interactions/**/*.stories.@(ts|tsx|js|jsx|mdx)',
],
addons: [
'@storybook/addon-links',
'@storybook/addon-essentials',
'@storybook/addon-interactions',
],
framework: {
name: '@storybook/react-vite',
options: {},
},
viteFinal: (config) => {
return {
...config,
optimizeDeps: {
...config.optimizeDeps,
include: [
...(config.optimizeDeps?.include ?? []),
'react-element-to-jsx-string',
'core-js/modules/es.regexp.flags.js',
'react-colorful',
],
},
/*
This might look complex but all we're doing is removing the default set of React Vite plugins
and adding them back in, but with the `jsxRuntime: 'classic'` option.
TODO: When we've upgraded to React 18 all of this shouldn't be necessary anymore
*/
plugins: [...withoutReactPlugins(config.plugins), vitePluginReact({ jsxRuntime: 'classic' })],
};
},
};
// recursively remove all plugins from the React Vite plugin
const withoutReactPlugins = (plugins: PluginOption[] = []) =>
plugins.map((plugin) => {
if (Array.isArray(plugin)) {
return withoutReactPlugins(plugin);
}
if (
plugin &&
'name' in plugin &&
['vite:react-jsx', 'vite:react-babel', 'vite:react-refresh'].includes(plugin.name)
) {
return false;
}
return plugin;
});
export default config;

View File

@ -0,0 +1,4 @@
<script>
// eslint-disable-next-line no-undef
window.global = window;
</script>

View File

@ -0,0 +1,248 @@
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';
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;
};
export const decorators = [
(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 loaders = [async () => ({ globalValue: 1 })];
export const argTypes = { color: { control: 'color' } };
export const args = { color: 'red' };

View File

@ -5,14 +5,14 @@
"keywords": [ "keywords": [
"storybook" "storybook"
], ],
"homepage": "https://github.com/storybookjs/storybook/tree/main/lib/ui", "homepage": "https://github.com/storybookjs/storybook/tree/main/code/ui/manager",
"bugs": { "bugs": {
"url": "https://github.com/storybookjs/storybook/issues" "url": "https://github.com/storybookjs/storybook/issues"
}, },
"repository": { "repository": {
"type": "git", "type": "git",
"url": "https://github.com/storybookjs/storybook.git", "url": "https://github.com/storybookjs/storybook.git",
"directory": "lib/ui" "directory": "code/ui/manager"
}, },
"funding": { "funding": {
"type": "opencollective", "type": "opencollective",

View File

@ -1,7 +1,7 @@
import React, { FC, useMemo } from 'react'; import React, { FC, useMemo } from 'react';
import sizeMe from 'react-sizeme'; import sizeMe from 'react-sizeme';
import { State } from '@storybook/api'; import { type State } from '@storybook/api';
import { Symbols } from '@storybook/components'; import { Symbols } from '@storybook/components';
import { Route } from '@storybook/router'; import { Route } from '@storybook/router';
import { Global, createGlobal, styled } from '@storybook/theming'; import { Global, createGlobal, styled } from '@storybook/theming';

View File

@ -1,8 +1,8 @@
import global from 'global'; import global from 'global';
import React, { Component, FC } from 'react'; import React, { Component, FC } from 'react';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { Collection } from '@storybook/addons'; import type { Collection } from '@storybook/addons';
import { State } from '@storybook/api'; import type { State } from '@storybook/api';
import { Sidebar, SidebarProps } from '../sidebar/Sidebar'; import { Sidebar, SidebarProps } from '../sidebar/Sidebar';
import Panel from '../panel/panel'; import Panel from '../panel/panel';
import { Preview } from '../preview/preview'; import { Preview } from '../preview/preview';

View File

@ -496,7 +496,7 @@ class Layout extends Component<LayoutProps, LayoutState> {
}; };
render() { render() {
const { children, bounds, options, theme, viewMode, panelCount } = this.props; const { children, bounds, options, viewMode, panelCount } = this.props;
const { isDragging, resizerNav, resizerPanel } = this.state; const { isDragging, resizerNav, resizerPanel } = this.state;
const margin = 0; const margin = 0;

View File

@ -15,18 +15,12 @@ export default {
decorators: [ decorators: [
((StoryFn, c) => { ((StoryFn, c) => {
const mocked = true; const mocked = true;
const height = 900;
const width = 1200;
if (isChromatic) { if (isChromatic()) {
store.local.set(`storybook-layout`, {}); store.local.set(`storybook-layout`, {});
} }
const props = { const props = mocked ? mockProps : realProps;
height,
width,
...(mocked ? mockProps : realProps),
};
return ( return (
<div style={{ minHeight: 900, minWidth: 1200 }}> <div style={{ minHeight: 900, minWidth: 1200 }}>

View File

@ -1,6 +1,6 @@
import React, { Fragment, ComponentType, FC } from 'react'; import React, { Fragment, ComponentType, FC } from 'react';
import { State } from '@storybook/api'; import type { State } from '@storybook/api';
import * as S from './container'; import * as S from './container';
export interface DesktopProps { export interface DesktopProps {

View File

@ -1,5 +1,5 @@
import React, { Component, Children, ComponentType, FC, ReactNode } from 'react'; import React, { Component, Children, ComponentType, FC, ReactNode } from 'react';
import { State, ActiveTabs } from '@storybook/api'; import { type State, ActiveTabs } from '@storybook/api';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { TabButton } from '@storybook/components'; import { TabButton } from '@storybook/components';

View File

@ -1,8 +1,8 @@
import React, { FC, SyntheticEvent } from 'react'; import React, { FC, SyntheticEvent } from 'react';
import { State } from '@storybook/api'; import { type State } from '@storybook/api';
import { Link } from '@storybook/router'; import { Link } from '@storybook/router';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { Icons, IconButton, IconsProps } from '@storybook/components'; import { Icons, IconButton, type IconsProps } from '@storybook/components';
import { transparentize } from 'polished'; import { transparentize } from 'polished';
const DEFAULT_ICON_COLOUR = '#66BF3C' as const; const DEFAULT_ICON_COLOUR = '#66BF3C' as const;

View File

@ -2,7 +2,7 @@ import React from 'react';
import { LocationProvider } from '@storybook/router'; import { LocationProvider } from '@storybook/router';
import NotificationList from './NotificationList'; import NotificationList from './NotificationList';
import itemMeta, * as itemStories from './NotificationItem.stories'; import itemMeta, * as itemStories from './NotificationItem.stories.jsx';
export default { export default {
component: NotificationList, component: NotificationList,

View File

@ -2,7 +2,7 @@ import React, { Component, Fragment, ReactElement } from 'react';
import { shortcutToHumanString } from '@storybook/api/shortcut'; import { shortcutToHumanString } from '@storybook/api/shortcut';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { Tabs, Icons, IconButton } from '@storybook/components'; import { Tabs, Icons, IconButton } from '@storybook/components';
import { State } from '@storybook/api'; import type { State } from '@storybook/api';
const DesktopOnlyIconButton = styled(IconButton)({ const DesktopOnlyIconButton = styled(IconButton)({
// Hides full screen icon at mobile breakpoint defined in app.js // Hides full screen icon at mobile breakpoint defined in app.js

View File

@ -1,5 +1,5 @@
import { types, Addon } from '@storybook/addons'; import { types, type Addon } from '@storybook/addons';
import { API, State } from '@storybook/api'; import type { API, State } from '@storybook/api';
import { PreviewProps } from './utils/types'; import { PreviewProps } from './utils/types';
export const previewProps: PreviewProps = { export const previewProps: PreviewProps = {

View File

@ -2,9 +2,9 @@
import React, { Fragment, useMemo, useEffect, useRef } from 'react'; import React, { Fragment, useMemo, useEffect, useRef } from 'react';
import { Helmet } from 'react-helmet-async'; import { Helmet } from 'react-helmet-async';
import { API, Consumer, Combo, merge } from '@storybook/api'; import { type API, Consumer, type Combo, merge } from '@storybook/api';
import { SET_CURRENT_STORY } from '@storybook/core-events'; import { SET_CURRENT_STORY } from '@storybook/core-events';
import { addons, types, Addon } from '@storybook/addons'; import { addons, types, type Addon } from '@storybook/addons';
import { Loader } from '@storybook/components'; import { Loader } from '@storybook/components';
import { Location } from '@storybook/router'; import { Location } from '@storybook/router';

View File

@ -3,11 +3,11 @@ import React, { Fragment, useMemo, FunctionComponent } from 'react';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { FlexBar, IconButton, Icons, Separator, TabButton, TabBar } from '@storybook/components'; import { FlexBar, IconButton, Icons, Separator, TabButton, TabBar } from '@storybook/components';
import { Consumer, Combo, API, State, merge, LeafEntry } from '@storybook/api'; import { Consumer, type Combo, type API, type State, merge, type LeafEntry } from '@storybook/api';
import { shortcutToHumanString } from '@storybook/api/shortcut'; import { shortcutToHumanString } from '@storybook/api/shortcut';
import { addons, Addon, types } from '@storybook/addons'; import { addons, type Addon, types } from '@storybook/addons';
import { Location, RenderData } from '@storybook/router'; import { Location, type RenderData } from '@storybook/router';
import { zoomTool } from './tools/zoom'; import { zoomTool } from './tools/zoom';
import * as S from './utils/components'; import * as S from './utils/components';

View File

@ -1,7 +1,7 @@
import React from 'react'; import React from 'react';
import { IconButton, Icons } from '@storybook/components'; import { IconButton, Icons } from '@storybook/components';
import { Consumer, Combo } from '@storybook/api'; import { Consumer, type Combo } from '@storybook/api';
import { Addon } from '@storybook/addons'; import type { Addon } from '@storybook/addons';
const menuMapper = ({ api, state }: Combo) => ({ const menuMapper = ({ api, state }: Combo) => ({
isVisible: state.layout.showPanel, isVisible: state.layout.showPanel,

View File

@ -2,8 +2,8 @@ import global from 'global';
import React from 'react'; import React from 'react';
import copy from 'copy-to-clipboard'; import copy from 'copy-to-clipboard';
import { getStoryHref, IconButton, Icons } from '@storybook/components'; import { getStoryHref, IconButton, Icons } from '@storybook/components';
import { Consumer, Combo } from '@storybook/api'; import { Consumer, type Combo } from '@storybook/api';
import { Addon } from '@storybook/addons'; import type { Addon } from '@storybook/addons';
const { PREVIEW_URL, document } = global; const { PREVIEW_URL, document } = global;

View File

@ -1,8 +1,8 @@
import global from 'global'; import global from 'global';
import React from 'react'; import React from 'react';
import { getStoryHref, IconButton, Icons } from '@storybook/components'; import { getStoryHref, IconButton, Icons } from '@storybook/components';
import { Consumer, Combo } from '@storybook/api'; import { Consumer, type Combo } from '@storybook/api';
import { Addon } from '@storybook/addons'; import type { Addon } from '@storybook/addons';
const { PREVIEW_URL } = global; const { PREVIEW_URL } = global;

View File

@ -1,7 +1,7 @@
import React from 'react'; import React from 'react';
import { IconButton, Icons, Separator } from '@storybook/components'; import { IconButton, Icons, Separator } from '@storybook/components';
import { Consumer, Combo } from '@storybook/api'; import { Consumer, type Combo } from '@storybook/api';
import { Addon } from '@storybook/addons'; import type { Addon } from '@storybook/addons';
const menuMapper = ({ api, state }: Combo) => ({ const menuMapper = ({ api, state }: Combo) => ({
isVisible: state.layout.showNav, isVisible: state.layout.showNav,

View File

@ -1,7 +1,7 @@
import React, { ComponentProps, useState } from 'react'; import React, { ComponentProps, useState } from 'react';
import { IconButton, Icons } from '@storybook/components'; import { IconButton, Icons } from '@storybook/components';
import { Consumer, Combo } from '@storybook/api'; import { Consumer, type Combo } from '@storybook/api';
import { Addon } from '@storybook/addons'; import type { Addon } from '@storybook/addons';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { FORCE_REMOUNT } from '@storybook/core-events'; import { FORCE_REMOUNT } from '@storybook/core-events';

View File

@ -1,7 +1,7 @@
import React, { Component, SyntheticEvent, useCallback, MouseEventHandler } from 'react'; import React, { Component, SyntheticEvent, useCallback, MouseEventHandler } from 'react';
import { Icons, IconButton, Separator } from '@storybook/components'; import { Icons, IconButton, Separator } from '@storybook/components';
import { Addon } from '@storybook/addons'; import type { Addon } from '@storybook/addons';
const initialZoom = 1 as const; const initialZoom = 1 as const;
@ -48,30 +48,32 @@ const Zoom = React.memo<{
export { Zoom, ZoomConsumer, ZoomProvider }; export { Zoom, ZoomConsumer, ZoomProvider };
const ZoomWrapper = React.memo<{ set: Function; value: number }>(({ set, value }) => { const ZoomWrapper = React.memo<{ set: (zoomLevel: number) => void; value: number }>(
const zoomIn = useCallback( ({ set, value }) => {
(e: SyntheticEvent) => { const zoomIn = useCallback(
e.preventDefault(); (e: SyntheticEvent) => {
set(0.8 * value); e.preventDefault();
}, set(0.8 * value);
[set, value] },
); [set, value]
const zoomOut = useCallback( );
(e: SyntheticEvent) => { const zoomOut = useCallback(
e.preventDefault(); (e: SyntheticEvent) => {
set(1.25 * value); e.preventDefault();
}, set(1.25 * value);
[set, value] },
); [set, value]
const reset = useCallback( );
(e) => { const reset = useCallback(
e.preventDefault(); (e) => {
set(initialZoom); e.preventDefault();
}, set(initialZoom);
[set, initialZoom] },
); [set, initialZoom]
return <Zoom key="zoom" {...{ zoomIn, zoomOut, reset }} />; );
}); return <Zoom key="zoom" {...{ zoomIn, zoomOut, reset }} />;
}
);
export const zoomTool: Addon = { export const zoomTool: Addon = {
title: 'zoom', title: 'zoom',

View File

@ -1,5 +1,5 @@
import { FunctionComponent, ReactNode } from 'react'; import { FunctionComponent, ReactNode } from 'react';
import { State, API, LeafEntry } from '@storybook/api'; import type { State, API, LeafEntry } from '@storybook/api';
import { StoryId } from '@storybook/csf'; import { StoryId } from '@storybook/csf';
export type ViewMode = State['viewMode']; export type ViewMode = State['viewMode'];

View File

@ -22,7 +22,7 @@ const selected = {
const simple: Record<string, RefType> = { const simple: Record<string, RefType> = {
storybook_internal: { storybook_internal: {
title: null, title: undefined,
id: 'storybook_internal', id: 'storybook_internal',
url: 'iframe.html', url: 'iframe.html',
ready: true, ready: true,

View File

@ -79,6 +79,7 @@ export const LinkAndText: Story = () => {
title: 'My title', title: 'My title',
url: 'https://example.com', url: 'https://example.com',
image: null, image: null,
target: undefined,
}, },
}} }}
> >
@ -97,6 +98,7 @@ export const OnlyText: Story = () => {
title: 'My title', title: 'My title',
url: null, url: null,
image: null, image: null,
target: undefined,
}, },
}} }}
> >
@ -115,6 +117,7 @@ export const LongText: Story = () => {
title: 'My title is way to long to actually fit', title: 'My title is way to long to actually fit',
url: null, url: null,
image: null, image: null,
target: undefined,
}, },
}} }}
> >
@ -133,6 +136,7 @@ export const CustomTitle: Story = () => {
title: '<span style="color:red">My custom title</span>', title: '<span style="color:red">My custom title</span>',
url: null, url: null,
image: null, image: null,
target: undefined,
}, },
}} }}
> >
@ -151,6 +155,7 @@ export const CustomBrandImage: Story = () => {
title: 'My Title', title: 'My Title',
url: 'https://example.com', url: 'https://example.com',
image: 'https://storybook.js.org/images/placeholders/150x22.png', image: 'https://storybook.js.org/images/placeholders/150x22.png',
target: undefined,
}, },
}} }}
> >
@ -169,6 +174,7 @@ export const CustomBrandImageTall: Story = () => {
title: 'My Title', title: 'My Title',
url: 'https://example.com', url: 'https://example.com',
image: 'https://storybook.js.org/images/placeholders/100x150.png', image: 'https://storybook.js.org/images/placeholders/100x150.png',
target: undefined,
}, },
}} }}
> >
@ -187,6 +193,7 @@ export const CustomBrandImageUnsizedSVG: Story = () => {
title: 'My Title', title: 'My Title',
url: 'https://example.com', url: 'https://example.com',
image: 'https://s.cdpn.io/91525/potofgold.svg', image: 'https://s.cdpn.io/91525/potofgold.svg',
target: undefined,
}, },
}} }}
> >
@ -202,9 +209,10 @@ export const NoBrand: Story = () => {
theme={{ theme={{
...theme, ...theme,
brand: { brand: {
title: null, title: undefined,
url: null, url: null,
image: null, image: null,
target: undefined,
}, },
}} }}
> >

View File

@ -1,7 +1,7 @@
import { expect } from '@storybook/jest'; import { expect } from '@storybook/jest';
import React, { Fragment, FunctionComponent } from 'react'; import React, { Fragment, FunctionComponent } from 'react';
import { WithTooltip, TooltipLinkList, Icons } from '@storybook/components'; import { TooltipLinkList } from '@storybook/components';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { within, userEvent, screen } from '@storybook/testing-library'; import { within, userEvent, screen } from '@storybook/testing-library';
import { MenuItemIcon, SidebarMenu, ToolbarMenu } from './Menu'; import { MenuItemIcon, SidebarMenu, ToolbarMenu } from './Menu';
@ -73,7 +73,6 @@ Expanded.play = async ({ canvasElement }) => {
export const ExpandedWithoutReleaseNotes = () => { export const ExpandedWithoutReleaseNotes = () => {
const menu = useMenu( const menu = useMenu(
{ {
// @ts-expect-error (Converted from ts-ignore)
getShortcutKeys: () => ({}), getShortcutKeys: () => ({}),
getAddonsShortcuts: () => ({}), getAddonsShortcuts: () => ({}),
versionUpdateAvailable: () => false, versionUpdateAvailable: () => false,

View File

@ -1,4 +1,4 @@
import React, { useMemo, useState, ComponentProps, FC } from 'react'; import React, { useMemo, ComponentProps, FC } from 'react';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { transparentize } from 'polished'; import { transparentize } from 'polished';

View File

@ -28,6 +28,7 @@ const generateStories = ({ title, refId }: { title: string; refId?: string }): S
const docsId = `${rootId}-${hypenatedComponentName}--docs`; const docsId = `${rootId}-${hypenatedComponentName}--docs`;
const storyBase: HashEntry[] = [ const storyBase: HashEntry[] = [
// @ts-expect-error the missing fields are deprecated and replaced by the type prop
{ {
type: 'root', type: 'root',
id: rootId, id: rootId,
@ -37,6 +38,7 @@ const generateStories = ({ title, refId }: { title: string; refId?: string }): S
children: [componentId], children: [componentId],
startCollapsed: false, startCollapsed: false,
}, },
// @ts-expect-error the missing fields are deprecated and replaced by the type prop
{ {
type: 'component', type: 'component',
id: componentId, id: componentId,
@ -46,6 +48,7 @@ const generateStories = ({ title, refId }: { title: string; refId?: string }): S
children: [docsId], children: [docsId],
parent: rootId, parent: rootId,
}, },
// @ts-expect-error the missing fields are deprecated and replaced by the type prop
{ {
type: 'docs', type: 'docs',
id: docsId, id: docsId,

View File

@ -0,0 +1,23 @@
import React from 'react';
export default {
title: 'UI/Addon Panel',
};
export const AllAddons = () => <div>By default all addon panels are rendered</div>;
/*
TODO: this story currently breaks the whole Storybook UI (including the manager).
Current findings:
- Only happens when actions below are disabled, not when a11y is.
- Is related to panels and addon tabs.
- Commenting out code/lib/components/src/tabs/tabs.tsx#L186 fixes the issue.
- ... this line: {list.map(({ id, active, render }) => render({ key: id, active }))}
- The error is most likely the shenanigans we do at code/lib/components/src/tabs/tabs.tsx#childrenToList
export const FilteredAddons = () => <div>By default all addon panels are rendered</div>;
FilteredAddons.parameters = {
a11y: { disable: true },
actions: { disable: true },
};
*/

View File

@ -1,6 +1,6 @@
import React, { FC } from 'react'; import React, { FC } from 'react';
import memoize from 'memoizerific'; import memoize from 'memoizerific';
import { Consumer, Combo } from '@storybook/api'; import { Consumer, type Combo } from '@storybook/api';
import AddonPanel from '../components/panel/panel'; import AddonPanel from '../components/panel/panel';

Some files were not shown because too many files have changed in this diff Show More