Merge branch 'next' into pr/joshwooding/18365

This commit is contained in:
Norbert de Langen 2022-06-07 09:59:31 +02:00
commit 482676fd82
No known key found for this signature in database
GPG Key ID: FD0E78AF9A837762
102 changed files with 1503 additions and 644 deletions

View File

@ -265,31 +265,32 @@ jobs:
steps:
- git-shallow-clone/checkout_advanced:
clone_options: '--depth 1 --verbose'
- attach_workspace:
at: .
- run:
name: Running local registry
command: yarn local-registry --port 6000 --open
background: true
- run:
name: Wait for registry
command: yarn wait-on http://localhost:6000
- run:
name: run e2e tests cra
command: yarn test:e2e-framework --pnp cra
- run:
name: run e2e tests vue
command: yarn test:e2e-framework --pnp sfcVue
- run:
name: prep artifacts
when: always
command: zip -r /tmp/storybook-e2e-testing-out.zip /tmp/storybook-e2e-testing
- store_artifacts:
path: /tmp/cypress-record
destination: cypress
- store_artifacts:
path: /tmp/storybook-e2e-testing-out.zip
destination: e2e
# TODO: we disabled this one because it keeps failing, we should fix it, and enable it again!
# - attach_workspace:
# at: .
# - run:
# name: Running local registry
# command: yarn local-registry --port 6000 --open
# background: true
# - run:
# name: Wait for registry
# command: yarn wait-on http://localhost:6000
# - run:
# name: run e2e tests cra
# command: yarn test:e2e-framework --pnp cra
# - run:
# name: run e2e tests vue
# command: yarn test:e2e-framework --pnp sfcVue
# - run:
# name: prep artifacts
# when: always
# command: zip -r /tmp/storybook-e2e-testing-out.zip /tmp/storybook-e2e-testing
# - store_artifacts:
# path: /tmp/cypress-record
# destination: cypress
# - store_artifacts:
# path: /tmp/storybook-e2e-testing-out.zip
# destination: e2e
e2e-tests-examples:
executor:
class: small

View File

@ -14,6 +14,25 @@
},
{
"pattern": "https://stackblitz.com/*"
},
{
"pattern": "https://*.chromatic.com"
},
{
"pattern": "https://www.chromatic.com/build?*"
},
{
"pattern": "http://*.nodeca.com"
},
{
"pattern": "http://definitelytyped.org/*"
},
{
"pattern": "https://yoursite.com/*"
},
{
"pattern": "https://my-specific-domain.com"
}
]
],
"aliveStatusCodes": [429, 200]
}

View File

@ -1,6 +1,7 @@
<h1>Migration</h1>
- [From version 6.4.x to 6.5.0](#from-version-64x-to-650)
- [Vue 3 upgrade](#vue-3-upgrade)
- [React18 new root API](#react18-new-root-api)
- [Renamed isToolshown to showToolbar](#renamed-istoolshown-to-showtoolbar)
- [Deprecated register.js](#deprecated-registerjs)
@ -202,6 +203,10 @@
## From version 6.4.x to 6.5.0
### Vue 3 upgrade
Storybook 6.5 supports Vue 3 out of the box when you install it fresh. However, if you're upgrading your project from a previous version, you'll need to [follow the steps for opting-in to webpack 5](#webpack-5).
### React18 new root API
React 18 introduces a [new root API](https://reactjs.org/blog/2022/03/08/react-18-upgrade-guide.html#updates-to-client-rendering-apis). Starting in 6.5, Storybook for React will auto-detect your react version and use the new root API automatically if you're on React18.
@ -360,6 +365,7 @@ In 6.5, the final titles would be:
- `NoTitle.stories.js` => `Custom/NoTitle`
- `Title.stories.js` => `Custom/Bar`
<!-- markdown-link-check-disable -->
## From version 6.3.x to 6.4.0
### Automigrate
@ -748,7 +754,29 @@ The `--static-dir` flag has been deprecated and will be removed in Storybook 7.0
### Webpack 5
Storybook 6.3 brings opt-in support for building both your project and the manager UI with webpack 5. To do so:
Storybook 6.3 brings opt-in support for building both your project and the manager UI with webpack 5. To do so, there are two ways:
1 - Upgrade command
If you're upgrading your Storybook version, run this command, which will both upgrade your dependencies but also detect whether you should migrate to webpack5 builders and apply the changes automatically:
```shell
npx sb upgrade
```
2 - Automigrate command
If you don't want to change your Storybook version but want Storybook to detect whether you should migrate to webpack5 builders and apply the changes automatically:
```shell
npx sb automigrate
```
3 - Manually
If either methods did not work or you just want to proceed manually, do the following steps:
Install the dependencies:
```shell
yarn add @storybook/builder-webpack5 @storybook/manager-webpack5 --dev
@ -2881,3 +2909,4 @@ If you **are** using these addons, it takes two steps to migrate:
import { action } from '@storybook/addon-actions';
import { linkTo } from '@storybook/addon-links';
```
<!-- markdown-link-check-enable -->

View File

@ -51,8 +51,8 @@ It allows you to browse a component library, view the different states of each c
<p align="center">
View README for:<br/>
<a href="https://github.com/storybookjs/storybook/blob/main/README.md" title="latest"><img alt="latest" src="https://img.shields.io/npm/v/@storybook/core/latest.svg" /></a>
<a href="https://github.com/storybookjs/storybook/blob/next/README.md" title="next"><img alt="next" src="https://img.shields.io/npm/v/@storybook/core/next.svg" /></a>
<a href="https://github.com/storybookjs/storybook/blob/main/README.md" title="latest"><img alt="latest" src="https://img.shields.io/npm/v/@storybook/core/latest?style=for-the-badge&logo=storybook&logoColor=ffffff&color=ff4785" /></a>
<a href="https://github.com/storybookjs/storybook/blob/next/README.md" title="next"><img alt="next" src="https://img.shields.io/npm/v/@storybook/core/next?style=for-the-badge&logo=storybook&logoColor=ffffff&color=purple" /></a>
</p>
## Table of contents
@ -92,19 +92,19 @@ For additional help, join us in the [Storybook Discord](https://discord.gg/story
### Supported Frameworks
| Framework | Demo | |
| -------------------------------------------------------------- | ------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------- |
| [React](app/react) | [v6.4.x](https://storybookjs.netlify.com/official-storybook/?path=/story/*) | [![React](https://img.shields.io/npm/dm/@storybook/react.svg)](app/react) |
| [Vue](app/vue) | [v6.4.x](https://storybookjs.netlify.com/vue-kitchen-sink/) | [![Vue](https://img.shields.io/npm/dm/@storybook/vue.svg)](app/vue) |
| [Angular](app/angular) | [v6.4.x](https://storybookjs.netlify.com/angular-cli/) | [![Angular](https://img.shields.io/npm/dm/@storybook/angular.svg)](app/angular) |
| [Web components](app/web-components) | [v6.4.x](https://storybookjs.netlify.com/web-components-kitchen-sink/) | [![Svelte](https://img.shields.io/npm/dm/@storybook/web-components.svg)](app/web-components) |
| [React Native](https://github.com/storybookjs/react-native) | - | [![React Native](https://img.shields.io/npm/dm/@storybook/react-native.svg)](https://github.com/storybookjs/react-native) |
| [HTML](app/html) | [v6.4.x](https://storybookjs.netlify.com/html-kitchen-sink/) | [![HTML](https://img.shields.io/npm/dm/@storybook/html.svg)](app/html) |
| [Ember](app/ember) | [v6.4.x](https://storybookjs.netlify.com/ember-cli/) | [![Ember](https://img.shields.io/npm/dm/@storybook/ember.svg)](app/ember) |
| [Svelte](app/svelte) | [v6.4.x](https://storybookjs.netlify.com/svelte-kitchen-sink/) | [![Svelte](https://img.shields.io/npm/dm/@storybook/svelte.svg)](app/svelte) |
| [Preact](app/preact) | [v6.4.x](https://storybookjs.netlify.com/preact-kitchen-sink/) | [![Preact](https://img.shields.io/npm/dm/@storybook/preact.svg)](app/preact) |
| [Marionette.js](https://github.com/storybookjs/marionette) | - | [![Marionette.js](https://img.shields.io/npm/dm/@storybook/marionette.svg)](https://github.com/storybookjs/marionette) |
| [Android, iOS, Flutter](https://github.com/storybookjs/native) | [v6.4.x](https://storybookjs.github.io/native/@storybook/native-flutter-example/index.html) | [![Native](https://img.shields.io/npm/dm/@storybook/native.svg)](https://github.com/storybookjs/native) |
| Framework | Demo | |
| -------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------- |
| [React](app/react) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/react/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/official-storybook/?path=/story/*) | [![React](https://img.shields.io/npm/dm/@storybook/react?style=flat-square&color=eee)](app/react) |
| [Vue](app/vue) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/vue/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/vue-kitchen-sink/) | [![Vue](https://img.shields.io/npm/dm/@storybook/vue?style=flat-square&color=eee)](app/vue) |
| [Angular](app/angular) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/angular/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/angular-cli/) | [![Angular](https://img.shields.io/npm/dm/@storybook/angular?style=flat-square&color=eee)](app/angular) |
| [Web components](app/web-components) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/web-components/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/web-components-kitchen-sink/) | [![Svelte](https://img.shields.io/npm/dm/@storybook/web-components?style=flat-square&color=eee)](app/web-components) |
| [React Native](https://github.com/storybookjs/react-native) | - | [![React Native](https://img.shields.io/npm/dm/@storybook/react-native?style=flat-square&color=eee)](https://github.com/storybookjs/react-native) |
| [HTML](app/html) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/html/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/html-kitchen-sink/) | [![HTML](https://img.shields.io/npm/dm/@storybook/html?style=flat-square&color=eee)](app/html) |
| [Ember](app/ember) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/ember/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/ember-cli/) | [![Ember](https://img.shields.io/npm/dm/@storybook/ember?style=flat-square&color=eee)](app/ember) |
| [Svelte](app/svelte) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/svelte/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/svelte-kitchen-sink/) | [![Svelte](https://img.shields.io/npm/dm/@storybook/svelte?style=flat-square&color=eee)](app/svelte) |
| [Preact](app/preact) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/preact/latest?style=flat-square&color=blue&label)](https://storybookjs.netlify.com/preact-kitchen-sink/) | [![Preact](https://img.shields.io/npm/dm/@storybook/preact?style=flat-square&color=eee)](app/preact) |
| [Marionette.js](https://github.com/storybookjs/marionette) | - | [![Marionette.js](https://img.shields.io/npm/dm/@storybook/marionette?style=flat-square&color=eee)](https://github.com/storybookjs/marionette) |
| [Android, iOS, Flutter](https://github.com/storybookjs/native) | [![Storybook demo](https://img.shields.io/npm/v/@storybook/native/latest?style=flat-square&color=blue&label)](https://storybookjs.github.io/native/@storybook/native-flutter-example/index.html) | [![Native](https://img.shields.io/npm/dm/@storybook/native?style=flat-square&color=eee)](https://github.com/storybookjs/native) |
### Sub Projects

View File

@ -81,12 +81,11 @@ there gathering upvotes and "me too" comments. We need a way to make sure that
these bugs get addressed.
For every non-PATCH release, we nominate a small number of bugs that must be
addressed before a release can go out by adding them to the milestone. For example, here's a list of blocking bugs [for the 3.2 milestone](https://github.com/storybookjs/storybook/milestone/3).
addressed before a release can go out by adding them to the milestone. For example, here's a list of blocking bugs [for the 6.5 milestone](https://github.com/storybookjs/storybook/milestone/75).
Adding bugs to the milestone helps people looking for good ways to contribute,
or to understand what is blocking the release so they can actually do something
about it. Discussion about which bugs are critical happens in the `#maintenance`
channel [in our Slack](https://now-examples-slackin-rrirkqohko.now.sh/) [![Storybook Slack](https://now-examples-slackin-rrirkqohko.now.sh/badge.svg)](https://now-examples-slackin-rrirkqohko.now.sh/)
about it. Discussion about which bugs are critical happens in the [`#maintenance` channel](https://discord.com/channels/486522875931656193/490070912448724992) in our Discord Server
If you're experiencing a bug, the best way to make sure that it gets attention
is to upvote it by adding a "thumbs-up" reaction in Github. This way important

View File

@ -2,9 +2,9 @@
## Supported Versions
| Version | Supported |
| ---------- | ------------------ |
| 6.3, 6.4 | :white_check_mark: |
| Version | Supported |
| --------------- | ------------------ |
| 6.3, 6.4, 6.5 | :white_check_mark: |
## Reporting a Vulnerability

View File

@ -30,7 +30,7 @@ describe('Vision Simulator', () => {
await waitFor(() => expect(screen.getByTitle('Vision simulator')).toBeInTheDocument());
});
it('should display tooltip on click', async () => {
it.skip('should display tooltip on click', async () => {
// given
render(<ThemedVisionSimulator />);
await waitFor(() => expect(screen.getByTitle('Vision simulator')).toBeInTheDocument());
@ -45,7 +45,7 @@ describe('Vision Simulator', () => {
);
});
it('should set filter', async () => {
it.skip('should set filter', async () => {
// given
render(<ThemedVisionSimulator />);
await waitFor(() => expect(screen.getByTitle('Vision simulator')).toBeInTheDocument());

View File

@ -66,7 +66,7 @@
"@storybook/core-events": "6.5.0-rc.1",
"@storybook/csf": "0.0.2--canary.4566f4d.1",
"@storybook/docs-tools": "6.5.0-rc.1",
"@storybook/mdx1-csf": "canary",
"@storybook/mdx1-csf": "^0.0.1",
"@storybook/node-logger": "6.5.0-rc.1",
"@storybook/postinstall": "6.5.0-rc.1",
"@storybook/preview-web": "6.5.0-rc.1",
@ -86,11 +86,11 @@
},
"devDependencies": {
"@babel/core": "^7.12.10",
"@storybook/mdx2-csf": "canary",
"@storybook/mdx2-csf": "^0.0.3",
"@types/util-deprecate": "^1.0.0"
},
"peerDependencies": {
"@storybook/mdx2-csf": "*",
"@storybook/mdx2-csf": "^0.0.3",
"react": "^16.8.0 || ^17.0.0 || ^18.0.0",
"react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0"
},

View File

@ -19,6 +19,7 @@ export async function babel(config: TransformOptions, options: Options) {
return {
...config,
overrides: [
...(config?.overrides || []),
{
test: reactDocgen === 'react-docgen' ? /\.(mjs|tsx?|jsx?)$/ : /\.(mjs|jsx?)$/,
plugins: [

View File

@ -1,4 +1,4 @@
export { render, renderToDOM } from './render';
export { decorateStory } from './decorateStory';
export { decorateStory as applyDecorators } from './decorateStory';
export const parameters = { framework: 'vue' };

View File

@ -1,4 +1,4 @@
export { render, renderToDOM } from './render';
export { decorateStory } from './decorateStory';
export { decorateStory as applyDecorators } from './decorateStory';
export const parameters = { framework: 'vue3' };

View File

@ -50,13 +50,22 @@ Once you've gone through the prompts, your `package.json` should look like:
### Build system
We'll need to add the necessary dependencies and make some adjustments. Run the following commands:
We'll need to add the necessary dependencies and make some adjustments. Run the following command to install the required dependencies:
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/storybook-write-addon-install-dependencies.yarn.js.mdx',
'common/storybook-write-addon-install-dependencies.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Initialize a local Storybook instance to allow you to test your addon.
```shell
# Installs React and Babel CLI
yarn add react react-dom @babel/cli
# Adds Storybook:
npx sb init
```
@ -246,4 +255,4 @@ To dive deeper, we recommend Storybook's [creating an addon](https://storybook.j
### Addon kit
To help you jumpstart the addon development, the Storybook maintainers created an [`addon-kit`](https://github.com/storybookjs/addon-kit), use it to bootstrap your next addon.
To help you jumpstart the addon development, the Storybook maintainers created an [`addon-kit`](https://github.com/storybookjs/addon-kit), use it to bootstrap your next addon.

View File

@ -5,7 +5,9 @@ title: 'CLI options'
Storybook comes with two CLI utilities: `start-storybook` and `build-storybook`.
<div class="aside">
Storybook collects completely anonymous data to help us improve user experience. Participation is optional, and you may [opt-out](../configure/telemetry.md#how-to-opt-out) if you'd not like to share any information.
</div>
Pass these commands the following options to alter Storybook's behavior.

View File

@ -40,16 +40,34 @@ You can import `.json` files and have them expanded to a JavaScript object:
<!-- prettier-ignore-end -->
If you want to know the exact details of the Webpack config, the best way is to run either of the following:
If you want to know the exact details of the Webpack config, the best way is to run either of the following commands:
```shell
For development mode:
## Development mode
yarn start-storybook --debug-webpack
<!-- prettier-ignore-start -->
## Production mode
yarn build-storybook --debug-webpack
```
<CodeSnippets
paths={[
'common/storybook-debug-webpack-dev.yarn.js.mdx',
'common/storybook-debug-webpack-dev.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
For production mode:
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/storybook-debug-webpack-prod.yarn.js.mdx',
'common/storybook-debug-webpack-prod.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
### Code splitting
@ -193,11 +211,11 @@ When working with TypeScript projects, the default Webpack configuration may fai
<!-- prettier-ignore-end -->
<div class="aside">
💡 Learn more about Storybook's <a href="./typescript">built-in TypeScript support</a> or see <a href="https://github.com/storybookjs/storybook/issues/14087">this issue</a> for more information.
💡 Learn more about Storybook's <a href="../configure/typescript">built-in TypeScript support</a> or see <a href="https://github.com/storybookjs/storybook/issues/14087">this issue</a> for more information.
</div>
#### Learn more about builders
- [Vite builder](./vite.md) for bundling with Vite
- [Webpack builder](./webpack.md) for bundling with Webpack
- [Builder API](./builder-api.md) for building a Storybook builder
- Webpack builder for bundling with Webpack
- [Builder API](./builder-api.md) for building a Storybook builder

View File

@ -42,7 +42,11 @@ Additionally, if you need Storybook specific styles that are separate from your
```
### Nx with Angular 13
If you're working with Storybook and [NX libraries](https://nx.dev/structure/library-types), you can extend your project's configuration (i.e., `project.json`) and provide the application's styles. For example:
If you're working with Storybook and [Nx libraries](https://nx.dev/structure/library-types),
you can extend your project's configuration (i.e., `project.json`) and provide the application's styles.
For earlier Nx versions (prior to `14.1.8`), your configuration would look like this:
```json
"build-storybook": {
@ -56,11 +60,35 @@ If you're working with Storybook and [NX libraries](https://nx.dev/structure/lib
},
"projectBuildConfig": "example-lib:build-storybook",
"styles": ["apps/example-app/src/styles.scss"]
}
}
```
Starting with version `14.1.8`, Nx uses the Storybook builder directly, which means any configuration supplied to the builder also applies to the NX setup. If you're working with a library, you'll need to configure the styling options ( e.g., preprocessors) inside the `build-storybook` `options` configuration object. For example:
```json
"storybook": {
"executor": "@storybook/angular:start-storybook",
"options": {
"configDir": "apps/example-lib/.storybook",
"browserTarget": "example-lib:build-storybook",
},
},
"configurations": {
"ci": {
"quiet": true
"build-storybook": {
"executor": "@storybook/angular:build-storybook",
"outputs": ["{options.outputPath}"],
"options": {
"outputDir": "dist/storybook/example-lib",
"configDir": "apps/example-lib/.storybook",
"browserTarget": "example-lib:build-storybook",
"styles": [".storybook/custom-styles.scss"],
"stylePreprocessorOptions": {
"includePaths": [
"libs/design-system/src/lib"
]
}
}
}
}
```
When Nx runs, it will load Storybook's configuration and styling based on the `storybook`'s [`browserTarget`](https://nx.dev/storybook/extra-topics-for-angular-projects#setting-up-browsertarget).

View File

@ -9,6 +9,12 @@ If you supply an environment variable prefixed with `STORYBOOK_`, it will be ava
STORYBOOK_THEME=red STORYBOOK_DATA_KEY=12345 npm run storybook
```
<div class="aside">
💡 Do not store any secrets (e.g., private API keys) or other types of sensitive information in your Storybook. Environment variables are embedded into the build, meaning anyone can view them by inspecting your files.
</div>
Then we can access these environment variables anywhere inside our preview JavaScript code like below:
<!-- prettier-ignore-start -->

View File

@ -178,7 +178,7 @@ You can also use Storybook's API to configure your project with TypeScript. Unde
| --------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| `stories` | The array of globs that indicates the [location of your story files](#configure-story-loading), relative to `main.ts` |
| `staticDirs` | Sets a list of directories of [static files](./images-and-assets.md#serving-static-files-via-storybook-configuration) to be loaded by Storybook <br/> `staticDirs:['../public']` |
| `addons` | Sets the list of [addons](https://storybook.js.org/addons/) loaded by Storybook <br/> `addons:['@storybook/addon-essentials']` |
| `addons` | Sets the list of [addons](https://storybook.js.org/addons/) loaded by Storybook <br/> `addons:['@storybook/addon-essentials']` |
| `typescript` | Configures how Storybook handles [TypeScript files](./typescript.md) <br/> `typescript: { check: false, checkOptions: {} }` |
| `framework` | Configures Storybook based on a set of framework-specific settings <br/> `framework:'@storybook/svelte'` |
| `core` | Configures Storybook's internal features.<br/> `core: { builder: 'webpack5' }` |
@ -186,6 +186,7 @@ You can also use Storybook's API to configure your project with TypeScript. Unde
| `refs` | Configures [Storybook composition](../sharing/storybook-composition.md) <br/> `refs:{ example: { title: 'ExampleStorybook', url:'https://your-url.com' } }` |
| `logLevel` | Configures Storybook's logs in the browser terminal. Useful for debugging <br/> `logLevel: 'debug'` |
| `webpackFinal` | Customize Storybook's [Webpack](./webpack.md) setup <br/> `webpackFinal: async (config:any) => { return config; }` |
| `env` | Defines custom Storybook [environment variables](./environment-variables.md#using-storybook-configuration). <br/> `env: (config) => ({...config, EXAMPLE_VAR: 'Example var' }),` |
## Configure story rendering

View File

@ -6,18 +6,21 @@ Contribute a new feature or bug fix to [Storybook's monorepo](https://github.com
## Initial setup
First [fork](https://docs.github.com/en/github/getting-started-with-github/quickstart/fork-a-repo) the Storybook repository then clone and build your fork locally. Run the following commands:
Start by [forking](https://docs.github.com/en/github/getting-started-with-github/quickstart/fork-a-repo) the Storybook monorepo and cloning it locally.
```shell
git clone https://github.com/your-username/storybook.git
cd storybook
yarn
yarn bootstrap --core
```
Navigate to the `storybook` directory and install the required dependencies with the following commands:
```shell
yarn && yarn bootstrap --core
```
## Run tests & examples
Once you've completed the [initial setup](#run-tests-&-examples), you should have a fully functional version of Storybook built on your local machine. Before making any code changes, it's helpful to verify that everything is working as it should. More specifically, the test suite and examples.
Once you've completed the [initial setup](#initial-setup), you should have a fully functional version of Storybook built on your local machine. Before making any code changes, it's helpful to verify that everything is working as it should. More specifically, the test suite and examples.
Run the following command to execute the tests:
@ -28,8 +31,7 @@ yarn test
Once the tests finish, check if the examples are working with the following commands:
```shell
cd examples/cra-ts-essentials
yarn storybook
cd examples/cra-ts-essentials && yarn storybook
```
<div class="aside">
@ -106,7 +108,7 @@ Storybook's monorepo is set up to rely on end-to-end testing with [Cypress](http
Before submitting your contribution, run the test suite one last time with:
```sh
```shell
yarn test
```
@ -133,7 +135,7 @@ We encourage bug reports to include reproductions. In the same way that it's pos
To do so, run the following command in the root of the monorepo:
```sh
```shell
npx sb@next link https://github.com/your-username/your-project.git
```
@ -141,7 +143,7 @@ This command creates a project `../storybook-repros/your-project`, and automatic
If you already have a reproduction on your local machine, you can similarly link it to your monorepo dev setup with the `--local` flag:
```sh
```shell
npx sb@next link --local /path/to/local-repro-directory
```

View File

@ -89,11 +89,15 @@ Go through the rest of the documentation and repeat the process.
Before submitting your contribution, we advise you to check your work against the Storybook website. Doing this prevents last-minute issues with the documentation and is also an excellent way for the maintainers to merge faster once you submit the pull request. However, failing to do so will lead one of the maintainers to notify you that your contribution has an issue.
Start by forking [frontpage repo](https://github.com/storybookjs/frontpage). Then, clone and install the dependencies with the following commands:
Start by forking [frontpage repo](https://github.com/storybookjs/frontpage) and cloning it locally.
```shell
git clone https://github.com/your-username/frontpage.git
cd frontpage
```
Navigate to the `frontpage` directory and install the required dependencies with the following command:
```shell
yarn
```

View File

@ -18,13 +18,16 @@ The interactions are written using a Storybook-instrumented version of [Testing
Since Interactions is still experimental, it doesn't yet ship with Storybook by default. As such, you'll have to install it. You may also want to add our wrappers for Testing Library and Jest.
```shell
# With npm
npm install @storybook/addon-interactions @storybook/jest @storybook/testing-library --save-dev
<!-- prettier-ignore-start -->
# With yarn
yarn add --dev @storybook/addon-interactions @storybook/jest @storybook/testing-library
```
<CodeSnippets
paths={[
'common/storybook-addon-interactions-addon-full-install.yarn.js.mdx',
'common/storybook-addon-interactions-addon-full-install.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Next, update [`.storybook/main.js`](../configure/overview.md#configure-story-rendering) to the following:

View File

@ -18,13 +18,16 @@ If you ran `sb init` to include Storybook in your project, the Essentials addon
If you're upgrading from a previous Storybook version, you'll need to run the following command in your terminal:
```shell
#With npm
npm install -D @storybook/addon-essentials
<!-- prettier-ignore-start -->
#With yarn
yarn add -D @storybook/addon-essentials
```
<CodeSnippets
paths={[
'common/storybook-addon-essentials-install.yarn.js.mdx',
'common/storybook-addon-essentials-install.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Update your Storybook configuration (in [`.storybook/main.js`](../configure/overview.md#configure-story-rendering)) to include the Essentials addon.
@ -44,14 +47,16 @@ Essentials is "zero-config”. It comes with a recommended configuration out of
If you need to reconfigure any of the [individual Essentials addons](https://storybook.js.org/addons/tag/essentials), install them manually by following the installation instructions, register them in your Storybook configuration file (i.e., [`.storybook/main.js`](../configure/overview.md#configure-story-rendering)) and adjust the configuration to suit your needs. For example:
```shell
#With npm
npm install -D @storybook/addon-actions
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/storybook-addon-actions-install.yarn.js.mdx',
'common/storybook-addon-actions-install.npm.js.mdx',
]}
/>
#With yarn
yarn add -D @storybook/addon-actions
```
<!-- prettier-ignore-end -->
<!-- prettier-ignore-start -->

View File

@ -472,3 +472,16 @@ export default {
},
};
```
### Why isn't Storybook's test runner working?
There's an issue with Storybook's test runner and the latest version of Jest (i.e., version 28), which prevents it from running effectively. As a workaround, you can downgrade Jest to the previous stable version (i.e., version 27), and you'll be able to run it. See the following [issue](https://github.com/storybookjs/test-runner/issues/99) for more information.
### How does Storybook handles enviroment variables?
Storybook has built-in support for [environment variables](./configure/environment-variables.md). By default, environment variables are only available in Node.js code and are not available in the browser as some variables should be kept secret (e.g., API keys) and **not** exposed to anyone visiting the published Storybook.
To expose a variable, you must preface its name with `STORYBOOK_`. So `STORYBOOK_API_URL` will be available in browser code but `API_KEY` will not. Additionally you can also customize which variables are exposed by setting the [`env`](./configure/environment-variables.md#using-storybook-configuration) field in the `.storybook/main.js` file.
Variables are set when JavaScript is compiled so when the development server is started or you build your Storybook. Environment variable files should not be committed to Git as they often contain secrets which are not safe to add to Git. Instead, add `.env.*` to your `.gitignore` file and set up the environment variables manually on your hosting provider (e.g., [GitHub](https://docs.github.com/en/actions/security-guides/encrypted-secrets#creating-encrypted-secrets-for-a-repository)).

View File

@ -90,7 +90,9 @@ Below are some of the most common installation issues and instructions on how to
<!-- prettier-ignore-end -->
<div class="aside">
Storybook collects completely anonymous data to help us improve user experience. Participation is optional, and you may [opt-out](../configure/telemetry.md#how-to-opt-out) if you'd not like to share any information.
</div>
If all else fails, try asking for [help](https://storybook.js.org/support)

View File

@ -21,7 +21,8 @@ First, we'll need to build Storybook as a static web application. The functional
paths={[
'angular/custom-build-script-production.script-for-builder.js.mdx',
'angular/build-storybook-production-mode.with-builder.js.mdx',
'common/build-storybook-production-mode.js.mdx',
'common/build-storybook-production-mode.yarn.js.mdx',
'common/build-storybook-production-mode.npm.js.mdx',
]}
/>
@ -49,13 +50,16 @@ To get started, sign up with your GitHub, GitLab, Bitbucket, or email and genera
Next, install the [Chromatic CLI](https://www.npmjs.com/package/chromatic) package from npm:
```shell
# With npm
npm install --save-dev chromatic
<!-- prettier-ignore-start -->
# With yarn
yarn add --dev chromatic
```
<CodeSnippets
paths={[
'common/chromatic-install.yarn.js.mdx',
'common/chromatic-install.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Run the following command after the package finishes installing. Make sure that you replace `your-project-token` with your own project token.

View File

@ -1,7 +0,0 @@
```shell
# With yarn
yarn build-storybook
# With npm
npm run build-storybook
```

View File

@ -0,0 +1,3 @@
```shell
npm run build-storybook
```

View File

@ -0,0 +1,3 @@
```shell
yarn build-storybook
```

View File

@ -2,7 +2,7 @@
// preview.js
// All stories expect a theme arg
export const argTypes = { theme: { control: { options: ['light', 'dark'] } } };
export const argTypes = { theme: { control: 'select', options: ['light', 'dark'] } };
// The default value of the theme arg to all stories
export const args = { theme: 'light' };

View File

@ -0,0 +1,3 @@
```shell
npm install chromatic --save-dev
```

View File

@ -0,0 +1,3 @@
```shell
yarn add --dev chromatic
```

View File

@ -0,0 +1,12 @@
```md
<!-- Changelog.stories.mdx -->
import { Meta } from "@storybook/addon-docs";
import Changelog from "../CHANGELOG.md";
<Meta title="Changelog" />
<Changelog />
```

View File

@ -0,0 +1,3 @@
```shell
npm install @storybook/addon-actions --save-dev
```

View File

@ -0,0 +1,3 @@
```shell
yarn add --dev @storybook/addon-actions
```

View File

@ -0,0 +1,3 @@
```shell
npm install @storybook/addon-actions --save-dev
```

View File

@ -0,0 +1,3 @@
```shell
yarn add --dev @storybook/addon-essentials
```

View File

@ -0,0 +1,3 @@
```shell
npm install @storybook/addon-storyshots --save-dev
```

View File

@ -0,0 +1,3 @@
```shell
yarn add --dev @storybook/addon-storyshots
```

View File

@ -0,0 +1,3 @@
```shell
npm run storybook -- --debug-webpack
```

View File

@ -0,0 +1,3 @@
```shell
yarn storybook --debug-webpack
```

View File

@ -0,0 +1,3 @@
```shell
npm run build-storybook -- --debug-webpack
```

View File

@ -0,0 +1,3 @@
```shell
yarn build-storybook --debug-webpack
```

View File

@ -0,0 +1,22 @@
```js
// .storybook/main.js|ts
module.exports = {
stories: [
//👇 Changes the load order of our stories. First loads the Changelog page
'../src/Changelog.stories.mdx',
'../stories/**/*.stories.mdx',
'../stories/**/*.stories.@(js|jsx|ts|tsx)'
],
addons: [
'@storybook/addon-links',
'@storybook/addon-essentials',
{
name: '@storybook/addon-docs',
options: {
transcludeMarkdown: true, //👈 Set transcludeMarkdown to true
},
},
],
};
```

View File

@ -2,8 +2,8 @@
// .storybook/main.js
module.exports = {
stories: [],
addons: [],
staticDirs: ['../public'],
stories: ['../src/**/*.stories.mdx', '../src/**/*.stories.@(js|jsx|ts|tsx)'],
addons: ['@storybook/addon-links', '@storybook/addon-essentials'],
staticDirs: ['../public'], //👈 Configures the static asset folder in Storybook
};
```

View File

@ -0,0 +1,3 @@
```shell
npx msw init public/
```

View File

@ -0,0 +1,3 @@
```shell
npm install msw msw-storybook-addon --save-dev
```

View File

@ -0,0 +1,3 @@
```shell
yarn add --dev msw msw-storybook-addon
```

View File

@ -0,0 +1,25 @@
```js
// .storybook/preview.js
import { initialize, mswDecorator } from 'msw-storybook-addon';
/*
* Initializes MSW
* See https://github.com/mswjs/msw-storybook-addon#configuring-msw
* to learn how to customize it
*/
initialize();
export const parameters = {
actions: { argTypesRegex: '^on[A-Z].*' },
controls: {
matchers: {
color: /(background|color)$/i,
date: /Date$/,
},
},
};
// Provide the MSW addon decorator globally
export const decorators = [mswDecorator];
```

View File

@ -1,3 +1,3 @@
```shell
npm install @storybook/test-runner jest --save-dev
npm install @storybook/test-runner jest@27 --save-dev
```

View File

@ -1,3 +1,3 @@
```shell
yarn add --dev @storybook/test-runner jest
yarn add --dev @storybook/test-runner jest@27
```

View File

@ -13,7 +13,7 @@ module.exports = {
// Merge custom configuration into the default config
return mergeConfig(config, {
// Use the same "resolve" configuration as your app
resolve: (await import('../vite.config.js')).default.resolve
resolve: (await import('../vite.config.js')).default.resolve,
// Add dependencies to pre-optimization
optimizeDeps: {
include: ['storybook-dark-mode'],

View File

@ -0,0 +1,3 @@
```shell
npm install react react-dom @babel/cli
```

View File

@ -0,0 +1,3 @@
```shell
yarn add react react-dom @babel/cli
```

Binary file not shown.

After

Width:  |  Height:  |  Size: 38 KiB

View File

@ -211,6 +211,35 @@ Write your documentation as you usually would, and your existing SCSS code block
<!-- prettier-ignore-end -->
### Creating a Changelog story
One common use case for MDX-only docs stories is importing a project's `CHANGELOG.md` into an MDX story, so that users can easily refer to the CHANGELOG via a documentation node in Storybook.
First, ensure that `transcludeMarkdown` is set to `true` in `main.js`:
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/storybook-main-enable-transcludemarkdown.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Then, import the markdown and treat the imported file as a component in the MDX file:
```mdx
import { Meta } from "@storybook/addon-docs";
import Changelog from "../CHANGELOG.md";
<Meta title="Changelog" />
<Changelog />
```
![Changelog markdown in an MDX story](./changelog-mdx-md-transcludemarkdown-optimized.png)
## Linking to other stories and pages
When writing MDX, you may want to provide links to other stories or documentation pages and sections. You can use the `path` query string.
@ -276,4 +305,4 @@ Update your Storybook configuration (in `.storybook/main.js|ts`) and add the `pr
]}
/>
<!-- prettier-ignore-end -->
<!-- prettier-ignore-end -->

View File

@ -2,7 +2,7 @@
title: 'Building pages with Storybook'
---
Storybook helps you build any component, from small “atomic” components to composed pages. But as you move up the component hierarchy toward the level of pages, you end up dealing with more complexity.
Storybook helps you build any component, from small “atomic” components to composed pages. But as you move up the component hierarchy toward the page level, you deal with more complexity.
There are many ways to build pages in Storybook. Here are common patterns and solutions.
@ -47,7 +47,7 @@ When you are building screens in this way, it is typical that the inputs of a co
<!-- prettier-ignore-end -->
In such cases it is natural to use [args composition](./args.md#args-composition) to build the stories for the page based on the stories of the sub-components:
In such cases, it is natural to use [args composition](./args.md#args-composition) to build the stories for the page based on the stories of the sub-components:
<!-- prettier-ignore-start -->
@ -72,13 +72,57 @@ If you need to render a connected component in Storybook, you can mock the netwo
### Mocking providers
If you are using a provider that supplies data via the context, you can wrap your story in a decorator that provides a mocked version of that provider. For example, in the [Screens](https://storybook.js.org/tutorials/intro-to-storybook/react/en/screen/) chapter of the Intro to Storybook tutorial, we mock a Redux provider with mock data.
Suppose you are using a provider that supplies data via the context. In that case, you can wrap your story in a decorator that provides a mocked version of that provider. For example, in the [Screens](https://storybook.js.org/tutorials/intro-to-storybook/react/en/screen/) chapter of the Intro to Storybook tutorial, we mock a Redux provider with mock data.
### Mocking API Services
Connected applications such as Twitter, Instagram, amongst others, are everywhere, consuming data either from REST or GraphQL endpoints. If you're working in an application that relies on either of these data providers, you can add Mock Service Worker (MSW) via [Storybook's MSW addon](https://storybook.js.org/addons/msw-storybook-addon) to mock data alongside your app and stories.
Connected applications such as Twitter, Instagram, amongst others, are everywhere, consuming data from REST or GraphQL endpoints. Suppose you're working in an application that relies on either of these data providers. In that case, you can add Mock Service Worker (MSW) via [Storybook's MSW addon](https://storybook.js.org/addons/msw-storybook-addon) to mock data alongside your app and stories.
[Mock Service Worker](https://mswjs.io/) is an API mocking library. It relies on service workers to capture network requests and provides mocked data in response. The MSW addon adds this functionality into Storybook, allowing you to mock API requests in your stories.
[Mock Service Worker](https://mswjs.io/) is an API mocking library. It relies on service workers to capture network requests and provides mocked data in response. The MSW addon adds this functionality into Storybook, allowing you to mock API requests in your stories. Below is an overview of how to set up and use the addon.
Run the following commands to install MSW, the addon, and generate a mock service worker.
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/storybook-msw-install.yarn.js.mdx',
'common/storybook-msw-install.npm.js.mdx',
'common/storybook-msw-generate.msw.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
<div class="aside">
💡 If you're working with Angular, you'll need to adjust the command to save the mock service worker file in a different directory (e.g., `src`).
</div>
Update your `.storybook/preview.js` file and enable the addon via a [global decorator](./decorators.md#global-decorators).
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/storybook-preview-register-msw-addon.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Finally, update your [`.storybook/main.js|ts`](../configure/overview.md#using-storybook-api) to allow Storybook to load the generated mock service worker file as follows:
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/storybook-main-with-single-static-dir.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
#### Mocking REST requests with MSW addon
@ -157,7 +201,7 @@ To test your screen with the GraphQL mocked data, you could write the following
It is also possible to mock imports directly, as you might in a unit test, using Webpacks aliasing. It's advantageous if your component makes network requests directly with third-party libraries.
We're going to use [isomorphic-fetch](https://www.npmjs.com/package/isomorphic-fetch) as an example.
We'll use [isomorphic-fetch](https://www.npmjs.com/package/isomorphic-fetch) as an example.
Inside a directory called `__mocks__`, create a new file called
`isomorphic-fetch.js` with the following code:
@ -223,7 +267,7 @@ Like the [import mocking](##mocking-imports) above, once you have a mock, you
It's possible to avoid mocking the dependencies of connected "container" components entirely by passing them around via props or React context. However, it requires a strict split of the container and presentational component logic. For example, if you have a component responsible for data fetching logic and rendering DOM, it will need to be mocked as previously described.
Its common to import and embed container components amongst presentational components. However, as we discovered earlier, to render them within Storybook, well likely have to mock their dependencies or the imports themselves.
Its common to import and embed container components amongst presentational components. However, as we discovered earlier, well likely have to mock their dependencies or the imports to render them within Storybook.
Not only can this quickly grow to become a tedious task, but its also challenging to mock container components that use local states. So, instead of importing containers directly, a solution to this problem is to create a React context that provides the container components. It allows you to freely embed container components as usual, at any level in the component hierarchy without worrying about subsequently mocking their dependencies; since we can swap out the containers themselves with their mocked presentational counterpart.
@ -238,7 +282,7 @@ ProfilePageContext.js
<div class="aside">
Its also often helpful to set up a “global” container context (perhaps named `GlobalContainerContext`) for container components that may be rendered on every page of your app and adding them to the top level of your application. While its possible to place every container within this global context, it should only provide globally required containers.
Its also often helpful to set up a “global” container context (perhaps named `GlobalContainerContext`) for container components that may be rendered on every page of your app and add them to the top level of your application. While its possible to place every container within this global context, it should only provide globally required containers.
</div>
@ -284,7 +328,7 @@ In the context of Storybook, instead of providing container components through c
<div class="aside">
If the same context applies to all `ProfilePage` stories, we can also use a [decorator](./decorators.md).
If the same context applies to all `ProfilePage` stories, we can use a [decorator](./decorators.md).
</div>

View File

@ -10,13 +10,16 @@ We recommend installing Storybook's [`addon-interactions`](https://storybook.js.
Run the following command to install the addon and the required dependencies.
```shell
# With npm
npm install @storybook/addon-interactions @storybook/testing-library --save-dev
<!-- prettier-ignore-start -->
# With yarn
yarn add --dev @storybook/addon-interactions @storybook/testing-library
```
<CodeSnippets
paths={[
'common/storybook-addon-interactions-addon-full-install.yarn.js.mdx',
'common/storybook-addon-interactions-addon-full-install.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Update your Storybook configuration (in `.storybook/main.js`) to include the interactions addon.

View File

@ -14,13 +14,16 @@ Storybook is a helpful tool for snapshot testing because every story is essentia
Run the following command to install Storyshots:
```shell
# With npm
npm install @storybook/addon-storyshots --save-dev
<!-- prettier-ignore-start -->
# With yarn
yarn add --dev @storybook/addon-storyshots
```
<CodeSnippets
paths={[
'common/storybook-addon-storyshots-install.yarn.js.mdx',
'common/storybook-addon-storyshots-install.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
Add a test file to your environment with the following contents to configure Storyshots:

View File

@ -20,13 +20,18 @@ To get started, sign up with your [GitHub](https://github.com/), [GitLab](https:
Next, install the [chromatic](https://www.npmjs.com/package/chromatic) CLI package from npm:
```shell
# With npm
npm install chromatic --save-dev
<!-- prettier-ignore-start -->
<CodeSnippets
paths={[
'common/chromatic-install.yarn.js.mdx',
'common/chromatic-install.npm.js.mdx',
]}
/>
<!-- prettier-ignore-end -->
# With yarn
yarn add --dev chromatic
```
Run the following command after the package finishes installing:

View File

@ -20,20 +20,4 @@ Live examples of these:
- [HTML](https://storybookjs.netlify.com/html-kitchen-sink/)
- [Svelte](https://storybookjs.netlify.com/svelte-kitchen-sink/)
- [Ember](https://storybookjs.netlify.com/ember-cli/)
- [Preact](https://storybookjs.netlify.com/preact-kitchen-sink/)
### 5.0
- [React Official](https://release-5-0--storybooks-official.netlify.com/)
- [Vue](https://release-5-0--storybooks-vue.netlify.com/)
- [Angular](https://release-5-0--storybooks-angular.netlify.com/)
- [HTML](https://release-5-0--storybooks-html.netlify.com/)
- [Svelte](https://release-5-0--storybooks-svelte.netlify.com/)
- [Ember](https://release-5-0--storybooks-ember.netlify.com/)
- [Preact](https://release-5-0--storybooks-preact.netlify.com/)
### 3.4
- [React Official](https://release-3-4--storybooks-official.netlify.com)
- [Vue](https://release-3-4--storybooks-vue.netlify.com/)
- [Angular](https://release-3-4--storybooks-angular.netlify.com/)
- [Preact](https://storybookjs.netlify.com/preact-kitchen-sink/)

View File

@ -11,6 +11,10 @@ const config: StorybookConfig = {
titlePrefix: 'Demo',
files: '*.stories.(js|ts|tsx|mdx)',
},
{
directory: '../src/addon-docs',
files: '*.stories.mdx',
},
],
logLevel: 'debug',
addons: [

View File

@ -4040,3 +4040,21 @@ exports[`Storyshots Demo/Examples / Emoji Button With Args 1`] = `
With args
</button>
`;
exports[`Storyshots Docs/ButtonMdx Basic 1`] = `
<button
type="button"
>
Click me
</button>
`;
exports[`Storyshots Docs/ButtonMdx Controls 1`] = `
<button
type="button"
>
Hello
</button>
`;

View File

@ -10,6 +10,7 @@ module.exports = {
disableTelemetry: true,
},
features: {
storyStoreV7: !global.navigator?.userAgent?.match?.('jsdom'),
buildStoriesJson: true,
channelOptions: { allowFunction: false, maxDepth: 10 },
},

View File

@ -13,6 +13,7 @@ module.exports = {
disableTelemetry: true,
},
features: {
storyStoreV7: true,
buildStoriesJson: true,
},
};

View File

@ -20,5 +20,6 @@ module.exports = {
staticDirs: ['../public'],
features: {
buildStoriesJson: true,
storyStoreV7: !global.navigator?.userAgent?.match?.('jsdom'),
},
};

View File

@ -10,4 +10,5 @@ export const parameters = {
docs: {
iframeHeight: '60px',
},
globalParameter: 'globalParameter',
};

View File

@ -30,7 +30,7 @@ export default {
}
</script>
<style>
<style scoped>
#app {
font-family: 'Avenir', Helvetica, Arial, sans-serif;
-webkit-font-smoothing: antialiased;

View File

@ -8,9 +8,9 @@ exports[`Storyshots Core/Parameters passed to story 1`] = `
"docs": {
"iframeHeight": "60px"
},
"framework": "vue",
"globalParameter": "globalParameter",
"chapterParameter": "chapterParameter",
"framework": "vue",
"componentParameter": "componentParameter",
"storyParameter": "storyParameter",
"__isArgsStory": true,
"__id": "core-parameters--passed-to-story",

View File

@ -53,8 +53,8 @@ exports[`Storyshots Custom/Decorator for Vue With Data 1`] = `
"docs": {
"iframeHeight": "60px"
},
"framework": "vue",
"globalParameter": "globalParameter",
"framework": "vue",
"__isArgsStory": true,
"__id": "custom-decorator-for-vue--with-data",
"args": {},

View File

@ -1,15 +1,10 @@
import { addParameters } from '@storybook/vue';
const globalParameter = 'globalParameter';
const chapterParameter = 'chapterParameter';
const componentParameter = 'componentParameter';
const storyParameter = 'storyParameter';
addParameters({ globalParameter });
export default {
title: 'Core/Parameters',
parameters: {
chapterParameter,
componentParameter,
},
};

View File

@ -20,8 +20,13 @@ export const entries = async (_: unknown, options: any) => {
return result;
};
export const babel = async (config: any, options: any) => {
// FIXME: Add this to overrides to only apply to story files
config.plugins.push('babel-plugin-named-exports-order');
return config;
};
export const babel = async (config: any, options: any) => ({
...config,
overrides: [
...(config?.overrides || []),
{
test: /\.(story|stories).*$/,
plugins: [require.resolve('babel-plugin-named-exports-order')],
},
],
});

View File

@ -6,7 +6,7 @@ import { angular12 } from './angular12';
// eslint-disable-next-line global-require, jest/no-mocks-import
jest.mock('fs-extra', () => require('../../../../../__mocks__/fs-extra'));
const checkCra5 = async ({ packageJson, main }) => {
const checkAngular12 = async ({ packageJson, main }) => {
// eslint-disable-next-line global-require
require('fs-extra').__setMockFiles({
[path.join('.storybook', 'main.js')]: `module.exports = ${JSON.stringify(main)};`,
@ -25,7 +25,7 @@ describe('angular12 fix', () => {
};
it('should fail', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson,
main: {},
})
@ -36,7 +36,7 @@ describe('angular12 fix', () => {
const packageJson = { dependencies: { '@storybook/react': '^6.2.0' } };
it('should no-op', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson,
main: {},
})
@ -52,7 +52,7 @@ describe('angular12 fix', () => {
describe('webpack5 builder', () => {
it('should no-op', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson,
main: { core: { builder: 'webpack5' } },
})
@ -62,7 +62,7 @@ describe('angular12 fix', () => {
describe('custom builder', () => {
it('should no-op', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson,
main: { core: { builder: 'storybook-builder-vite' } },
})
@ -72,7 +72,7 @@ describe('angular12 fix', () => {
describe('webpack4 builder', () => {
it('should add webpack5 builder', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson,
main: { core: { builder: 'webpack4' } },
})
@ -85,7 +85,7 @@ describe('angular12 fix', () => {
describe('no builder', () => {
it('should add webpack5 builder', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson,
main: {},
})
@ -99,7 +99,7 @@ describe('angular12 fix', () => {
describe('no angular dependency', () => {
it('should no-op', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson: {},
main: {},
})
@ -109,7 +109,7 @@ describe('angular12 fix', () => {
describe('angular11 dependency', () => {
it('should no-op', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson: {
dependencies: {
'@angular/core': '11',
@ -128,7 +128,7 @@ describe('angular12 fix', () => {
};
it('should no-op', async () => {
await expect(
checkCra5({
checkAngular12({
packageJson,
main: {},
})

View File

@ -1,10 +1,19 @@
import { cra5 } from './cra5';
import { webpack5 } from './webpack5';
import { angular12 } from './angular12';
import { vue3 } from './vue3';
import { mainjsFramework } from './mainjsFramework';
import { eslintPlugin } from './eslint-plugin';
import { builderVite } from './builder-vite';
import { Fix } from '../types';
export * from '../types';
export const fixes: Fix[] = [cra5, webpack5, angular12, mainjsFramework, eslintPlugin, builderVite];
export const fixes: Fix[] = [
cra5,
webpack5,
angular12,
vue3,
mainjsFramework,
eslintPlugin,
builderVite,
];

View File

@ -0,0 +1,139 @@
/* eslint-disable no-underscore-dangle */
import path from 'path';
import { JsPackageManager } from '../../js-package-manager';
import { vue3 } from './vue3';
// eslint-disable-next-line global-require, jest/no-mocks-import
jest.mock('fs-extra', () => require('../../../../../__mocks__/fs-extra'));
const checkVue3 = async ({ packageJson, main }) => {
// eslint-disable-next-line global-require
require('fs-extra').__setMockFiles({
[path.join('.storybook', 'main.js')]: `module.exports = ${JSON.stringify(main)};`,
});
const packageManager = {
retrievePackageJson: () => ({ dependencies: {}, devDependencies: {}, ...packageJson }),
} as JsPackageManager;
return vue3.check({ packageManager });
};
describe('vue3 fix', () => {
describe('sb < 6.3', () => {
describe('vue3 dependency', () => {
const packageJson = {
dependencies: { '@storybook/vue': '^6.2.0', vue: '^3.0.0' },
};
it('should fail', async () => {
await expect(
checkVue3({
packageJson,
main: {},
})
).rejects.toThrow();
});
});
describe('no vue dependency', () => {
const packageJson = { dependencies: { '@storybook/vue': '^6.2.0' } };
it('should no-op', async () => {
await expect(
checkVue3({
packageJson,
main: {},
})
).resolves.toBeFalsy();
});
});
});
describe('sb 6.3 - 7.0', () => {
describe('vue3 dependency', () => {
const packageJson = {
dependencies: { '@storybook/vue': '^6.3.0', vue: '^3.0.0' },
};
describe('webpack5 builder', () => {
it('should no-op', async () => {
await expect(
checkVue3({
packageJson,
main: { core: { builder: 'webpack5' } },
})
).resolves.toBeFalsy();
});
});
describe('custom builder', () => {
it('should no-op', async () => {
await expect(
checkVue3({
packageJson,
main: { core: { builder: 'storybook-builder-vite' } },
})
).resolves.toBeFalsy();
});
});
describe('webpack4 builder', () => {
it('should add webpack5 builder', async () => {
await expect(
checkVue3({
packageJson,
main: { core: { builder: 'webpack4' } },
})
).resolves.toMatchObject({
vueVersion: '^3.0.0',
storybookVersion: '^6.3.0',
});
});
});
describe('no builder', () => {
it('should add webpack5 builder', async () => {
await expect(
checkVue3({
packageJson,
main: {},
})
).resolves.toMatchObject({
vueVersion: '^3.0.0',
storybookVersion: '^6.3.0',
});
});
});
});
describe('no vue dependency', () => {
it('should no-op', async () => {
await expect(
checkVue3({
packageJson: {},
main: {},
})
).resolves.toBeFalsy();
});
});
describe('vue2 dependency', () => {
it('should no-op', async () => {
await expect(
checkVue3({
packageJson: {
dependencies: {
vue: '2',
},
},
main: {},
})
).resolves.toBeFalsy();
});
});
});
describe('sb 7.0+', () => {
describe('vue3 dependency', () => {
const packageJson = {
dependencies: { '@storybook/vue': '^7.0.0-alpha.0', vue: '^3.0.0' },
};
it('should no-op', async () => {
await expect(
checkVue3({
packageJson,
main: {},
})
).resolves.toBeFalsy();
});
});
});
});

View File

@ -0,0 +1,60 @@
import chalk from 'chalk';
import dedent from 'ts-dedent';
import semver from '@storybook/semver';
import { ConfigFile } from '@storybook/csf-tools';
import { Fix } from '../types';
import { webpack5 } from './webpack5';
interface Vue3RunOptions {
vueVersion: string;
storybookVersion: string;
main: ConfigFile;
}
/**
* Is the user upgrading to Vue3?
*
* If so:
* - Run webpack5 fix
*/
export const vue3: Fix<Vue3RunOptions> = {
id: 'vue3',
async check({ packageManager }) {
const packageJson = packageManager.retrievePackageJson();
const { dependencies, devDependencies } = packageJson;
const vueVersion = dependencies.vue || devDependencies.vue;
const vueCoerced = semver.coerce(vueVersion)?.version;
if (!vueCoerced || semver.lt(vueCoerced, '3.0.0')) {
return null;
}
const builderInfo = await webpack5.checkWebpack5Builder(packageJson);
return builderInfo ? { vueVersion, ...builderInfo } : null;
},
prompt({ vueVersion, storybookVersion }) {
const vueFormatted = chalk.cyan(`Vue ${vueVersion}`);
const sbFormatted = chalk.cyan(`Storybook ${storybookVersion}`);
return dedent`
We've detected you are running ${vueFormatted} with Storybook.
${sbFormatted} runs webpack4 by default, which is incompatible.
In order to work with your version of Vue, we need to install Storybook's ${chalk.cyan(
'webpack5 builder'
)}.
More info: ${chalk.yellow(
'https://github.com/storybookjs/storybook/blob/next/MIGRATION.md#vue3-upgrade'
)}
`;
},
async run(options) {
return webpack5.run({
...options,
result: { webpackVersion: null, ...options.result },
});
},
};

View File

@ -18,39 +18,45 @@ export const automigrate = async ({ fixId, dryRun, yes }: FixOptions = {}) => {
const packageManager = JsPackageManagerFactory.getPackageManager();
const filtered = fixId ? fixes.filter((f) => f.id === fixId) : fixes;
logger.info('🔎 checking possible migrations..');
for (let i = 0; i < filtered.length; i += 1) {
const f = fixes[i] as Fix;
logger.info(`🔎 checking '${chalk.cyan(f.id)}'`);
const result = await f.check({ packageManager });
if (result) {
logger.info(`🔎 found a '${chalk.cyan(f.id)}' migration:`);
logger.info();
const message = f.prompt(result);
logger.info(
boxen(message, { borderStyle: 'round', padding: 1, borderColor: '#F1618C' } as any)
);
const runAnswer =
yes || dryRun
? { fix: false }
: await prompts([
{
type: 'confirm',
name: 'fix',
message: `Do you want to run the '${chalk.cyan(f.id)}' fix on your project?`,
},
]);
let runAnswer: { fix: boolean };
if (dryRun) {
runAnswer = { fix: false };
} else if (yes) {
runAnswer = { fix: true };
} else {
runAnswer = await prompts({
type: 'confirm',
name: 'fix',
message: `Do you want to run the '${chalk.cyan(f.id)}' migration on your project?`,
});
}
if (runAnswer.fix) {
try {
await f.run({ result, packageManager, dryRun });
logger.info(`fixed ${chalk.cyan(f.id)}`);
logger.info(`ran ${chalk.cyan(f.id)} migration`);
} catch (error) {
logger.info(`❌ error in ${chalk.cyan(f.id)}:`);
logger.info(`❌ error when running ${chalk.cyan(f.id)} migration:`);
logger.info(error.message);
logger.info();
}
} else {
logger.info(`Skipping the ${chalk.cyan(f.id)} fix.`);
logger.info(`Skipping the ${chalk.cyan(f.id)} migration.`);
logger.info();
logger.info(
`If you change your mind, run '${chalk.cyan('npx storybook@next automigrate')}'`
@ -58,4 +64,8 @@ export const automigrate = async ({ fixId, dryRun, yes }: FixOptions = {}) => {
}
}
}
logger.info();
logger.info('✅ migration check successfully ran');
logger.info();
};

View File

@ -10,7 +10,7 @@ type TsConfig = {
export function getAngularAppTsConfigPath() {
const angularJson = readFileAsJson('angular.json', true);
const { defaultProject } = angularJson;
const defaultProject = getDefaultProjectName(angularJson);
const tsConfigPath = angularJson.projects[defaultProject].architect.build.options.tsConfig;
if (!tsConfigPath || !fs.existsSync(path.resolve(tsConfigPath))) {
@ -50,9 +50,33 @@ export function editStorybookTsConfig(tsconfigPath: string) {
writeFileAsJson(tsconfigPath, tsConfigJson);
}
export function isDefaultProjectSet() {
const angularJson = readFileAsJson('angular.json', true);
return angularJson && !!angularJson.defaultProject;
export function getDefaultProjectName(angularJson: any): string | undefined {
const { defaultProject, projects } = angularJson;
if (projects?.storybook) {
return 'storybook';
}
if (defaultProject) {
return defaultProject;
}
const firstProjectName = projects ? Object.keys(projects)[0] : undefined;
if (firstProjectName) {
return firstProjectName;
}
return undefined;
}
export function checkForProjects() {
const { projects } = readFileAsJson('angular.json', true);
if (!projects || Object.keys(projects).length === 0) {
throw new Error(
'Could not find a project in your Angular workspace. \nAdd a project and re-run the installation'
);
}
}
export async function getBaseTsConfigName() {

View File

@ -1,7 +1,7 @@
import path from 'path';
import semver from '@storybook/semver';
import {
isDefaultProjectSet,
checkForProjects,
editStorybookTsConfig,
getAngularAppTsConfigJson,
getAngularAppTsConfigPath,
@ -28,11 +28,8 @@ function editAngularAppTsConfig() {
}
const generator: Generator = async (packageManager, npmOptions, options) => {
if (!isDefaultProjectSet()) {
throw new Error(
'Could not find a default project in your Angular workspace.\nSet a defaultProject in your angular.json and re-run the installation.'
);
}
checkForProjects();
const angularVersion = semver.coerce(
packageManager.retrievePackageJson().dependencies['@angular/core']
)?.version;

View File

@ -66,21 +66,6 @@ const installStorybook = (
commonJs: options.commonJs,
};
const end = () => {
if (!options.skipInstall) {
packageManager.installDependencies();
}
logger.log('\nTo run your Storybook, type:\n');
codeLog([packageManager.getRunStorybookCommand()]);
logger.log('\nFor more information visit:', chalk.cyan('https://storybook.js.org'));
// Add a new line for the clear visibility.
logger.log();
};
const REACT_NATIVE_REPO = 'https://github.com/storybookjs/react-native';
const runGenerator: () => Promise<void> = () => {
switch (projectType) {
case ProjectType.ALREADY_HAS_STORYBOOK:
@ -96,18 +81,17 @@ const installStorybook = (
case ProjectType.UPDATE_PACKAGE_ORGANIZATIONS:
return updateOrganisationsGenerator(packageManager, options.parser, npmOptions)
.then(() => null) // commandLog doesn't like to see output
.then(commandLog('Upgrading your project to the new Storybook packages.\n'))
.then(end);
.then(commandLog('Upgrading your project to the new Storybook packages.\n'));
case ProjectType.REACT_SCRIPTS:
return reactScriptsGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Create React App" based project'))
.then(end);
return reactScriptsGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Create React App" based project')
);
case ProjectType.REACT:
return reactGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "React" app\n'))
.then(end);
return reactGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "React" app\n')
);
case ProjectType.REACT_NATIVE: {
return (
@ -124,112 +108,103 @@ const installStorybook = (
]) as Promise<{ server: boolean }>)
)
.then(({ server }) => reactNativeGenerator(packageManager, npmOptions, server))
.then(commandLog('Adding Storybook support to your "React Native" app\n'))
.then(end)
.then(() => {
logger.log(chalk.red('NOTE: installation is not 100% automated.'));
logger.log(`To quickly run Storybook, replace contents of your app entry with:\n`);
codeLog(["export {default} from './storybook';"]);
logger.log('\n For more in information, see the github readme:\n');
logger.log(chalk.cyan(REACT_NATIVE_REPO));
logger.log();
});
.then(commandLog('Adding Storybook support to your "React Native" app\n'));
}
case ProjectType.METEOR:
return meteorGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Meteor" app\n'))
.then(end);
return meteorGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Meteor" app\n')
);
case ProjectType.WEBPACK_REACT:
return webpackReactGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Webpack React" app\n'))
.then(end);
return webpackReactGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Webpack React" app\n')
);
case ProjectType.REACT_PROJECT:
return reactGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "React" library\n'))
.then(end);
return reactGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "React" library\n')
);
case ProjectType.SFC_VUE:
return sfcVueGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Single File Components Vue" app\n'))
.then(end);
return sfcVueGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Single File Components Vue" app\n')
);
case ProjectType.VUE:
return vueGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Vue" app\n'))
.then(end);
return vueGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Vue" app\n')
);
case ProjectType.VUE3:
return vue3Generator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Vue 3" app\n'))
.then(end);
return vue3Generator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Vue 3" app\n')
);
case ProjectType.ANGULAR:
return angularGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Angular" app\n'))
.then(end);
return angularGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Angular" app\n')
);
case ProjectType.EMBER:
return emberGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Ember" app\n'))
.then(end);
return emberGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Ember" app\n')
);
case ProjectType.MITHRIL:
return mithrilGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Mithril" app\n'))
.then(end);
return mithrilGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Mithril" app\n')
);
case ProjectType.MARIONETTE:
return marionetteGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Marionette.js" app\n'))
.then(end);
return marionetteGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Marionette.js" app\n')
);
case ProjectType.MARKO:
return markoGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Marko" app\n'))
.then(end);
return markoGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Marko" app\n')
);
case ProjectType.HTML:
return htmlGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "HTML" app\n'))
.then(end);
return htmlGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "HTML" app\n')
);
case ProjectType.WEB_COMPONENTS:
return webComponentsGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "web components" app\n'))
.then(end);
return webComponentsGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "web components" app\n')
);
case ProjectType.RIOT:
return riotGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "riot.js" app\n'))
.then(end);
return riotGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "riot.js" app\n')
);
case ProjectType.PREACT:
return preactGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Preact" app\n'))
.then(end);
return preactGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Preact" app\n')
);
case ProjectType.SVELTE:
return svelteGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Svelte" app\n'))
.then(end);
return svelteGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Svelte" app\n')
);
case ProjectType.RAX:
return raxGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Rax" app\n'))
.then(end);
return raxGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Rax" app\n')
);
case ProjectType.AURELIA:
return aureliaGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Aurelia" app\n'))
.then(end);
return aureliaGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Aurelia" app\n')
);
case ProjectType.SERVER:
return serverGenerator(packageManager, npmOptions, generatorOptions)
.then(commandLog('Adding Storybook support to your "Server" app\n'))
.then(end);
return serverGenerator(packageManager, npmOptions, generatorOptions).then(
commandLog('Adding Storybook support to your "Server" app\n')
);
case ProjectType.UNSUPPORTED:
paddedLog(`We detected a project type that we don't support yet.`);
@ -294,7 +269,7 @@ const projectTypeInquirer = async (
export async function initiate(options: CommandOptions, pkg: Package): Promise<void> {
const packageManager = JsPackageManagerFactory.getPackageManager(options.useNpm);
const welcomeMessage = 'sb init - the simplest way to add a Storybook to your project.';
const welcomeMessage = 'storybook init - the simplest way to add a Storybook to your project.';
logger.log(chalk.inverse(`\n ${welcomeMessage} \n`));
if (!options.disableTelemetry) {
@ -337,7 +312,6 @@ export async function initiate(options: CommandOptions, pkg: Package): Promise<v
} catch (ex) {
done(ex.message);
process.exit(1);
return;
}
done();
@ -346,5 +320,28 @@ export async function initiate(options: CommandOptions, pkg: Package): Promise<v
...(isEsm ? { commonJs: true } : undefined),
});
await automigrate();
if (!options.skipInstall) {
packageManager.installDependencies();
}
await automigrate({ yes: process.env.CI === 'true' });
logger.log('\nTo run your Storybook, type:\n');
codeLog([packageManager.getRunStorybookCommand()]);
logger.log('\nFor more information visit:', chalk.cyan('https://storybook.js.org'));
if (projectType === ProjectType.REACT_NATIVE) {
const REACT_NATIVE_REPO = 'https://github.com/storybookjs/react-native';
logger.log();
logger.log(chalk.red('NOTE: installation is not 100% automated.'));
logger.log(`To quickly run Storybook, replace contents of your app entry with:\n`);
codeLog(["export {default} from './storybook';"]);
logger.log('\n For more in information, see the github readme:\n');
logger.log(chalk.cyan(REACT_NATIVE_REPO));
logger.log();
}
// Add a new line for the clear visibility.
logger.log();
}

View File

@ -13,6 +13,11 @@ export interface Parameters {
autoDetect?: boolean;
/** Dependencies to add before building Storybook */
additionalDeps?: string[];
/** Files to add before installing Storybook */
additionalFiles?: {
path: string;
contents: string;
}[];
/** Add typescript dependency and creates a tsconfig.json file */
typescript?: boolean;
/** Merge configurations to main.js before running the tests */
@ -119,7 +124,7 @@ const baseAngular: Parameters = {
framework: 'angular',
name: 'angular',
version: 'latest',
generator: `npx -p @angular/cli@{{version}} ng new {{appName}} --routing=true --minimal=true --style=scss --skipInstall=true --strict`,
generator: `npx -p @angular/cli@{{version}} ng new {{appName}} --routing=true --minimal=true --style=scss --skip-install=true --strict`,
};
export const angular10: Parameters = {
@ -155,7 +160,7 @@ export const angular13: Parameters = {
export const angular_modern_inline_rendering: Parameters = {
...baseAngular,
name: 'angular_modern_inline_rendering',
additionalDeps: ['jest', '@storybook/test-runner'],
additionalDeps: ['jest@27', '@storybook/test-runner'],
mainOverrides: {
features: {
storyStoreV7: true,

View File

@ -1,5 +1,5 @@
import path from 'path';
import { readJSON, writeJSON } from 'fs-extra';
import { readJSON, writeJSON, outputFile } from 'fs-extra';
import shell, { ExecOptions } from 'shelljs';
import chalk from 'chalk';
import { cra, cra_typescript } from './configs';
@ -22,6 +22,11 @@ export interface Parameters {
ensureDir?: boolean;
/** Dependencies to add before building Storybook */
additionalDeps?: string[];
/** Files to add before installing Storybook */
additionalFiles?: {
path: string;
contents: string;
}[];
/** Add typescript dependency and creates a tsconfig.json file */
typescript?: boolean;
}
@ -136,6 +141,16 @@ const generate = async ({ cwd, name, appName, version, generator }: Options) =>
);
};
const addAdditionalFiles = async ({ additionalFiles, cwd }: Options) => {
logger.info(`⤵️ Adding required files`);
await Promise.all(
additionalFiles.map(async (file) => {
await outputFile(path.resolve(cwd, file.path), file.contents, { encoding: 'UTF-8' });
})
);
};
const initStorybook = async ({ cwd, autoDetect = true, name, e2e }: Options) => {
const type = autoDetect ? '' : `--type ${name}`;
const linkable = e2e ? '' : '--linkable';
@ -228,6 +243,7 @@ export const createAndInit = async (
logger.log();
await doTask(generate, { ...options, cwd: options.creationPath });
await doTask(addAdditionalFiles, { ...options, cwd }, !!options.additionalFiles);
if (e2e) {
await doTask(addPackageResolutions, options);
}

View File

@ -1,6 +1,6 @@
import React, { FC } from 'react';
import pickBy from 'lodash/pickBy';
import { styled, ignoreSsrWarning } from '@storybook/theming';
import { styled } from '@storybook/theming';
import { opacify, transparentize, darken, lighten } from 'polished';
import { includeConditionalArg } from '@storybook/csf';
import { once } from '@storybook/client-logger';
@ -111,20 +111,20 @@ export const TableWrapper = styled.table<{
marginLeft: inAddonPanel ? 0 : 1,
marginRight: inAddonPanel ? 0 : 1,
[`tr:first-child${ignoreSsrWarning}`]: {
[`td:first-child${ignoreSsrWarning}, th:first-child${ignoreSsrWarning}`]: {
[`tr:first-child`]: {
[`td:first-child, th:first-child`]: {
borderTopLeftRadius: inAddonPanel ? 0 : theme.appBorderRadius,
},
[`td:last-child${ignoreSsrWarning}, th:last-child${ignoreSsrWarning}`]: {
[`td:last-child, th:last-child`]: {
borderTopRightRadius: inAddonPanel ? 0 : theme.appBorderRadius,
},
},
[`tr:last-child${ignoreSsrWarning}`]: {
[`td:first-child${ignoreSsrWarning}, th:first-child${ignoreSsrWarning}`]: {
[`tr:last-child`]: {
[`td:first-child, th:first-child`]: {
borderBottomLeftRadius: inAddonPanel ? 0 : theme.appBorderRadius,
},
[`td:last-child${ignoreSsrWarning}, th:last-child${ignoreSsrWarning}`]: {
[`td:last-child, th:last-child`]: {
borderBottomRightRadius: inAddonPanel ? 0 : theme.appBorderRadius,
},
},
@ -172,7 +172,7 @@ export const TableWrapper = styled.table<{
: lighten(0.05, theme.background.content),
}
: {
[`&:not(:first-child${ignoreSsrWarning})`]: {
[`&:not(:first-child)`]: {
borderTopWidth: 1,
borderTopStyle: 'solid',
borderTopColor:

View File

@ -1,5 +1,5 @@
import React, { ComponentProps, FunctionComponent } from 'react';
import { styled, ThemeProvider, convert, themes, ignoreSsrWarning } from '@storybook/theming';
import { styled, ThemeProvider, convert, themes } from '@storybook/theming';
import { EmptyBlock } from './EmptyBlock';
import { SyntaxHighlighter } from '../syntaxhighlighter/lazy-syntaxhighlighter';
@ -52,7 +52,7 @@ const SourceSkeletonPlaceholder = styled.div<{}>(({ theme }) => ({
marginTop: 1,
width: '60%',
[`&:first-child${ignoreSsrWarning}`]: {
[`&:first-child`]: {
margin: 0,
},
}));

View File

@ -1,5 +1,5 @@
import React, { FunctionComponent } from 'react';
import { styled, ignoreSsrWarning } from '@storybook/theming';
import { styled } from '@storybook/theming';
const toNumber = (input: any) => (typeof input === 'number' ? input : Number(input));
@ -19,7 +19,7 @@ const Container = styled.div<ContainerProps>(
marginLeft: col * theme.layoutMargin,
verticalAlign: 'inherit',
},
[`& > *:first-child${ignoreSsrWarning}`]: {
[`& > *:first-child`]: {
marginLeft: 0,
},
}
@ -27,7 +27,7 @@ const Container = styled.div<ContainerProps>(
'& > *': {
marginTop: row * theme.layoutMargin,
},
[`& > *:first-child${ignoreSsrWarning}`]: {
[`& > *:first-child`]: {
marginTop: 0,
},
},

View File

@ -15,9 +15,6 @@ import { Placeholder } from '../placeholder/placeholder';
import { FlexBar } from '../bar/bar';
import { TabButton } from '../bar/button';
const ignoreSsrWarning =
'/* emotion-disable-server-rendering-unsafe-selector-warning-please-do-not-use-this-the-warning-exists-for-a-reason */';
export interface WrapperProps {
bordered?: boolean;
absolute?: boolean;
@ -87,7 +84,7 @@ const Content = styled.div<ContentProps>(
bottom: 0 + (bordered ? 1 : 0),
top: 40 + (bordered ? 1 : 0),
overflow: 'auto',
[`& > *:first-child${ignoreSsrWarning}`]: {
[`& > *:first-child`]: {
position: 'absolute',
left: 0 + (bordered ? 1 : 0),
right: 0 + (bordered ? 1 : 0),

View File

@ -6,6 +6,7 @@ import { PackageJson } from '../types';
interface StorybookInfo {
framework: string;
version: string;
frameworkPackage: string;
configDir?: string;
mainConfig?: string;
previewConfig?: string;
@ -57,7 +58,7 @@ const getFrameworkInfo = (packageJson: PackageJson) => {
);
}
return { framework, version };
return { framework, version, frameworkPackage: pkg };
};
const validConfigExtensions = ['ts', 'js', 'tsx', 'jsx', 'mjs', 'cjs'];

View File

@ -68,7 +68,7 @@
"fs-extra": "^9.0.1",
"global": "^4.4.0",
"globby": "^11.0.2",
"ip": "^1.1.5",
"ip": "^2.0.0",
"lodash": "^4.17.21",
"node-fetch": "^2.6.7",
"open": "^8.4.0",

View File

@ -7,7 +7,7 @@ import dedent from 'ts-dedent';
export function openInBrowser(address: string) {
getDefaultBrowser(async (err: any, res: any) => {
try {
if (res.isChrome || res.isChromium) {
if (res && (res.isChrome || res.isChromium)) {
// We use betterOpn for Chrome because it is better at handling which chrome tab
// or window the preview loads in.
betterOpn(address);

View File

@ -48,7 +48,7 @@
"@babel/traverse": "^7.12.11",
"@babel/types": "^7.12.11",
"@storybook/csf": "0.0.2--canary.4566f4d.1",
"@storybook/mdx1-csf": "canary",
"@storybook/mdx1-csf": "^0.0.1",
"core-js": "^3.8.2",
"fs-extra": "^9.0.1",
"global": "^4.4.0",
@ -56,12 +56,12 @@
"ts-dedent": "^2.0.0"
},
"devDependencies": {
"@storybook/mdx2-csf": "canary",
"@storybook/mdx2-csf": "^0.0.3",
"@types/fs-extra": "^9.0.6",
"js-yaml": "^3.14.1"
},
"peerDependencies": {
"@storybook/mdx2-csf": "*"
"@storybook/mdx2-csf": "^0.0.3"
},
"peerDependenciesMeta": {
"@storybook/mdx2-csf": {

View File

@ -1,7 +1,18 @@
import dedent from 'ts-dedent';
import global from 'global';
import { SynchronousPromise } from 'synchronous-promise';
import Events from '@storybook/core-events';
import {
CONFIG_ERROR,
FORCE_REMOUNT,
FORCE_RE_RENDER,
GLOBALS_UPDATED,
RESET_STORY_ARGS,
SET_GLOBALS,
STORY_ARGS_UPDATED,
STORY_INDEX_INVALIDATED,
UPDATE_GLOBALS,
UPDATE_STORY_ARGS,
} from '@storybook/core-events';
import { logger } from '@storybook/client-logger';
import { addons, Channel } from '@storybook/addons';
import { AnyFramework, StoryId, ProjectAnnotations, Args, Globals } from '@storybook/csf';
@ -80,13 +91,13 @@ export class Preview<TFramework extends AnyFramework> {
}
setupListeners() {
this.serverChannel?.on(Events.STORY_INDEX_INVALIDATED, this.onStoryIndexChanged.bind(this));
this.serverChannel?.on(STORY_INDEX_INVALIDATED, this.onStoryIndexChanged.bind(this));
this.channel.on(Events.UPDATE_GLOBALS, this.onUpdateGlobals.bind(this));
this.channel.on(Events.UPDATE_STORY_ARGS, this.onUpdateArgs.bind(this));
this.channel.on(Events.RESET_STORY_ARGS, this.onResetArgs.bind(this));
this.channel.on(Events.FORCE_RE_RENDER, this.onForceReRender.bind(this));
this.channel.on(Events.FORCE_REMOUNT, this.onForceRemount.bind(this));
this.channel.on(UPDATE_GLOBALS, this.onUpdateGlobals.bind(this));
this.channel.on(UPDATE_STORY_ARGS, this.onUpdateArgs.bind(this));
this.channel.on(RESET_STORY_ARGS, this.onResetArgs.bind(this));
this.channel.on(FORCE_RE_RENDER, this.onForceReRender.bind(this));
this.channel.on(FORCE_REMOUNT, this.onForceRemount.bind(this));
}
getProjectAnnotationsOrRenderError(
@ -144,7 +155,7 @@ export class Preview<TFramework extends AnyFramework> {
}
emitGlobals() {
this.channel.emit(Events.SET_GLOBALS, {
this.channel.emit(SET_GLOBALS, {
globals: this.storyStore.globals.get() || {},
globalTypes: this.storyStore.projectAnnotations.globalTypes || {},
});
@ -227,7 +238,7 @@ export class Preview<TFramework extends AnyFramework> {
await Promise.all(this.storyRenders.map((r) => r.rerender()));
this.channel.emit(Events.GLOBALS_UPDATED, {
this.channel.emit(GLOBALS_UPDATED, {
globals: this.storyStore.globals.get(),
initialGlobals: this.storyStore.globals.initialGlobals,
});
@ -238,7 +249,7 @@ export class Preview<TFramework extends AnyFramework> {
await Promise.all(this.storyRenders.filter((r) => r.id === storyId).map((r) => r.rerender()));
this.channel.emit(Events.STORY_ARGS_UPDATED, {
this.channel.emit(STORY_ARGS_UPDATED, {
storyId,
args: this.storyStore.args.get(storyId),
});
@ -344,6 +355,6 @@ export class Preview<TFramework extends AnyFramework> {
this.previewEntryError = err;
logger.error(reason);
logger.error(err);
this.channel.emit(Events.CONFIG_ERROR, err);
this.channel.emit(CONFIG_ERROR, err);
}
}

View File

@ -76,8 +76,9 @@ describe('PreviewWeb', () => {
const preview = new PreviewWeb();
const docsRoot = window.document.createElement('div');
// @ts-ignore
preview.view.prepareForDocs.mockReturnValue(docsRoot);
(
preview.view.prepareForDocs as any as jest.Mock<typeof preview.view.prepareForDocs>
).mockReturnValue(docsRoot);
componentOneExports.default.parameters.docs.container.mockImplementationOnce(() =>
React.createElement('div', {}, 'INSIDE')
);

View File

@ -1,5 +1,12 @@
import { EventEmitter } from 'events';
import Events from '@storybook/core-events';
import {
DOCS_RENDERED,
STORY_ERRORED,
STORY_MISSING,
STORY_RENDERED,
STORY_RENDER_PHASE_CHANGED,
STORY_THREW_EXCEPTION,
} from '@storybook/core-events';
import { StoryIndex } from '@storybook/store';
import { RenderPhase } from './PreviewWeb';
@ -99,15 +106,15 @@ export const waitForEvents = (
// the async parts, so we need to listen for the "done" events
export const waitForRender = () =>
waitForEvents([
Events.STORY_RENDERED,
Events.DOCS_RENDERED,
Events.STORY_THREW_EXCEPTION,
Events.STORY_ERRORED,
Events.STORY_MISSING,
STORY_RENDERED,
DOCS_RENDERED,
STORY_THREW_EXCEPTION,
STORY_ERRORED,
STORY_MISSING,
]);
export const waitForRenderPhase = (phase: RenderPhase) =>
waitForEvents([Events.STORY_RENDER_PHASE_CHANGED], ({ newPhase }) => newPhase === phase);
waitForEvents([STORY_RENDER_PHASE_CHANGED], ({ newPhase }) => newPhase === phase);
// A little trick to ensure that we always call the real `setTimeout` even when timers are mocked
const realSetTimeout = setTimeout;

File diff suppressed because it is too large Load Diff

View File

@ -1,7 +1,24 @@
import deprecate from 'util-deprecate';
import dedent from 'ts-dedent';
import global from 'global';
import Events, { IGNORED_EXCEPTION } from '@storybook/core-events';
import {
CURRENT_STORY_WAS_SET,
IGNORED_EXCEPTION,
PRELOAD_STORIES,
PREVIEW_KEYDOWN,
SET_CURRENT_STORY,
SET_STORIES,
STORY_ARGS_UPDATED,
STORY_CHANGED,
STORY_ERRORED,
STORY_MISSING,
STORY_PREPARED,
STORY_RENDER_PHASE_CHANGED,
STORY_SPECIFIED,
STORY_THREW_EXCEPTION,
STORY_UNCHANGED,
UPDATE_QUERY_PARAMS,
} from '@storybook/core-events';
import { logger } from '@storybook/client-logger';
import { AnyFramework, StoryId, ProjectAnnotations, Args, Globals } from '@storybook/csf';
import type {
@ -64,9 +81,9 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
globalWindow.onkeydown = this.onKeydown.bind(this);
this.channel.on(Events.SET_CURRENT_STORY, this.onSetCurrentStory.bind(this));
this.channel.on(Events.UPDATE_QUERY_PARAMS, this.onUpdateQueryParams.bind(this));
this.channel.on(Events.PRELOAD_STORIES, this.onPreloadStories.bind(this));
this.channel.on(SET_CURRENT_STORY, this.onSetCurrentStory.bind(this));
this.channel.on(UPDATE_QUERY_PARAMS, this.onUpdateQueryParams.bind(this));
this.channel.on(PRELOAD_STORIES, this.onPreloadStories.bind(this));
}
initializeWithProjectAnnotations(projectAnnotations: WebProjectAnnotations<TFramework>) {
@ -87,7 +104,7 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
initializeWithStoryIndex(storyIndex: StoryIndex): PromiseLike<void> {
return super.initializeWithStoryIndex(storyIndex).then(() => {
if (!global.FEATURES?.storyStoreV7) {
this.channel.emit(Events.SET_STORIES, this.storyStore.getSetStoriesPayload());
this.channel.emit(SET_STORIES, this.storyStore.getSetStoriesPayload());
}
return this.selectSpecifiedStory();
@ -130,9 +147,9 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
}
this.urlStore.setSelection({ storyId, viewMode });
this.channel.emit(Events.STORY_SPECIFIED, this.urlStore.selection);
this.channel.emit(STORY_SPECIFIED, this.urlStore.selection);
this.channel.emit(Events.CURRENT_STORY_WAS_SET, this.urlStore.selection);
this.channel.emit(CURRENT_STORY_WAS_SET, this.urlStore.selection);
await this.renderSelection({ persistedArgs: args });
}
@ -161,7 +178,7 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
super.onStoriesChanged({ importFn, storyIndex });
if (!global.FEATURES?.storyStoreV7) {
this.channel.emit(Events.SET_STORIES, await this.storyStore.getSetStoriesPayload());
this.channel.emit(SET_STORIES, await this.storyStore.getSetStoriesPayload());
}
if (this.urlStore.selection) {
@ -176,15 +193,15 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
if (!this.currentRender?.disableKeyListeners && !focusInInput(event)) {
// We have to pick off the keys of the event that we need on the other side
const { altKey, ctrlKey, metaKey, shiftKey, key, code, keyCode } = event;
this.channel.emit(Events.PREVIEW_KEYDOWN, {
this.channel.emit(PREVIEW_KEYDOWN, {
event: { altKey, ctrlKey, metaKey, shiftKey, key, code, keyCode },
});
}
}
onSetCurrentStory(selection: Selection) {
this.urlStore.setSelection(selection);
this.channel.emit(Events.CURRENT_STORY_WAS_SET, this.urlStore.selection);
this.urlStore.setSelection({ viewMode: 'story', ...selection });
this.channel.emit(CURRENT_STORY_WAS_SET, this.urlStore.selection);
this.renderSelection();
}
@ -290,7 +307,7 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
// Don't re-render the story if nothing has changed to justify it
if (lastRender && !storyIdChanged && !implementationChanged && !viewModeChanged) {
this.currentRender = lastRender;
this.channel.emit(Events.STORY_UNCHANGED, storyId);
this.channel.emit(STORY_UNCHANGED, storyId);
this.view.showMain();
return;
}
@ -301,11 +318,11 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
// If we are rendering something new (as opposed to re-rendering the same or first story), emit
if (lastSelection && (storyIdChanged || viewModeChanged)) {
this.channel.emit(Events.STORY_CHANGED, storyId);
this.channel.emit(STORY_CHANGED, storyId);
}
if (global.FEATURES?.storyStoreV7) {
this.channel.emit(Events.STORY_PREPARED, {
this.channel.emit(STORY_PREPARED, {
id: storyId,
parameters,
initialArgs,
@ -318,7 +335,7 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
// If the implementation changed, or args were persisted, the args may have changed,
// and the STORY_PREPARED event above may not be respected.
if (implementationChanged || persistedArgs) {
this.channel.emit(Events.STORY_ARGS_UPDATED, { storyId, args });
this.channel.emit(STORY_ARGS_UPDATED, { storyId, args });
}
if (selection.viewMode === 'docs' || parameters.docsOnly) {
@ -411,20 +428,20 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
renderMissingStory() {
this.view.showNoPreview();
this.channel.emit(Events.STORY_MISSING);
this.channel.emit(STORY_MISSING);
}
renderStoryLoadingException(storySpecifier: StorySpecifier, err: Error) {
logger.error(`Unable to load story '${storySpecifier}':`);
logger.error(err);
this.view.showErrorDisplay(err);
this.channel.emit(Events.STORY_MISSING, storySpecifier);
this.channel.emit(STORY_MISSING, storySpecifier);
}
// renderException is used if we fail to render the story and it is uncaught by the app layer
renderException(storyId: StoryId, err: Error) {
this.channel.emit(Events.STORY_THREW_EXCEPTION, err);
this.channel.emit(Events.STORY_RENDER_PHASE_CHANGED, { newPhase: 'errored', storyId });
this.channel.emit(STORY_THREW_EXCEPTION, err);
this.channel.emit(STORY_RENDER_PHASE_CHANGED, { newPhase: 'errored', storyId });
// Ignored exceptions exist for control flow purposes, and are typically handled elsewhere.
if (err !== IGNORED_EXCEPTION) {
@ -438,8 +455,8 @@ export class PreviewWeb<TFramework extends AnyFramework> extends Preview<TFramew
// wrong -- for instance returned the wrong thing from a story
renderError(storyId: StoryId, { title, description }: { title: string; description: string }) {
logger.error(`Error rendering story ${title}: ${description}`);
this.channel.emit(Events.STORY_ERRORED, { title, description });
this.channel.emit(Events.STORY_RENDER_PHASE_CHANGED, { newPhase: 'errored', storyId });
this.channel.emit(STORY_ERRORED, { title, description });
this.channel.emit(STORY_RENDER_PHASE_CHANGED, { newPhase: 'errored', storyId });
this.view.showErrorDisplay({
message: title,
stack: description,

View File

@ -1,4 +1,6 @@
import slash from 'slash';
import dedent from 'ts-dedent';
import { once } from '@storybook/client-logger';
// FIXME: types duplicated type from `core-common', to be
// removed when we remove v6 back-compat.
@ -48,11 +50,24 @@ function pathJoin(paths: string[]): string {
return paths.join('/').replace(slashes, '/');
}
export const userOrAutoTitleFromSpecifier = (fileName: string, entry: NormalizedStoriesSpecifier, userTitle?: string) => {
export const userOrAutoTitleFromSpecifier = (
fileName: string | number,
entry: NormalizedStoriesSpecifier,
userTitle?: string
) => {
const { directory, importPathMatcher, titlePrefix = '' } = entry || {};
// On Windows, backslashes are used in paths, which can cause problems here
// slash makes sure we always handle paths with unix-style forward slash
const normalizedFileName = slash(fileName);
if (typeof fileName === 'number') {
once.warn(dedent`
CSF Auto-title received a numeric fileName. This typically happens when
webpack is mis-configured in production mode. To force webpack to produce
filenames, set optimization.moduleIds = "named" in your webpack config.
`);
}
const normalizedFileName = slash(String(fileName));
if (importPathMatcher.exec(normalizedFileName)) {
if (!userTitle) {
@ -74,11 +89,15 @@ export const userOrAutoTitleFromSpecifier = (fileName: string, entry: Normalized
return undefined;
};
export const userOrAutoTitle = (fileName: string, storiesEntries: NormalizedStoriesSpecifier[], userTitle?: string) => {
export const userOrAutoTitle = (
fileName: string,
storiesEntries: NormalizedStoriesSpecifier[],
userTitle?: string
) => {
for (let i = 0; i < storiesEntries.length; i += 1) {
const title = userOrAutoTitleFromSpecifier(fileName, storiesEntries[i], userTitle);
if (title) return title;
}
return userTitle || undefined;
};
};

View File

@ -27,6 +27,8 @@ export const getMonorepoType = (): MonorepoType => {
if (monorepoType) {
return monorepoType;
}
if (!fs.existsSync(path.join(projectRootPath, 'package.json'))) return undefined;
const packageJson = fs.readJsonSync(path.join(projectRootPath, 'package.json')) as PackageJson;

View File

@ -187,9 +187,13 @@ export const computeStorybookMetadata = async ({
const hasStorybookEslint = !!allDependencies['eslint-plugin-storybook'];
const storybookInfo = getStorybookInfo(packageJson);
const storybookVersion =
storybookPackages[storybookInfo.frameworkPackage]?.version || storybookInfo.version;
return {
...metadata,
storybookVersion: storybookInfo.version,
storybookVersion,
language,
storybookPackages,
framework: {

View File

@ -7,13 +7,21 @@ import ReactDOM from 'react-dom';
import { Location, LocationProvider, useNavigate } from '@storybook/router';
import { Provider as ManagerProvider, Combo } from '@storybook/api';
import { ThemeProvider, ensure as ensureTheme } from '@storybook/theming';
import {
ThemeProvider,
ensure as ensureTheme,
CacheProvider,
createCache,
} from '@storybook/theming';
import { HelmetProvider } from 'react-helmet-async';
import App from './app';
import Provider from './provider';
const emotionCache = createCache({ key: 'sto' });
emotionCache.compat = true;
const { DOCS_MODE } = global;
// @ts-ignore
@ -66,15 +74,17 @@ const Main: FC<{ provider: Provider }> = ({ provider }) => {
: !state.storiesFailed && !state.storiesConfigured;
return (
<ThemeProvider key="theme.provider" theme={ensureTheme(state.theme)}>
<App
key="app"
viewMode={state.viewMode}
layout={isLoading ? { ...state.layout, showPanel: false } : state.layout}
panelCount={panelCount}
docsOnly={story && story.parameters && story.parameters.docsOnly}
/>
</ThemeProvider>
<CacheProvider value={emotionCache}>
<ThemeProvider key="theme.provider" theme={ensureTheme(state.theme)}>
<App
key="app"
viewMode={state.viewMode}
layout={isLoading ? { ...state.layout, showPanel: false } : state.layout}
panelCount={panelCount}
docsOnly={story && story.parameters && story.parameters.docsOnly}
/>
</ThemeProvider>
</CacheProvider>
);
}}
</ManagerProvider>

View File

@ -57,6 +57,7 @@
"build-storybooks": "cross-env STORYBOOK_DISPLAY_WARNING=true DISPLAY_WARNING=true node -r esm ./scripts/build-storybooks.js",
"changelog": "pr-log --sloppy --cherry-pick",
"changelog:next": "pr-log --sloppy --since-prerelease",
"clean:dist": "del **/dist",
"coverage": "codecov",
"danger": "danger",
"generate-repros": "zx scripts/repros-generator/index.mjs",
@ -228,6 +229,7 @@
"core-js": "^3.21.1",
"cross-env": "^7.0.3",
"danger": "^10.6.2",
"del-cli": "^4.0.1",
"detect-port": "^1.3.0",
"downlevel-dts": "^0.6.0",
"dts-bundle-generator": "^6.2.0",

13
scripts/bootstrap.js vendored
View File

@ -76,6 +76,15 @@ function run() {
},
order: 1,
}),
cleanup: createTask({
name: `Remove compiled dist directories ${chalk.gray('(cleanup)')}`,
defaultValue: false,
option: '--cleanup',
command: () => {
spawn('npm run clean:dist');
},
order: 0,
}),
reset: createTask({
name: `Clean repository ${chalk.red('(reset)')}`,
defaultValue: false,
@ -142,7 +151,7 @@ function run() {
const groups = {
main: ['core'],
buildtasks: ['install', 'build', 'manager'],
devtasks: ['dev', 'registry', 'reset'],
devtasks: ['dev', 'registry', 'cleanup', 'reset'],
};
Object.keys(tasks)
@ -203,7 +212,7 @@ function run() {
if (sure) {
return list;
}
throw new Error('problem is between keyboard and chair');
throw new Error('Cleanup canceled');
});
}
return list;

View File

@ -11,8 +11,8 @@ const cleaningProcess = spawn('git', [
'clean',
'-xdf',
'-n',
'--exclude=".vscode"',
'--exclude=".idea"',
'--exclude="/.vscode"',
'--exclude="/.idea"',
]);
cleaningProcess.stdout.on('data', (data) => {

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