+
+> migration guide: This page documents the method to configure storybook introduced recently in 5.3.0, consult the [migration guide](https://github.com/storybookjs/storybook/blob/next/MIGRATION.md) if you want to migrate to this format of configuring storybook.
+
+Storybook Docs transforms your Storybook stories into world-class component documentation. Storybook Docs for Vue 3 supports [DocsPage](../docs/docspage.md) for auto-generated docs, and [MDX](../docs/mdx.md) for rich long-form docs.
+
+To learn more about Storybook Docs, read the [general documentation](../README.md). To learn the Vue 3 specifics, read on!
+
+- [Installation](#installation)
+- [Preset options](#preset-options)
+- [DocsPage](#docspage)
+- [Props tables](#props-tables)
+- [MDX](#mdx)
+- [Inline Stories](#inline-stories)
+- [More resources](#more-resources)
+
+## Installation
+
+First add the package. Make sure that the versions for your `@storybook/*` packages match:
+
+```sh
+yarn add -D @storybook/addon-docs@next
+```
+
+Then add the following to your `.storybook/main.js` addons:
+
+```js
+module.exports = {
+ addons: ['@storybook/addon-docs'],
+};
+```
+
+## Preset options
+
+The `addon-docs` preset for Vue has a configuration option that can be used to configure [`vue-docgen-api`](https://github.com/vue-styleguidist/vue-styleguidist/tree/dev/packages/vue-docgen-api), a tool which extracts information from Vue components. Here's an example of how to use the preset with options for Vue app:
+
+```js
+const path = require('path');
+
+module.exports = {
+ addons: [
+ {
+ name: '@storybook/addon-docs',
+ options: {
+ vueDocgenOptions: {
+ alias: {
+ '@': path.resolve(__dirname, '../'),
+ },
+ },
+ },
+ },
+ ],
+};
+```
+
+The `vueDocgenOptions` is an object for configuring `vue-docgen-api`. See [`vue-docgen-api`'s docs](https://github.com/vue-styleguidist/vue-styleguidist/tree/dev/packages/vue-docgen-api#options-docgenoptions) for available configuration options.
+
+## DocsPage
+
+When you [install docs](#installation) you should get basic [DocsPage](../docs/docspage.md) documentation automagically for all your stories, available in the `Docs` tab of the Storybook UI.
+
+## Props tables
+
+Getting [Props tables](../docs/props-tables.md) for your components requires a few more steps. Docs for Vue relies on [`vue-docgen-loader`](https://github.com/pocka/vue-docgen-loader). It supports `props`, `events`, and `slots` as first class prop types.
+
+Finally, be sure to fill in the `component` field in your story metadata:
+
+```ts
+import { InfoButton } from './InfoButton.vue';
+
+export default {
+ title: 'InfoButton',
+ component: InfoButton,
+};
+```
+
+If you haven't upgraded from `storiesOf`, you can use a parameter to do the same thing:
+
+```ts
+import { storiesOf } from '@storybook/vue';
+import { InfoButton } from './InfoButton.vue';
+
+storiesOf('InfoButton', module)
+ .addParameters({ component: InfoButton })
+ .add( ... );
+```
+
+## MDX
+
+[MDX](../docs/mdx.md) is a convenient way to document your components in Markdown and embed documentation components, such as stories and props tables, inline.
+
+Docs has peer dependencies on `react` and `babel-loader`. If you want to write stories in MDX, you'll need to add these dependencies as well:
+
+```sh
+yarn add -D react babel-loader
+```
+
+Then update your `.storybook/main.js` to make sure you load MDX files:
+
+```js
+module.exports = {
+ stories: ['../src/stories/**/*.stories.@(js|mdx)'],
+};
+```
+
+Finally, you can create MDX files like this:
+
+```md
+import { Meta, Story, ArgsTable } from '@storybook/addon-docs/blocks';
+import { InfoButton } from './InfoButton.vue';
+
+
+
+# InfoButton
+
+Some **markdown** description, or whatever you want.
+
+{{
+ components: { InfoButton },
+ template: '',
+}}
+
+## ArgsTable
+
+
+```
+
+Yes, it's redundant to declare `component` twice. [Coming soon](https://github.com/storybookjs/storybook/issues/8685).
+
+## Inline Stories
+
+Storybook Docs renders all Vue stories inside IFrames, with a default height of `60px` (configurable using the `docs.iframeHeight` story parameter).
+
+Starting in 5.3, you can also render stories inline, and in 6.0 this has become the default behavior. To render inline, update `.storybook/preview.js`:
+
+```js
+import { addParameters } from '@storybook/vue';
+
+addParameters({
+ docs: {
+ inlineStories: true,
+ },
+});
+```
+
+## More resources
+
+Want to learn more? Here are some more articles on Storybook Docs:
+
+- References: [DocsPage](../docs/docspage.md) / [MDX](../docs/mdx.md) / [FAQ](../docs/faq.md) / [Recipes](../docs/recipes.md) / [Theming](../docs/theming.md) / [Props](../docs/props-tables.md)
+- Announcements: [Vision](https://medium.com/storybookjs/storybook-docs-sneak-peak-5be78445094a) / [DocsPage](https://medium.com/storybookjs/storybook-docspage-e185bc3622bf) / [MDX](https://medium.com/storybookjs/rich-docs-with-storybook-mdx-61bc145ae7bc) / [Framework support](https://medium.com/storybookjs/storybook-docs-for-new-frameworks-b1f6090ee0ea)
+- Example: [Storybook Design System](https://github.com/storybookjs/design-system)
diff --git a/addons/docs/web-components/README.md b/addons/docs/web-components/README.md
deleted file mode 100644
index 0f5cfb8c641..00000000000
--- a/addons/docs/web-components/README.md
+++ /dev/null
@@ -1,107 +0,0 @@
-
');
- });
-
- it('can work with compiled code', () => {
- expect(render([{}], context)).toBe(true);
- // does only work in true mode, and not in jest mode
- });
-
- it('will be possible to compile a template before rendering it', () => {
- const compiledTemplate = asCompiledCode('
');
+ });
+
+ it('can work with compiled code', () => {
+ expect(render([{}])).toBe(true);
+ // does only work in true mode, and not in jest mode
+ });
+
+ it('will be possible to compile a template before rendering it', () => {
+ const compiledTemplate = asCompiledCode('
HACKED : {opts.hacked} ; simple test ({opts.test || 'without parameter'}). Oh, by the way ({opts.riotValue || '... well, nothing'})
",
+ boundAs: 'mustBeUniquePlease',
+ },
+ ],
+ template:
+ '',
+ tagConstructor: function tagConstructor() {
+ this.hacked = true;
+ },
+ })
+ ).toBe(true);
+
+ expect(rootElement.innerHTML).toMatchSnapshot();
+ });
+});
diff --git a/app/riot/src/client/preview/render.js b/app/riot/src/client/preview/render.ts
similarity index 84%
rename from app/riot/src/client/preview/render.js
rename to app/riot/src/client/preview/render.ts
index f17ee259638..c5338a4f235 100644
--- a/app/riot/src/client/preview/render.js
+++ b/app/riot/src/client/preview/render.ts
@@ -9,6 +9,12 @@ export default function renderMain({
name,
showMain = () => {},
showError = () => {},
+}: {
+ storyFn: Function;
+ kind: string;
+ name: string;
+ showMain: () => any;
+ showError: (input: { title: string; description: string }) => void;
}) {
showMain();
unregister('#root');
diff --git a/app/riot/src/client/preview/rendering/compiledButUnmounted.js b/app/riot/src/client/preview/rendering/compiledButUnmounted.ts
similarity index 56%
rename from app/riot/src/client/preview/rendering/compiledButUnmounted.js
rename to app/riot/src/client/preview/rendering/compiledButUnmounted.ts
index 5a8d92e47be..47d8f3ee3bc 100644
--- a/app/riot/src/client/preview/rendering/compiledButUnmounted.js
+++ b/app/riot/src/client/preview/rendering/compiledButUnmounted.ts
@@ -1,5 +1,5 @@
import { mount } from 'riot';
-export default function renderCompiledButUnmounted(component) {
+export default function renderCompiledButUnmounted(component: any) {
mount('root', component.tagName, component.opts || {});
}
diff --git a/app/riot/src/client/preview/rendering/index.js b/app/riot/src/client/preview/rendering/index.ts
similarity index 93%
rename from app/riot/src/client/preview/rendering/index.js
rename to app/riot/src/client/preview/rendering/index.ts
index 752053ba02e..67bf624371a 100644
--- a/app/riot/src/client/preview/rendering/index.js
+++ b/app/riot/src/client/preview/rendering/index.ts
@@ -2,7 +2,7 @@ import renderCompiledButUnmounted from './compiledButUnmounted';
import renderStringified from './stringified';
import renderRaw from './raw';
-export function render(component) {
+export function render(component: any) {
if (typeof component === 'string') {
renderRaw(component);
return true;
diff --git a/app/riot/src/client/preview/rendering/raw.js b/app/riot/src/client/preview/rendering/raw.ts
similarity index 88%
rename from app/riot/src/client/preview/rendering/raw.js
rename to app/riot/src/client/preview/rendering/raw.ts
index f706cd8876b..8ed2fdebf32 100644
--- a/app/riot/src/client/preview/rendering/raw.js
+++ b/app/riot/src/client/preview/rendering/raw.ts
@@ -2,7 +2,7 @@ import { mount, tag2 as tag } from 'riot';
import compiler from 'riot-compiler';
import { alreadyCompiledMarker, getRidOfRiotNoise } from '../compileStageFunctions';
-export default function renderRaw(sourceCode) {
+export default function renderRaw(sourceCode: string) {
const tag2 = tag;
// eslint-disable-next-line no-eval
eval(
diff --git a/app/riot/src/client/preview/rendering/stringified.js b/app/riot/src/client/preview/rendering/stringified.ts
similarity index 93%
rename from app/riot/src/client/preview/rendering/stringified.js
rename to app/riot/src/client/preview/rendering/stringified.ts
index 4a7a732b34a..fcde052b227 100644
--- a/app/riot/src/client/preview/rendering/stringified.js
+++ b/app/riot/src/client/preview/rendering/stringified.ts
@@ -4,7 +4,7 @@ import compiler from 'riot-compiler';
import { document } from 'global';
import { alreadyCompiledMarker, getRidOfRiotNoise, setConstructor } from '../compileStageFunctions';
-function guessRootName(stringified) {
+function guessRootName(stringified: string) {
const whiteSpaceLocation = stringified.indexOf(' ', stringified.indexOf('<') + 1);
const firstWhitespace = whiteSpaceLocation === -1 ? stringified.length : whiteSpaceLocation;
const supposedName = stringified.trim().match(/^<[^ >]+\/>$/)
@@ -17,7 +17,7 @@ function guessRootName(stringified) {
return matchingBuiltInTag === 'HTMLUnknownElement' ? supposedName : 'root';
}
-function compileText(code, rootName) {
+function compileText(code: string, rootName: string) {
const sourceCodeEndOfHtml =
(Math.min(code.indexOf('`,
tagConstructor,
+}: {
+ tags: any[];
+ template: string;
+ tagConstructor: any;
}) {
const tag2 = tag;
tags.forEach((input) => {
diff --git a/app/riot/src/server/build.js b/app/riot/src/server/build.ts
similarity index 100%
rename from app/riot/src/server/build.js
rename to app/riot/src/server/build.ts
diff --git a/app/riot/src/server/framework-preset-riot.js b/app/riot/src/server/framework-preset-riot.ts
similarity index 81%
rename from app/riot/src/server/framework-preset-riot.js
rename to app/riot/src/server/framework-preset-riot.ts
index 820e02e015d..47b0b810534 100644
--- a/app/riot/src/server/framework-preset-riot.js
+++ b/app/riot/src/server/framework-preset-riot.ts
@@ -1,4 +1,6 @@
-export function webpack(config) {
+import { Configuration } from 'webpack';
+
+export function webpack(config: Configuration): Configuration {
return {
...config,
module: {
diff --git a/app/riot/src/server/index.js b/app/riot/src/server/index.ts
similarity index 100%
rename from app/riot/src/server/index.js
rename to app/riot/src/server/index.ts
diff --git a/app/riot/src/server/options.js b/app/riot/src/server/options.ts
similarity index 55%
rename from app/riot/src/server/options.js
rename to app/riot/src/server/options.ts
index fbaea49b857..e5a06e2140a 100644
--- a/app/riot/src/server/options.js
+++ b/app/riot/src/server/options.ts
@@ -1,7 +1,7 @@
-import packageJson from '../../package.json';
+import { sync } from 'read-pkg-up';
export default {
- packageJson,
+ packageJson: sync({ cwd: __dirname }).packageJson,
framework: 'riot',
frameworkPresets: [require.resolve('./framework-preset-riot.js')],
};
diff --git a/app/riot/src/typings.d.ts b/app/riot/src/typings.d.ts
new file mode 100644
index 00000000000..f5fca913e27
--- /dev/null
+++ b/app/riot/src/typings.d.ts
@@ -0,0 +1,7 @@
+declare module 'global';
+declare module 'riot-compiler';
+declare module 'riot' {
+ const tag2: Function;
+ const mount: Function;
+ const unregister: Function;
+}
diff --git a/app/riot/standalone.js b/app/riot/standalone.js
index 1b1febe0d3b..d11a82f7995 100644
--- a/app/riot/standalone.js
+++ b/app/riot/standalone.js
@@ -1,5 +1,5 @@
const build = require('@storybook/core/standalone');
-const frameworkOptions = require('./dist/server/options').default;
+const frameworkOptions = require('./dist/cjs/server/options').default;
async function buildStandalone(options) {
return build(options, frameworkOptions);
diff --git a/app/riot/tsconfig.json b/app/riot/tsconfig.json
new file mode 100644
index 00000000000..bc50e94eb18
--- /dev/null
+++ b/app/riot/tsconfig.json
@@ -0,0 +1,14 @@
+{
+ "extends": "../../tsconfig.json",
+ "compilerOptions": {
+ "rootDir": "./src",
+ "types": ["webpack-env", "node", "jest"],
+ "resolveJsonModule": true
+ },
+ "include": [
+ "src/**/*"
+ ],
+ "exclude": [
+ "src/**/*.test.*"
+ ]
+}
diff --git a/app/server/README.md b/app/server/README.md
index 8da84acf454..ea0285b5de7 100644
--- a/app/server/README.md
+++ b/app/server/README.md
@@ -286,7 +286,7 @@ For control over how `@storybook/server` fetches Html from the server you can pr
```javascript
// .storybook/preview.js
-const fetchStoryHtml = async (url, path, params) => {
+const fetchStoryHtml = async (url, path, params, context) => {
// Custom fetch implementation
// ....
return html;
@@ -303,9 +303,10 @@ export const parameters = {
`fetchStoryHtml` should be an async function with the following signature
```javascript
-type FetchStoryHtmlType = (url: string, id: string, params: any) => Promise;
+type FetchStoryHtmlType = (url: string, id: string, params: any, context: StoryContext) => Promise;
```
* url: Server url configured by the `parameters.server.url`
* id: Id of the story being rendered given by `parameters.server.id`
* params: Merged story params `parameters.server.params`and story args
+ * context: The context of the story
diff --git a/app/server/bin/build.js b/app/server/bin/build.js
index 26142ec0af2..6ef07a26a9f 100755
--- a/app/server/bin/build.js
+++ b/app/server/bin/build.js
@@ -1,4 +1,4 @@
#!/usr/bin/env node
process.env.NODE_ENV = process.env.NODE_ENV || 'production';
-require('../dist/server/build');
+require('../dist/cjs/server/build');
diff --git a/app/server/bin/index.js b/app/server/bin/index.js
index 2e96258ce63..8aab7bbceb2 100755
--- a/app/server/bin/index.js
+++ b/app/server/bin/index.js
@@ -1,3 +1,3 @@
#!/usr/bin/env node
-require('../dist/server');
+require('../dist/cjs/server');
diff --git a/app/server/package.json b/app/server/package.json
index b0dec677ba2..ca9c9303821 100644
--- a/app/server/package.json
+++ b/app/server/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/server",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "Storybook for Server: View HTML snippets from a server in isolation with Hot Reloading.",
"keywords": [
"storybook"
@@ -15,12 +15,13 @@
"directory": "app/server"
},
"license": "MIT",
- "main": "dist/client/index.js",
- "types": "dist/client/index.d.ts",
+ "main": "dist/cjs/client/index.js",
+ "module": "dist/esm/client/index.js",
+ "types": "dist/ts3.9/client/index.d.ts",
"typesVersions": {
"<3.8": {
"*": [
- "ts3.4/*"
+ "dist/ts3.4/*"
]
}
},
@@ -34,35 +35,36 @@
"dist/**/*",
"README.md",
"*.js",
- "*.d.ts",
- "ts3.4/**/*"
+ "*.d.ts"
],
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "6.2.0-alpha.5",
- "@storybook/api": "6.2.0-alpha.5",
- "@storybook/client-api": "6.2.0-alpha.5",
- "@storybook/core": "6.2.0-alpha.5",
- "@storybook/node-logger": "6.2.0-alpha.5",
- "@types/webpack-env": "^1.15.3",
- "core-js": "^3.0.1",
- "global": "^4.3.2",
- "react": "16.13.1",
- "react-dom": "16.13.1",
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/api": "6.2.0-beta.14",
+ "@storybook/client-api": "6.2.0-beta.14",
+ "@storybook/core": "6.2.0-beta.14",
+ "@storybook/core-common": "6.2.0-beta.14",
+ "@storybook/node-logger": "6.2.0-beta.14",
+ "@types/webpack-env": "^1.16.0",
+ "core-js": "^3.8.2",
+ "global": "^4.4.0",
+ "react": "16.14.0",
+ "react-dom": "16.14.0",
+ "read-pkg-up": "^7.0.1",
"regenerator-runtime": "^0.13.7",
"safe-identifier": "^0.4.1",
"ts-dedent": "^2.0.0"
},
"devDependencies": {
- "fs-extra": "^9.0.0"
+ "fs-extra": "^9.0.1"
},
"engines": {
- "node": ">=8.0.0"
+ "node": ">=10.13.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/app/server/src/client/preview/index.ts b/app/server/src/client/preview/index.ts
index 9a5ac6ebbda..9c8154de92f 100644
--- a/app/server/src/client/preview/index.ts
+++ b/app/server/src/client/preview/index.ts
@@ -30,7 +30,8 @@ export const {
addParameters,
clearDecorators,
setAddon,
- forceReRender,
getStorybook,
raw,
} = api.clientApi;
+
+export const { forceReRender } = api;
diff --git a/app/server/src/client/preview/render.ts b/app/server/src/client/preview/render.ts
index b2194caa514..b3a38490ae7 100644
--- a/app/server/src/client/preview/render.ts
+++ b/app/server/src/client/preview/render.ts
@@ -6,9 +6,9 @@ import { RenderContext, FetchStoryHtmlType } from './types';
const rootElement = document.getElementById('root');
-const defaultFetchStoryHtml: FetchStoryHtmlType = async (url, path, params) => {
+const defaultFetchStoryHtml: FetchStoryHtmlType = async (url, path, params, storyContext) => {
const fetchUrl = new URL(`${url}/${path}`);
- fetchUrl.search = new URLSearchParams(params).toString();
+ fetchUrl.search = new URLSearchParams({ ...storyContext.globals, ...params }).toString();
const response = await fetch(fetchUrl);
return response.text();
@@ -52,6 +52,7 @@ export async function renderMain({
showError,
forceRender,
parameters,
+ storyContext,
storyFn,
args,
argTypes,
@@ -65,8 +66,8 @@ export async function renderMain({
} = parameters;
const fetchId = storyId || id;
- const fetchParams = { ...params, ...storyArgs };
- const element = await fetchStoryHtml(url, fetchId, fetchParams);
+ const storyParams = { ...params, ...storyArgs };
+ const element = await fetchStoryHtml(url, fetchId, storyParams, storyContext);
showMain();
if (typeof element === 'string') {
diff --git a/app/server/src/client/preview/types.ts b/app/server/src/client/preview/types.ts
index 8238b46d042..7d371f2d7fe 100644
--- a/app/server/src/client/preview/types.ts
+++ b/app/server/src/client/preview/types.ts
@@ -1,8 +1,15 @@
-export { RenderContext } from '@storybook/core';
+import { StoryContext } from '@storybook/addons';
+
+export type { RenderContext } from '@storybook/core';
export type StoryFnServerReturnType = any;
-export type FetchStoryHtmlType = (url: string, id: string, params: any) => Promise;
+export type FetchStoryHtmlType = (
+ url: string,
+ id: string,
+ params: any,
+ context: StoryContext
+) => Promise;
export interface IStorybookStory {
name: string;
diff --git a/app/server/src/lib/compiler/json-to-csf-compiler.test.ts b/app/server/src/lib/compiler/json-to-csf-compiler.test.ts
index 503cf198a99..a99dcc45a90 100644
--- a/app/server/src/lib/compiler/json-to-csf-compiler.test.ts
+++ b/app/server/src/lib/compiler/json-to-csf-compiler.test.ts
@@ -15,6 +15,7 @@ describe('json-to-csf-compiler', () => {
fs.readdirSync(transformFixturesDir)
.filter((fileName: string) => inputRegExp.test(fileName))
.forEach((fixtureFile: string) => {
+ // eslint-disable-next-line jest/valid-title
it(fixtureFile, async () => {
const inputPath = path.join(transformFixturesDir, fixtureFile);
const code = await generate(inputPath);
diff --git a/app/server/src/server/framework-preset-server.ts b/app/server/src/server/framework-preset-server.ts
index 2604c82e8b0..e8bc1b379db 100644
--- a/app/server/src/server/framework-preset-server.ts
+++ b/app/server/src/server/framework-preset-server.ts
@@ -3,22 +3,11 @@ import { Configuration } from 'webpack';
import path from 'path';
export function webpack(config: Configuration) {
- return {
- ...config,
- module: {
- ...config.module,
- rules: [
- ...config.module.rules,
- {
- type: 'javascript/auto',
- test: /\.stories\.json$/,
- use: [
- {
- loader: path.resolve(__dirname, './loader.js'),
- },
- ],
- },
- ],
- },
- };
+ config.module.rules.push({
+ type: 'javascript/auto',
+ test: /\.stories\.json$/,
+ use: path.resolve(__dirname, './loader.js'),
+ });
+
+ return config;
}
diff --git a/app/server/src/server/options.ts b/app/server/src/server/options.ts
index ee538806a87..ba3f398a385 100644
--- a/app/server/src/server/options.ts
+++ b/app/server/src/server/options.ts
@@ -1,7 +1,8 @@
-const packageJson = require('../../package.json');
+import { sync } from 'read-pkg-up';
+import { LoadOptions } from '@storybook/core-common';
export default {
- packageJson,
+ packageJson: sync({ cwd: __dirname }).packageJson,
framework: 'server',
frameworkPresets: [require.resolve('./framework-preset-server.js')],
-};
+} as LoadOptions;
diff --git a/app/server/src/typings.d.ts b/app/server/src/typings.d.ts
index 690e93343de..d8f7c6f660a 100644
--- a/app/server/src/typings.d.ts
+++ b/app/server/src/typings.d.ts
@@ -1,4 +1,3 @@
-declare module '@storybook/core/*';
declare module 'global';
// will be provided by the webpack define plugin
diff --git a/app/server/standalone.js b/app/server/standalone.js
index 1b1febe0d3b..d11a82f7995 100644
--- a/app/server/standalone.js
+++ b/app/server/standalone.js
@@ -1,5 +1,5 @@
const build = require('@storybook/core/standalone');
-const frameworkOptions = require('./dist/server/options').default;
+const frameworkOptions = require('./dist/cjs/server/options').default;
async function buildStandalone(options) {
return build(options, frameworkOptions);
diff --git a/app/server/tsconfig.json b/app/server/tsconfig.json
index 82ce44329cc..13f32ad6309 100644
--- a/app/server/tsconfig.json
+++ b/app/server/tsconfig.json
@@ -2,7 +2,7 @@
"extends": "../../tsconfig.json",
"compilerOptions": {
"rootDir": "./src",
- "types": ["webpack-env"]
+ "types": ["webpack-env", "node"]
},
"include": ["src/**/*"],
"exclude": ["src/__tests__/**/*"]
diff --git a/app/standalone.test.ts b/app/standalone.test.ts
new file mode 100644
index 00000000000..416a5753593
--- /dev/null
+++ b/app/standalone.test.ts
@@ -0,0 +1,34 @@
+import build from '@storybook/core/standalone';
+
+jest.mock('@storybook/core/standalone');
+
+describe.each([
+ ['angular'],
+ ['aurelia'],
+ ['ember'],
+ ['html'],
+ ['marionette'],
+ ['marko'],
+ ['mithril'],
+ ['preact'],
+ ['rax'],
+ ['react'],
+ ['riot'],
+ ['server'],
+ ['svelte'],
+ ['vue'],
+ ['vue3'],
+ ['web-components'],
+])('%s', (app) => {
+ it('should run standalone', async () => {
+ // eslint-disable-next-line import/no-dynamic-require, global-require
+ const storybook = require(`@storybook/${app}/standalone`);
+
+ await storybook({
+ mode: 'static',
+ outputDir: '',
+ });
+
+ expect(build).toHaveBeenCalled();
+ });
+});
diff --git a/app/svelte/bin/build.js b/app/svelte/bin/build.js
index 410ac35d845..8b8db43a663 100755
--- a/app/svelte/bin/build.js
+++ b/app/svelte/bin/build.js
@@ -2,4 +2,4 @@
process.env.NODE_ENV = process.env.NODE_ENV || 'production';
-require('../dist/server/build');
+require('../dist/cjs/server/build');
diff --git a/app/svelte/bin/index.js b/app/svelte/bin/index.js
index 2e96258ce63..8aab7bbceb2 100755
--- a/app/svelte/bin/index.js
+++ b/app/svelte/bin/index.js
@@ -1,3 +1,3 @@
#!/usr/bin/env node
-require('../dist/server');
+require('../dist/cjs/server');
diff --git a/app/svelte/package.json b/app/svelte/package.json
index 39a3450a057..d44e3a5fc44 100644
--- a/app/svelte/package.json
+++ b/app/svelte/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/svelte",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "Storybook for Svelte: Develop Svelte Component in isolation with Hot Reloading.",
"keywords": [
"storybook"
@@ -15,12 +15,13 @@
"directory": "app/svelte"
},
"license": "MIT",
- "main": "dist/client/index.js",
- "types": "dist/client/index.d.ts",
+ "main": "dist/cjs/client/index.js",
+ "module": "dist/esm/client/index.js",
+ "types": "dist/ts3.9/client/index.d.ts",
"typesVersions": {
"<3.8": {
"*": [
- "ts3.4/*"
+ "dist/ts3.4/*"
]
}
},
@@ -34,38 +35,39 @@
"dist/**/*",
"README.md",
"*.js",
- "*.d.ts",
- "ts3.4/**/*"
+ "*.d.ts"
],
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "6.2.0-alpha.5",
- "@storybook/core": "6.2.0-alpha.5",
- "core-js": "^3.0.1",
- "global": "^4.3.2",
- "react": "16.13.1",
- "react-dom": "16.13.1",
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/core": "6.2.0-beta.14",
+ "@storybook/core-common": "6.2.0-beta.14",
+ "core-js": "^3.8.2",
+ "global": "^4.4.0",
+ "react": "16.14.0",
+ "react-dom": "16.14.0",
+ "read-pkg-up": "^7.0.1",
"regenerator-runtime": "^0.13.7",
- "sveltedoc-parser": "^3.0.4",
+ "sveltedoc-parser": "^4.1.0",
"ts-dedent": "^2.0.0"
},
"devDependencies": {
- "@types/webpack-env": "^1.15.3",
- "svelte": "^3.18.1",
- "svelte-loader": "^2.13.4"
+ "@types/webpack-env": "^1.16.0",
+ "svelte": "^3.31.2",
+ "svelte-loader": "^3.0.0"
},
"peerDependencies": {
"@babel/core": "*",
"svelte": "^3.1.0",
- "svelte-loader": "^2.9.1"
+ "svelte-loader": "*"
},
"engines": {
- "node": ">=8.0.0"
+ "node": ">=10.13.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/app/svelte/src/client/preview/PreviewRender.svelte b/app/svelte/src/client/preview/PreviewRender.svelte
new file mode 100644
index 00000000000..40c027d6c80
--- /dev/null
+++ b/app/svelte/src/client/preview/PreviewRender.svelte
@@ -0,0 +1,37 @@
+
+
\ No newline at end of file
diff --git a/app/svelte/src/client/preview/SlotDecorator.svelte b/app/svelte/src/client/preview/SlotDecorator.svelte
new file mode 100644
index 00000000000..bd51e06f427
--- /dev/null
+++ b/app/svelte/src/client/preview/SlotDecorator.svelte
@@ -0,0 +1,30 @@
+
+{#if decorator}
+
+
+
+{:else}
+
+{/if}
\ No newline at end of file
diff --git a/app/svelte/src/client/preview/decorators.ts b/app/svelte/src/client/preview/decorators.ts
new file mode 100644
index 00000000000..1f9af116602
--- /dev/null
+++ b/app/svelte/src/client/preview/decorators.ts
@@ -0,0 +1,100 @@
+import { StoryFn, DecoratorFunction, StoryContext } from '@storybook/addons';
+import SlotDecorator from './SlotDecorator.svelte';
+
+const defaultContext: StoryContext = {
+ id: 'unspecified',
+ name: 'unspecified',
+ kind: 'unspecified',
+ parameters: {},
+ args: {},
+ argTypes: {},
+ globals: {},
+};
+
+/**
+ * Check if an object is a svelte component.
+ * @param obj Object
+ */
+function isSvelteComponent(obj: any) {
+ return obj.prototype && obj.prototype.$destroy !== undefined;
+}
+
+/**
+ * Handle component loaded with esm or cjs.
+ * @param obj object
+ */
+function unWrap(obj: any) {
+ return obj && obj.default ? obj.default : obj;
+}
+
+/**
+ * Transform a story to be compatible with the PreviewRender component.
+ *
+ * - `() => MyComponent` is translated to `() => ({ Component: MyComponent })`
+ * - `() => ({})` is translated to `() => ({ Component: })`
+ * - A decorator component is wrapped with SlotDecorator. The decorated component is inject through
+ * a
+ *
+ * @param context StoryContext
+ * @param story the current story
+ * @param originalStory the story decorated by the current story
+ */
+function prepareStory(context: StoryContext, story: any, originalStory?: any) {
+ let result = unWrap(story);
+ if (isSvelteComponent(result)) {
+ // wrap the component
+ result = {
+ Component: result,
+ };
+ }
+
+ if (originalStory) {
+ // inject the new story as a wrapper of the original story
+ result = {
+ Component: SlotDecorator,
+ props: {
+ decorator: unWrap(result.Component),
+ decoratorProps: result.props,
+ component: unWrap(originalStory.Component),
+ props: originalStory.props,
+ on: originalStory.on,
+ },
+ };
+ } else {
+ let cpn = result.Component;
+ if (!cpn) {
+ // if the component is not defined, get it from parameters
+ cpn = context.parameters.component;
+ }
+ result.Component = unWrap(cpn);
+ }
+ return result;
+}
+
+export function decorateStory(storyFn: any, decorators: any[]) {
+ return decorators.reduce(
+ (previousStoryFn: StoryFn, decorator: DecoratorFunction) => (
+ context: StoryContext = defaultContext
+ ) => {
+ let story;
+ const decoratedStory = decorator(
+ ({ parameters, ...innerContext }: StoryContext = {} as StoryContext) => {
+ story = previousStoryFn({ ...context, ...innerContext });
+ return story;
+ },
+ context
+ );
+
+ if (!story) {
+ story = previousStoryFn(context);
+ }
+
+ if (!decoratedStory || decoratedStory === story) {
+ return story;
+ }
+
+ return prepareStory(context, decoratedStory, story);
+ },
+ (context: StoryContext) => prepareStory(context, storyFn(context))
+ );
+}
diff --git a/app/svelte/src/client/preview/index.ts b/app/svelte/src/client/preview/index.ts
index abc4d66ddb3..cbfed51c166 100644
--- a/app/svelte/src/client/preview/index.ts
+++ b/app/svelte/src/client/preview/index.ts
@@ -1,9 +1,10 @@
import { start } from '@storybook/core/client';
+import { decorateStory } from './decorators';
import './globals';
import render from './render';
-const { configure: coreConfigure, clientApi, forceReRender } = start(render);
+const { configure: coreConfigure, clientApi, forceReRender } = start(render, { decorateStory });
export const {
setAddon,
@@ -15,8 +16,8 @@ export const {
} = clientApi;
const framework = 'svelte';
-export const storiesOf = (...args: any) =>
- clientApi.storiesOf(...args).addParameters({ framework });
-export const configure = (...args: any) => coreConfigure(framework, ...args);
+export const storiesOf = (kind: string, m: any) =>
+ clientApi.storiesOf(kind, m).addParameters({ framework });
+export const configure = (loadable: any, m: any) => coreConfigure(framework, loadable, m);
export { forceReRender };
diff --git a/app/svelte/src/client/preview/render.ts b/app/svelte/src/client/preview/render.ts
index cea239a2d15..ba072e1400a 100644
--- a/app/svelte/src/client/preview/render.ts
+++ b/app/svelte/src/client/preview/render.ts
@@ -1,7 +1,6 @@
-import { detach, insert, noop } from 'svelte/internal';
import { document } from 'global';
-import dedent from 'ts-dedent';
-import { MountViewArgs, RenderContext } from './types';
+import { RenderContext } from './types';
+import PreviewRender from './PreviewRender.svelte';
type Component = any;
@@ -15,104 +14,21 @@ function cleanUpPreviousStory() {
previousComponent = null;
}
-function createSlotFn(element: any) {
- return [
- function createSlot() {
- return {
- c: noop,
- m: function mount(target: any, anchor: any) {
- insert(target, element, anchor);
- },
- d: function destroy(detaching: boolean) {
- if (detaching) {
- detach(element);
- }
- },
- l: noop,
- };
- },
- ];
-}
-
-function createSlots(slots: Record): Record {
- return Object.entries(slots).reduce((acc, [slotName, element]) => {
- acc[slotName] = createSlotFn(element);
- return acc;
- }, {} as Record);
-}
-
-function mountView({ Component, target, props, on, Wrapper, WrapperData }: MountViewArgs) {
- let component: Component;
-
- if (Wrapper) {
- const fragment = document.createDocumentFragment();
- component = new Component({ target: fragment, props });
-
- const wrapper = new Wrapper({
- target,
- props: {
- ...WrapperData,
- $$slots: createSlots({ default: fragment }),
- $$scope: {},
- },
- });
- component.$on('destroy', () => {
- wrapper.$destroy(true);
- });
- } else {
- component = new Component({ target, props });
- }
-
- if (on) {
- // Attach svelte event listeners.
- Object.keys(on).forEach((eventName) => {
- component.$on(eventName, on[eventName]);
- });
- }
-
- previousComponent = component;
-}
-
export default function render({ storyFn, kind, name, showMain, showError }: RenderContext) {
- const {
- /** @type {SvelteComponent} */
- Component,
- /** @type {any} */
- props,
- /** @type {{[string]: () => {}}} Attach svelte event handlers */
- on,
- Wrapper,
- WrapperData,
- } = storyFn();
-
cleanUpPreviousStory();
- const DefaultCompatComponent = Component ? Component.default || Component : undefined;
- const DefaultCompatWrapper = Wrapper ? Wrapper.default || Wrapper : undefined;
-
- if (!DefaultCompatComponent) {
- showError({
- title: `Expecting a Svelte component from the story: "${name}" of "${kind}".`,
- description: dedent`
- Did you forget to return the Svelte component configuration from the story?
- Use "() => ({ Component: YourComponent, data: {} })"
- when defining the story.
- `,
- });
-
- return;
- }
const target = document.getElementById('root');
target.innerHTML = '';
- mountView({
- Component: DefaultCompatComponent,
+ previousComponent = new PreviewRender({
target,
- props,
- on,
- Wrapper: DefaultCompatWrapper,
- WrapperData,
+ props: {
+ storyFn,
+ name,
+ kind,
+ showError,
+ },
});
showMain();
diff --git a/app/svelte/src/client/preview/types.ts b/app/svelte/src/client/preview/types.ts
index 4382e18ca0f..0299eb23f8e 100644
--- a/app/svelte/src/client/preview/types.ts
+++ b/app/svelte/src/client/preview/types.ts
@@ -1,4 +1,4 @@
-export { RenderContext } from '@storybook/core';
+export type { RenderContext } from '@storybook/core';
export interface ShowErrorArgs {
title: string;
diff --git a/app/svelte/src/server/framework-preset-svelte.ts b/app/svelte/src/server/framework-preset-svelte.ts
index d29eaf8ec51..bf17ac7a948 100644
--- a/app/svelte/src/server/framework-preset-svelte.ts
+++ b/app/svelte/src/server/framework-preset-svelte.ts
@@ -1,6 +1,14 @@
-import { Configuration } from 'webpack'; // eslint-disable-line
+// eslint-disable-next-line import/no-extraneous-dependencies
+import { Configuration } from 'webpack';
+import type { Options } from '@storybook/core-common';
+
+export async function webpack(config: Configuration, options: Options): Promise {
+ const { preprocess = undefined, loader = {} } = await options.presets.apply(
+ 'svelteOptions',
+ {} as any,
+ options
+ );
-export function webpack(config: Configuration) {
return {
...config,
module: {
@@ -10,7 +18,7 @@ export function webpack(config: Configuration) {
{
test: /\.(svelte|html)$/,
loader: require.resolve('svelte-loader'),
- options: {},
+ options: { preprocess, ...loader },
},
],
},
diff --git a/app/svelte/src/server/options.ts b/app/svelte/src/server/options.ts
index 64c8aa3cd58..ee8c399d9af 100644
--- a/app/svelte/src/server/options.ts
+++ b/app/svelte/src/server/options.ts
@@ -1,7 +1,8 @@
-const packageJson = require('../../package.json');
+import { sync } from 'read-pkg-up';
+import { LoadOptions } from '@storybook/core-common';
export default {
- packageJson,
+ packageJson: sync({ cwd: __dirname }).packageJson,
framework: 'svelte',
frameworkPresets: [require.resolve('./framework-preset-svelte.js')],
-};
+} as LoadOptions;
diff --git a/app/svelte/src/typings.d.ts b/app/svelte/src/typings.d.ts
index 6288cba4b09..19424af371e 100644
--- a/app/svelte/src/typings.d.ts
+++ b/app/svelte/src/typings.d.ts
@@ -1,2 +1,2 @@
-declare module '@storybook/core/*';
declare module 'global';
+declare module '*.svelte';
\ No newline at end of file
diff --git a/app/svelte/standalone.js b/app/svelte/standalone.js
index 1b1febe0d3b..d11a82f7995 100644
--- a/app/svelte/standalone.js
+++ b/app/svelte/standalone.js
@@ -1,5 +1,5 @@
const build = require('@storybook/core/standalone');
-const frameworkOptions = require('./dist/server/options').default;
+const frameworkOptions = require('./dist/cjs/server/options').default;
async function buildStandalone(options) {
return build(options, frameworkOptions);
diff --git a/app/svelte/tsconfig.json b/app/svelte/tsconfig.json
index 29fcd6ad6a2..08bdb7764f7 100644
--- a/app/svelte/tsconfig.json
+++ b/app/svelte/tsconfig.json
@@ -2,7 +2,7 @@
"extends": "../../tsconfig.json",
"compilerOptions": {
"rootDir": "./src",
- "types": ["webpack-env"],
+ "types": ["webpack-env", "node"],
"resolveJsonModule": true
},
"include": [
diff --git a/app/vue/bin/build.js b/app/vue/bin/build.js
index 26142ec0af2..6ef07a26a9f 100755
--- a/app/vue/bin/build.js
+++ b/app/vue/bin/build.js
@@ -1,4 +1,4 @@
#!/usr/bin/env node
process.env.NODE_ENV = process.env.NODE_ENV || 'production';
-require('../dist/server/build');
+require('../dist/cjs/server/build');
diff --git a/app/vue/bin/index.js b/app/vue/bin/index.js
index 2e96258ce63..8aab7bbceb2 100755
--- a/app/vue/bin/index.js
+++ b/app/vue/bin/index.js
@@ -1,3 +1,3 @@
#!/usr/bin/env node
-require('../dist/server');
+require('../dist/cjs/server');
diff --git a/app/vue/package.json b/app/vue/package.json
index 9f628705df6..3c4f5dd0108 100644
--- a/app/vue/package.json
+++ b/app/vue/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/vue",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "Storybook for Vue: Develop Vue Component in isolation with Hot Reloading.",
"keywords": [
"storybook"
@@ -15,12 +15,13 @@
"directory": "app/vue"
},
"license": "MIT",
- "main": "dist/client/index.js",
- "types": "dist/client/index.d.ts",
+ "main": "dist/cjs/client/index.js",
+ "module": "dist/esm/client/index.js",
+ "types": "dist/ts3.9/client/index.d.ts",
"typesVersions": {
"<3.8": {
"*": [
- "ts3.4/*"
+ "dist/ts3.4/*"
]
}
},
@@ -34,48 +35,47 @@
"dist/**/*",
"README.md",
"*.js",
- "*.d.ts",
- "ts3.4/**/*"
+ "*.d.ts"
],
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "6.2.0-alpha.5",
- "@storybook/core": "6.2.0-alpha.5",
- "@types/webpack-env": "^1.15.3",
- "core-js": "^3.0.1",
- "global": "^4.3.2",
- "react": "16.13.1",
- "react-dom": "16.13.1",
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/core": "6.2.0-beta.14",
+ "@storybook/core-common": "6.2.0-beta.14",
+ "@types/webpack-env": "^1.16.0",
+ "core-js": "^3.8.2",
+ "global": "^4.4.0",
+ "react": "16.14.0",
+ "react-dom": "16.14.0",
+ "read-pkg-up": "^7.0.1",
"regenerator-runtime": "^0.13.7",
"ts-dedent": "^2.0.0",
- "ts-loader": "^6.2.2",
- "vue-docgen-api": "^4.33.1",
+ "ts-loader": "^8.0.14",
+ "vue-docgen-api": "^4.34.2",
"vue-docgen-loader": "^1.5.0",
- "webpack": "^4.44.2"
+ "webpack": "4"
},
"devDependencies": {
- "@types/node": "^14.0.10",
- "@types/webpack": "^4.41.24",
- "vue": "^2.6.8",
- "vue-loader": "^15.7.0",
- "vue-template-compiler": "^2.6.8"
+ "@types/node": "^14.14.20",
+ "vue": "^2.6.12",
+ "vue-loader": "^15.9.6",
+ "vue-template-compiler": "^2.6.12"
},
"peerDependencies": {
"@babel/core": "*",
"babel-loader": "^7.0.0 || ^8.0.0",
"css-loader": "*",
- "ts-loader": "^6.2.2",
"vue": "^2.6.8",
"vue-loader": "^15.7.0",
"vue-template-compiler": "^2.6.8"
},
"engines": {
- "node": ">=8.0.0"
+ "node": ">=10.13.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/app/vue/src/client/preview/index.ts b/app/vue/src/client/preview/index.ts
index b00eb4385d1..06b5202bb69 100644
--- a/app/vue/src/client/preview/index.ts
+++ b/app/vue/src/client/preview/index.ts
@@ -123,8 +123,10 @@ export const storiesOf: ClientApi['storiesOf'] = (kind, m) => {
};
export const configure: ClientApi['configure'] = (...args) => api.configure(framework, ...args);
-export const addDecorator: ClientApi['addDecorator'] = api.clientApi.addDecorator;
-export const addParameters: ClientApi['addParameters'] = api.clientApi.addParameters;
+export const addDecorator: ClientApi['addDecorator'] = api.clientApi
+ .addDecorator as ClientApi['addDecorator'];
+export const addParameters: ClientApi['addParameters'] = api.clientApi
+ .addParameters as ClientApi['addParameters'];
export const clearDecorators: ClientApi['clearDecorators'] = api.clientApi.clearDecorators;
export const setAddon: ClientApi['setAddon'] = api.clientApi.setAddon;
export const forceReRender: ClientApi['forceReRender'] = api.forceReRender;
diff --git a/app/vue/src/client/preview/types-6-0.ts b/app/vue/src/client/preview/types-6-0.ts
index 3807ea00e78..66dbbd4ca8e 100644
--- a/app/vue/src/client/preview/types-6-0.ts
+++ b/app/vue/src/client/preview/types-6-0.ts
@@ -2,7 +2,7 @@ import { Component, AsyncComponent } from 'vue';
import { Args as DefaultArgs, Annotations, BaseMeta, BaseStory } from '@storybook/addons';
import { StoryFnVueReturnType } from './types';
-export { Args, ArgTypes, Parameters, StoryContext } from '@storybook/addons';
+export type { Args, ArgTypes, Parameters, StoryContext } from '@storybook/addons';
type VueComponent = Component | AsyncComponent;
type VueReturnType = StoryFnVueReturnType;
diff --git a/app/vue/src/client/preview/types.ts b/app/vue/src/client/preview/types.ts
index 94442a01779..b2e8b22ec0c 100644
--- a/app/vue/src/client/preview/types.ts
+++ b/app/vue/src/client/preview/types.ts
@@ -1,6 +1,6 @@
import { Component } from 'vue';
-export { RenderContext } from '@storybook/core';
+export type { RenderContext } from '@storybook/core';
export interface ShowErrorArgs {
title: string;
diff --git a/app/vue/src/server/framework-preset-vue.ts b/app/vue/src/server/framework-preset-vue.ts
index bf6f79b53ad..e81c5267681 100644
--- a/app/vue/src/server/framework-preset-vue.ts
+++ b/app/vue/src/server/framework-preset-vue.ts
@@ -1,40 +1,29 @@
+/* eslint-disable no-param-reassign */
import VueLoaderPlugin from 'vue-loader/lib/plugin';
-import { Configuration } from 'webpack';
+import type { Configuration } from 'webpack';
export function webpack(config: Configuration) {
- return {
- ...config,
- plugins: [...config.plugins, new VueLoaderPlugin()],
- module: {
- ...config.module,
- rules: [
- ...config.module.rules,
- {
- test: /\.vue$/,
- loader: require.resolve('vue-loader'),
- options: {},
+ config.plugins.push(new VueLoaderPlugin());
+ config.module.rules.push({
+ test: /\.vue$/,
+ loader: require.resolve('vue-loader'),
+ options: {},
+ });
+ config.module.rules.push({
+ test: /\.tsx?$/,
+ use: [
+ {
+ loader: require.resolve('ts-loader'),
+ options: {
+ transpileOnly: true,
+ appendTsSuffixTo: [/\.vue$/],
},
- {
- test: /\.tsx?$/,
- use: [
- {
- loader: require.resolve('ts-loader'),
- options: {
- transpileOnly: true,
- appendTsSuffixTo: [/\.vue$/],
- },
- },
- ],
- },
- ],
- },
- resolve: {
- ...config.resolve,
- extensions: [...config.resolve.extensions, '.vue'],
- alias: {
- ...config.resolve.alias,
- vue$: require.resolve('vue/dist/vue.esm.js'),
},
- },
- };
+ ],
+ });
+
+ config.resolve.extensions.push('.vue');
+ config.resolve.alias = { ...config.resolve.alias, vue$: require.resolve('vue/dist/vue.esm.js') };
+
+ return config;
}
diff --git a/app/vue/src/server/options.ts b/app/vue/src/server/options.ts
index 46df1823ce8..e57b0499481 100644
--- a/app/vue/src/server/options.ts
+++ b/app/vue/src/server/options.ts
@@ -1,7 +1,8 @@
-const packageJson = require('../../package.json');
+import { sync } from 'read-pkg-up';
+import { LoadOptions } from '@storybook/core-common';
export default {
- packageJson,
+ packageJson: sync({ cwd: __dirname }).packageJson,
framework: 'vue',
frameworkPresets: [require.resolve('./framework-preset-vue.js')],
-};
+} as LoadOptions;
diff --git a/app/vue/src/typings.d.ts b/app/vue/src/typings.d.ts
index ae854f4da05..87f498f0480 100644
--- a/app/vue/src/typings.d.ts
+++ b/app/vue/src/typings.d.ts
@@ -1,4 +1,3 @@
-declare module '@storybook/core/*';
declare module 'global';
// todo check for correct types
declare module 'webpack/lib/RuleSet';
diff --git a/app/vue/standalone.js b/app/vue/standalone.js
index 1b1febe0d3b..d11a82f7995 100644
--- a/app/vue/standalone.js
+++ b/app/vue/standalone.js
@@ -1,5 +1,5 @@
const build = require('@storybook/core/standalone');
-const frameworkOptions = require('./dist/server/options').default;
+const frameworkOptions = require('./dist/cjs/server/options').default;
async function buildStandalone(options) {
return build(options, frameworkOptions);
diff --git a/app/vue/tsconfig.json b/app/vue/tsconfig.json
index 29fcd6ad6a2..08bdb7764f7 100644
--- a/app/vue/tsconfig.json
+++ b/app/vue/tsconfig.json
@@ -2,7 +2,7 @@
"extends": "../../tsconfig.json",
"compilerOptions": {
"rootDir": "./src",
- "types": ["webpack-env"],
+ "types": ["webpack-env", "node"],
"resolveJsonModule": true
},
"include": [
diff --git a/app/vue/types-6-0.d.ts b/app/vue/types-6-0.d.ts
index 6ed7da8e519..b5946b39a8d 100644
--- a/app/vue/types-6-0.d.ts
+++ b/app/vue/types-6-0.d.ts
@@ -1 +1 @@
-export * from './dist/client/preview/types-6-0.d';
+export * from './dist/ts3.9/client/preview/types-6-0.d';
diff --git a/app/vue3/README.md b/app/vue3/README.md
new file mode 100644
index 00000000000..3582c7266b8
--- /dev/null
+++ b/app/vue3/README.md
@@ -0,0 +1,41 @@
+# Storybook for Vue 3
+
+Storybook for Vue 3 is a UI development environment for your Vue 3 components.
+With it, you can visualize different states of your UI components and develop them interactively.
+
+
+
+Storybook runs outside of your app.
+So you can develop UI components in isolation without worrying about app specific dependencies and requirements.
+
+## Getting Started
+
+```sh
+cd my-vue3-app
+npx -p @storybook/cli sb init
+```
+
+For more information visit: [storybook.js.org](https://storybook.js.org)
+
+---
+
+Storybook also comes with a lot of [addons](https://storybook.js.org/docs/vue3/configure/storybook-addons) and a great API to customize as you wish.
+You can also build a [static version](https://storybook.js.org/docs/vue3/workflows/publish-storybook) of your storybook and deploy it anywhere you want.
+
+## Extending the Vue application
+
+Storybook creates a [Vue 3 application](https://v3.vuejs.org/api/application-api.html#application-api) for your component preview, which can be imported as `import { app } from '@storybook/vue3'`.
+
+When using global custom components (`app.component`), directives (`app.directive`), extensions (`app.use`), or other application methods, you will need to configure those in the `./storybook/preview.js` file.
+
+For example:
+
+```js
+// .storybook/preview.js
+
+import { app } from '@storybook/vue3';
+
+app.use(MyPlugin);
+app.component('my-component', MyComponent);
+app.mixin({ /* My mixin */ });
+```
diff --git a/app/vue3/bin/build.js b/app/vue3/bin/build.js
new file mode 100755
index 00000000000..6ef07a26a9f
--- /dev/null
+++ b/app/vue3/bin/build.js
@@ -0,0 +1,4 @@
+#!/usr/bin/env node
+
+process.env.NODE_ENV = process.env.NODE_ENV || 'production';
+require('../dist/cjs/server/build');
diff --git a/app/vue3/bin/index.js b/app/vue3/bin/index.js
new file mode 100755
index 00000000000..8aab7bbceb2
--- /dev/null
+++ b/app/vue3/bin/index.js
@@ -0,0 +1,3 @@
+#!/usr/bin/env node
+
+require('../dist/cjs/server');
diff --git a/app/vue3/package.json b/app/vue3/package.json
new file mode 100644
index 00000000000..5cf68b53980
--- /dev/null
+++ b/app/vue3/package.json
@@ -0,0 +1,80 @@
+{
+ "name": "@storybook/vue3",
+ "version": "6.2.0-beta.14",
+ "description": "Storybook for Vue 3: Develop Vue 3 Components in isolation with Hot Reloading.",
+ "keywords": [
+ "storybook"
+ ],
+ "homepage": "https://github.com/storybookjs/storybook/tree/master/app/vue3",
+ "bugs": {
+ "url": "https://github.com/storybookjs/storybook/issues"
+ },
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/storybookjs/storybook.git",
+ "directory": "app/vue3"
+ },
+ "license": "MIT",
+ "main": "dist/cjs/client/index.js",
+ "module": "dist/esm/client/index.js",
+ "types": "dist/ts3.9/client/index.d.ts",
+ "typesVersions": {
+ "<3.8": {
+ "*": [
+ "dist/ts3.4/*"
+ ]
+ }
+ },
+ "bin": {
+ "build-storybook": "./bin/build.js",
+ "start-storybook": "./bin/index.js",
+ "storybook-server": "./bin/index.js"
+ },
+ "files": [
+ "bin/**/*",
+ "dist/**/*",
+ "README.md",
+ "*.js",
+ "*.d.ts"
+ ],
+ "scripts": {
+ "prepare": "node ../../scripts/prepare.js"
+ },
+ "dependencies": {
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/core": "6.2.0-beta.14",
+ "@storybook/core-common": "6.2.0-beta.14",
+ "@types/webpack-env": "^1.16.0",
+ "core-js": "^3.8.2",
+ "global": "^4.4.0",
+ "react": "16.14.0",
+ "react-dom": "16.14.0",
+ "read-pkg-up": "^7.0.1",
+ "regenerator-runtime": "^0.13.7",
+ "ts-dedent": "^2.0.0",
+ "ts-loader": "^8.0.14",
+ "vue-docgen-api": "^4.34.2",
+ "vue-docgen-loader": "^1.5.0",
+ "webpack": "4"
+ },
+ "devDependencies": {
+ "@types/node": "^14.14.20",
+ "@vue/compiler-sfc": "^3.0.0",
+ "vue": "^3.0.0",
+ "vue-loader": "^16.0.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "*",
+ "@vue/compiler-sfc": "^3.0.0",
+ "babel-loader": "^7.0.0 || ^8.0.0",
+ "vue": "^3.0.0",
+ "vue-loader": "^16.0.0"
+ },
+ "engines": {
+ "node": ">=10.13.0"
+ },
+ "publishConfig": {
+ "access": "public"
+ },
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
+}
diff --git a/app/vue3/src/client/index.ts b/app/vue3/src/client/index.ts
new file mode 100644
index 00000000000..57d0e7c7f30
--- /dev/null
+++ b/app/vue3/src/client/index.ts
@@ -0,0 +1,17 @@
+export {
+ storiesOf,
+ setAddon,
+ addDecorator,
+ addParameters,
+ configure,
+ getStorybook,
+ forceReRender,
+ raw,
+ app,
+} from './preview';
+
+export * from './preview/types-6-0';
+
+if (module && module.hot && module.hot.decline) {
+ module.hot.decline();
+}
diff --git a/app/vue3/src/client/preview/globals.ts b/app/vue3/src/client/preview/globals.ts
new file mode 100644
index 00000000000..8ccb267b341
--- /dev/null
+++ b/app/vue3/src/client/preview/globals.ts
@@ -0,0 +1,4 @@
+import { window } from 'global';
+
+window.STORYBOOK_REACT_CLASSES = {};
+window.STORYBOOK_ENV = 'vue3';
diff --git a/app/vue3/src/client/preview/index.ts b/app/vue3/src/client/preview/index.ts
new file mode 100644
index 00000000000..8dfacbc5713
--- /dev/null
+++ b/app/vue3/src/client/preview/index.ts
@@ -0,0 +1,117 @@
+import type { ConcreteComponent, Component, ComponentOptions, App } from 'vue';
+import { h } from 'vue';
+import { start } from '@storybook/core/client';
+import {
+ ClientStoryApi,
+ StoryFn,
+ DecoratorFunction,
+ StoryContext,
+ Loadable,
+} from '@storybook/addons';
+
+import './globals';
+import { IStorybookSection, StoryFnVueReturnType } from './types';
+
+import render, { storybookApp } from './render';
+
+/*
+ This normalizes a functional component into a render method in ComponentOptions.
+
+ The concept is taken from Vue 3's `defineComponent` but changed from creating a `setup`
+ method on the ComponentOptions so end-users don't need to specify a "thunk" as a decorator.
+ */
+function normalizeFunctionalComponent(options: ConcreteComponent): ComponentOptions {
+ return typeof options === 'function' ? { render: options, name: options.name } : options;
+}
+
+function prepare(story: StoryFnVueReturnType, innerStory?: ConcreteComponent): Component | null {
+ if (story == null) {
+ return null;
+ }
+
+ if (innerStory) {
+ return {
+ // Normalize so we can always spread an object
+ ...normalizeFunctionalComponent(story),
+ components: { story: innerStory },
+ };
+ }
+
+ return {
+ render() {
+ return h(story);
+ },
+ };
+}
+
+const defaultContext: StoryContext = {
+ id: 'unspecified',
+ name: 'unspecified',
+ kind: 'unspecified',
+ parameters: {},
+ args: {},
+ argTypes: {},
+ globals: {},
+};
+
+function decorateStory(
+ storyFn: StoryFn,
+ decorators: DecoratorFunction[]
+): StoryFn {
+ return decorators.reduce(
+ (decorated: StoryFn, decorator) => (
+ context: StoryContext = defaultContext
+ ) => {
+ let story;
+
+ const decoratedStory = decorator(
+ ({ parameters, ...innerContext }: StoryContext = {} as StoryContext) => {
+ story = decorated({ ...context, ...innerContext });
+ return story;
+ },
+ context
+ );
+
+ if (!story) {
+ story = decorated(context);
+ }
+
+ if (decoratedStory === story) {
+ return story;
+ }
+
+ return prepare(decoratedStory, story);
+ },
+ (context) => prepare(storyFn(context))
+ );
+}
+const framework = 'vue3';
+
+interface ClientApi extends ClientStoryApi {
+ setAddon(addon: any): void;
+ configure(loader: Loadable, module: NodeModule): void;
+ getStorybook(): IStorybookSection[];
+ clearDecorators(): void;
+ forceReRender(): void;
+ raw: () => any; // todo add type
+ load: (...args: any[]) => void;
+ app: App;
+}
+
+const api = start(render, { decorateStory });
+
+export const storiesOf: ClientApi['storiesOf'] = (kind, m) => {
+ return (api.clientApi.storiesOf(kind, m) as ReturnType).addParameters({
+ framework,
+ });
+};
+
+export const configure: ClientApi['configure'] = (...args) => api.configure(framework, ...args);
+export const { addDecorator } = api.clientApi;
+export const { addParameters } = api.clientApi;
+export const { clearDecorators } = api.clientApi;
+export const { setAddon } = api.clientApi;
+export const { forceReRender } = api;
+export const { getStorybook } = api.clientApi;
+export const { raw } = api.clientApi;
+export const app: ClientApi['app'] = storybookApp;
diff --git a/app/vue3/src/client/preview/render.ts b/app/vue3/src/client/preview/render.ts
new file mode 100644
index 00000000000..6a9cf7e1e64
--- /dev/null
+++ b/app/vue3/src/client/preview/render.ts
@@ -0,0 +1,56 @@
+import dedent from 'ts-dedent';
+import { createApp, h, shallowRef, ComponentPublicInstance } from 'vue';
+import { RenderContext, StoryFnVueReturnType } from './types';
+
+const activeStoryComponent = shallowRef(null);
+
+let root: ComponentPublicInstance | null = null;
+
+export const storybookApp = createApp({
+ // If an end-user calls `unmount` on the app, we need to clear our root variable
+ unmounted() {
+ root = null;
+ },
+
+ setup() {
+ return () => {
+ if (!activeStoryComponent.value)
+ throw new Error('No Vue 3 Story available. Was it set correctly?');
+ return h(activeStoryComponent.value);
+ };
+ },
+});
+
+export default function render({
+ storyFn,
+ kind,
+ name,
+ args,
+ showMain,
+ showError,
+ showException,
+ forceRender,
+}: RenderContext) {
+ storybookApp.config.errorHandler = showException;
+
+ const element: StoryFnVueReturnType = storyFn();
+
+ if (!element) {
+ showError({
+ title: `Expecting a Vue component from the story: "${name}" of "${kind}".`,
+ description: dedent`
+ Did you forget to return the Vue component from the story?
+ Use "() => ({ template: '' })" or "() => ({ components: MyComp, template: '' })" when defining the story.
+ `,
+ });
+ return;
+ }
+
+ showMain();
+
+ activeStoryComponent.value = element;
+
+ if (!root) {
+ root = storybookApp.mount('#root');
+ }
+}
diff --git a/app/vue3/src/client/preview/types-6-0.ts b/app/vue3/src/client/preview/types-6-0.ts
new file mode 100644
index 00000000000..fb12070637d
--- /dev/null
+++ b/app/vue3/src/client/preview/types-6-0.ts
@@ -0,0 +1,25 @@
+import { ConcreteComponent } from 'vue';
+import { Args as DefaultArgs, Annotations, BaseMeta, BaseStory } from '@storybook/addons';
+import { StoryFnVueReturnType } from './types';
+
+export type { Args, ArgTypes, Parameters, StoryContext } from '@storybook/addons';
+
+type VueComponent = ConcreteComponent;
+type VueReturnType = StoryFnVueReturnType;
+
+/**
+ * Metadata to configure the stories for a component.
+ *
+ * @see [Default export](https://storybook.js.org/docs/formats/component-story-format/#default-export)
+ */
+export type Meta = BaseMeta & Annotations;
+
+/**
+ * Story function that represents a component example.
+ *
+ * @see [Named Story exports](https://storybook.js.org/docs/formats/component-story-format/#named-story-exports)
+ */
+export type Story = BaseStory &
+ Annotations;
+
+export type Decorators = Story['decorators'];
diff --git a/app/vue3/src/client/preview/types.ts b/app/vue3/src/client/preview/types.ts
new file mode 100644
index 00000000000..b25133f953e
--- /dev/null
+++ b/app/vue3/src/client/preview/types.ts
@@ -0,0 +1,20 @@
+import { ConcreteComponent } from 'vue';
+
+export type { RenderContext } from '@storybook/core';
+
+export interface ShowErrorArgs {
+ title: string;
+ description: string;
+}
+
+export type StoryFnVueReturnType = ConcreteComponent;
+
+export interface IStorybookStory {
+ name: string;
+ render: () => any;
+}
+
+export interface IStorybookSection {
+ kind: string;
+ stories: IStorybookStory[];
+}
diff --git a/app/vue3/src/server/build.ts b/app/vue3/src/server/build.ts
new file mode 100755
index 00000000000..d8abf06a439
--- /dev/null
+++ b/app/vue3/src/server/build.ts
@@ -0,0 +1,4 @@
+import { buildStatic } from '@storybook/core/server';
+import options from './options';
+
+buildStatic(options);
diff --git a/app/vue3/src/server/framework-preset-vue3.ts b/app/vue3/src/server/framework-preset-vue3.ts
new file mode 100644
index 00000000000..5061efe35be
--- /dev/null
+++ b/app/vue3/src/server/framework-preset-vue3.ts
@@ -0,0 +1,47 @@
+import { VueLoaderPlugin } from 'vue-loader';
+import { Configuration, DefinePlugin } from 'webpack';
+
+export function webpack(config: Configuration): Configuration {
+ return {
+ ...config,
+ plugins: [
+ ...config.plugins,
+ new VueLoaderPlugin(),
+ new DefinePlugin({
+ __VUE_OPTIONS_API__: JSON.stringify(true),
+ __VUE_PROD_DEVTOOLS__: JSON.stringify(true),
+ }),
+ ],
+ module: {
+ ...config.module,
+ rules: [
+ ...config.module.rules,
+ {
+ test: /\.vue$/,
+ loader: require.resolve('vue-loader'),
+ options: {},
+ },
+ {
+ test: /\.tsx?$/,
+ use: [
+ {
+ loader: require.resolve('ts-loader'),
+ options: {
+ transpileOnly: true,
+ appendTsSuffixTo: [/\.vue$/],
+ },
+ },
+ ],
+ },
+ ],
+ },
+ resolve: {
+ ...config.resolve,
+ extensions: [...config.resolve.extensions, '.vue'],
+ alias: {
+ ...config.resolve.alias,
+ vue$: require.resolve('vue/dist/vue.esm-bundler.js'),
+ },
+ },
+ };
+}
diff --git a/app/vue3/src/server/index.ts b/app/vue3/src/server/index.ts
new file mode 100755
index 00000000000..774d96025a8
--- /dev/null
+++ b/app/vue3/src/server/index.ts
@@ -0,0 +1,4 @@
+import { buildDev } from '@storybook/core/server';
+import options from './options';
+
+buildDev(options);
diff --git a/app/vue3/src/server/options.ts b/app/vue3/src/server/options.ts
new file mode 100644
index 00000000000..539771517c8
--- /dev/null
+++ b/app/vue3/src/server/options.ts
@@ -0,0 +1,8 @@
+import { sync } from 'read-pkg-up';
+import { LoadOptions } from '@storybook/core-common';
+
+export default {
+ packageJson: sync({ cwd: __dirname }).packageJson,
+ framework: 'vue3',
+ frameworkPresets: [require.resolve('./framework-preset-vue3')],
+} as LoadOptions;
diff --git a/app/vue3/src/typings.d.ts b/app/vue3/src/typings.d.ts
new file mode 100644
index 00000000000..64a098b2547
--- /dev/null
+++ b/app/vue3/src/typings.d.ts
@@ -0,0 +1,7 @@
+declare module 'global';
+// todo check for correct types
+declare module 'webpack/lib/RuleSet';
+
+declare module 'vue-loader' {
+ export const VueLoaderPlugin
+}
diff --git a/app/vue3/standalone.js b/app/vue3/standalone.js
new file mode 100644
index 00000000000..d11a82f7995
--- /dev/null
+++ b/app/vue3/standalone.js
@@ -0,0 +1,8 @@
+const build = require('@storybook/core/standalone');
+const frameworkOptions = require('./dist/cjs/server/options').default;
+
+async function buildStandalone(options) {
+ return build(options, frameworkOptions);
+}
+
+module.exports = buildStandalone;
diff --git a/app/vue3/tsconfig.json b/app/vue3/tsconfig.json
new file mode 100644
index 00000000000..08bdb7764f7
--- /dev/null
+++ b/app/vue3/tsconfig.json
@@ -0,0 +1,14 @@
+{
+ "extends": "../../tsconfig.json",
+ "compilerOptions": {
+ "rootDir": "./src",
+ "types": ["webpack-env", "node"],
+ "resolveJsonModule": true
+ },
+ "include": [
+ "src/**/*"
+ ],
+ "exclude": [
+ "src/**/*.test.*"
+ ]
+}
diff --git a/app/vue3/types-6-0.d.ts b/app/vue3/types-6-0.d.ts
new file mode 100644
index 00000000000..b5946b39a8d
--- /dev/null
+++ b/app/vue3/types-6-0.d.ts
@@ -0,0 +1 @@
+export * from './dist/ts3.9/client/preview/types-6-0.d';
diff --git a/app/web-components/README.md b/app/web-components/README.md
index 842a6d988af..7baf1c3c0af 100644
--- a/app/web-components/README.md
+++ b/app/web-components/README.md
@@ -52,9 +52,9 @@ if (module.hot) {
# Setup es6/7 dependencies
-By default storybook only works with precompiled es5 code but as most web components themselves and their libs are distributed as es7 you will need to manually mark those packages as "needs transpilation".
+By default storybook only works with precompiled ES5 code but as most web components themselves and their libs are distributed as ES2017 you will need to manually mark those packages as "needs transpilation".
-For example if you have a library called `my-library` which is in es7 then you can add it like so
+For example if you have a library called `my-library` which is in ES2017 then you can add it like so
```js
// .storybook/main.js
diff --git a/app/web-components/bin/build.js b/app/web-components/bin/build.js
index 26142ec0af2..6ef07a26a9f 100755
--- a/app/web-components/bin/build.js
+++ b/app/web-components/bin/build.js
@@ -1,4 +1,4 @@
#!/usr/bin/env node
process.env.NODE_ENV = process.env.NODE_ENV || 'production';
-require('../dist/server/build');
+require('../dist/cjs/server/build');
diff --git a/app/web-components/bin/index.js b/app/web-components/bin/index.js
index 2e96258ce63..8aab7bbceb2 100755
--- a/app/web-components/bin/index.js
+++ b/app/web-components/bin/index.js
@@ -1,3 +1,3 @@
#!/usr/bin/env node
-require('../dist/server');
+require('../dist/cjs/server');
diff --git a/app/web-components/package.json b/app/web-components/package.json
index a0ccb09ebd5..f82fc02c0d9 100644
--- a/app/web-components/package.json
+++ b/app/web-components/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/web-components",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "Storybook for web-components: View web components snippets in isolation with Hot Reloading.",
"keywords": [
"lit-html",
@@ -17,12 +17,13 @@
"directory": "app/web-components"
},
"license": "MIT",
- "main": "dist/client/index.js",
- "types": "dist/client/index.d.ts",
+ "main": "dist/cjs/client/index.js",
+ "module": "dist/esm/client/index.js",
+ "types": "dist/ts3.9/client/index.d.ts",
"typesVersions": {
"<3.8": {
"*": [
- "ts3.4/*"
+ "dist/ts3.4/*"
]
}
},
@@ -36,8 +37,7 @@
"dist/**/*",
"README.md",
"*.js",
- "*.d.ts",
- "ts3.4/**/*"
+ "*.d.ts"
],
"scripts": {
"prepare": "node ../../scripts/prepare.js"
@@ -45,21 +45,23 @@
"dependencies": {
"@babel/plugin-syntax-dynamic-import": "^7.8.3",
"@babel/plugin-syntax-import-meta": "^7.10.4",
- "@babel/preset-env": "^7.12.1",
- "@storybook/addons": "6.2.0-alpha.5",
- "@storybook/client-api": "6.2.0-alpha.5",
- "@storybook/core": "6.2.0-alpha.5",
- "@types/webpack-env": "^1.15.3",
+ "@babel/preset-env": "^7.12.11",
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/client-api": "6.2.0-beta.14",
+ "@storybook/core": "6.2.0-beta.14",
+ "@storybook/core-common": "6.2.0-beta.14",
+ "@types/webpack-env": "^1.16.0",
"babel-plugin-bundled-import-meta": "^0.3.1",
- "core-js": "^3.0.1",
- "global": "^4.3.2",
- "react": "16.13.1",
- "react-dom": "16.13.1",
+ "core-js": "^3.8.2",
+ "global": "^4.4.0",
+ "react": "16.14.0",
+ "react-dom": "16.14.0",
+ "read-pkg-up": "^7.0.1",
"regenerator-runtime": "^0.13.7",
"ts-dedent": "^2.0.0"
},
"devDependencies": {
- "lit-html": "^1.0.0"
+ "lit-html": "^1.3.0"
},
"peerDependencies": {
"@babel/core": "*",
@@ -67,10 +69,10 @@
"lit-html": "^1.0.0"
},
"engines": {
- "node": ">=8.0.0"
+ "node": ">=10.13.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/app/web-components/src/client/index.ts b/app/web-components/src/client/index.ts
index 0ab72efcf90..b2379641137 100644
--- a/app/web-components/src/client/index.ts
+++ b/app/web-components/src/client/index.ts
@@ -16,6 +16,8 @@ export {
isValidMetaData,
} from './customElements';
+export * from './preview/types-6-0';
+
// TODO: disable HMR and do full page loads because of customElements.define
if (module && module.hot && module.hot.decline) {
module.hot.decline();
diff --git a/app/web-components/src/client/preview/index.ts b/app/web-components/src/client/preview/index.ts
index b62b7446ab1..7193f399353 100644
--- a/app/web-components/src/client/preview/index.ts
+++ b/app/web-components/src/client/preview/index.ts
@@ -26,8 +26,10 @@ export const storiesOf: ClientApi['storiesOf'] = (kind, m) => {
};
export const configure: ClientApi['configure'] = (...args) => api.configure(framework, ...args);
-export const addDecorator: ClientApi['addDecorator'] = api.clientApi.addDecorator;
-export const addParameters: ClientApi['addParameters'] = api.clientApi.addParameters;
+export const addDecorator: ClientApi['addDecorator'] = api.clientApi
+ .addDecorator as ClientApi['addDecorator'];
+export const addParameters: ClientApi['addParameters'] = api.clientApi
+ .addParameters as ClientApi['addParameters'];
export const clearDecorators: ClientApi['clearDecorators'] = api.clientApi.clearDecorators;
export const setAddon: ClientApi['setAddon'] = api.clientApi.setAddon;
export const forceReRender: ClientApi['forceReRender'] = api.forceReRender;
diff --git a/app/web-components/src/client/preview/types-6-0.ts b/app/web-components/src/client/preview/types-6-0.ts
new file mode 100644
index 00000000000..c7077eaea1b
--- /dev/null
+++ b/app/web-components/src/client/preview/types-6-0.ts
@@ -0,0 +1,19 @@
+import { Args as DefaultArgs, Annotations, BaseMeta, BaseStory } from '@storybook/addons';
+import { StoryFnHtmlReturnType } from './types';
+
+export type { Args, ArgTypes, Parameters, StoryContext } from '@storybook/addons';
+
+/**
+ * Metadata to configure the stories for a component.
+ *
+ * @see [Default export](https://storybook.js.org/docs/formats/component-story-format/#default-export)
+ */
+export type Meta = BaseMeta & Annotations;
+
+/**
+ * Story function that represents a component example.
+ *
+ * @see [Named Story exports](https://storybook.js.org/docs/formats/component-story-format/#named-story-exports)
+ */
+export type Story = BaseStory &
+ Annotations;
diff --git a/app/web-components/src/client/preview/types.ts b/app/web-components/src/client/preview/types.ts
index 59574c42812..d3380b6686d 100644
--- a/app/web-components/src/client/preview/types.ts
+++ b/app/web-components/src/client/preview/types.ts
@@ -1,7 +1,8 @@
// eslint-disable-next-line import/no-extraneous-dependencies
import { TemplateResult, SVGTemplateResult } from 'lit-element';
-export { RenderContext } from '@storybook/core';
+export type { RenderContext } from '@storybook/core';
+export { Args, ArgTypes, Parameters, StoryContext } from '@storybook/addons';
export type StoryFnHtmlReturnType = string | Node | TemplateResult | SVGTemplateResult;
diff --git a/app/web-components/src/server/framework-preset-web-components.ts b/app/web-components/src/server/framework-preset-web-components.ts
index a63675e4650..8e14b519def 100644
--- a/app/web-components/src/server/framework-preset-web-components.ts
+++ b/app/web-components/src/server/framework-preset-web-components.ts
@@ -2,45 +2,38 @@
import { Configuration } from 'webpack';
export function webpack(config: Configuration) {
- return {
- ...config,
- module: {
- ...config.module,
- rules: [
- ...config.module.rules,
- {
- test: [
- new RegExp(`src(.*)\\.js$`),
- new RegExp(`packages(\\/|\\\\)*(\\/|\\\\)src(\\/|\\\\)(.*)\\.js$`),
- new RegExp(`node_modules(\\/|\\\\)lit-html(.*)\\.js$`),
- new RegExp(`node_modules(\\/|\\\\)lit-element(.*)\\.js$`),
- new RegExp(`node_modules(\\/|\\\\)@open-wc(.*)\\.js$`),
- new RegExp(`node_modules(\\/|\\\\)@polymer(.*)\\.js$`),
- new RegExp(`node_modules(\\/|\\\\)@vaadin(.*)\\.js$`),
- ],
- use: {
- loader: require.resolve('babel-loader'),
- options: {
- plugins: [
- require.resolve('@babel/plugin-syntax-dynamic-import'),
- require.resolve('@babel/plugin-syntax-import-meta'),
- // webpack does not support import.meta.url yet, so we rewrite them in babel
- [require.resolve('babel-plugin-bundled-import-meta'), { importStyle: 'baseURI' }],
- ],
- presets: [
- [
- require.resolve('@babel/preset-env'),
- {
- useBuiltIns: 'entry',
- corejs: 3,
- },
- ],
- ],
- babelrc: false,
+ config.module.rules.push({
+ test: [
+ new RegExp(`src(.*)\\.js$`),
+ new RegExp(`packages(\\/|\\\\)*(\\/|\\\\)src(\\/|\\\\)(.*)\\.js$`),
+ new RegExp(`node_modules(\\/|\\\\)lit-html(.*)\\.js$`),
+ new RegExp(`node_modules(\\/|\\\\)lit-element(.*)\\.js$`),
+ new RegExp(`node_modules(\\/|\\\\)@open-wc(.*)\\.js$`),
+ new RegExp(`node_modules(\\/|\\\\)@polymer(.*)\\.js$`),
+ new RegExp(`node_modules(\\/|\\\\)@vaadin(.*)\\.js$`),
+ ],
+ use: {
+ loader: require.resolve('babel-loader'),
+ options: {
+ plugins: [
+ require.resolve('@babel/plugin-syntax-dynamic-import'),
+ require.resolve('@babel/plugin-syntax-import-meta'),
+ // webpack does not support import.meta.url yet, so we rewrite them in babel
+ [require.resolve('babel-plugin-bundled-import-meta'), { importStyle: 'baseURI' }],
+ ],
+ presets: [
+ [
+ require.resolve('@babel/preset-env'),
+ {
+ useBuiltIns: 'entry',
+ corejs: 3,
},
- },
- },
- ],
+ ],
+ ],
+ babelrc: false,
+ },
},
- };
+ });
+
+ return config;
}
diff --git a/app/web-components/src/server/options.ts b/app/web-components/src/server/options.ts
index ea9b2ee7aba..0a63c8168cb 100644
--- a/app/web-components/src/server/options.ts
+++ b/app/web-components/src/server/options.ts
@@ -1,7 +1,8 @@
-const packageJson = require('../../package.json');
+import { sync } from 'read-pkg-up';
+import { LoadOptions } from '@storybook/core-common';
export default {
- packageJson,
+ packageJson: sync({ cwd: __dirname }).packageJson,
framework: 'web-components',
- frameworkPresets: [require.resolve('./framework-preset-web-components.js')],
-};
+ frameworkPresets: [require.resolve('./framework-preset-web-components')],
+} as LoadOptions;
diff --git a/app/web-components/src/typings.d.ts b/app/web-components/src/typings.d.ts
index 690e93343de..d8f7c6f660a 100644
--- a/app/web-components/src/typings.d.ts
+++ b/app/web-components/src/typings.d.ts
@@ -1,4 +1,3 @@
-declare module '@storybook/core/*';
declare module 'global';
// will be provided by the webpack define plugin
diff --git a/app/web-components/standalone.js b/app/web-components/standalone.js
index 1b1febe0d3b..d11a82f7995 100644
--- a/app/web-components/standalone.js
+++ b/app/web-components/standalone.js
@@ -1,5 +1,5 @@
const build = require('@storybook/core/standalone');
-const frameworkOptions = require('./dist/server/options').default;
+const frameworkOptions = require('./dist/cjs/server/options').default;
async function buildStandalone(options) {
return build(options, frameworkOptions);
diff --git a/app/web-components/tsconfig.json b/app/web-components/tsconfig.json
index 82ce44329cc..13f32ad6309 100644
--- a/app/web-components/tsconfig.json
+++ b/app/web-components/tsconfig.json
@@ -2,7 +2,7 @@
"extends": "../../tsconfig.json",
"compilerOptions": {
"rootDir": "./src",
- "types": ["webpack-env"]
+ "types": ["webpack-env", "node"]
},
"include": ["src/**/*"],
"exclude": ["src/__tests__/**/*"]
diff --git a/app/web-components/types-6-0.d.ts b/app/web-components/types-6-0.d.ts
new file mode 100644
index 00000000000..b5946b39a8d
--- /dev/null
+++ b/app/web-components/types-6-0.d.ts
@@ -0,0 +1 @@
+export * from './dist/ts3.9/client/preview/types-6-0.d';
diff --git a/dev-kits/addon-decorator/package.json b/dev-kits/addon-decorator/package.json
index 1d39680114c..1ea6ce4dfc5 100644
--- a/dev-kits/addon-decorator/package.json
+++ b/dev-kits/addon-decorator/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/addon-decorator",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "decorator addon for storybook",
"keywords": [
"addon",
@@ -18,26 +18,20 @@
"directory": "addons/actions"
},
"license": "MIT",
- "main": "dist/index.js",
- "types": "dist/index.d.ts",
- "typesVersions": {
- "<3.8": {
- "*": [
- "ts3.4/*"
- ]
- }
- },
+ "main": "dist/cjs/index.js",
+ "module": "dist/esm/index.js",
+ "types": "dist/ts3.9/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "6.2.0-alpha.5",
- "@storybook/client-api": "6.2.0-alpha.5",
- "core-js": "^3.0.1",
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/client-api": "6.2.0-beta.14",
+ "core-js": "^3.8.2",
"global": "^4.4.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/dev-kits/addon-parameter/package.json b/dev-kits/addon-parameter/package.json
index 444af683ddf..dfd5fbf18d3 100644
--- a/dev-kits/addon-parameter/package.json
+++ b/dev-kits/addon-parameter/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/addon-parameter",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "parameter addon for storybook",
"keywords": [
"addon",
@@ -18,32 +18,26 @@
"directory": "dev-kit/addon-parameter"
},
"license": "MIT",
- "main": "dist/index.js",
- "types": "dist/index.d.ts",
- "typesVersions": {
- "<3.8": {
- "*": [
- "ts3.4/*"
- ]
- }
- },
+ "main": "dist/cjs/index.js",
+ "module": "dist/esm/index.js",
+ "types": "dist/ts3.9/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "6.2.0-alpha.5",
- "@storybook/api": "6.2.0-alpha.5",
- "@storybook/client-logger": "6.2.0-alpha.5",
- "@storybook/components": "6.2.0-alpha.5",
- "@storybook/core-events": "6.2.0-alpha.5",
- "@storybook/theming": "6.2.0-alpha.5",
- "core-js": "^3.0.1",
- "global": "^4.3.2",
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/api": "6.2.0-beta.14",
+ "@storybook/client-logger": "6.2.0-beta.14",
+ "@storybook/components": "6.2.0-beta.14",
+ "@storybook/core-events": "6.2.0-beta.14",
+ "@storybook/theming": "6.2.0-beta.14",
+ "core-js": "^3.8.2",
+ "global": "^4.4.0",
"react": "^16.8.0 || ^17.0.0",
"ts-dedent": "^2.0.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/dev-kits/addon-preview-wrapper/package.json b/dev-kits/addon-preview-wrapper/package.json
index b8c0a85f1b8..5d422e69a77 100644
--- a/dev-kits/addon-preview-wrapper/package.json
+++ b/dev-kits/addon-preview-wrapper/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/addon-preview-wrapper",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "preview wrapper addon for storybook",
"keywords": [
"addon",
@@ -18,24 +18,18 @@
"directory": "dev-kit/addon-preview-wrappe"
},
"license": "MIT",
- "main": "dist/index.js",
- "types": "dist/index.d.ts",
- "typesVersions": {
- "<3.8": {
- "*": [
- "ts3.4/*"
- ]
- }
- },
+ "main": "dist/cjs/index.js",
+ "module": "dist/esm/index.js",
+ "types": "dist/ts3.9/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "6.2.0-alpha.5",
+ "@storybook/addons": "6.2.0-beta.14",
"react": "^16.8.0 || ^17.0.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/dev-kits/addon-roundtrip/package.json b/dev-kits/addon-roundtrip/package.json
index ae2e2c79c55..0eb59493765 100644
--- a/dev-kits/addon-roundtrip/package.json
+++ b/dev-kits/addon-roundtrip/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/addon-roundtrip",
- "version": "6.2.0-alpha.5",
+ "version": "6.2.0-beta.14",
"description": "roundtrip addon for storybook",
"keywords": [
"addon",
@@ -18,33 +18,27 @@
"directory": "dev-kit/addon-roundtrip"
},
"license": "MIT",
- "main": "dist/index.js",
- "types": "dist/index.d.ts",
- "typesVersions": {
- "<3.8": {
- "*": [
- "ts3.4/*"
- ]
- }
- },
+ "main": "dist/cjs/index.js",
+ "module": "dist/esm/index.js",
+ "types": "dist/ts3.9/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "6.2.0-alpha.5",
- "@storybook/api": "6.2.0-alpha.5",
- "@storybook/client-api": "6.2.0-alpha.5",
- "@storybook/client-logger": "6.2.0-alpha.5",
- "@storybook/components": "6.2.0-alpha.5",
- "@storybook/core-events": "6.2.0-alpha.5",
- "@storybook/theming": "6.2.0-alpha.5",
- "core-js": "^3.0.1",
- "global": "^4.3.2",
+ "@storybook/addons": "6.2.0-beta.14",
+ "@storybook/api": "6.2.0-beta.14",
+ "@storybook/client-api": "6.2.0-beta.14",
+ "@storybook/client-logger": "6.2.0-beta.14",
+ "@storybook/components": "6.2.0-beta.14",
+ "@storybook/core-events": "6.2.0-beta.14",
+ "@storybook/theming": "6.2.0-beta.14",
+ "core-js": "^3.8.2",
+ "global": "^4.4.0",
"react": "^16.8.0 || ^17.0.0",
"ts-dedent": "^2.0.0"
},
"publishConfig": {
"access": "public"
},
- "gitHead": "850a0d8b98f2ddab1cd4d652e9beac11e21a369c"
+ "gitHead": "45957443c70b308214c9eb52cb048e7e488153c6"
}
diff --git a/docs/addons/addon-catalog.md b/docs/addons/addon-catalog.md
new file mode 100644
index 00000000000..fc02fdf06cb
--- /dev/null
+++ b/docs/addons/addon-catalog.md
@@ -0,0 +1,83 @@
+---
+title: 'Add to the addon catalog'
+---
+
+Storybook addons are listed in the [catalog](/addons) and distributed via npm. The catalog is populated by querying npm's registry for Storybook-specific metadata in `package.json`.
+
+Add your addon to the catalog by publishing an npm package that follows these requirements:
+
+- `package.json` with [module information](writing-addons.md#get-started) and [addon metadata](#addon-metadata)
+- `README.md` file with installation and configuration instructions
+- `/dist` directory containing transpiled ES5 code
+- `preset.js` file written as an ES5 module at the root level
+
+
+
+Get a refresher on how to [write a Storybook addon](./writing-addons.md).
+
+
+
+## Addon metadata
+
+We rely on metadata to organize your addon in the catalog. You must add the storybook-addons as the first keyword, followed by your addon's category. Additional keywords will be used in search and as tags.
+
+| Property | Description | Example |
+| ------------- | -------------------------------------- | ------------------------------------------------------------------------- |
+| `name` | Addon package name | storybook-addon-outline |
+| `description` | Addon description | Outline all elements with CSS to help with layout placement and alignment |
+| `author` | Name of the author | winkerVSbecks |
+| `keywords` | List of keywords to describe the addon | `["storybook-addons","style","debug"]` |
+| `repository` | Addon repository | `{"type": "git","url": "https://github.com/someone/my-addon" }` |
+
+Customize your addon's appearance by adding the `storybook` property with the following fields.
+
+| Property | Description | Example |
+| ----------------------- | --------------------------------- | ------------------------------------- |
+| `displayName` | Display name | Outline |
+| `icon` | Link to custom icon for the addon | https://yoursite.com/outline-icon.png |
+| `unsupportedFrameworks` | List of unsupported frameworks | `["vue"]` |
+| `supportedFrameworks` | List of supported frameworks | `["react", "angular"]` |
+
+
+Use the table below as a reference when filling in the values for both the `supportedFrameworks` and `unsupportedFrameworks` metadata properties.
+
+| react | vue | angular |
+|----------------|------------|--------------|
+| web-components | ember | html |
+| mithril | marko | svelte |
+| riot | preact | rax |
+| aurelia | marionette | react-native |
+
+
+Note: Make sure to copy each item exactly as listed so that we can properly index your addon in our catalog.
+
+
+```json
+{
+ // package.json
+
+ "name": "storybook-addon-outline",
+ "version": "1.0.0",
+ "description": "Outline all elements with CSS to help with layout placement and alignment",
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/chromaui/storybook-outline"
+ },
+ "author": "winkerVSbecks",
+ "keywords": ["storybook-addons", "style", "debug", "layout", "css"],
+ "storybook": {
+ "displayName": "Outline",
+ "unsupportedFrameworks": ["Vue"],
+ "supportedFrameworks": ["React", "Angular"],
+ "icon": "https://yoursite.com/outline-icon.png"
+ }
+}
+```
+
+The `package.json` above appears like below in the catalog. See an example of a production package.json [here](https://github.com/chromaui/storybook-outline/blob/main/package.json).
+
+
+
+#### How long does it take for my addon to show up in the catalog?
+
+Once you publish the addon, it will appear in the catalog. There may be a delay between the time you publish your addon and when it's listed in the catalog. If your addon doesn't show up within 24 hours, [open an issue](https://github.com/storybookjs/frontpage/issues).
diff --git a/docs/addons/addon-display.png b/docs/addons/addon-display.png
new file mode 100644
index 00000000000..042025f5f35
Binary files /dev/null and b/docs/addons/addon-display.png differ
diff --git a/docs/addons/addon-knowledge-base.md b/docs/addons/addon-knowledge-base.md
index e3ba7c34f1f..fd4ff6e0a9b 100644
--- a/docs/addons/addon-knowledge-base.md
+++ b/docs/addons/addon-knowledge-base.md
@@ -26,7 +26,7 @@ Then when adding a story, you can pass a disabled parameter.
diff --git a/docs/addons/addon-types.md b/docs/addons/addon-types.md
index 21f0fb3f3ee..ccf5d774bb7 100644
--- a/docs/addons/addon-types.md
+++ b/docs/addons/addon-types.md
@@ -45,6 +45,11 @@ Use this boilerplate code to add a new `button` to Storybook's Toolbar:
+
+
+The icon element used in the example loads the icons from the @storybook/components package. See [here](../workflows/faq.md#what-icons-are-available-for-my-toolbar-or-my-addon) the list of available icons that you can use.
+
+
### Tabs
diff --git a/docs/addons/addons-api.md b/docs/addons/addons-api.md
index b24c4eca9c3..4167551de22 100644
--- a/docs/addons/addons-api.md
+++ b/docs/addons/addons-api.md
@@ -201,6 +201,8 @@ Let's say you've got a story like this:
diff --git a/docs/addons/install-addons.md b/docs/addons/install-addons.md
index d62a10ed067..5d52d4e3849 100644
--- a/docs/addons/install-addons.md
+++ b/docs/addons/install-addons.md
@@ -41,7 +41,7 @@ Storybook preset addons are grouped collections of specific `babel`,`webpack` an
For example, to use SCSS styling, run the following command to install the addon and the required dependencies:
```sh
-yarn add -D @storybook/preset-scss css-loader sass-loader style-loader
+yarn add -D @storybook/preset-scss css-loader sass sass-loader style-loader
```
Next, update [`.storybook/main.js`](../configure/overview.md#configure-story-rendering) to the following:
diff --git a/docs/addons/storybook-toolbar.png b/docs/addons/storybook-toolbar.png
index 48cdcc8825f..1a83a7fa412 100644
Binary files a/docs/addons/storybook-toolbar.png and b/docs/addons/storybook-toolbar.png differ
diff --git a/docs/addons/writing-addons.md b/docs/addons/writing-addons.md
index 78d233a7860..cdccbd4998a 100644
--- a/docs/addons/writing-addons.md
+++ b/docs/addons/writing-addons.md
@@ -183,6 +183,8 @@ When Storybook was initialized it provided a small set of examples stories. Chan
@@ -217,7 +219,7 @@ This auto-registers the addon without any additional configuration from the user
Now that you've seen how to create a bare-bones addon, let's see how to share it with the community. Before we begin, make sure your addon meets the following requirements:
-- `package.json file` with metadata about the addon
+- `package.json` file with metadata about the addon
- Peer dependencies of `react` and `@storybook/addons`
- `preset.js` file at the root level written as an ES5 module
- `src` directory containing the ES6 addon code
@@ -225,13 +227,15 @@ Now that you've seen how to create a bare-bones addon, let's see how to share it
- [GitHub](https://github.com/) account to host your code
- [NPM](https://www.npmjs.com/) account to publish the addon
-For example, check out [storybook-addon-outline](https://www.npmjs.com/package/storybook-addon-outline) to see a project that meets these requirements.
+Reference the [storybook-addon-outline](https://www.npmjs.com/package/storybook-addon-outline) to see a project that meets these requirements.
+
+Learn how to [add to the addon catalog](./addon-catalog.md).
### More guides and tutorials
In the previous example, we introduced the structure of an addon, but barely scratched the surface of what addons can do.
-To dive deeper we recommend [Learn Storybookโs โcreating addonsโ](https://www.learnstorybook.com/intro-to-storybook/react/en/creating-addons/) tutorial. Itโs an excellent walkthrough that covers the same ground as the above introduction, but goes further and leads you through the full process of creating a realistic addon.
+To dive deeper we recommend Storybook's [creating an addon](https://storybook.js.org/tutorials/create-an-addon/) tutorial. Itโs an excellent walkthrough that covers the same ground as the above introduction, but goes further and leads you through the full process of creating a realistic addon.
[How to build a Storybook addon](https://www.chromatic.com/blog/how-to-build-a-storybook-addon/) shows you how to create a standalone addon in great detail.
diff --git a/docs/addons/writing-presets.md b/docs/addons/writing-presets.md
index f7cd12d2473..bde2651b6aa 100644
--- a/docs/addons/writing-presets.md
+++ b/docs/addons/writing-presets.md
@@ -177,6 +177,24 @@ If it doesn't exist yet, create a file `.storybook/main.js`:
+### Preview/Manager templates
+
+It's also possible to to programmatically modify the preview head/body HTML using a preset, similar to the way `preview-head.html`/`preview-body.html` can be used to [configure story rendering](../configure/story-rendering.md). The `previewHead` and `previewBody` functions accept a string, which is the existing head/body, and return a modified string.
+
+For example, the following snippet adds a style tag to the preview head programatically:
+
+
+
+
+
+
+
+Similarly, the `managerHead` can be used to modify the surrounding "manager" UI, analogous to `manager-head.html`.
+
## Sharing advanced configuration
Change your `main.js` file to:
diff --git a/docs/api/argtypes.md b/docs/api/argtypes.md
index 5112246008c..59c8bc41fb6 100644
--- a/docs/api/argtypes.md
+++ b/docs/api/argtypes.md
@@ -85,6 +85,12 @@ These values--description, table.type, and controls.type--get merged over the de
In particular, this would render a row with a modified description, a type display with a dropdown that shows the detail, and no control.
+
+
+As it happens with other properties such as `args`, `argTypes` can be overridden in a single story.
+
+
+
#### Using argTypes in addons
If you want to access the argTypes of the current component inside an addon, you can use the `useArgTypes` hook from the `@storybook/api` package:
diff --git a/docs/api/cli-options.md b/docs/api/cli-options.md
index 669eaebc161..81278e78762 100644
--- a/docs/api/cli-options.md
+++ b/docs/api/cli-options.md
@@ -12,24 +12,30 @@ Pass these commands the following options to alter Storybook's behavior.
Usage: start-storybook [options]
```
-| Options | Description | Example |
-| ------------------------------ | :--------------------------------------------------------------------------------------------------------------------------------------------- | :---------------------------------------------- |
-| --help | Output usage information | `start-storybook --help` |
-| -V, --version | Output the version number | `start-storybook -V` |
-| -p, --port [number] | Port to run Storybook | `start-storybook -p 9009` |
-| -h, --host [string] | Host to run Storybook | `start-storybook -h http://my-host.com` |
-| -s, --static-dir `` | Directory where to load static files from, comma-separated list | `start-storybook -s public` |
-| -c, --config-dir [dir-name] | Directory where to load Storybook configurations from | `start-storybook -c .storybook` |
-| --https | Serve Storybook over HTTPS. Note: You must provide your own certificate information. | `start-storybook --https` |
-| --ssl-ca `` | Provide an SSL certificate authority. (Optional with --https, required if using a self-signed certificate) | `start-storybook --ssl-ca my-certificate` |
-| --ssl-cert `` | Provide an SSL certificate. (Required with --https) | `start-storybook --ssl-cert my-ssl-certificate` |
-| --ssl-key `` | Provide an SSL key. (Required with --https) | `start-storybook --ssl-key my-ssl-key` |
-| --smoke-test | Exit after successful start | `start-storybook --smoke-test` |
-| --ci | CI mode (skip interactive prompts, don't open browser) | `start-storybook --ci` |
-| --quiet | Suppress verbose build output | `start-storybook --quiet` |
-| --no-dll | Do not use dll reference (no-op) | `start-storybook --no-dll` |
-| --debug-webpack | Display final webpack configurations for debugging purposes | `start-storybook --debug-webpack` |
-| --docs | Starts Storybook in documentation mode. Learn more about it in [here](../writing-docs/build-documentation.md#preview-storybooks-documentation) | `start-storybook --docs` |
+| Options | Description | Example |
+| ---------------------------------- | :--------------------------------------------------------------------------------------------------------------------------------------------- | :-------------------------------------------------------- |
+| --help | Output usage information | `start-storybook --help` |
+| -V, --version | Output the version number | `start-storybook -V` |
+| -p, --port [number] | Port to run Storybook | `start-storybook -p 9009` |
+| -h, --host [string] | Host to run Storybook | `start-storybook -h http://my-host.com` |
+| -s, --static-dir `` | Directory where to load static files from, comma-separated list | `start-storybook -s public` |
+| -c, --config-dir [dir-name] | Directory where to load Storybook configurations from | `start-storybook -c .storybook` |
+| --https | Serve Storybook over HTTPS. Note: You must provide your own certificate information. | `start-storybook --https` |
+| --ssl-ca `` | Provide an SSL certificate authority. (Optional with --https, required if using a self-signed certificate) | `start-storybook --ssl-ca my-certificate` |
+| --ssl-cert `` | Provide an SSL certificate. (Required with --https) | `start-storybook --ssl-cert my-ssl-certificate` |
+| --ssl-key `` | Provide an SSL key. (Required with --https) | `start-storybook --ssl-key my-ssl-key` |
+| --smoke-test | Exit after successful start | `start-storybook --smoke-test` |
+| --ci | CI mode (skip interactive prompts, don't open browser) | `start-storybook --ci` |
+| --quiet | Suppress verbose build output | `start-storybook --quiet` |
+| --no-dll | Do not use dll reference (no-op) | `start-storybook --no-dll` |
+| --debug-webpack | Display final webpack configurations for debugging purposes | `start-storybook --debug-webpack` |
+| `--webpack-stats-json ` | Write Webpack Stats JSON to disk | `start-storybook --webpack-stats-json /tmp/webpack-stats` |
+| --docs | Starts Storybook in documentation mode. Learn more about it in [here](../writing-docs/build-documentation.md#preview-storybooks-documentation) | `start-storybook --docs` |
+| --no-manager-cache | Disables Storybook's manager caching mechanism. See note below. | `start-storybook --no-manager-cache` |
+
+
+๐ก NOTE: Use the --no-manager-cache flag with caution. As it disables the internal caching mechanism and can severely impact your Storybook's loading time.
+
## build-storybook
@@ -37,26 +43,21 @@ Usage: start-storybook [options]
Usage: build-storybook [options]
```
-
+| Options | Description | Example |
+| ---------------------------------- | :---------------------------------------------------------------------------------------------------------------------------------------------- | :-------------------------------------------------------- |
+| -h, --help | Output usage information | `build-storybook --help` |
+| -V, --version | Output the version number | `build-storybook -V` |
+| -s, --static-dir `` | Directory where to load static files from, comma-separated list | `build-storybook -s public` |
+| -o, --output-dir [dir-name] | Directory where to store built files | `build-storybook -o /my-deployed-storybook` |
+| -c, --config-dir [dir-name] | Directory where to load Storybook configurations from | `build-storybook -c .storybook` |
+| -w, --watch | Enables watch mode | `build-storybook -w` |
+| --loglevel [level] | Controls level of logging during build. Can be one of: [silly, verbose, info (default), warn, error, silent] | `build-storybook --loglevel warn` |
+| --quiet | Suppress verbose build output | `build-storybook --quiet` |
+| --no-dll | Do not use dll reference (no-op) | `build-storybook --no-dll` |
+| --debug-webpack | Display final webpack configurations for debugging purposes | `build-storybook --debug-webpack` |
+| `--webpack-stats-json ` | Write Webpack Stats JSON to disk | `start-storybook --webpack-stats-json /tmp/webpack-stats` |
+| --docs | Builds Storybook in documentation mode. Learn more about it in [here](../writing-docs/build-documentation.md#publish-storybooks-documentation)) | `build-storybook --docs` |
-
Troubleshooting routing issues with Storybook 6.0
-
- If you are building your Storybook and you encounter an issue where you cannot change the route in the sidebar, try building Storybook with the `--no-dll` flag and see if it solves the problem. If so, adjust your `build-storybook` script accordingly to include this flag. We would like to point out that your build process will run slower than usual when using this flag.
-
- If you want, you can take a look at the following issue to get an in depth description of what is currently happening with your built Storybook.
-
-
-
-| Options | Description | Example |
-| ------------------------------ | :---------------------------------------------------------------------------------------------------------------------------------------------- | :------------------------------------------ |
-| -h, --help | Output usage information | `build-storybook --help` |
-| -V, --version | Output the version number | `build-storybook -V` |
-| -s, --static-dir `` | Directory where to load static files from, comma-separated list | `build-storybook -s public` |
-| -o, --output-dir [dir-name] | Directory where to store built files | `build-storybook -o /my-deployed-storybook` |
-| -c, --config-dir [dir-name] | Directory where to load Storybook configurations from | `build-storybook -c .storybook` |
-| -w, --watch | Enables watch mode | `build-storybook -w` |
-| --loglevel [level] | Controls level of logging during build. Can be one of: [silly, verbose, info (default), warn, error, silent] | `build-storybook --loglevel warn` |
-| --quiet | Suppress verbose build output | `build-storybook --quiet` |
-| --no-dll | Do not use dll reference (no-op) | `build-storybook --no-dll` |
-| --debug-webpack | Display final webpack configurations for debugging purposes | `build-storybook --debug-webpack` |
-| --docs | Builds Storybook in documentation mode. Learn more about it in [here](../writing-docs/build-documentation.md#publish-storybooks-documentation)) | `build-storybook --docs` |
+
+๐ก NOTE: If you're using npm instead of yarn to publish Storybook, the commands work slightly different. For example, npm run build-storybook -- -o ./path/to/build.
+
diff --git a/docs/api/csf.md b/docs/api/csf.md
index 6a99e369ecf..884ffda6905 100644
--- a/docs/api/csf.md
+++ b/docs/api/csf.md
@@ -41,6 +41,7 @@ With CSF, every named export in the file represents a story function by default.
@@ -83,6 +84,8 @@ Consider Storybookโs ["Button" example](../writing-stories/introduction.md#def
@@ -95,10 +98,13 @@ Now consider the same example, re-written with args:
+
At first blush this might seem no better than the original example. However, if we add the [Docs addon](https://github.com/storybookjs/storybook/tree/master/addons/docs) and configure the [Actions addon](https://github.com/storybookjs/storybook/tree/master/addons/actions) appropriately, we can write:
@@ -107,6 +113,8 @@ At first blush this might seem no better than the original example. However, if
@@ -168,6 +176,7 @@ Consider the following story file:
diff --git a/docs/api/mdx.md b/docs/api/mdx.md
index 927bf750c11..96d938dbbda 100644
--- a/docs/api/mdx.md
+++ b/docs/api/mdx.md
@@ -39,6 +39,7 @@ For example, here's the story from `Checkbox` example above, rewritten in CSF:
diff --git a/docs/api/new-frameworks.md b/docs/api/new-frameworks.md
index 650d4e6ed13..471825ce9dd 100644
--- a/docs/api/new-frameworks.md
+++ b/docs/api/new-frameworks.md
@@ -81,10 +81,10 @@ When developing your own framework that is not published by storybook, you can p
```ts
// my-framework/src/server/options.ts
-const packageJson = require('../../package.json');
+import { sync } from 'read-pkg-up';
export default {
- packageJson,
+ packageJson: sync({ cwd: __dirname }).packageJson,
framework: 'my-framework',
frameworkPath: '@my-framework/storybook',
frameworkPresets: [require.resolve('./framework-preset-my-framework.js')],
diff --git a/docs/configure/environment-variables.md b/docs/configure/environment-variables.md
index 5c241f058a5..2ffb5a1d9a3 100644
--- a/docs/configure/environment-variables.md
+++ b/docs/configure/environment-variables.md
@@ -29,6 +29,51 @@ If using the environment variables as attributes or values in JavaScript, you ma
+### Using .env files
+
+You can also use `.env` files to change Storybook's behavior in different modes. For example, if you add a `.env` file to your project with the following:
+
+```
+STORYBOOK_DATA_KEY=12345
+```
+
+Then you can access this environment variable anywhere, even within your stories:
+
+
+
+
+
+
+
+
+You can also use specific files for specific modes. Add a .env.development or .env.production to apply different values to your environment variables.
+
+
You can also pass these environment variables when you are [building your Storybook](../workflows/publish-storybook.md) with `build-storybook`.
-Then they'll be hard coded to the static version of your Storybook.
\ No newline at end of file
+Then they'll be hard coded to the static version of your Storybook.
+
+### Using environment variables to choose the browser
+
+Storybook allows you to choose the browser you want to preview your stories. Either through an `.env` file entry or directly in your `storybook` script.
+
+The table below lists the available options:
+
+| Browser | Example |
+|----------|----------------------|
+| Safari | `BROWSER="safari"` |
+| Firefox | `BROWSER="firefox"` |
+| Chromium | `BROWSER="chromium"` |
+
+
+Note: By default Storybook will open a new Chrome window as part of its startup process. If you don't have Chrome installed, make sure to include one of the following options, or set your default browser accordingly.
+
\ No newline at end of file
diff --git a/docs/configure/features-and-behavior.md b/docs/configure/features-and-behavior.md
index 2dc5b27049f..b6b0596f7c5 100644
--- a/docs/configure/features-and-behavior.md
+++ b/docs/configure/features-and-behavior.md
@@ -22,11 +22,17 @@ The following table details how to use the API values:
| **showNav** | Boolean |Display panel that shows a list of stories |`true` |
| **showPanel** | Boolean |Display panel that shows addon configurations |`true` |
| **panelPosition** | String/Object |Where to show the addon panel |`bottom` or `right` |
-| **sidebarAnimations** | Boolean |Sidebar tree animations |`true` |
| **enableShortcuts** | Boolean |Enable/disable shortcuts |`true` |
| **isToolshown** | String |Show/hide tool bar |`true` |
| **theme** | Object |Storybook Theme, see next section |`undefined` |
| **selectedPanel** | String |Id to select an addon panel |`my-panel` |
-| **initialActive** | String |Select the default active tab on Mobile. |`sidebar` or `canvas` or `addons` |
-| **showRoots** | Boolean |Display the top-level grouping as a "root" in the sidebar |`false` |
+| **initialActive** | String |Select the default active tab on Mobile |`sidebar` or `canvas` or `addons` |
+| **sidebar** | Object |Sidebar options, see below |`{ showRoots: false }` |
+The following options are configurable under the `sidebar` namespace:
+
+| Name | Type | Description | Example Value |
+| ----------------------|:-------------:|:-------------------------------------------------------------:|:----------------------------------------------:|
+| **showRoots** | Boolean |Display the top-level nodes as a "root" in the sidebar |`false` |
+| **collapsedRoots** | Array |Set of root node IDs to visually collapse by default |`['misc', 'other']` |
+| **renderLabel** | Function |Create a custom label for tree nodes; must return a ReactNode |`(item) => {item.name}`|
diff --git a/docs/configure/images-and-assets.md b/docs/configure/images-and-assets.md
index 7d6e9c7b836..5cf29bfe91f 100644
--- a/docs/configure/images-and-assets.md
+++ b/docs/configure/images-and-assets.md
@@ -1,8 +1,8 @@
---
-title: 'Images and assets'
+title: 'Images, fonts, and assets'
---
-Components often rely on images, videos, and other assets to render as the user expects. There are many ways to use these assets in your story files.
+Components often rely on images, videos, fonts, and other assets to render as the user expects. There are many ways to use these assets in your story files.
### Import assets into stories
@@ -15,6 +15,9 @@ Afterwards you can use any asset in your stories:
@@ -22,15 +25,25 @@ Afterwards you can use any asset in your stories:
### Serving static files via Storybook
-We recommend serving static files via Storybook to ensure that your components always have the assets they need to load.
+We recommend serving static files via Storybook to ensure that your components always have the assets they need to load. This technique is recommended for assets that your components often use like logos, fonts, and icons.
Configure a directory (or a list of directories) where your assets live when starting Storybook. Use the`-s` flag in your npm script like so:
```json
{
- "scripts": {
- "start-storybook": "start-storybook -s ./public -p 9001"
- }
+ "scripts": {
+ "start-storybook": "start-storybook -s ./public -p 9001"
+ }
+}
+```
+
+Or when building your Storybook with `build-storybook`:
+
+```json
+{
+ "scripts": {
+ "build-storybook": "build-storybook -s public"
+ }
}
```
@@ -41,6 +54,8 @@ Here `./public` is your static directory. Now use it in a component or story lik
@@ -50,9 +65,18 @@ You can also pass a list of directories separated by commas without spaces inste
```json
{
- "scripts": {
- "start-storybook": "start-storybook -s ./public,./static -p 9001"
- }
+ "scripts": {
+ "start-storybook": "start-storybook -s ./public,./static -p 9001"
+ }
+}
+```
+The same can be applied when you're building your Storybook.
+
+```json
+{
+ "scripts": {
+ "build-storybook": "build-storybook -s ./public,./static -p 9001"
+ }
}
```
@@ -65,6 +89,8 @@ Upload your files to an online CDN and reference them. In this example weโre u
@@ -78,4 +104,4 @@ In this case, you need to have all your images and media files with relative pat
If you load static content via importing, this is automatic and you do not have to do anything.
-If you are serving assets in a [static directory](#serving-static-files-via-storybook) along with your Storybook, then you need to use relative paths to load images or use the base element.
+If you are serving assets in a [static directory](#serving-static-files-via-storybook) along with your Storybook, then you need to use relative paths to load images or use the base element.
\ No newline at end of file
diff --git a/docs/configure/overview.md b/docs/configure/overview.md
index 114464dcdc7..b4d0f1cd284 100644
--- a/docs/configure/overview.md
+++ b/docs/configure/overview.md
@@ -31,6 +31,10 @@ The `main.js` configuration file is a [preset](../addons/addon-types.md) and as
- `webpackFinal` - custom [webpack configuration](./webpack.md#extending-storybooks-webpack-config).
- `babel` - custom [babel configuration](./babel.md).
+
+ Tip: Customize your default story by referencing it first in the `stories` array.
+
+
## Configure story loading
By default, Storybook will load stories from your project based on a glob (pattern matching string) in `.storybook/main.js` that matches all files in your project with extension `.stories.js`. The intention is you colocate a story file with the component it documents.
diff --git a/docs/configure/sidebar-and-urls.md b/docs/configure/sidebar-and-urls.md
index 161ad961aa0..bcf9cb2d316 100644
--- a/docs/configure/sidebar-and-urls.md
+++ b/docs/configure/sidebar-and-urls.md
@@ -10,11 +10,11 @@ We recommend using a nesting scheme that mirrors the filesystem path of the comp
## Roots
-By default, Storybook will treat your highest level of groups as โrootsโ--which are displayed in the UI as โsectionsโ of the hierarchy. Lower level groups are displayed as expandable items in the hierarchy:
+By default, Storybook will treat your top-level nodes as โrootsโ. Roots are displayed in the UI as โsectionsโ of the hierarchy. Lower level groups are displayed as folders:

-If youโd prefer all groups to be expandable, you can set the `showRoots` option to `false` in [`./storybook/manager.js`](./overview.md#configure-story-rendering):
+If youโd prefer to show top-level nodes as folders rather than roots, you can set the `sidebar.showRoots` option to `false` in [`./storybook/manager.js`](./overview.md#configure-story-rendering):
@@ -30,7 +30,6 @@ If youโd prefer all groups to be expandable, you can set the `showRoots` optio
As a CSF file is a JavaScript file, the exports (including the default export) can be generated dynamically. In particular you can use the `__dirname` variable to generate the title based on the path name (this example uses the paths.macro):
-
-
## Permalinking to stories
By default, Storybook generates an `id` for each story based on the component title and the story name. This `id` in particular is used in the URL for each story and that URL can serve as a permalink (especially when you [publish](../workflows/publish-storybook.md) your Storybook).
diff --git a/docs/configure/story-layout.md b/docs/configure/story-layout.md
index a86e6e85046..377bd1aa100 100644
--- a/docs/configure/story-layout.md
+++ b/docs/configure/story-layout.md
@@ -2,7 +2,9 @@
title: 'Story layout'
---
-The `layout` [global parameter](../writing-stories/parameters.md) allows you to configure how stories are positioned in Storybook's Canvas tab.
+The `layout` [parameter](../writing-stories/parameters.md) allows you to configure how stories are positioned in Storybook's Canvas tab.
+
+## Global layout
You can add the parameter to your [`./storybook/preview.js`](./overview.md#configure-story-rendering), like so:
@@ -24,4 +26,30 @@ In the example above, Storybook will center all stories in the UI. `layout` acce
- `fullscreen`: allow the component to expand to the full width and height of the Canvas
- `padded`: Add extra padding around the component
-If you want to use your own styles, or require a more granular approach we recommend using [decorators](../writing-stories/decorators.md) instead.
+## Component layout
+
+You can also set it at a component level like so:
+
+
+
+
+
+
+
+## Story layout
+
+Or even apply it to specific stories like so:
+
+
+
+
+
+
diff --git a/docs/configure/theming.md b/docs/configure/theming.md
index 48f5245d487..7083ce5a117 100644
--- a/docs/configure/theming.md
+++ b/docs/configure/theming.md
@@ -98,9 +98,16 @@ Finally we'll need to import the theme into Storybook. Create a new file called
-
-If the theme is not shown when Storybook starts, update your storybook scripts to include the --no-manager-cache flag.
-
+
+Adjust your `storybook` script in your package.json and include the [`--no-manager-cache`](../api/cli-options.md#start-storybook) flag. For instance:
+
+```json
+{
+ "scripts":{
+ "storybook": "start-storybook -p 6006 --no-manager-cache",
+ },
+}
+```
Now your custom theme will replace Storybook's default theme and you'll see a similar set of changes in the UI.
@@ -232,4 +239,4 @@ Or with template literals:
]}
/>
-
+
\ No newline at end of file
diff --git a/docs/essentials/addon-controls-args-variant-optimized.png b/docs/essentials/addon-controls-args-variant-optimized.png
new file mode 100644
index 00000000000..d7d2d15abae
Binary files /dev/null and b/docs/essentials/addon-controls-args-variant-optimized.png differ
diff --git a/docs/essentials/addon-controls-args-variant-string.png b/docs/essentials/addon-controls-args-variant-string.png
new file mode 100644
index 00000000000..a7795014521
Binary files /dev/null and b/docs/essentials/addon-controls-args-variant-string.png differ
diff --git a/docs/essentials/auto-generated-controls/ember.mdx b/docs/essentials/auto-generated-controls/ember.mdx
index f8c954f9c56..bf21b7d1ab6 100644
--- a/docs/essentials/auto-generated-controls/ember.mdx
+++ b/docs/essentials/auto-generated-controls/ember.mdx
@@ -1,32 +1,30 @@
-To use auto-detected controls with Ember, you must fill in the `component` field in your story metadata:
-
-```ts
-import { Button } from './Button';
-
-export default {
- title: 'Button',
- component: 'button', // name of your button component
-};
-```
-
-Storybook uses this to auto-generate the `ArgTypes` for your component based on docgen information created by [ember-cli-addon-docs-yuidoc](https://github.com/ember-learn/ember-cli-addon-docs-yuidoc#documenting-components).
-
-You'll need to register that in `.storybook/preview.js`:
-
-```js
-import { setJSONDoc } from '@storybook/addon-docs/ember';
-import docJson from '../storybook-docgen/index.json';
-setJSONDoc(docJson);
-```
-
-Storybook for Ember relies on [@storybook/ember-cli-storybook addon](https://github.com/storybookjs/ember-cli-storybook), to extract documentation comments from your component source files. If you're using Storybook with Ember, you should already have this addon installed, you will just need to enable it by adding the following config block in your `ember-cli-build.js` file:
+Storybook for Ember relies on [@storybook/ember-cli-storybook addon](https://github.com/storybookjs/ember-cli-storybook), to extract documentation comments from your component source files. If you're using Storybook with Ember, you should already have this addon installed, and you will just need to enable it by adding the following config block in your `ember-cli-build.js` file:
```js
let app = new EmberApp(defaults, {
- 'ember-cli-storybook': {
+ '@storybook/ember-cli-storybook': {
enableAddonDocsIntegration: true,
},
});
```
-Now, running the ember-cli server will generate a JSON documentation file at `/storybook-docgen/index.json`. Since generation of this file is tied into the ember-cli build, it will get regenerated every time component files are saved. For details on documenting your components, check out the examples in the addon that powers the generation [ember-cli-addon-docs-yuidoc](https://github.com/ember-learn/ember-cli-addon-docs-yuidoc#documenting-components).
+Now, running the ember-cli server will generate a JSON documentation file at `/dist/storybook-docgen/index.json`. Since generation of this file is tied into the ember-cli build, it will get regenerated every time component files are saved. For details on documenting your components, check out the examples in the addon that powers the generation [ember-cli-addon-docs-yuidoc](https://github.com/ember-learn/ember-cli-addon-docs-yuidoc#documenting-components).
+
+Storybook uses this file to auto-generate the `ArgTypes` for your component based on docgen information created by [ember-cli-addon-docs-yuidoc](https://github.com/ember-learn/ember-cli-addon-docs-yuidoc#documenting-components).
+
+You'll need to register that in `.storybook/preview.js`:
+
+```js
+import { setJSONDoc } from '@storybook/addon-docs/ember';
+import docJson from '../dist/storybook-docgen/index.json';
+setJSONDoc(docJson);
+```
+
+Finally, to use auto-detected controls with Ember, you must fill in the `component` field in your story metadata:
+
+```ts
+export default {
+ title: 'Button',
+ component: 'button', // name of your button component from docgen-json file (index.js)
+};
+```
diff --git a/docs/essentials/controls.md b/docs/essentials/controls.md
index e96102da9ba..0bbc296af07 100644
--- a/docs/essentials/controls.md
+++ b/docs/essentials/controls.md
@@ -44,43 +44,65 @@ By default, Storybook will choose a control for each arg based on the initial va
-For instance, suppose you have a `backgroundColor` arg on your story:
+For instance, suppose you have a `variant` arg on your story that should be `primary` or `secondary`:
-By default, Storybook will render a free text input for the `backgroundColor` arg:
+By default, Storybook will render a free text input for the `variant` arg:
-
+
-This works as long as you type a valid string into the auto-generated text control, but it's not the best UI for picking a color. Letโs replace it with Storybookโs color picker component.
-We can specify which controls get used by declaring a custom [argType](../api/argtypes.md) for the `backgroundColor` property. ArgTypes encode basic metadata for args, such as name, description, defaultValue for an arg. These get automatically filled in by Storybook Docs.
+This works as long as you type a valid string into the auto-generated text control, but it's not the best UI for our scenario, given that the component only accepts `primary` or `secondary` as variants. Letโs replace it with Storybookโs radio component.
-ArgTypes can also contain arbitrary annotations which can be overridden by the user. Since `backgroundColor` is a property of the component, let's put that annotation on the default export.
+We can specify which controls get used by declaring a custom [argType](../api/argtypes.md) for the `variant` property. ArgTypes encode basic metadata for args, such as name, description, defaultValue for an arg. These get automatically filled in by Storybook Docs.
+
+ArgTypes can also contain arbitrary annotations which can be overridden by the user. Since `variant` is a property of the component, let's put that annotation on the default export.
-This replaces the input with a color picker for a more intuitive developer experience.
+This replaces the input with a radio group for a more intuitive experience.
-
+
+
+## Custom control type matchers
+
+For a few types, Controls will automatically infer them by using [regex](https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/RegExp). You can change the matchers for a regex that suits you better.
+
+| Data type | Default regex | Description |
+| :---------: | :-----------------------: | :-------------------------------------------------------: |
+| **color** | `/(background\|color)$/i` | Will display a color picker UI for the args that match it |
+| **date** | `/Date$/` | Will display a date picker UI for the args that match it |
+
+
+To do so, use the `matchers` property in `controls` parameter:
+
+
+
+
+
## Fully custom args
@@ -92,6 +114,9 @@ Up until now, we only used auto-generated controls based on the component we're
paths={[
'react/table-story-fully-customize-controls.js.mdx',
'react/table-story-fully-customize-controls.mdx.mdx',
+ 'vue/table-story-fully-customize-controls.2.js.mdx',
+ 'vue/table-story-fully-customize-controls.3.js.mdx',
+ 'angular/table-story-fully-customize-controls.ts.mdx',
]}
/>
@@ -103,9 +128,13 @@ By default, Storybook will add controls for all args that:
- Appear in the list of args for your story.
-You can determine the control by using `argTypes` in each case.
+Using `argTypes`, you can change the display and behavior of each control.
-As they can be complex cases:
+### Dealing with complex values
+
+You'll notice when dealing with non-primitive values, you'll run into some limitations. The most obvious issue is that not every value can be represented as part of the `args` param in the URL, losing the ability to share and deeplink to such a state. Beyond that, complex values such as JSX cannot be synchronized between the manager (e.g. Controls addon) and the preview (your story).
+
+One way to deal with this is to use primitive values (e.g. strings) as arg values, and using a story template to convert these values to their complex counterpart before rendering. This isn't the nicest way to do it (see below), but certainly the most flexible.
@@ -113,27 +142,29 @@ As they can be complex cases:
paths={[
'react/component-story-custom-args-complex.js.mdx',
'react/component-story-custom-args-complex.ts.mdx',
- 'react/component-story-custom-args-complex.mdx.mdx'
+ 'react/component-story-custom-args-complex.mdx.mdx',
+ 'vue/component-story-custom-args-complex.2.js.mdx',
+ 'vue/component-story-custom-args-complex.3.js.mdx',
+ 'angular/component-story-custom-args-complex.ts.mdx',
]}
/>
-
-Or even with certain types of elements, such as icons:
+Unless you need the flexibility of a function, an easier way to map primitives to complex values before rendering is to define a `mapping`. Additionally, you can specify `control.labels` to configure custom labels for your checkbox, radio or select input.
+Note that both `mapping` and `control.labels` don't have to be exhaustive. If the currently selected option is not listed, it will be used verbatim.
+
## Configuration
The Controls addon can be configured in two ways:
@@ -149,40 +180,22 @@ Here is the full list of available controls you can use:
| Data Type | Control Type | Description | Options |
| :---------- | :----------: | :------------------------------------------------------------- | :------------: |
-| **array** | array | serialize array into a comma-separated string inside a textbox | separator |
| **boolean** | boolean | checkbox input | - |
| **number** | number | a numeric text box input | min, max, step |
| | range | a range slider input | min, max, step |
| **object** | object | json editor text input | - |
-| **enum** | radio | radio buttons input | options |
-| | inline-radio | inline radio buttons input | options |
-| | check | multi-select checkbox input | options |
-| | inline-check | multi-select inline checkbox input | options |
-| | select | select dropdown input | options |
-| | multi-select | multi-select dropdown input | options |
+| **array** | object | json editor text input | - |
+| | file | a file input that gives you a array of urls | accept |
+| **enum** | radio | radio buttons input | - |
+| | inline-radio | inline radio buttons input | - |
+| | check | multi-select checkbox input | - |
+| | inline-check | multi-select inline checkbox input | - |
+| | select | select dropdown input | - |
+| | multi-select | multi-select dropdown input | - |
| **string** | text | simple text input | - |
| | color | color picker input that assumes strings are color values | - |
| | date | date picker input | - |
-If you need to customize a control to use a enum data type in your story, for instance the `inline-radio` you can do it like so:
-
-
-
-
-
-
-
-
-If you don't provide a specific one, it defaults to:
-- a radio type for enums with 5 or less elements
-- a select control type with more than 5 elements
-
-
If you need to customize a control for a number data type in your story, you can do it like so:
@@ -226,7 +239,7 @@ And here's what the resulting UI looks like:
### Disable controls for specific properties
-Asides from the features already documented here. Controls can also be disabled for individual properties.
+Aside from the features already documented here, Controls can also be disabled for individual properties.
Suppose you want to disable Controls for a property called `foo` in a component's story. The following example illustrates how:
@@ -250,9 +263,18 @@ Resulting in the following change in Storybook UI:
/>
+The previous example also removed the prop documentation from the table. In some cases this is fine, however sometimes you might want to still render the prop documentation but without a control. The following example illustrates how:
+
+
+
- As with other Storybook properties, such as [decorators](../writing-stories/decorators.md) the same principle can also be applied at a story-level for more granular cases.
+As with other Storybook properties, such as [decorators](../writing-stories/decorators.md) the same principle can also be applied at a story-level for more granular cases.
@@ -269,3 +291,22 @@ If you don't plan to handle the control args inside your Story, you can remove t
/>
+
+## Filtering controls
+
+In some cases, you may want to either only present a few controls in the controls panel, or present all controls except a small set.
+
+To make this possible, you can use optional `include` and `exclude` configuration fields in the `controls` parameter, which can be set to either an array of strings, or a regular expression.
+
+Consider the following story snippets:
+
+
+
+
+
+
diff --git a/docs/essentials/introduction.md b/docs/essentials/introduction.md
index c22a2366693..65994ae85db 100644
--- a/docs/essentials/introduction.md
+++ b/docs/essentials/introduction.md
@@ -11,6 +11,28 @@ A major strength of Storybook are [addons](/addons/) that extend Storybookโs U
- [Backgrounds](./backgrounds.md)
- [Toolbars & globals](./toolbars-and-globals.md)
+### Installation
+If you're running `sb init` to add Storybook to your project, the essentials package (`@storybook/addon-essentials`) is already installed and configured for you . You can skip the rest of this section.
+
+If you're upgrading from a previous Storybook version, you'll need to run the following command in your terminal:
+
+```shell
+npm install --save-dev @storybook/addon-essentials
+```
+
+
+๐ก Note: If you're using yarn as a package manager, you'll need to adjust the command accordingly.
+
+
+Update your Storybook configuration (in `.storybook/main.js`) to include the essentials addon.
+
+```js
+module.exports = {
+ addons: ['@storybook/addon-essentials'],
+};
+```
+
+
### Configuration
Essentials is "zero configโ, it comes with a recommended configuration out of the box.
@@ -40,27 +62,3 @@ As an example, if the background addon wasn't necessary to your work, you would
You can use the following keys for each individual addon: `actions`, `backgrounds`, `controls`, `docs`, `viewport`, `toolbars`.
-
-### Upgrading from previous versions
-
-If you're upgrading from a previous Storybook version, you will need to add the `@storybook/addon-essentials` package manually.
-
-In your terminal run the following command:
-
-```shell
-npm install --save-dev @storybook/addon-essentials
-```
-
-
-
-If you're using yarn, you'll need to adjust the command accordingly.
-
-
-
-Update your Storybook configuration (in `.storybook/main.js`) to include the essentials addon.
-
-```js
-module.exports = {
- addons: ['@storybook/addon-essentials'],
-};
-```
diff --git a/docs/essentials/toolbars-and-globals.md b/docs/essentials/toolbars-and-globals.md
index f3722027e2a..7fbddfa8b6f 100644
--- a/docs/essentials/toolbars-and-globals.md
+++ b/docs/essentials/toolbars-and-globals.md
@@ -66,6 +66,12 @@ In your [`.storybook/preview.js`](../configure/overview.md#configure-story-rende
+
+
+The icon element used in the examples loads the icons from the @storybook/components package. See [here](../workflows/faq.md#what-icons-are-available-for-my-toolbar-or-my-addon) the list of available icons that you can use.
+
+
+
By adding the configuration element `right`, the text will be displayed on the right side in the toolbar menu, once you connect it to a decorator.
Here's a list of the configuration options available.
diff --git a/docs/essentials/viewport.md b/docs/essentials/viewport.md
index b7c37e924f1..a2c01708133 100644
--- a/docs/essentials/viewport.md
+++ b/docs/essentials/viewport.md
@@ -122,6 +122,7 @@ You can change your story through [parameters](../writing-stories/parameters.md)
'react/my-component-story-configure-viewports.js.mdx',
'react/my-component-story-configure-viewports.mdx.mdx',
'vue/my-component-story-configure-viewports.js.mdx',
+ 'angular/my-component-story-configure-viewports.ts.mdx',
]}
/>
diff --git a/docs/get-started/browse-stories.md b/docs/get-started/browse-stories.md
index 9305dbc7ab3..900074121cc 100644
--- a/docs/get-started/browse-stories.md
+++ b/docs/get-started/browse-stories.md
@@ -15,7 +15,17 @@ A `*.stories.js` file defines all the stories for a component. Each story has a
/>
-Navigate between stories by clicking on them in the sidebar or use keyboard shortcuts (for instance use opt/alt + โ๏ธ โถ๏ธ). Try the sidebar search to find a story by name.
+
+Navigate between stories by clicking on them in the sidebar. Try the sidebar search to find a story by name.
+
+Or use keyboard shortcuts. Click on the Storybook's menu to see the list of shortcuts available.
+
+
## Toolbar
diff --git a/docs/get-started/conclusion.md b/docs/get-started/conclusion.md
index 0af65c0769c..79d662b19f4 100644
--- a/docs/get-started/conclusion.md
+++ b/docs/get-started/conclusion.md
@@ -4,7 +4,7 @@ title: 'Conclusion'
Congratulations! You learned the basics. Storybook is the most popular tool for UI component development and documentation. Youโll be able to transfer these skills to thousands of companies that use Storybook to build UIs including GitHub, Airbnb, and Stripe.
-If youโd like to learn workflows for building app UIs with Storybook, check out the in-depth guides on [Learn Storybook](https://www.learnstorybook.com/). Continue reading for detailed information on how to use Storybook APIs.
+If youโd like to learn workflows for building app UIs with Storybook, check out our in-depth guides over at the [tutorials](https://storybook.js.org/tutorials/) page. Continue reading for detailed information on how to use Storybook APIs.
- [How to write stories](../writing-stories/introduction.md)
- [How to document components and design systems](../writing-docs/introduction.md)
diff --git a/docs/get-started/examples.md b/docs/get-started/examples.md
index a25f893b870..692e77adc2b 100644
--- a/docs/get-started/examples.md
+++ b/docs/get-started/examples.md
@@ -11,7 +11,7 @@ This is a curated list of Storybooks for your inspiration.
- [Official Storybook](https://next--storybookjs.netlify.app/official-storybook/): Storybook application UI
- [Storybook Design System](https://master--5ccbc373887ca40020446347.chromatic.com/): Reusable components that adhere to Storybook's design language
- [Marketing and docs](https://master--5be26744d2f6250024a9117d.chromatic.com/): Main website that has stories for components and pages.
-- [Learn Storybook](https://master--5cf841a3f3e3d200208ffc74.chromatic.com/): Tutorial site that has stories for components and pages.
+- [Storybook tutorials](https://master--5cf841a3f3e3d200208ffc74.chromatic.com/): Tutorial site that has stories for components and pages.
## Websites
@@ -48,6 +48,9 @@ Learn how leading teams build design systems.
- [AnyVision UI](http://storybook.anyvision.co/)
- [Skyscanner Backpack](https://backpack.github.io/storybook/)
- [GitLab UI](https://gitlab-org.gitlab.io/gitlab-ui)
+- [Grafana UI](https://developers.grafana.com/ui/latest/index.html)
+- [PX Blue](https://pxblue-components.github.io/)
+- [Audi](https://react.ui.audi/)
-```shell
-npx sb init
-```
+
+
+
sb init is not made for empty projects
-Storybook needs to be installed into a project that is already setup with a framework. It will not work on an empty project. There are many ways to bootstrap an app in given framework including:
+Storybook needs to be installed into a project that is already setup with a framework. It will not work on an empty project. There are many ways to bootstrap an app in a given framework including:
- ๐ฆ [Create React App](https://reactjs.org/docs/create-a-new-react-app.html)
- ๐ฆ [Vue CLI](https://cli.vuejs.org/)
+- ๐ฆ [Ember CLI](https://guides.emberjs.com/release/getting-started/quick-start/)
- Or any other tooling available.
@@ -29,11 +45,18 @@ The command above will make the following changes to your local environment:
- ๐ Add the default Storybook configuration.
- ๐ Add some boilerplate stories to get you started.
-Check that everything worked by running:
+Depending on your framework, first build your app and then check that everything worked by running:
-```shell
-npm run storybook
-```
+
+
+
+
+
It will start Storybook locally and output the address. Depending on your system configuration, it will automatically open the address in a new browser tab and you'll be greeted by a welcome screen.
@@ -50,15 +73,7 @@ Now that you installed Storybook successfully, letโs take a look at a story th
Troubleshooting
-You can also setup Storybook manually through the Storybook CLI.
-
-You can use the `--type` flag to tell Storybook to configure itself based on the flag.
-
-For instance you can use:
-
-- `--type react` to setup Storybook with the React configuration options.
-- `--type vue` to setup Storybook with the Vue configuration options.
-- `--type angular` to setup Storybook with the Angular configuration options.
+Below is a curated list to get you unblocked while adding Storybook to your project.
diff --git a/docs/get-started/installation-command-section/angular.mdx b/docs/get-started/installation-command-section/angular.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/angular.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/ember.mdx b/docs/get-started/installation-command-section/ember.mdx
new file mode 100644
index 00000000000..64c73fb41b8
--- /dev/null
+++ b/docs/get-started/installation-command-section/ember.mdx
@@ -0,0 +1,12 @@
+Use the Storybook CLI to install it with a couple of commands. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx -p @storybook/cli sb init
+
+# Add Ember storybook adapter
+ember install @storybook/ember-cli-storybook
+
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/html.mdx b/docs/get-started/installation-command-section/html.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/html.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/marko.mdx b/docs/get-started/installation-command-section/marko.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/marko.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/mithril.mdx b/docs/get-started/installation-command-section/mithril.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/mithril.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/preact.mdx b/docs/get-started/installation-command-section/preact.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/preact.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/rax.mdx b/docs/get-started/installation-command-section/rax.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/rax.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/react.mdx b/docs/get-started/installation-command-section/react.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/react.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/riot.mdx b/docs/get-started/installation-command-section/riot.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/riot.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/svelte.mdx b/docs/get-started/installation-command-section/svelte.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/svelte.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-command-section/vue.mdx b/docs/get-started/installation-command-section/vue.mdx
new file mode 100644
index 00000000000..2965c6d482c
--- /dev/null
+++ b/docs/get-started/installation-command-section/vue.mdx
@@ -0,0 +1,18 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing Vue 2 projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+For Vue 3 projects, run this inside your projectโs root directory:
+
+```shell
+# Add Storybook ๐ฅ pre-release ๐ฅ version:
+
+npx sb@next init
+```
+
+This command installs the latest pre-release version of Storybook with a new package (๐ฆ [`@storybook/vue3`](https://www.npmjs.com/package/@storybook/vue3)), created specifically for Vue 3.
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
diff --git a/docs/get-started/installation-command-section/web-components.mdx b/docs/get-started/installation-command-section/web-components.mdx
new file mode 100644
index 00000000000..ea17dd6c384
--- /dev/null
+++ b/docs/get-started/installation-command-section/web-components.mdx
@@ -0,0 +1,8 @@
+Use the Storybook CLI to install it in a single command. Run this inside your existing projectโs root directory:
+
+```shell
+# Add Storybook:
+npx sb init
+```
+
+If you run into issues with the installation, check the troubleshooting section below for guidance on how to solve it.
\ No newline at end of file
diff --git a/docs/get-started/installation-problems/angular.mdx b/docs/get-started/installation-problems/angular.mdx
index 9cb61e6cbc0..cf304275a40 100644
--- a/docs/get-started/installation-problems/angular.mdx
+++ b/docs/get-started/installation-problems/angular.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Angular framework](../../app/angular/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type angular` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Angular framework](../../app/angular/README.md).
diff --git a/docs/get-started/installation-problems/ember.mdx b/docs/get-started/installation-problems/ember.mdx
index bc83291e9fb..507f78b2897 100644
--- a/docs/get-started/installation-problems/ember.mdx
+++ b/docs/get-started/installation-problems/ember.mdx
@@ -1 +1,14 @@
-If there's an installation problem, check the [README for the Ember framework](../../app/ember/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type ember` flag when you initialize Storybook in your project.
+
+- If you see the following message during the `@storybook/ember-cli-storybook` installation process:
+
+ ```shell
+ The ember generate entity-name command requires an entity name to be specified.
+
+ For more details, use ember help.
+ ```
+
+ This is just a warning! If you followed the instructions provided, everything should be properly configured.
+
+
+- If you run into a installation problem, check the [README for the Ember framework](../../app/ember/README.md) for additional instructions.
\ No newline at end of file
diff --git a/docs/get-started/installation-problems/html.mdx b/docs/get-started/installation-problems/html.mdx
index a5e32107104..fea82a08911 100644
--- a/docs/get-started/installation-problems/html.mdx
+++ b/docs/get-started/installation-problems/html.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the HTML framework](../../app/html/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type html` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Html framework](../../app/html/README.md).
diff --git a/docs/get-started/installation-problems/marko.mdx b/docs/get-started/installation-problems/marko.mdx
index f6c02bccc6f..ec4168801c7 100644
--- a/docs/get-started/installation-problems/marko.mdx
+++ b/docs/get-started/installation-problems/marko.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Marko framework](../../app/marko/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type marko` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Marko framework](../../app/marko/README.md).
diff --git a/docs/get-started/installation-problems/mithril.mdx b/docs/get-started/installation-problems/mithril.mdx
index 201cbd380e6..7e2d8f66869 100644
--- a/docs/get-started/installation-problems/mithril.mdx
+++ b/docs/get-started/installation-problems/mithril.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Mithril framework](../../app/mithril/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type mithril` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Mithril framework](../../app/mithril/README.md).
diff --git a/docs/get-started/installation-problems/preact.mdx b/docs/get-started/installation-problems/preact.mdx
index d33de4b3111..8341dc81cf1 100644
--- a/docs/get-started/installation-problems/preact.mdx
+++ b/docs/get-started/installation-problems/preact.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Preact framework](../../app/preact/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type mithril` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Preact framework](../../app/preact/README.md).
diff --git a/docs/get-started/installation-problems/rax.mdx b/docs/get-started/installation-problems/rax.mdx
index ccca04d2f5f..8c8a34c5f79 100644
--- a/docs/get-started/installation-problems/rax.mdx
+++ b/docs/get-started/installation-problems/rax.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Rax framework](../../app/rax/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type rax` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Rax framework](../../app/rax/README.md).
diff --git a/docs/get-started/installation-problems/react.mdx b/docs/get-started/installation-problems/react.mdx
index 07c07d03d6f..f368e563afb 100644
--- a/docs/get-started/installation-problems/react.mdx
+++ b/docs/get-started/installation-problems/react.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the React framework](../../app/react/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type react` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the React framework](../../app/react/README.md).
diff --git a/docs/get-started/installation-problems/riot.mdx b/docs/get-started/installation-problems/riot.mdx
index e6ec6ecf364..320277cef28 100644
--- a/docs/get-started/installation-problems/riot.mdx
+++ b/docs/get-started/installation-problems/riot.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Riot framework](../../app/riot/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type riot` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Riot framework](../../app/riot/README.md).
diff --git a/docs/get-started/installation-problems/svelte.mdx b/docs/get-started/installation-problems/svelte.mdx
index 0d0776b2a41..c684ea74bbd 100644
--- a/docs/get-started/installation-problems/svelte.mdx
+++ b/docs/get-started/installation-problems/svelte.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Svelte framework](../../app/svelte/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type svelte` flag when you initialize Storybook in your project.
+- If there's an installation problem, check the [README for the Svelte framework](../../app/svelte/README.md).
diff --git a/docs/get-started/installation-problems/vue.mdx b/docs/get-started/installation-problems/vue.mdx
index 80137e14615..fa56a020cf6 100644
--- a/docs/get-started/installation-problems/vue.mdx
+++ b/docs/get-started/installation-problems/vue.mdx
@@ -1 +1,3 @@
-If there's an installation problem, check the [README for the Vue framework](../../app/vue/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type vue` flag when you initialize Storybook in your Vue 2 project.
+- If you run into a installation problem, check the [README for the Vue 2 framework](../../app/vue/README.md) or [Readme for the Vue 3](../../app/vue/README.md) for additional instructions.
+- Vue 3 support is still under active development and we encourage feedback and improvements. Check the [contribution guidelines to help us improve it](../../CONTRIBUTING.md).
\ No newline at end of file
diff --git a/docs/get-started/installation-problems/web-components.mdx b/docs/get-started/installation-problems/web-components.mdx
index 679fa60eb08..2c152e09665 100644
--- a/docs/get-started/installation-problems/web-components.mdx
+++ b/docs/get-started/installation-problems/web-components.mdx
@@ -1 +1,2 @@
-If there's an installation problem, check the [README for the Web Components framework](../../app/web-components/README.md).
+- You can also setup Storybook manually through the Storybook CLI. Add the `--type web_components` flag when you initialize Storybook in your project. web_components
+- If there's an installation problem, check the [README for the Web Components framework](../../app/web-components/README.md).
diff --git a/docs/get-started/introduction.md b/docs/get-started/introduction.md
index dadde776b69..9336b250795 100644
--- a/docs/get-started/introduction.md
+++ b/docs/get-started/introduction.md
@@ -23,6 +23,6 @@ Storybook integrates with most popular JavaScript UI frameworks and (experimenta
If you want to learn more about the component-driven approach that Storybook enables, this [site](http://componentdriven.org) is a good place to start.
-If you want a guided tutorial through building a simple application with Storybook in your framework and language, [LearnStorybook](https://www.learnstorybook.com/) has your back.
+If you want a guided tutorial through building a simple application with Storybook in your framework and language, our [tutorials](https://storybook.js.org/tutorials/) have your back.
Read on to learn Storybook basics and API!
diff --git a/docs/get-started/setup.md b/docs/get-started/setup.md
index d5b6861edcd..2c5c08d523a 100644
--- a/docs/get-started/setup.md
+++ b/docs/get-started/setup.md
@@ -13,6 +13,10 @@ Pick a simple component from your project, like a Button, and write a `.stories.
'react/your-component.js.mdx',
'react/your-component.ts.mdx',
'angular/your-component.ts.mdx',
+ 'vue/your-component.2.js.mdx',
+ 'vue/your-component.3.js.mdx',
+ 'svelte/your-component.js.mdx',
+ 'web-components/your-component.js.mdx',
]}
/>
@@ -60,6 +64,8 @@ Use [decorators](../writing-stories/decorators.md) to โwrapโ every story in
@@ -97,14 +103,21 @@ If you have global imported styles, create a file called [`.storybook/preview.js
- Add external CSS or fonts in the <head>
+ Add external CSS or webfonts in the <head>
-Alternatively if you want to inject a CSS link tag to the `` directly (or some other resource like a font link), you can use [`.storybook/preview-head.html`](../configure/story-rendering.md#adding-to-<head>) to add arbitrary HTML.
+Alternatively, if you want to inject a CSS link tag to the `` directly (or some other resource like a webfont link), you can use [`.storybook/preview-head.html`](../configure/story-rendering.md#adding-to-<head>) to add arbitrary HTML.
+
+
+
+
+ Load fonts or images from a local directory
+
+If you're referencing fonts or images from a local directory, you'll need to configure the Storybook script to [serve the static files](../configure/images-and-assets).
## Load assets and resources
-If you want to link to static files in your project or stories (e.g. `/fonts/XYZ.woff`), use the `-s path/to/folder` to specify a static folder to serve from when you start up Storybook. To do so, edit the `storybook` and `build-storybook` scripts in `package.json`.
+If you want to [link to static files](../configure/images-and-assets.md) in your project or stories (e.g. `/fonts/XYZ.woff`), use the `-s path/to/folder` to specify a static folder to serve from when you start up Storybook. To do so, edit the `storybook` and `build-storybook` scripts in `package.json`.
We recommend serving external resources and assets requested in your components statically with Storybook. This ensures that assets are always available to your stories.
diff --git a/docs/get-started/storybook-keyboard-shortcuts-optimized.mp4 b/docs/get-started/storybook-keyboard-shortcuts-optimized.mp4
new file mode 100644
index 00000000000..db8aa9d9653
Binary files /dev/null and b/docs/get-started/storybook-keyboard-shortcuts-optimized.mp4 differ
diff --git a/docs/get-started/whats-a-story.md b/docs/get-started/whats-a-story.md
index 0090e26e717..c8e66028068 100644
--- a/docs/get-started/whats-a-story.md
+++ b/docs/get-started/whats-a-story.md
@@ -18,6 +18,10 @@ Letโs start with the `Button` component. A story is a function that describes
'react/button-story.ts.mdx',
'react/button-story.mdx.mdx',
'angular/button-story.ts.mdx',
+ 'vue/button-story.js.mdx',
+ 'svelte/button-story.js.mdx',
+ 'svelte/button-story.mdx.mdx',
+ 'web-components/button-story.js.mdx',
]}
/>
@@ -29,19 +33,20 @@ View the rendered `Button` by clicking on it in the Storybook sidebar.
The above story definition can be further improved to take advantage of [Storybookโs โargsโ](../writing-stories/args.md) concept. Args describes the arguments to Button in a machine readable way. This unlocks Storybookโs superpower of altering and composing arguments dynamically.
-```js
-// We create a โtemplateโ of how args map to rendering
+
-const Template = (args) => ;
+
-// Each story then reuses that template
-export const Primary = Template.bind({});
-
-Primary.args = {
- primary: true,
- label: 'Button',
-};
-```
+

diff --git a/docs/snippets/angular/app-story-with-mock.ts.mdx b/docs/snippets/angular/app-story-with-mock.ts.mdx
new file mode 100644
index 00000000000..62ebba01239
--- /dev/null
+++ b/docs/snippets/angular/app-story-with-mock.ts.mdx
@@ -0,0 +1,34 @@
+```ts
+// App.stories.ts
+
+import { Story, Meta } from '@storybook/angular/types-6-0';
+import { AppComponent } from './app.component';
+
+export default {
+ component: AppComponent,
+ title: 'App',
+} as Meta;
+
+const Template: Story = (args) => ({
+ component: AppComponent,
+ props: args,
+});
+
+export const Success = Template.bind({});
+Success.parameters = {
+ fetch: {
+ json: {
+ JavaScript: 3390991,
+ 'C++': 44974,
+ TypeScript: 15530,
+ CoffeeScript: 12253,
+ Python: 9383,
+ C: 5341,
+ Shell: 5115,
+ HTML: 3420,
+ CSS: 3171,
+ Makefile: 189,
+ },
+ },
+};
+```
diff --git a/docs/snippets/angular/button-component-with-proptypes.ts.mdx b/docs/snippets/angular/button-component-with-proptypes.ts.mdx
index 85e5aa0f12a..b9bcdbfbb95 100644
--- a/docs/snippets/angular/button-component-with-proptypes.ts.mdx
+++ b/docs/snippets/angular/button-component-with-proptypes.ts.mdx
@@ -1,13 +1,11 @@
```ts
// my-button.component.ts
-import { Component, Input} from '@angular/core';
+import { Component, Input } from '@angular/core';
@Component({
selector: 'my-button',
- template: `