storybook/presets/react-webpack/src/framework-preset-react.ts
2022-05-11 23:28:31 +08:00

84 lines
2.7 KiB
TypeScript

import path from 'path';
import ReactRefreshWebpackPlugin from '@pmmmwh/react-refresh-webpack-plugin';
import { logger } from '@storybook/node-logger';
import type { StorybookConfig, Options, Preset } from '@storybook/core-webpack';
import type { ReactOptions } from './types';
const useFastRefresh = async (options: Options) => {
const isDevelopment = options.configType === 'DEVELOPMENT';
const framework = await options.presets.apply<Preset>('framework');
const reactOptions = (typeof framework === 'object' ? framework.options : {}) as ReactOptions;
return isDevelopment && (reactOptions.fastRefresh || process.env.FAST_REFRESH === 'true');
};
export const babel: StorybookConfig['babel'] = async (config, options) => {
if (!(await useFastRefresh(options))) return config;
return {
...config,
plugins: [
[require.resolve('react-refresh/babel'), {}, 'storybook-react-refresh'],
...(config.plugins || []),
],
};
};
const storybookReactDirName = path.dirname(
require.resolve('@storybook/preset-react-webpack/package.json')
);
// TODO: improve node_modules detection
const context = storybookReactDirName.includes('node_modules')
? path.join(storybookReactDirName, '../../') // Real life case, already in node_modules
: path.join(storybookReactDirName, '../../node_modules'); // SB Monorepo
const hasJsxRuntime = () => {
try {
require.resolve('react/jsx-runtime', { paths: [context] });
return true;
} catch (e) {
return false;
}
};
export const babelDefault: StorybookConfig['babelDefault'] = async (config) => {
const presetReactOptions = hasJsxRuntime() ? { runtime: 'automatic' } : {};
return {
...config,
presets: [
...(config?.presets || []),
[require.resolve('@babel/preset-react'), presetReactOptions],
require.resolve('@babel/preset-flow'),
],
plugins: [...(config?.plugins || []), require.resolve('babel-plugin-add-react-displayname')],
};
};
export const webpackFinal: StorybookConfig['webpackFinal'] = async (config, options) => {
if (!(await useFastRefresh(options))) return config;
// matches the name of the plugin in CRA.
const hasReactRefresh = config.plugins.find((p) => p.constructor.name === 'ReactRefreshPlugin');
if (hasReactRefresh) {
logger.warn("=> React refresh is already set. You don't need to set the option");
return config;
}
logger.info('=> Using React fast refresh');
return {
...config,
plugins: [
...config.plugins,
// Storybook uses webpack-hot-middleware https://github.com/storybookjs/storybook/issues/14114
new ReactRefreshWebpackPlugin({
overlay: {
sockIntegration: 'whm',
},
}),
],
};
};