improve vue integration

This commit is contained in:
Oliver Hoff 2018-12-20 03:50:08 +01:00
parent 03c4c0f67a
commit 9f0ce22c72
5 changed files with 89 additions and 86 deletions

View File

@ -2,12 +2,15 @@
import Vue from 'vue'; import Vue from 'vue';
function getRenderedTree(story, context) { function getRenderedTree(story, context) {
const storyElement = story.render(context); const component = story.render(context);
const Constructor = Vue.extend(storyElement); const vm = new Vue({
const vm = new Constructor().$mount(); render(h) {
return h(component);
},
});
return vm.$el; return vm.$mount().$el;
} }
export default getRenderedTree; export default getRenderedTree;

View File

@ -1,30 +1,72 @@
import { start } from '@storybook/core/client'; import { start } from '@storybook/core/client';
import Vue from 'vue';
import './globals'; import './globals';
import render from './render'; import render from './render';
const createWrapperComponent = Target => ({ export const WRAPS = 'STORYBOOK_WRAPS';
functional: true, export const VALUES = 'STORYBOOK_VALUES';
render(h, c) {
return h(Target, c.data, c.children);
},
});
const decorateStory = (getStory, decorators) =>
decorators.reduce(
(decorated, decorator) => context => {
const story = () => decorated(context);
let decoratedStory = decorator(story, context);
if (typeof decoratedStory === 'string') { function extractProps(component) {
decoratedStory = { template: decoratedStory }; return Object.entries(component.options.props || {})
.map(([name, def]) => ({ [name]: def.default }))
.reduce((wrap, prop) => ({ ...wrap, ...prop }), {});
}
function prepare(rawStory, innerStory) {
let story = rawStory;
// eslint-disable-next-line no-underscore-dangle
if (!story._isVue) {
if (typeof story === 'string') {
story = { template: story };
}
if (innerStory) {
story.components = { ...(story.components || {}), story: innerStory };
}
story = Vue.extend(story);
} else if (story.options[WRAPS]) {
return story;
}
return Vue.extend({
[WRAPS]: story,
[VALUES]: { ...(innerStory ? innerStory.options[VALUES] : {}), ...extractProps(story) },
functional: true,
render(h, { data, parent, children }) {
return h(
story,
{
...data,
props: { ...(data.props || {}), ...(parent.$root[VALUES] || {}) },
},
children
);
},
});
}
function decorateStory(getStory, decorators) {
return decorators.reduce(
(decorated, decorator) => context => {
let story;
const decoratedStory = decorator(() => {
story = decorated(context);
return story;
}, context);
if (!story) {
story = decorated(context);
} }
decoratedStory.components = decoratedStory.components || {}; if (decoratedStory === story) {
decoratedStory.components.story = createWrapperComponent(story()); return story;
return decoratedStory; }
return prepare(decoratedStory, story);
}, },
getStory context => prepare(getStory(context))
); );
}
const { clientApi, configApi, forceReRender } = start(render, { decorateStory }); const { clientApi, configApi, forceReRender } = start(render, { decorateStory });

View File

@ -1,24 +1,17 @@
import { stripIndents } from 'common-tags'; import { stripIndents } from 'common-tags';
import Vue from 'vue'; import Vue from 'vue';
import { VALUES } from '.';
let root = null; let root = null;
function getComponentProxy(component) { function renderRoot(component) {
return Object.entries(component.props || {})
.map(([name, def]) => ({ [name]: def.default }))
.reduce((wrap, prop) => ({ ...wrap, ...prop }), {});
}
function renderRoot(component, proxy) {
root = new Vue({ root = new Vue({
el: '#root', el: '#root',
beforeCreate() { data() {
this.proxy = proxy; return { [VALUES]: component.options[VALUES] };
}, },
render(h) { render(h) {
const props = this.proxy; return h('div', { attrs: { id: 'root' } }, [h(component)]);
return h('div', { attrs: { id: 'root' } }, [h(component, { props })]);
}, },
}); });
} }
@ -49,15 +42,12 @@ export default function render({
showMain(); showMain();
const proxy = getComponentProxy(component);
// at component creation || refresh by HMR // at component creation || refresh by HMR
if (!root || !forceRender) { if (!root || !forceRender) {
if (root) root.$destroy(); if (root) root.$destroy();
renderRoot(component, proxy); renderRoot(component);
} else { } else {
root.proxy = proxy; root[VALUES] = component.options[VALUES];
root.$forceUpdate();
} }
} }

View File

@ -7,28 +7,12 @@ exports[`Storyshots Addon|Centered rounded 1`] = `
<div <div
style="margin: auto; max-height: 100%;" style="margin: auto; max-height: 100%;"
> >
<div <button
style="position: fixed; top: 0px; left: 0px; bottom: 0px; right: 0px; display: flex; align-items: center; overflow: auto;" class="button rounded"
style="color: rgb(66, 185, 131); border-color: #42b983;"
> >
<div A Button with rounded edges!
style="margin: auto; max-height: 100%;" </button>
>
<div
style="position: fixed; top: 0px; left: 0px; bottom: 0px; right: 0px; display: flex; align-items: center; overflow: auto;"
>
<div
style="margin: auto; max-height: 100%;"
>
<button
class="button rounded"
style="color: rgb(66, 185, 131); border-color: #42b983;"
>
A Button with rounded edges!
</button>
</div>
</div>
</div>
</div>
</div> </div>
</div> </div>
`; `;

View File

@ -5,21 +5,13 @@ exports[`Storyshots Custom|Decorator for Vue render 1`] = `
style="border: medium solid blue;" style="border: medium solid blue;"
> >
<div <div
style="border: medium solid blue;" style="border: medium solid red;"
> >
<div <button
style="border: medium solid blue;" class="button"
> >
<div renders component: MyButton!
style="border: medium solid red;" </button>
>
<button
class="button"
>
renders component: MyButton!
</button>
</div>
</div>
</div> </div>
</div> </div>
`; `;
@ -29,22 +21,14 @@ exports[`Storyshots Custom|Decorator for Vue template 1`] = `
style="border: medium solid blue;" style="border: medium solid blue;"
> >
<div <div
style="border: medium solid blue;" style="border: medium solid red;"
> >
<div <button
style="border: medium solid blue;" class="button"
style="color: rgb(66, 185, 131); border-color: #42b983;"
> >
<div MyButton with template!
style="border: medium solid red;" </button>
>
<button
class="button"
style="color: rgb(66, 185, 131); border-color: #42b983;"
>
MyButton with template!
</button>
</div>
</div>
</div> </div>
</div> </div>
`; `;