mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 10:51:05 +08:00
197 lines
5.5 KiB
TypeScript
197 lines
5.5 KiB
TypeScript
import fs from 'fs-extra';
|
|
import remarkSlug from 'remark-slug';
|
|
import remarkExternalLinks from 'remark-external-links';
|
|
import { dedent } from 'ts-dedent';
|
|
|
|
import type {
|
|
CoreCommon_IndexerOptions,
|
|
CoreCommon_StoryIndexer,
|
|
DocsOptions,
|
|
Options,
|
|
} from '@storybook/types';
|
|
import type { CsfPluginOptions } from '@storybook/csf-plugin';
|
|
import { loadCsf } from '@storybook/csf-tools';
|
|
|
|
// for frameworks that are not working with react, we need to configure
|
|
// the jsx to transpile mdx, for now there will be a flag for that
|
|
// for more complex solutions we can find alone that we need to add '@babel/plugin-transform-react-jsx'
|
|
type BabelParams = {
|
|
babelOptions?: any;
|
|
mdxBabelOptions?: any;
|
|
configureJSX?: boolean;
|
|
};
|
|
function createBabelOptions({ babelOptions, mdxBabelOptions, configureJSX }: BabelParams) {
|
|
const babelPlugins = mdxBabelOptions?.plugins || babelOptions?.plugins || [];
|
|
|
|
const filteredBabelPlugins = babelPlugins.filter((p: any) => {
|
|
const name = Array.isArray(p) ? p[0] : p;
|
|
if (typeof name === 'string') {
|
|
return !name.includes('plugin-transform-react-jsx');
|
|
}
|
|
return true;
|
|
});
|
|
|
|
const jsxPlugin = [
|
|
require.resolve('@babel/plugin-transform-react-jsx'),
|
|
{ pragma: 'React.createElement', pragmaFrag: 'React.Fragment' },
|
|
];
|
|
const plugins = configureJSX ? [...filteredBabelPlugins, jsxPlugin] : babelPlugins;
|
|
return {
|
|
// don't use the root babelrc by default (users can override this in mdxBabelOptions)
|
|
babelrc: false,
|
|
configFile: false,
|
|
...babelOptions,
|
|
...mdxBabelOptions,
|
|
plugins,
|
|
};
|
|
}
|
|
|
|
async function webpack(
|
|
webpackConfig: any = {},
|
|
options: Options &
|
|
BabelParams & {
|
|
/** @deprecated */
|
|
sourceLoaderOptions: any;
|
|
csfPluginOptions: CsfPluginOptions | null;
|
|
transcludeMarkdown: boolean;
|
|
} /* & Parameters<
|
|
typeof createCompiler
|
|
>[0] */
|
|
) {
|
|
const resolvedBabelLoader = await options.presets.apply('babelLoaderRef');
|
|
|
|
const { module = {} } = webpackConfig;
|
|
|
|
// it will reuse babel options that are already in use in storybook
|
|
// also, these babel options are chained with other presets.
|
|
const {
|
|
babelOptions,
|
|
mdxBabelOptions,
|
|
configureJSX = true,
|
|
csfPluginOptions = {},
|
|
sourceLoaderOptions = null,
|
|
transcludeMarkdown = false,
|
|
} = options;
|
|
|
|
const mdxLoaderOptions = {
|
|
// whether to skip storybook files, useful for docs only mdx or md files
|
|
skipCsf: true,
|
|
remarkPlugins: [remarkSlug, remarkExternalLinks],
|
|
};
|
|
|
|
if (sourceLoaderOptions) {
|
|
throw new Error(dedent`
|
|
Addon-docs no longer uses source-loader in 7.0.
|
|
|
|
To update your configuration, please see migration instructions here:
|
|
|
|
https://github.com/storybookjs/storybook/blob/next/MIGRATION.md#dropped-source-loader--storiesof-static-snippets
|
|
`);
|
|
}
|
|
|
|
const mdxLoader = require.resolve('@storybook/mdx2-csf/loader');
|
|
|
|
let rules = module.rules || [];
|
|
if (transcludeMarkdown) {
|
|
rules = [
|
|
...rules.filter((rule: any) => rule.test?.toString() !== '/\\.md$/'),
|
|
{
|
|
test: /\.md$/,
|
|
use: [
|
|
{
|
|
loader: resolvedBabelLoader,
|
|
options: createBabelOptions({ babelOptions, mdxBabelOptions, configureJSX }),
|
|
},
|
|
{
|
|
loader: mdxLoader,
|
|
options: mdxLoaderOptions,
|
|
},
|
|
],
|
|
},
|
|
];
|
|
}
|
|
|
|
const result = {
|
|
...webpackConfig,
|
|
plugins: [
|
|
...(webpackConfig.plugins || []),
|
|
// eslint-disable-next-line global-require
|
|
...(csfPluginOptions ? [require('@storybook/csf-plugin').webpack(csfPluginOptions)] : []),
|
|
],
|
|
|
|
module: {
|
|
...module,
|
|
rules: [
|
|
...rules,
|
|
{
|
|
test: /(stories|story)\.mdx$/,
|
|
use: [
|
|
{
|
|
loader: resolvedBabelLoader,
|
|
options: createBabelOptions({ babelOptions, mdxBabelOptions, configureJSX }),
|
|
},
|
|
{
|
|
loader: mdxLoader,
|
|
options: {
|
|
...mdxLoaderOptions,
|
|
skipCsf: false,
|
|
},
|
|
},
|
|
],
|
|
},
|
|
{
|
|
test: /\.mdx$/,
|
|
exclude: /(stories|story)\.mdx$/,
|
|
use: [
|
|
{
|
|
loader: resolvedBabelLoader,
|
|
options: createBabelOptions({ babelOptions, mdxBabelOptions, configureJSX }),
|
|
},
|
|
{
|
|
loader: mdxLoader,
|
|
options: mdxLoaderOptions,
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
};
|
|
|
|
return result;
|
|
}
|
|
|
|
const storyIndexers = (indexers: CoreCommon_StoryIndexer[] | null) => {
|
|
const mdxIndexer = async (fileName: string, opts: CoreCommon_IndexerOptions) => {
|
|
let code = (await fs.readFile(fileName, 'utf-8')).toString();
|
|
const { compile } = await import('@storybook/mdx2-csf');
|
|
code = await compile(code, {});
|
|
return loadCsf(code, { ...opts, fileName }).parse();
|
|
};
|
|
return [
|
|
{
|
|
test: /(stories|story)\.mdx$/,
|
|
indexer: mdxIndexer,
|
|
},
|
|
...(indexers || []),
|
|
];
|
|
};
|
|
|
|
const docs = (docsOptions: DocsOptions) => {
|
|
return {
|
|
...docsOptions,
|
|
enabled: true,
|
|
defaultName: 'Docs',
|
|
docsPage: true,
|
|
};
|
|
};
|
|
|
|
/*
|
|
* This is a workaround for https://github.com/Swatinem/rollup-plugin-dts/issues/162
|
|
* something down the dependency chain is using typescript namespaces, which are not supported by rollup-plugin-dts
|
|
*/
|
|
const webpackX = webpack as any;
|
|
const storyIndexersX = webpack as any;
|
|
const docsX = docs as any;
|
|
|
|
export { webpackX as webpack, storyIndexersX as storyIndexers, docsX as docs };
|