import global from 'global'; import { StoryId, AnyFramework, toId, isExportStory, Parameters, StoryFn, storyNameFromExport, } from '@storybook/csf'; import { NormalizedProjectAnnotations, Path, StoryIndex, ModuleExports, StoryStore, Story, autoTitle, sortStoriesV6, } from '@storybook/store'; const { STORIES = [] } = global; export interface GetStorybookStory { name: string; render: StoryFn; } export interface GetStorybookKind { kind: string; fileName: string; stories: GetStorybookStory[]; } export class StoryStoreFacade { projectAnnotations: NormalizedProjectAnnotations; stories: StoryIndex['stories']; csfExports: Record; constructor() { this.projectAnnotations = { loaders: [], decorators: [], parameters: {}, argsEnhancers: [], argTypesEnhancers: [], }; this.stories = {}; this.csfExports = {}; } // This doesn't actually import anything because the client-api loads fully // on startup, but this is a shim after all. importFn(path: Path) { const moduleExports = this.csfExports[path]; if (!moduleExports) throw new Error(`Unknown path: ${path}`); return moduleExports; } fetchStoryIndex(store: StoryStore) { const fileNameOrder = Object.keys(this.csfExports); const storySortParameter = this.projectAnnotations.parameters?.options?.storySort; const storyEntries = Object.entries(this.stories); // Add the kind parameters and global parameters to each entry const sortableV6: [StoryId, Story, Parameters, Parameters][] = storyEntries.map( ([storyId, { importPath }]) => { const exports = this.csfExports[importPath]; const csfFile = store.processCSFFileWithCache(exports, exports.default.title); return [ storyId, store.storyFromCSFFile({ storyId, csfFile }), csfFile.meta.parameters, this.projectAnnotations.parameters, ]; } ); // NOTE: the sortStoriesV6 version returns the v7 data format. confusing but more convenient! const sortedV7 = sortStoriesV6(sortableV6, storySortParameter, fileNameOrder); const stories = sortedV7.reduce((acc, s) => { acc[s.id] = s; return acc; }, {} as StoryIndex['stories']); return { v: 3, stories }; } clearFilenameExports(fileName: Path) { if (!this.csfExports[fileName]) { return; } // Clear this module's stories from the storyList and existing exports Object.entries(this.stories).forEach(([id, { importPath }]) => { if (importPath === fileName) { delete this.stories[id]; } }); // We keep this as an empty record so we can use it to maintain component order this.csfExports[fileName] = {}; } // NOTE: we could potentially share some of this code with the stories.json generation addStoriesFromExports(fileName: Path, fileExports: ModuleExports) { // if the export haven't changed since last time we added them, this is a no-op if (this.csfExports[fileName] === fileExports) { return; } // OTOH, if they have changed, let's clear them out first this.clearFilenameExports(fileName); const { default: defaultExport, __namedExportsOrder, ...namedExports } = fileExports; // eslint-disable-next-line prefer-const let { id: componentId, title } = defaultExport || {}; title = title || autoTitle(fileName, STORIES); if (!title) { throw new Error( `Unexpected default export without title in '${fileName}': ${JSON.stringify( fileExports.default )}` ); } this.csfExports[fileName] = { ...fileExports, default: { ...defaultExport, title, parameters: { fileName, ...defaultExport.parameters, }, }, }; Object.entries(namedExports) .filter(([key]) => isExportStory(key, defaultExport)) .forEach(([key, storyExport]: [string, any]) => { const exportName = storyNameFromExport(key); const id = storyExport.parameters?.__id || toId(componentId || title, exportName); const name = (typeof storyExport !== 'function' && storyExport.name) || storyExport.storyName || storyExport.story?.name || exportName; this.stories[id] = { id, name, title, importPath: fileName, }; }); } }