storybook/lib/client-api/src/StoryStoreFacade.ts
2021-09-21 17:51:55 +08:00

163 lines
4.5 KiB
TypeScript

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<TFramework extends AnyFramework> {
name: string;
render: StoryFn<TFramework>;
}
export interface GetStorybookKind<TFramework extends AnyFramework> {
kind: string;
fileName: string;
stories: GetStorybookStory<TFramework>[];
}
export class StoryStoreFacade<TFramework extends AnyFramework> {
projectAnnotations: NormalizedProjectAnnotations<TFramework>;
stories: StoryIndex['stories'];
csfExports: Record<Path, ModuleExports>;
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<TFramework>) {
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<TFramework>, Parameters, Parameters][] = storyEntries.map(
([storyId, { importPath }]) => {
const exports = this.csfExports[importPath];
const csfFile = store.processCSFFileWithCache<TFramework>(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,
};
});
}
}