mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 12:11:06 +08:00
163 lines
4.5 KiB
TypeScript
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,
|
|
};
|
|
});
|
|
}
|
|
}
|