mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-07 00:13:43 +08:00
Merge branch 'next' into shilman/vue2-rendering-improvements
This commit is contained in:
commit
8e76081b09
@ -477,7 +477,7 @@ jobs:
|
|||||||
executor:
|
executor:
|
||||||
class: medium+
|
class: medium+
|
||||||
name: sb_node_14_browsers
|
name: sb_node_14_browsers
|
||||||
parallelism: 7
|
parallelism: 8
|
||||||
steps:
|
steps:
|
||||||
- git-shallow-clone/checkout_advanced:
|
- git-shallow-clone/checkout_advanced:
|
||||||
clone_options: '--depth 1 --verbose'
|
clone_options: '--depth 1 --verbose'
|
||||||
@ -497,7 +497,7 @@ jobs:
|
|||||||
executor:
|
executor:
|
||||||
class: medium+
|
class: medium+
|
||||||
name: sb_node_14_browsers
|
name: sb_node_14_browsers
|
||||||
parallelism: 7
|
parallelism: 8
|
||||||
steps:
|
steps:
|
||||||
- git-shallow-clone/checkout_advanced:
|
- git-shallow-clone/checkout_advanced:
|
||||||
clone_options: '--depth 1 --verbose'
|
clone_options: '--depth 1 --verbose'
|
||||||
@ -513,7 +513,7 @@ jobs:
|
|||||||
executor:
|
executor:
|
||||||
class: medium+
|
class: medium+
|
||||||
name: sb_node_14_browsers
|
name: sb_node_14_browsers
|
||||||
parallelism: 7
|
parallelism: 8
|
||||||
steps:
|
steps:
|
||||||
- git-shallow-clone/checkout_advanced:
|
- git-shallow-clone/checkout_advanced:
|
||||||
clone_options: '--depth 1 --verbose'
|
clone_options: '--depth 1 --verbose'
|
||||||
@ -533,7 +533,7 @@ jobs:
|
|||||||
executor:
|
executor:
|
||||||
class: medium+
|
class: medium+
|
||||||
name: sb_node_14_browsers
|
name: sb_node_14_browsers
|
||||||
parallelism: 7
|
parallelism: 8
|
||||||
steps:
|
steps:
|
||||||
- git-shallow-clone/checkout_advanced:
|
- git-shallow-clone/checkout_advanced:
|
||||||
clone_options: '--depth 1 --verbose'
|
clone_options: '--depth 1 --verbose'
|
||||||
@ -549,7 +549,7 @@ jobs:
|
|||||||
executor:
|
executor:
|
||||||
class: medium+
|
class: medium+
|
||||||
name: sb_node_14_browsers
|
name: sb_node_14_browsers
|
||||||
parallelism: 7
|
parallelism: 8
|
||||||
steps:
|
steps:
|
||||||
- git-shallow-clone/checkout_advanced:
|
- git-shallow-clone/checkout_advanced:
|
||||||
clone_options: '--depth 1 --verbose'
|
clone_options: '--depth 1 --verbose'
|
||||||
@ -565,7 +565,7 @@ jobs:
|
|||||||
executor:
|
executor:
|
||||||
class: medium+
|
class: medium+
|
||||||
name: sb_playwright
|
name: sb_playwright
|
||||||
parallelism: 7
|
parallelism: 8
|
||||||
steps:
|
steps:
|
||||||
- git-shallow-clone/checkout_advanced:
|
- git-shallow-clone/checkout_advanced:
|
||||||
clone_options: '--depth 1 --verbose'
|
clone_options: '--depth 1 --verbose'
|
||||||
|
@ -3,7 +3,7 @@ import globalThis from 'global';
|
|||||||
export default {
|
export default {
|
||||||
component: globalThis.Components.Html,
|
component: globalThis.Components.Html,
|
||||||
args: {
|
args: {
|
||||||
contents: '<button>Click Me!</button>',
|
content: '<button>Click Me!</button>',
|
||||||
},
|
},
|
||||||
parameters: {
|
parameters: {
|
||||||
chromatic: { disable: true },
|
chromatic: { disable: true },
|
||||||
@ -12,7 +12,7 @@ export default {
|
|||||||
|
|
||||||
export const Options = {
|
export const Options = {
|
||||||
args: {
|
args: {
|
||||||
contents:
|
content:
|
||||||
'<button style="color: rgb(255, 255, 255); background-color: rgb(76, 175, 80);">Click me!</button>',
|
'<button style="color: rgb(255, 255, 255); background-color: rgb(76, 175, 80);">Click me!</button>',
|
||||||
},
|
},
|
||||||
parameters: {
|
parameters: {
|
||||||
|
@ -3,7 +3,7 @@ import globalThis from 'global';
|
|||||||
export default {
|
export default {
|
||||||
component: globalThis.Components.Html,
|
component: globalThis.Components.Html,
|
||||||
args: {
|
args: {
|
||||||
contents: '',
|
content: '',
|
||||||
},
|
},
|
||||||
parameters: {
|
parameters: {
|
||||||
chromatic: { disable: true },
|
chromatic: { disable: true },
|
||||||
@ -12,7 +12,7 @@ export default {
|
|||||||
|
|
||||||
export const Violations = {
|
export const Violations = {
|
||||||
args: {
|
args: {
|
||||||
contents: `
|
content: `
|
||||||
<div>
|
<div>
|
||||||
<p>empty heading</p>
|
<p>empty heading</p>
|
||||||
<h1></h1>
|
<h1></h1>
|
||||||
@ -39,7 +39,7 @@ export const Violations = {
|
|||||||
|
|
||||||
export const Passes = {
|
export const Passes = {
|
||||||
args: {
|
args: {
|
||||||
contents: `
|
content: `
|
||||||
<div>
|
<div>
|
||||||
<p>heading</p>
|
<p>heading</p>
|
||||||
<h1>heading 1</h1>
|
<h1>heading 1</h1>
|
||||||
|
@ -64,6 +64,7 @@ export function action(name: string, options: ActionOptions = {}): HandlerFuncti
|
|||||||
};
|
};
|
||||||
channel.emit(EVENT_ID, actionDisplayToEmit);
|
channel.emit(EVENT_ID, actionDisplayToEmit);
|
||||||
};
|
};
|
||||||
|
handler.isAction = true;
|
||||||
|
|
||||||
return handler;
|
return handler;
|
||||||
}
|
}
|
||||||
|
@ -37,7 +37,7 @@ const addSpies = (id: string, val: any, key?: string): any => {
|
|||||||
if (Array.isArray(val)) {
|
if (Array.isArray(val)) {
|
||||||
return val.map((item, index) => addSpies(id, item, `${key}[${index}]`));
|
return val.map((item, index) => addSpies(id, item, `${key}[${index}]`));
|
||||||
}
|
}
|
||||||
if (typeof val === 'function' && val.name === 'actionHandler') {
|
if (typeof val === 'function' && val.isAction) {
|
||||||
Object.defineProperty(val, 'name', { value: key, writable: false });
|
Object.defineProperty(val, 'name', { value: key, writable: false });
|
||||||
Object.defineProperty(val, '__storyId__', { value: id, writable: false });
|
Object.defineProperty(val, '__storyId__', { value: id, writable: false });
|
||||||
const spy = action(val);
|
const spy = action(val);
|
||||||
|
@ -14,8 +14,9 @@ test.describe('addon-docs', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test('should provide source snippet', async ({ page }) => {
|
test('should provide source snippet', async ({ page }) => {
|
||||||
|
// templateName is e.g. 'Vue-CLI (Default JS)'
|
||||||
test.skip(
|
test.skip(
|
||||||
/^vue3/.test(templateName),
|
/^(vue3|vue-cli)/i.test(templateName),
|
||||||
`Skipping ${templateName}, which does not support dynamic source snippets`
|
`Skipping ${templateName}, which does not support dynamic source snippets`
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -21,11 +21,7 @@ export const core: PresetProperty<'core', StorybookConfig> = async (config, opti
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export const typescript = async (
|
export const typescript: PresetProperty<'typescript', StorybookConfig> = async (config) => ({
|
||||||
config: StorybookConfig['typescript']
|
|
||||||
): Promise<StorybookConfig['typescript']> => {
|
|
||||||
return {
|
|
||||||
...config,
|
...config,
|
||||||
skipBabel: true,
|
skipBabel: true,
|
||||||
};
|
});
|
||||||
};
|
|
||||||
|
@ -20,3 +20,8 @@ export const core: PresetProperty<'core', StorybookConfig> = async (config, opti
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const typescript: PresetProperty<'typescript', StorybookConfig> = async (config) => ({
|
||||||
|
...config,
|
||||||
|
skipBabel: true,
|
||||||
|
});
|
||||||
|
@ -3,7 +3,7 @@ import { Generator } from '../types';
|
|||||||
|
|
||||||
const generator: Generator = async (packageManager, npmOptions, options) => {
|
const generator: Generator = async (packageManager, npmOptions, options) => {
|
||||||
await baseGenerator(packageManager, npmOptions, options, 'vue3', {
|
await baseGenerator(packageManager, npmOptions, options, 'vue3', {
|
||||||
extraPackages: ['vue-loader@^16.0.0'],
|
extraPackages: ['vue-loader@^17.0.0', '@vue/compiler-sfc@^3.2.0'],
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ const defaultOptions: FrameworkOptions = {
|
|||||||
staticDir: undefined,
|
staticDir: undefined,
|
||||||
addScripts: true,
|
addScripts: true,
|
||||||
addComponents: true,
|
addComponents: true,
|
||||||
addBabel: true,
|
addBabel: false,
|
||||||
addESLint: false,
|
addESLint: false,
|
||||||
extraMain: undefined,
|
extraMain: undefined,
|
||||||
framework: undefined,
|
framework: undefined,
|
||||||
|
@ -94,11 +94,39 @@ const svelteViteTemplates = {
|
|||||||
// }
|
// }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const vueCliTemplates = {
|
||||||
|
'vue-cli/default-js': {
|
||||||
|
name: 'Vue-CLI (Default JS)',
|
||||||
|
script: 'npx -p @vue/cli vue create . --default --packageManager=yarn --force --merge',
|
||||||
|
cadence: ['ci', 'daily', 'weekly'],
|
||||||
|
expected: {
|
||||||
|
framework: '@storybook/vue3-webpack5',
|
||||||
|
renderer: '@storybook/vue3',
|
||||||
|
builder: '@storybook/builder-webpack5',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
//
|
||||||
|
// FIXME: https://github.com/storybookjs/storybook/issues/19204
|
||||||
|
//
|
||||||
|
// 'vue-cli/vue2-default-js': {
|
||||||
|
// name: 'Vue-CLI (Vue2 JS)',
|
||||||
|
// script:
|
||||||
|
// 'npx -p @vue/cli vue create . --default --packageManager=yarn --force --merge --preset=Default\\ (Vue\\ 2)',
|
||||||
|
// cadence: ['ci', 'daily', 'weekly'],
|
||||||
|
// expected: {
|
||||||
|
// framework: '@storybook/vue-webpack5',
|
||||||
|
// renderer: '@storybook/vue',
|
||||||
|
// builder: '@storybook/builder-webpack5',
|
||||||
|
// },
|
||||||
|
// },
|
||||||
|
};
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
...craTemplates,
|
...craTemplates,
|
||||||
...reactViteTemplates,
|
...reactViteTemplates,
|
||||||
...vue3ViteTemplates,
|
...vue3ViteTemplates,
|
||||||
...svelteViteTemplates,
|
...svelteViteTemplates,
|
||||||
|
...vueCliTemplates,
|
||||||
// FIXME: missing documentation.json
|
// FIXME: missing documentation.json
|
||||||
// 'angular/latest': {
|
// 'angular/latest': {
|
||||||
// name: 'Angular (latest)',
|
// name: 'Angular (latest)',
|
||||||
|
@ -40,7 +40,7 @@ Object {
|
|||||||
"include": Array [
|
"include": Array [
|
||||||
"ROOT",
|
"ROOT",
|
||||||
],
|
],
|
||||||
"test": "/\\\\.(mjs|tsx?|jsx?)$/",
|
"test": "/\\\\.(mjs|jsx?)$/",
|
||||||
"use": Array [
|
"use": Array [
|
||||||
Object {
|
Object {
|
||||||
"loader": "NODE_MODULES/babel-loader/lib/index.js",
|
"loader": "NODE_MODULES/babel-loader/lib/index.js",
|
||||||
|
@ -39,7 +39,7 @@ Object {
|
|||||||
"include": Array [
|
"include": Array [
|
||||||
"ROOT",
|
"ROOT",
|
||||||
],
|
],
|
||||||
"test": "/\\\\.(mjs|tsx?|jsx?)$/",
|
"test": "/\\\\.(mjs|jsx?)$/",
|
||||||
"use": Array [
|
"use": Array [
|
||||||
Object {
|
Object {
|
||||||
"loader": "NODE_MODULES/babel-loader/lib/index.js",
|
"loader": "NODE_MODULES/babel-loader/lib/index.js",
|
||||||
|
@ -25,11 +25,15 @@ export const ForceReRender = {
|
|||||||
|
|
||||||
export const ChangeArgs = {
|
export const ChangeArgs = {
|
||||||
play: async ({ canvasElement, id }: PlayFunctionContext) => {
|
play: async ({ canvasElement, id }: PlayFunctionContext) => {
|
||||||
// const channel = globalThis.__STORYBOOK_ADDONS_CHANNEL__;
|
const channel = globalThis.__STORYBOOK_ADDONS_CHANNEL__;
|
||||||
const button = await within(canvasElement).findByRole('button');
|
const button = await within(canvasElement).findByRole('button');
|
||||||
await button.focus();
|
await button.focus();
|
||||||
await expect(button).toHaveFocus();
|
await expect(button).toHaveFocus();
|
||||||
|
|
||||||
|
// Vue3: https://github.com/storybookjs/storybook/issues/13913
|
||||||
|
// Svelte: https://github.com/storybookjs/storybook/issues/19205
|
||||||
|
if (['vue3', 'svelte'].includes(globalThis.storybookRenderer)) return;
|
||||||
|
|
||||||
// When we change the args to the button, it should not rerender
|
// When we change the args to the button, it should not rerender
|
||||||
await channel.emit('updateStoryArgs', { storyId: id, updatedArgs: { children: 'New Text' } });
|
await channel.emit('updateStoryArgs', { storyId: id, updatedArgs: { children: 'New Text' } });
|
||||||
await within(canvasElement).findByText(/New Text/);
|
await within(canvasElement).findByText(/New Text/);
|
||||||
|
@ -4,7 +4,7 @@ import { PlayFunctionContext } from '@storybook/csf';
|
|||||||
|
|
||||||
export default {
|
export default {
|
||||||
component: globalThis.Components.Pre,
|
component: globalThis.Components.Pre,
|
||||||
title: 'manual title',
|
title: 'lib/store/manual title',
|
||||||
args: { text: 'No content' },
|
args: { text: 'No content' },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -6,3 +6,4 @@ import { Form } from './Form.jsx';
|
|||||||
import { Html } from './Html.jsx';
|
import { Html } from './Html.jsx';
|
||||||
|
|
||||||
globalThis.Components = { Button, Pre, Form, Html };
|
globalThis.Components = { Button, Pre, Form, Html };
|
||||||
|
globalThis.storybookRenderer = 'react';
|
||||||
|
@ -6,3 +6,4 @@ import Form from './Form.svelte';
|
|||||||
import Html from './Html.svelte';
|
import Html from './Html.svelte';
|
||||||
|
|
||||||
globalThis.Components = { Button, Pre, Form, Html };
|
globalThis.Components = { Button, Pre, Form, Html };
|
||||||
|
globalThis.storybookRenderer = 'svelte';
|
||||||
|
54
code/renderers/vue/template/components/Button.vue
Normal file
54
code/renderers/vue/template/components/Button.vue
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<template>
|
||||||
|
<button type="button" :class="classes" @click="onClick" :style="style">{{ children }}</button>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import './button.css';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'my-button',
|
||||||
|
|
||||||
|
props: {
|
||||||
|
children: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
primary: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
type: String,
|
||||||
|
default: 'medium',
|
||||||
|
validator: function (value) {
|
||||||
|
return ['small', 'medium', 'large'].indexOf(value) !== -1;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
backgroundColor: {
|
||||||
|
type: String,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
classes() {
|
||||||
|
return {
|
||||||
|
'storybook-button': true,
|
||||||
|
'storybook-button--primary': this.primary,
|
||||||
|
'storybook-button--secondary': !this.primary,
|
||||||
|
[`storybook-button--${this.size}`]: true,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
style() {
|
||||||
|
return {
|
||||||
|
backgroundColor: this.backgroundColor,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
onClick() {
|
||||||
|
this.$emit('onClick');
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
44
code/renderers/vue/template/components/Form.vue
Normal file
44
code/renderers/vue/template/components/Form.vue
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
<template>
|
||||||
|
<form id="interaction-test-form" @submit.prevent="onSubmit">
|
||||||
|
<label>
|
||||||
|
Enter Value
|
||||||
|
<input type="text" data-testid="value" :value="value" required @click="setValue" />
|
||||||
|
</label>
|
||||||
|
<button type="submit">Submit</button>
|
||||||
|
<p v-if="complete">Completed!!</p>
|
||||||
|
</form>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: 'my-form',
|
||||||
|
|
||||||
|
props: {
|
||||||
|
onSuccess: {
|
||||||
|
type: Function,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
value: '',
|
||||||
|
complete: false,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
setValue(event) {
|
||||||
|
this.value = event.target.value;
|
||||||
|
},
|
||||||
|
onSubmit() {
|
||||||
|
this.onSuccess(this.value);
|
||||||
|
setTimeout(() => {
|
||||||
|
this.complete = true;
|
||||||
|
}, 500);
|
||||||
|
setTimeout(() => {
|
||||||
|
this.complete = false;
|
||||||
|
}, 1500);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
17
code/renderers/vue/template/components/Html.vue
Normal file
17
code/renderers/vue/template/components/Html.vue
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<template>
|
||||||
|
<div v-html="content"></div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'my-html',
|
||||||
|
|
||||||
|
props: {
|
||||||
|
content: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
29
code/renderers/vue/template/components/Pre.vue
Normal file
29
code/renderers/vue/template/components/Pre.vue
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
<template>
|
||||||
|
<pre data-testid="pre" :style="style">{{ finalText }}</pre>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: 'my-pre',
|
||||||
|
|
||||||
|
props: {
|
||||||
|
// deepscan-disable-next-line
|
||||||
|
style: {
|
||||||
|
type: Object,
|
||||||
|
},
|
||||||
|
object: {
|
||||||
|
type: Object,
|
||||||
|
},
|
||||||
|
text: {
|
||||||
|
type: String,
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
finalText() {
|
||||||
|
return this.object ? JSON.stringify(this.object, null, 2) : this.text;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
30
code/renderers/vue/template/components/button.css
Normal file
30
code/renderers/vue/template/components/button.css
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
.storybook-button {
|
||||||
|
font-family: 'Nunito Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
|
||||||
|
font-weight: 700;
|
||||||
|
border: 0;
|
||||||
|
border-radius: 3em;
|
||||||
|
cursor: pointer;
|
||||||
|
display: inline-block;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
.storybook-button--primary {
|
||||||
|
color: white;
|
||||||
|
background-color: #1ea7fd;
|
||||||
|
}
|
||||||
|
.storybook-button--secondary {
|
||||||
|
color: #333;
|
||||||
|
background-color: transparent;
|
||||||
|
box-shadow: rgba(0, 0, 0, 0.15) 0px 0px 0px 1px inset;
|
||||||
|
}
|
||||||
|
.storybook-button--small {
|
||||||
|
font-size: 12px;
|
||||||
|
padding: 10px 16px;
|
||||||
|
}
|
||||||
|
.storybook-button--medium {
|
||||||
|
font-size: 14px;
|
||||||
|
padding: 11px 20px;
|
||||||
|
}
|
||||||
|
.storybook-button--large {
|
||||||
|
font-size: 16px;
|
||||||
|
padding: 12px 24px;
|
||||||
|
}
|
9
code/renderers/vue/template/components/index.js
Normal file
9
code/renderers/vue/template/components/index.js
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import globalThis from 'global';
|
||||||
|
|
||||||
|
import Button from './Button.vue';
|
||||||
|
import Pre from './Pre.vue';
|
||||||
|
import Form from './Form.vue';
|
||||||
|
import Html from './Html.vue';
|
||||||
|
|
||||||
|
globalThis.Components = { Button, Pre, Form, Html };
|
||||||
|
globalThis.storybookRenderer = 'vue';
|
@ -6,3 +6,4 @@ import Form from './Form.vue';
|
|||||||
import Html from './Html.vue';
|
import Html from './Html.vue';
|
||||||
|
|
||||||
globalThis.Components = { Button, Pre, Form, Html };
|
globalThis.Components = { Button, Pre, Form, Html };
|
||||||
|
globalThis.storybookRenderer = 'vue3';
|
||||||
|
@ -13,7 +13,6 @@ import {
|
|||||||
import prompts from 'prompts';
|
import prompts from 'prompts';
|
||||||
import type { AbortController } from 'node-abort-controller';
|
import type { AbortController } from 'node-abort-controller';
|
||||||
import command from 'execa';
|
import command from 'execa';
|
||||||
import dedent from 'ts-dedent';
|
|
||||||
|
|
||||||
import { createOptions, getOptionsOrPrompt, OptionValues } from './utils/options';
|
import { createOptions, getOptionsOrPrompt, OptionValues } from './utils/options';
|
||||||
import { executeCLIStep } from './utils/cli-step';
|
import { executeCLIStep } from './utils/cli-step';
|
||||||
@ -222,6 +221,7 @@ function addEsbuildLoaderToStories(mainConfig: ConfigFile) {
|
|||||||
module: {
|
module: {
|
||||||
...config.modules,
|
...config.modules,
|
||||||
rules: [
|
rules: [
|
||||||
|
// Ensure esbuild-loader applies to all files in ./template-stories
|
||||||
{
|
{
|
||||||
test: [/\\/template-stories\\//],
|
test: [/\\/template-stories\\//],
|
||||||
loader: '${loaderPath}',
|
loader: '${loaderPath}',
|
||||||
@ -230,7 +230,11 @@ function addEsbuildLoaderToStories(mainConfig: ConfigFile) {
|
|||||||
target: 'es2015',
|
target: 'es2015',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
...config.module.rules,
|
// Ensure no other loaders from the framework apply
|
||||||
|
...config.module.rules.map(rule => ({
|
||||||
|
...rule,
|
||||||
|
exclude: [/\\/template-stories\\//].concat(rule.exclude || []),
|
||||||
|
})),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
})`;
|
})`;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
/* eslint-disable no-await-in-loop, no-restricted-syntax */
|
/* eslint-disable no-await-in-loop, no-restricted-syntax */
|
||||||
import { getJunitXml } from 'junit-xml';
|
import { getJunitXml } from 'junit-xml';
|
||||||
import { outputFile } from 'fs-extra';
|
import { outputFile, existsSync, readFile } from 'fs-extra';
|
||||||
import { join, resolve } from 'path';
|
import { join, resolve } from 'path';
|
||||||
|
|
||||||
import { createOptions, getOptionsOrPrompt } from './utils/options';
|
import { createOptions, getOptionsOrPrompt } from './utils/options';
|
||||||
@ -158,6 +158,13 @@ async function runTask(
|
|||||||
if (junit && !task.junit) await writeJunitXml(taskKey, templateKey, start, err);
|
if (junit && !task.junit) await writeJunitXml(taskKey, templateKey, start, err);
|
||||||
|
|
||||||
throw err;
|
throw err;
|
||||||
|
} finally {
|
||||||
|
const { junitFilename } = details;
|
||||||
|
if (existsSync(junitFilename)) {
|
||||||
|
const junitXml = await (await readFile(junitFilename)).toString();
|
||||||
|
const prefixedXml = junitXml.replace(/classname="(.*)"/g, `classname="${templateKey} $1"`);
|
||||||
|
await outputFile(junitFilename, prefixedXml);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10,7 +10,8 @@ export const chromatic: Task = {
|
|||||||
async run(templateKey, { sandboxDir, builtSandboxDir, junitFilename }) {
|
async run(templateKey, { sandboxDir, builtSandboxDir, junitFilename }) {
|
||||||
const tokenEnvVarName = `CHROMATIC_TOKEN_${templateKey.toUpperCase().replace(/\/|-/g, '_')}`;
|
const tokenEnvVarName = `CHROMATIC_TOKEN_${templateKey.toUpperCase().replace(/\/|-/g, '_')}`;
|
||||||
const token = process.env[tokenEnvVarName];
|
const token = process.env[tokenEnvVarName];
|
||||||
return exec(
|
|
||||||
|
await exec(
|
||||||
`npx chromatic \
|
`npx chromatic \
|
||||||
--exit-zero-on-changes \
|
--exit-zero-on-changes \
|
||||||
--storybook-build-dir=${builtSandboxDir} \
|
--storybook-build-dir=${builtSandboxDir} \
|
||||||
|
Loading…
x
Reference in New Issue
Block a user