Vue: Add first-class args support

This commit is contained in:
Michael Shilman 2020-06-10 21:53:26 +08:00
parent 4b18fb6510
commit 721406b8b5
4 changed files with 24 additions and 8 deletions

View File

@ -2,6 +2,7 @@
import Vue, { VueConstructor, ComponentOptions } from 'vue';
import { start } from '@storybook/core/client';
import {
Args,
ClientStoryApi,
StoryFn,
DecoratorFunction,
@ -13,7 +14,7 @@ import './globals';
import { IStorybookSection, StoryFnVueReturnType } from './types';
import render, { VALUES } from './render';
import { extractProps } from './util';
import { extractProps, propsFromArgs } from './util';
export const WRAPS = 'STORYBOOK_WRAPS';
@ -26,7 +27,11 @@ function prepare(
if (typeof rawStory === 'string') {
story = { template: rawStory };
} else if (rawStory != null) {
story = rawStory as ComponentOptions<Vue>;
const { args, props } = rawStory as ComponentOptions<Vue> & { args: Args };
story = {
...rawStory,
props: { ...propsFromArgs(args), ...props },
} as ComponentOptions<Vue>;
} else {
return null;
}

View File

@ -44,8 +44,10 @@ export default function render({
showMain();
// always refresh the component
root[COMPONENT] = element;
// at component creation || refresh by HMR // always refresh the component
if (!root[COMPONENT] || !forceRender) {
root[COMPONENT] = element;
}
// @ts-ignore https://github.com/storybookjs/storybook/pull/7578#discussion_r307986139
root[VALUES] = element.options[VALUES];

View File

@ -1,4 +1,5 @@
import { VueConstructor } from 'vue';
import { Args } from '@storybook/addons';
function getType(fn: Function) {
const match = fn && fn.toString().match(/^\s*function (\w+)/);
@ -21,3 +22,11 @@ export function extractProps(component: VueConstructor) {
.map(([name, prop]) => ({ [name]: resolveDefault(prop) }))
.reduce((wrap, prop) => ({ ...wrap, ...prop }), {});
}
export const propsFromArgs = (args: Args) => {
if (!args) return {};
return Object.entries(args).reduce((acc, [key, val]) => {
acc[key] = { default: val };
return acc;
}, {} as any);
};

View File

@ -1,3 +1,4 @@
import { propsFromArgs } from '@storybook/vue';
import MyButton from '../Button.vue';
export default {
@ -6,11 +7,10 @@ export default {
};
export const Rounded = (args) => ({
args,
components: { MyButton },
template: '<my-button :color="color" :rounded="rounded">A Button with rounded edges</my-button>',
data() {
return args;
},
template:
'<my-button v-bind:color="color" v-bind:rounded="rounded">A Button with rounded edges</my-button>',
});
Rounded.argTypes = {
rounded: { defaultValue: true },