/* eslint-disable camelcase */ /* eslint-disable no-underscore-dangle */ import { dedent } from 'ts-dedent'; import Vue from 'vue'; import type { Store_RenderContext, ArgsStoryFn } from '@storybook/types'; import { CombinedVueInstance } from 'vue/types/vue'; import type { VueFramework } from './types'; export const COMPONENT = 'STORYBOOK_COMPONENT'; export const VALUES = 'STORYBOOK_VALUES'; const map = new Map(); type Instance = CombinedVueInstance< Vue, { STORYBOOK_COMPONENT: any; STORYBOOK_VALUES: Record; }, object, object, Record >; const getRoot = (domElement: Element): Instance => { const cachedInstance = map.get(domElement); if (cachedInstance != null) return cachedInstance; // Create a dummy "target" underneath #storybook-root // that Vue2 will replace on first render with #storybook-vue-root const target = document.createElement('div'); domElement.appendChild(target); const instance: Instance = new Vue({ beforeDestroy() { map.delete(domElement); }, data() { return { [COMPONENT]: undefined, [VALUES]: {}, }; }, // @ts-expect-error What's going on here? (TS says that we should not return an array here, but the `h` directly) render(h) { map.set(domElement, instance); return this[COMPONENT] ? [h(this[COMPONENT])] : undefined; }, }); return instance; }; export const render: ArgsStoryFn = (args, context) => { const { id, component: Component, argTypes } = context; const component = Component as VueFramework['component'] & { __docgenInfo?: { displayName: string }; props: Record; }; if (!component) { throw new Error( `Unable to render story ${id} as the component annotation is missing from the default export` ); } let componentName = 'component'; // if there is a name property, we either use it or preprend with sb- in case it's an invalid name if (component.name) { // @ts-expect-error isReservedTag is an internal function from Vue, might be changed in future releases const isReservedTag = Vue.config.isReservedTag && Vue.config.isReservedTag(component.name); componentName = isReservedTag ? `sb-${component.name}` : component.name; } else if (component.__docgenInfo?.displayName) { // otherwise, we use the displayName from docgen, if present componentName = component.__docgenInfo?.displayName; } return { props: Object.keys(argTypes), components: { [componentName]: component }, template: `<${componentName} v-bind="$props" />`, }; }; export function renderToDOM( { title, name, storyFn, showMain, showError, showException, forceRemount, }: Store_RenderContext, domElement: Element ) { const root = getRoot(domElement); Vue.config.errorHandler = showException; const element = storyFn(); let mountTarget: Element | null; // Vue2 mount always replaces the mount target with Vue-generated DOM. // https://v2.vuejs.org/v2/api/#el:~:text=replaced%20with%20Vue%2Dgenerated%20DOM // We cannot mount to the domElement directly, because it would be replaced. That would // break the references to the domElement like canvasElement used in the play function. // Instead, we mount to a child element of the domElement, creating one if necessary. if (domElement.hasChildNodes()) { mountTarget = domElement.firstElementChild; } else { mountTarget = document.createElement('div'); domElement.appendChild(mountTarget); } if (!element) { showError({ title: `Expecting a Vue component from the story: "${name}" of "${title}".`, description: dedent` Did you forget to return the Vue component from the story? Use "() => ({ template: '' })" or "() => ({ components: MyComp, template: '' })" when defining the story. `, }); return; } // at component creation || refresh by HMR or switching stories if (!root[COMPONENT] || forceRemount) { root[COMPONENT] = element; } // @ts-expect-error https://github.com/storybookjs/storrybook/pull/7578#discussion_r307986139 root[VALUES] = { ...element.options[VALUES] }; if (!map.has(domElement)) { root.$mount(mountTarget ?? undefined); } showMain(); }