mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 18:31:08 +08:00
Builder-vite: Fix storysource support
This commit is contained in:
parent
5ed049fb8d
commit
ed975bb233
@ -52,6 +52,7 @@
|
||||
"@storybook/node-logger": "7.0.0-beta.47",
|
||||
"@storybook/preview": "7.0.0-beta.47",
|
||||
"@storybook/preview-api": "7.0.0-beta.47",
|
||||
"@storybook/source-loader": "7.0.0-beta.47",
|
||||
"@storybook/types": "7.0.0-beta.47",
|
||||
"browser-assert": "^1.2.1",
|
||||
"es-module-lexer": "^0.9.3",
|
||||
|
@ -4,3 +4,4 @@ export * from './strip-story-hmr-boundaries';
|
||||
export * from './code-generator-plugin';
|
||||
export * from './csf-plugin';
|
||||
export * from './external-globals-plugin';
|
||||
export * from './source-loader-plugin';
|
||||
|
106
code/lib/builder-vite/src/plugins/source-loader-plugin.ts
Normal file
106
code/lib/builder-vite/src/plugins/source-loader-plugin.ts
Normal file
@ -0,0 +1,106 @@
|
||||
import type { Plugin } from 'vite';
|
||||
import sourceLoaderTransform from '@storybook/source-loader';
|
||||
import MagicString from 'magic-string';
|
||||
import type { ExtendedOptions } from '../types';
|
||||
|
||||
const storyPattern = /\.stories\.[jt]sx?$/;
|
||||
const storySourcePattern = /var __STORY__ = "(.*)"/;
|
||||
const storySourceReplacement = '--STORY_SOURCE_REPLACEMENT--';
|
||||
|
||||
const mockClassLoader = (id: string) => ({
|
||||
// eslint-disable-next-line no-console
|
||||
emitWarning: (message: string) => console.warn(message),
|
||||
resourcePath: id,
|
||||
getOptions: () => ({ injectStoryParameters: true }),
|
||||
extension: `.${id.split('.').pop()}`,
|
||||
});
|
||||
|
||||
// HACK: Until we can support only node 15+ and use string.prototype.replaceAll
|
||||
const replaceAll = (str: string, search: string, replacement: string) => {
|
||||
return str.split(search).join(replacement);
|
||||
};
|
||||
|
||||
export function sourceLoaderPlugin(config: ExtendedOptions): Plugin | Plugin[] {
|
||||
if (config.configType === 'DEVELOPMENT') {
|
||||
return {
|
||||
name: 'storybook:source-loader-plugin',
|
||||
enforce: 'pre',
|
||||
async transform(src: string, id: string) {
|
||||
if (id.match(storyPattern)) {
|
||||
const code: string = await sourceLoaderTransform.call(mockClassLoader(id), src);
|
||||
const s = new MagicString(src);
|
||||
// Entirely replace with new code
|
||||
s.overwrite(0, src.length, code);
|
||||
|
||||
return {
|
||||
code: s.toString(),
|
||||
map: s.generateMap({ hires: true, source: id }),
|
||||
};
|
||||
}
|
||||
return undefined;
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
// In production, we need to be fancier, to avoid vite:define plugin from replacing values inside the `__STORY__` string
|
||||
const storySources = new WeakMap<ExtendedOptions, Map<string, string>>();
|
||||
|
||||
return [
|
||||
{
|
||||
name: 'storybook-vite-source-loader-plugin',
|
||||
enforce: 'pre',
|
||||
buildStart() {
|
||||
storySources.set(config, new Map());
|
||||
},
|
||||
async transform(src: string, id: string) {
|
||||
if (id.match(storyPattern)) {
|
||||
let code: string = await sourceLoaderTransform.call(mockClassLoader(id), src);
|
||||
// eslint-disable-next-line @typescript-eslint/naming-convention
|
||||
const [_, sourceString] = code.match(storySourcePattern) ?? [null, null];
|
||||
if (sourceString) {
|
||||
const map = storySources.get(config);
|
||||
map?.set(id, sourceString);
|
||||
|
||||
// Remove story source so that it is not processed by vite:define plugin
|
||||
code = replaceAll(code, sourceString, storySourceReplacement);
|
||||
}
|
||||
|
||||
const s = new MagicString(src);
|
||||
// Entirely replace with new code
|
||||
s.overwrite(0, src.length, code);
|
||||
|
||||
return {
|
||||
code: s.toString(),
|
||||
map: s.generateMap(),
|
||||
};
|
||||
}
|
||||
return undefined;
|
||||
},
|
||||
},
|
||||
{
|
||||
name: 'storybook-vite-source-loader-plugin-post',
|
||||
enforce: 'post',
|
||||
buildStart() {
|
||||
storySources.set(config, new Map());
|
||||
},
|
||||
async transform(src: string, id: string) {
|
||||
if (id.match(storyPattern)) {
|
||||
const s = new MagicString(src);
|
||||
const map = storySources.get(config);
|
||||
const storySourceStatement = map?.get(id);
|
||||
// Put the previously-extracted source back in
|
||||
if (storySourceStatement) {
|
||||
const newCode = replaceAll(src, storySourceReplacement, storySourceStatement);
|
||||
s.overwrite(0, src.length, newCode);
|
||||
}
|
||||
|
||||
return {
|
||||
code: s.toString(),
|
||||
map: s.generateMap(),
|
||||
};
|
||||
}
|
||||
return undefined;
|
||||
},
|
||||
},
|
||||
];
|
||||
}
|
@ -17,6 +17,7 @@ import {
|
||||
mdxPlugin,
|
||||
stripStoryHMRBoundary,
|
||||
externalGlobalsPlugin,
|
||||
sourceLoaderPlugin,
|
||||
} from './plugins';
|
||||
|
||||
import type { BuilderOptions } from './types';
|
||||
@ -77,6 +78,7 @@ export async function pluginConfig(options: Options) {
|
||||
|
||||
const plugins = [
|
||||
codeGeneratorPlugin(options),
|
||||
sourceLoaderPlugin(options),
|
||||
await csfPlugin(options),
|
||||
await mdxPlugin(options),
|
||||
injectExportOrderPlugin,
|
||||
|
@ -5195,6 +5195,7 @@ __metadata:
|
||||
"@storybook/node-logger": 7.0.0-beta.47
|
||||
"@storybook/preview": 7.0.0-beta.47
|
||||
"@storybook/preview-api": 7.0.0-beta.47
|
||||
"@storybook/source-loader": 7.0.0-beta.47
|
||||
"@storybook/types": 7.0.0-beta.47
|
||||
"@types/express": ^4.17.13
|
||||
"@types/node": ^16.0.0
|
||||
|
Loading…
x
Reference in New Issue
Block a user