mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-07 05:21:06 +08:00
Merge pull request #3919 from storybooks/tech/replace-html-webpack-plugin
CHANGE html-webpack-plugin for generate-page-plugin
This commit is contained in:
commit
0a33b0bca9
4
app/angular/src/server/angular-cli_config.js
vendored
4
app/angular/src/server/angular-cli_config.js
vendored
@ -80,7 +80,9 @@ export function applyAngularCliWebpackConfig(baseConfig, cliWebpackConfigOptions
|
|||||||
// cliStyleConfig.entry adds global style files to the webpack context
|
// cliStyleConfig.entry adds global style files to the webpack context
|
||||||
const entry = {
|
const entry = {
|
||||||
...baseConfig.entry,
|
...baseConfig.entry,
|
||||||
...cliStyleConfig.entry,
|
iframe: []
|
||||||
|
.concat(baseConfig.entry.iframe)
|
||||||
|
.concat(Object.values(cliStyleConfig.entry).reduce((acc, item) => acc.concat(item), [])),
|
||||||
};
|
};
|
||||||
|
|
||||||
const mod = {
|
const mod = {
|
||||||
|
@ -47,10 +47,11 @@
|
|||||||
"case-sensitive-paths-webpack-plugin": "^2.1.2",
|
"case-sensitive-paths-webpack-plugin": "^2.1.2",
|
||||||
"commander": "^2.17.0",
|
"commander": "^2.17.0",
|
||||||
"dotenv-webpack": "^1.5.7",
|
"dotenv-webpack": "^1.5.7",
|
||||||
|
"ejs": "^2.6.1",
|
||||||
"express": "^4.16.3",
|
"express": "^4.16.3",
|
||||||
"find-cache-dir": "^2.0.0",
|
"find-cache-dir": "^2.0.0",
|
||||||
|
"generate-page-webpack-plugin": "^1.0.0",
|
||||||
"global": "^4.3.2",
|
"global": "^4.3.2",
|
||||||
"html-webpack-plugin": "^3.2.0",
|
|
||||||
"json5": "^1.0.1",
|
"json5": "^1.0.1",
|
||||||
"prop-types": "^15.6.2",
|
"prop-types": "^15.6.2",
|
||||||
"raw-loader": "^0.5.1",
|
"raw-loader": "^0.5.1",
|
||||||
|
2
app/react-native/src/server/config.js
vendored
2
app/react-native/src/server/config.js
vendored
@ -116,7 +116,7 @@ export default function(configType, baseConfig, projectDir, configDir) {
|
|||||||
|
|
||||||
if (typeof customConfig === 'function') {
|
if (typeof customConfig === 'function') {
|
||||||
logger.info('=> Loading custom webpack config (full-control mode).');
|
logger.info('=> Loading custom webpack config (full-control mode).');
|
||||||
return customConfig(config, configType, defaultConfig);
|
return customConfig(config, configType, defaultConfig, configDir);
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info('=> Loading custom webpack config.');
|
logger.info('=> Loading custom webpack config.');
|
||||||
|
@ -4,12 +4,22 @@ import { getEnvironment } from 'universal-dotenv';
|
|||||||
import Dotenv from 'dotenv-webpack';
|
import Dotenv from 'dotenv-webpack';
|
||||||
import WatchMissingNodeModulesPlugin from 'react-dev-utils/WatchMissingNodeModulesPlugin';
|
import WatchMissingNodeModulesPlugin from 'react-dev-utils/WatchMissingNodeModulesPlugin';
|
||||||
import CaseSensitivePathsPlugin from 'case-sensitive-paths-webpack-plugin';
|
import CaseSensitivePathsPlugin from 'case-sensitive-paths-webpack-plugin';
|
||||||
import HtmlWebpackPlugin from 'html-webpack-plugin';
|
|
||||||
import { indexHtmlPath } from '@storybook/core/server';
|
|
||||||
import { version } from '../../../package.json';
|
|
||||||
import { includePaths, excludePaths, nodeModulesPaths } from './utils';
|
|
||||||
|
|
||||||
const getConfig = options => ({
|
import GeneratePagePlugin from 'generate-page-webpack-plugin';
|
||||||
|
|
||||||
|
import { getManagerHeadHtml } from '@storybook/core/server';
|
||||||
|
|
||||||
|
import { includePaths, excludePaths, nodeModulesPaths } from './utils';
|
||||||
|
import { version } from '../../../package.json';
|
||||||
|
|
||||||
|
const getConfig = options => {
|
||||||
|
const entriesMeta = {
|
||||||
|
manager: {
|
||||||
|
headHtmlSnippet: getManagerHeadHtml(options.configDir, process.env),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
mode: 'development',
|
mode: 'development',
|
||||||
devtool: '#cheap-module-eval-source-map',
|
devtool: '#cheap-module-eval-source-map',
|
||||||
entry: {
|
entry: {
|
||||||
@ -21,13 +31,21 @@ const getConfig = options => ({
|
|||||||
publicPath: '/',
|
publicPath: '/',
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
new HtmlWebpackPlugin({
|
new GeneratePagePlugin(
|
||||||
filename: 'index.html',
|
{
|
||||||
data: {
|
template: require.resolve('@storybook/core/dist/server/templates/index.html.ejs'),
|
||||||
version,
|
// eslint-disable-next-line global-require
|
||||||
|
parser: require('ejs'),
|
||||||
|
filename: entry => (entry === 'manager' ? 'index' : entry),
|
||||||
},
|
},
|
||||||
template: indexHtmlPath,
|
{
|
||||||
}),
|
data: { version },
|
||||||
|
headHtmlSnippet: entry =>
|
||||||
|
entriesMeta[entry] ? entriesMeta[entry].headHtmlSnippet : null,
|
||||||
|
bodyHtmlSnippet: entry =>
|
||||||
|
entriesMeta[entry] ? entriesMeta[entry].bodyHtmlSnippet : null,
|
||||||
|
}
|
||||||
|
),
|
||||||
new webpack.HotModuleReplacementPlugin(),
|
new webpack.HotModuleReplacementPlugin(),
|
||||||
new CaseSensitivePathsPlugin(),
|
new CaseSensitivePathsPlugin(),
|
||||||
new WatchMissingNodeModulesPlugin(nodeModulesPaths),
|
new WatchMissingNodeModulesPlugin(nodeModulesPaths),
|
||||||
@ -52,6 +70,7 @@ const getConfig = options => ({
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
});
|
};
|
||||||
|
};
|
||||||
|
|
||||||
export default getConfig;
|
export default getConfig;
|
||||||
|
@ -2,8 +2,9 @@ import path from 'path';
|
|||||||
import webpack from 'webpack';
|
import webpack from 'webpack';
|
||||||
import { getEnvironment } from 'universal-dotenv';
|
import { getEnvironment } from 'universal-dotenv';
|
||||||
import Dotenv from 'dotenv-webpack';
|
import Dotenv from 'dotenv-webpack';
|
||||||
import HtmlWebpackPlugin from 'html-webpack-plugin';
|
import GeneratePagePlugin from 'generate-page-webpack-plugin';
|
||||||
import { indexHtmlPath } from '@storybook/core/server';
|
|
||||||
|
import { getManagerHeadHtml } from '@storybook/core/dist/server/utils';
|
||||||
import { version } from '../../../package.json';
|
import { version } from '../../../package.json';
|
||||||
import { includePaths, excludePaths } from './utils';
|
import { includePaths, excludePaths } from './utils';
|
||||||
|
|
||||||
@ -26,13 +27,18 @@ const getConfig = options => {
|
|||||||
publicPath: '/',
|
publicPath: '/',
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
new HtmlWebpackPlugin({
|
new GeneratePagePlugin(
|
||||||
filename: 'index.html',
|
{
|
||||||
data: {
|
template: require.resolve('@storybook/core/dist/server/templates/index.html.ejs'),
|
||||||
version,
|
// eslint-disable-next-line global-require
|
||||||
|
parser: require('ejs'),
|
||||||
|
filename: entry => (entry === 'manager' ? 'index' : entry),
|
||||||
},
|
},
|
||||||
template: indexHtmlPath,
|
{
|
||||||
}),
|
data: { version },
|
||||||
|
headHtmlSnippet: getManagerHeadHtml(options.configDir, process.env),
|
||||||
|
}
|
||||||
|
),
|
||||||
new webpack.DefinePlugin({
|
new webpack.DefinePlugin({
|
||||||
'process.env.NODE_ENV': '"production"',
|
'process.env.NODE_ENV': '"production"',
|
||||||
storybookOptions: JSON.stringify(options),
|
storybookOptions: JSON.stringify(options),
|
||||||
|
3
app/react-native/src/server/middleware.js
vendored
3
app/react-native/src/server/middleware.js
vendored
@ -20,7 +20,8 @@ function getMiddleware(configDir) {
|
|||||||
return () => {};
|
return () => {};
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function({ projectDir, configDir, ...options }) {
|
export default function(options) {
|
||||||
|
const { projectDir, configDir } = options;
|
||||||
// Build the webpack configuration using the `baseConfig`
|
// Build the webpack configuration using the `baseConfig`
|
||||||
// custom `.babelrc` file and `webpack.config.js` files
|
// custom `.babelrc` file and `webpack.config.js` files
|
||||||
const environment = options.environment || 'DEVELOPMENT';
|
const environment = options.environment || 'DEVELOPMENT';
|
||||||
|
@ -11,7 +11,7 @@ setOptions({
|
|||||||
sortStoriesByKind: false,
|
sortStoriesByKind: false,
|
||||||
hierarchySeparator: /\./,
|
hierarchySeparator: /\./,
|
||||||
hierarchyRootSeparator: /\|/,
|
hierarchyRootSeparator: /\|/,
|
||||||
enableShortcuts: false,
|
enableShortcuts: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
function loadStories() {
|
function loadStories() {
|
||||||
|
@ -38,12 +38,13 @@
|
|||||||
"core-js": "^2.5.7",
|
"core-js": "^2.5.7",
|
||||||
"css-loader": "^1.0.0",
|
"css-loader": "^1.0.0",
|
||||||
"dotenv-webpack": "^1.5.7",
|
"dotenv-webpack": "^1.5.7",
|
||||||
|
"ejs": "^2.6.1",
|
||||||
"emotion": "^9.2.6",
|
"emotion": "^9.2.6",
|
||||||
"express": "^4.16.3",
|
"express": "^4.16.3",
|
||||||
"file-loader": "^1.1.11",
|
"file-loader": "^1.1.11",
|
||||||
"find-cache-dir": "^2.0.0",
|
"find-cache-dir": "^2.0.0",
|
||||||
|
"generate-page-webpack-plugin": "^1.0.0",
|
||||||
"global": "^4.3.2",
|
"global": "^4.3.2",
|
||||||
"html-webpack-plugin": "^3.2.0",
|
|
||||||
"interpret": "^1.1.0",
|
"interpret": "^1.1.0",
|
||||||
"json5": "^1.0.1",
|
"json5": "^1.0.1",
|
||||||
"postcss-flexbugs-fixes": "^4.1.0",
|
"postcss-flexbugs-fixes": "^4.1.0",
|
||||||
|
@ -4,7 +4,4 @@ const serverUtils = require('./dist/server/utils');
|
|||||||
const buildStatic = require('./dist/server/build-static');
|
const buildStatic = require('./dist/server/build-static');
|
||||||
const buildDev = require('./dist/server/build-dev');
|
const buildDev = require('./dist/server/build-dev');
|
||||||
|
|
||||||
module.exports = assign({}, defaultWebpackConfig, buildStatic, buildDev, serverUtils, {
|
module.exports = assign({}, defaultWebpackConfig, buildStatic, buildDev, serverUtils);
|
||||||
indexHtmlPath: require.resolve('./src/server/index.html.ejs'),
|
|
||||||
iframeHtmlPath: require.resolve('./src/server/iframe.html.ejs'),
|
|
||||||
});
|
|
||||||
|
@ -37,7 +37,7 @@ export function loadEnv(options = {}) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function getEntries(configDir) {
|
export function getEntries(configDir) {
|
||||||
const preview = [require.resolve('./polyfills'), require.resolve('./globals')];
|
const iframe = [require.resolve('./polyfills'), require.resolve('./globals')];
|
||||||
const manager = [require.resolve('./polyfills'), require.resolve('../../client/manager')];
|
const manager = [require.resolve('./polyfills'), require.resolve('../../client/manager')];
|
||||||
|
|
||||||
// Check whether a config.{ext} file exists inside the storybook
|
// Check whether a config.{ext} file exists inside the storybook
|
||||||
@ -47,7 +47,7 @@ export function getEntries(configDir) {
|
|||||||
throw new Error(`=> Create a storybook config file in "${configDir}/config.{ext}".`);
|
throw new Error(`=> Create a storybook config file in "${configDir}/config.{ext}".`);
|
||||||
}
|
}
|
||||||
|
|
||||||
preview.push(require.resolve(storybookConfigPath));
|
iframe.push(require.resolve(storybookConfigPath));
|
||||||
|
|
||||||
// Check whether addons.{ext} file exists inside the storybook.
|
// Check whether addons.{ext} file exists inside the storybook.
|
||||||
const storybookCustomAddonsPath = getInterpretedFile(path.resolve(configDir, 'addons'));
|
const storybookCustomAddonsPath = getInterpretedFile(path.resolve(configDir, 'addons'));
|
||||||
@ -56,5 +56,5 @@ export function getEntries(configDir) {
|
|||||||
manager.unshift(storybookCustomAddonsPath);
|
manager.unshift(storybookCustomAddonsPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
return { preview, manager };
|
return { iframe, manager };
|
||||||
}
|
}
|
||||||
|
@ -2,12 +2,10 @@ import path from 'path';
|
|||||||
import webpack from 'webpack';
|
import webpack from 'webpack';
|
||||||
import { getEnvironment } from 'universal-dotenv';
|
import { getEnvironment } from 'universal-dotenv';
|
||||||
import Dotenv from 'dotenv-webpack';
|
import Dotenv from 'dotenv-webpack';
|
||||||
import InterpolateHtmlPlugin from 'react-dev-utils/InterpolateHtmlPlugin';
|
|
||||||
import WatchMissingNodeModulesPlugin from 'react-dev-utils/WatchMissingNodeModulesPlugin';
|
import WatchMissingNodeModulesPlugin from 'react-dev-utils/WatchMissingNodeModulesPlugin';
|
||||||
import CaseSensitivePathsPlugin from 'case-sensitive-paths-webpack-plugin';
|
import CaseSensitivePathsPlugin from 'case-sensitive-paths-webpack-plugin';
|
||||||
import HtmlWebpackPlugin from 'html-webpack-plugin';
|
import GeneratePagePlugin from 'generate-page-webpack-plugin';
|
||||||
|
import { getPreviewHeadHtml, getManagerHeadHtml, getPreviewBodyHtml } from '../utils';
|
||||||
import { getPreviewHeadHtml, getManagerHeadHtml } from '../utils';
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
includePaths,
|
includePaths,
|
||||||
@ -21,13 +19,23 @@ import { version } from '../../../package.json';
|
|||||||
|
|
||||||
export default ({ configDir, quiet, babelOptions }) => {
|
export default ({ configDir, quiet, babelOptions }) => {
|
||||||
const entries = getEntries(configDir, true);
|
const entries = getEntries(configDir, true);
|
||||||
|
const entriesMeta = {
|
||||||
|
iframe: {
|
||||||
|
headHtmlSnippet: getPreviewHeadHtml(configDir, process.env),
|
||||||
|
bodyHtmlSnippet: getPreviewBodyHtml(),
|
||||||
|
},
|
||||||
|
manager: {
|
||||||
|
headHtmlSnippet: getManagerHeadHtml(configDir, process.env),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
mode: 'development',
|
mode: 'development',
|
||||||
devtool: 'cheap-module-source-map',
|
devtool: 'cheap-module-source-map',
|
||||||
entry: {
|
entry: {
|
||||||
...entries,
|
...entries,
|
||||||
preview: [
|
iframe: [
|
||||||
...entries.preview,
|
...entries.iframe,
|
||||||
`${require.resolve('webpack-hot-middleware/client')}?reload=true`,
|
`${require.resolve('webpack-hot-middleware/client')}?reload=true`,
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@ -42,26 +50,21 @@ export default ({ configDir, quiet, babelOptions }) => {
|
|||||||
publicPath: '',
|
publicPath: '',
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
new HtmlWebpackPlugin({
|
new GeneratePagePlugin(
|
||||||
filename: 'index.html',
|
{
|
||||||
chunks: ['manager'],
|
template: require.resolve('../templates/index.html.ejs'),
|
||||||
chunksSortMode: 'none',
|
// eslint-disable-next-line global-require
|
||||||
data: {
|
parser: require('ejs'),
|
||||||
managerHead: getManagerHeadHtml(configDir),
|
filename: entry => (entry === 'manager' ? 'index' : entry),
|
||||||
version,
|
|
||||||
},
|
},
|
||||||
template: require.resolve('../index.html.ejs'),
|
{
|
||||||
}),
|
data: { version },
|
||||||
new HtmlWebpackPlugin({
|
headHtmlSnippet: entry =>
|
||||||
filename: 'iframe.html',
|
entriesMeta[entry] ? entriesMeta[entry].headHtmlSnippet : null,
|
||||||
excludeChunks: ['manager'],
|
bodyHtmlSnippet: entry =>
|
||||||
chunksSortMode: 'none',
|
entriesMeta[entry] ? entriesMeta[entry].bodyHtmlSnippet : null,
|
||||||
data: {
|
}
|
||||||
previewHead: getPreviewHeadHtml(configDir),
|
),
|
||||||
},
|
|
||||||
template: require.resolve('../iframe.html.ejs'),
|
|
||||||
}),
|
|
||||||
new InterpolateHtmlPlugin(process.env),
|
|
||||||
new webpack.DefinePlugin(loadEnv()),
|
new webpack.DefinePlugin(loadEnv()),
|
||||||
new webpack.HotModuleReplacementPlugin(),
|
new webpack.HotModuleReplacementPlugin(),
|
||||||
new CaseSensitivePathsPlugin(),
|
new CaseSensitivePathsPlugin(),
|
||||||
|
@ -1,19 +1,31 @@
|
|||||||
import webpack from 'webpack';
|
import webpack from 'webpack';
|
||||||
import { getEnvironment } from 'universal-dotenv';
|
import { getEnvironment } from 'universal-dotenv';
|
||||||
import Dotenv from 'dotenv-webpack';
|
import Dotenv from 'dotenv-webpack';
|
||||||
import InterpolateHtmlPlugin from 'react-dev-utils/InterpolateHtmlPlugin';
|
import GeneratePagePlugin from 'generate-page-webpack-plugin';
|
||||||
import HtmlWebpackPlugin from 'html-webpack-plugin';
|
|
||||||
|
|
||||||
import { version } from '../../../package.json';
|
import { version } from '../../../package.json';
|
||||||
import { getPreviewHeadHtml, getManagerHeadHtml } from '../utils';
|
import { getPreviewHeadHtml, getManagerHeadHtml, getPreviewBodyHtml } from '../utils';
|
||||||
|
|
||||||
import { includePaths, excludePaths, loadEnv, nodePaths, getEntries } from './utils';
|
import { includePaths, excludePaths, loadEnv, nodePaths, getEntries } from './utils';
|
||||||
|
|
||||||
export default ({ configDir, babelOptions }) => ({
|
export default ({ configDir, babelOptions }) => {
|
||||||
|
const entries = getEntries(configDir);
|
||||||
|
|
||||||
|
const entriesMeta = {
|
||||||
|
iframe: {
|
||||||
|
headHtmlSnippet: getPreviewHeadHtml(configDir, process.env),
|
||||||
|
bodyHtmlSnippet: getPreviewBodyHtml(),
|
||||||
|
},
|
||||||
|
manager: {
|
||||||
|
headHtmlSnippet: getManagerHeadHtml(configDir, process.env),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
mode: 'production',
|
mode: 'production',
|
||||||
bail: true,
|
bail: true,
|
||||||
devtool: '#cheap-module-source-map',
|
devtool: '#cheap-module-source-map',
|
||||||
entry: getEntries(configDir),
|
entry: entries,
|
||||||
output: {
|
output: {
|
||||||
filename: 'static/[name].[chunkhash].bundle.js',
|
filename: 'static/[name].[chunkhash].bundle.js',
|
||||||
// Here we set the publicPath to ''.
|
// Here we set the publicPath to ''.
|
||||||
@ -24,26 +36,21 @@ export default ({ configDir, babelOptions }) => ({
|
|||||||
publicPath: '',
|
publicPath: '',
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
new HtmlWebpackPlugin({
|
new GeneratePagePlugin(
|
||||||
filename: 'index.html',
|
{
|
||||||
chunks: ['manager', 'runtime~manager'],
|
template: require.resolve('../templates/index.html.ejs'),
|
||||||
chunksSortMode: 'none',
|
// eslint-disable-next-line global-require
|
||||||
data: {
|
parser: require('ejs'),
|
||||||
managerHead: getManagerHeadHtml(configDir),
|
filename: entry => (entry === 'manager' ? 'index' : entry),
|
||||||
version,
|
|
||||||
},
|
},
|
||||||
template: require.resolve('../index.html.ejs'),
|
{
|
||||||
}),
|
data: { version },
|
||||||
new HtmlWebpackPlugin({
|
headHtmlSnippet: entry =>
|
||||||
filename: 'iframe.html',
|
entriesMeta[entry] ? entriesMeta[entry].headHtmlSnippet : null,
|
||||||
excludeChunks: ['manager', 'runtime~manager'],
|
bodyHtmlSnippet: entry =>
|
||||||
chunksSortMode: 'none',
|
entriesMeta[entry] ? entriesMeta[entry].bodyHtmlSnippet : null,
|
||||||
data: {
|
}
|
||||||
previewHead: getPreviewHeadHtml(configDir),
|
),
|
||||||
},
|
|
||||||
template: require.resolve('../iframe.html.ejs'),
|
|
||||||
}),
|
|
||||||
new InterpolateHtmlPlugin(process.env),
|
|
||||||
new webpack.DefinePlugin(loadEnv({ production: true })),
|
new webpack.DefinePlugin(loadEnv({ production: true })),
|
||||||
new webpack.DefinePlugin(getEnvironment().webpack),
|
new webpack.DefinePlugin(getEnvironment().webpack),
|
||||||
new Dotenv({ silent: true }),
|
new Dotenv({ silent: true }),
|
||||||
@ -89,4 +96,5 @@ export default ({ configDir, babelOptions }) => ({
|
|||||||
// https://twitter.com/wSokra/status/969679223278505985
|
// https://twitter.com/wSokra/status/969679223278505985
|
||||||
runtimeChunk: true,
|
runtimeChunk: true,
|
||||||
},
|
},
|
||||||
});
|
};
|
||||||
|
};
|
||||||
|
@ -1,89 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8">
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
||||||
<meta content="IE=edge" http-equiv="X-UA-Compatible" />
|
|
||||||
<base target="_parent">
|
|
||||||
<title>Storybook</title>
|
|
||||||
<link rel="shortcut icon" href="favicon.ico?v=1" />
|
|
||||||
<%= htmlWebpackPlugin.options.data.previewHead %>
|
|
||||||
<style>
|
|
||||||
:not(.sb-show-main) > .sb-main,
|
|
||||||
:not(.sb-show-nopreview) > .sb-nopreview,
|
|
||||||
:not(.sb-show-errordisplay) > .sb-errordisplay {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.sb-wrapper {
|
|
||||||
position: fixed;
|
|
||||||
top: 0;
|
|
||||||
bottom: 0;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
padding: 20px;
|
|
||||||
font-family: -apple-system, ".SFNSText-Regular", "San Francisco", Roboto, "Segoe UI", "Helvetica Neue", "Lucida Grande", sans-serif;
|
|
||||||
-webkit-font-smoothing: antialiased;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sb-heading {
|
|
||||||
font-size: 20px;
|
|
||||||
font-weight: 600;
|
|
||||||
letter-spacing: 0.2px;
|
|
||||||
margin: 10px 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.sb-nopreview {
|
|
||||||
display: flex;
|
|
||||||
align-content: center;
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sb-nopreview_main {
|
|
||||||
margin: auto;
|
|
||||||
padding: 30px;
|
|
||||||
border-radius: 10px;
|
|
||||||
background: rgba(0,0,0,0.03);
|
|
||||||
}
|
|
||||||
|
|
||||||
.sb-nopreview_heading {
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.sb-errordisplay {
|
|
||||||
background-color: rgb(187, 49, 49);
|
|
||||||
color: #FFF;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sb-errordisplay_code {
|
|
||||||
font-size: 14px;
|
|
||||||
width: 100vw;
|
|
||||||
overflow: auto;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
</head>
|
|
||||||
<body class="sb-show-main">
|
|
||||||
<div id="root" class="sb-main"></div>
|
|
||||||
|
|
||||||
<div class="sb-nopreview sb-wrapper">
|
|
||||||
<div class="sb-nopreview_main">
|
|
||||||
<h1 class="sb-nopreview_heading sb-heading">No Preview</h1>
|
|
||||||
<p>Sorry, but you either have no stories or none are selected somehow.</p>
|
|
||||||
<ul>
|
|
||||||
<li>Please check the storybook config.</li>
|
|
||||||
<li>Try reloading the page.</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="sb-errordisplay sb-wrapper">
|
|
||||||
<div id="error-message" class="sb-heading"></div>
|
|
||||||
<pre class="sb-errordisplay_code">
|
|
||||||
<code id="error-stack"></code>
|
|
||||||
</pre>
|
|
||||||
</div>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
@ -1,23 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8">
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
||||||
<meta name="storybook-version" content="<%= htmlWebpackPlugin.options.data.version %>">
|
|
||||||
<meta content="IE=edge" http-equiv="X-UA-Compatible" />
|
|
||||||
<title>Storybook</title>
|
|
||||||
<link rel="shortcut icon" href="favicon.ico?v=1" />
|
|
||||||
<%= htmlWebpackPlugin.options.data.managerHead %>
|
|
||||||
|
|
||||||
</head>
|
|
||||||
<body style="margin: 0;">
|
|
||||||
<style>
|
|
||||||
html, body {
|
|
||||||
overflow: hidden;
|
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
<div id="root"></div>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
16
lib/core/src/server/templates/base-manager-head.html
Normal file
16
lib/core/src/server/templates/base-manager-head.html
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
<style>
|
||||||
|
html, body {
|
||||||
|
overflow: hidden;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
if (window.parent !== window) {
|
||||||
|
window.__REACT_DEVTOOLS_GLOBAL_HOOK__ = window.parent.__REACT_DEVTOOLS_GLOBAL_HOOK__;
|
||||||
|
window.__VUE_DEVTOOLS_GLOBAL_HOOK__ = window.parent.__VUE_DEVTOOLS_GLOBAL_HOOK__;
|
||||||
|
}
|
||||||
|
</script>
|
17
lib/core/src/server/templates/base-preview-body.html
Normal file
17
lib/core/src/server/templates/base-preview-body.html
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<div class="sb-nopreview sb-wrapper">
|
||||||
|
<div class="sb-nopreview_main">
|
||||||
|
<h1 class="sb-nopreview_heading sb-heading">No Preview</h1>
|
||||||
|
<p>Sorry, but you either have no stories or none are selected somehow.</p>
|
||||||
|
<ul>
|
||||||
|
<li>Please check the storybook config.</li>
|
||||||
|
<li>Try reloading the page.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="sb-errordisplay sb-wrapper">
|
||||||
|
<div id="error-message" class="sb-heading"></div>
|
||||||
|
<pre class="sb-errordisplay_code">
|
||||||
|
<code id="error-stack"></code>
|
||||||
|
</pre>
|
||||||
|
</div>
|
62
lib/core/src/server/templates/base-preview-head.html
Normal file
62
lib/core/src/server/templates/base-preview-head.html
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
<base target="_parent">
|
||||||
|
|
||||||
|
<style>
|
||||||
|
:not(.sb-show-main) > .sb-main,
|
||||||
|
:not(.sb-show-nopreview) > .sb-nopreview,
|
||||||
|
:not(.sb-show-errordisplay) > .sb-errordisplay {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sb-wrapper {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
padding: 20px;
|
||||||
|
font-family: -apple-system, ".SFNSText-Regular", "San Francisco", Roboto, "Segoe UI", "Helvetica Neue", "Lucida Grande", sans-serif;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sb-heading {
|
||||||
|
font-size: 20px;
|
||||||
|
font-weight: 600;
|
||||||
|
letter-spacing: 0.2px;
|
||||||
|
margin: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sb-nopreview {
|
||||||
|
display: flex;
|
||||||
|
align-content: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sb-nopreview_main {
|
||||||
|
margin: auto;
|
||||||
|
padding: 30px;
|
||||||
|
border-radius: 10px;
|
||||||
|
background: rgba(0,0,0,0.03);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sb-nopreview_heading {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sb-errordisplay {
|
||||||
|
background-color: rgb(187, 49, 49);
|
||||||
|
color: #FFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sb-errordisplay_code {
|
||||||
|
font-size: 14px;
|
||||||
|
width: 100vw;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
if (window.parent !== window) {
|
||||||
|
window.__REACT_DEVTOOLS_GLOBAL_HOOK__ = window.parent.__REACT_DEVTOOLS_GLOBAL_HOOK__;
|
||||||
|
window.__VUE_DEVTOOLS_GLOBAL_HOOK__ = window.parent.__VUE_DEVTOOLS_GLOBAL_HOOK__;
|
||||||
|
}
|
||||||
|
</script>
|
56
lib/core/src/server/templates/index.html.ejs
Normal file
56
lib/core/src/server/templates/index.html.ejs
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
<meta content="IE=edge" http-equiv="X-UA-Compatible" />
|
||||||
|
<title>Storybook</title>
|
||||||
|
<link rel="shortcut icon" href="favicon.ico?v=1" />
|
||||||
|
|
||||||
|
<% if (options.links) { %>
|
||||||
|
<% for (item of options.links) { %>
|
||||||
|
<% if (typeof item === 'string' || item instanceof String) { item = { href: item, rel: 'stylesheet' } } %>
|
||||||
|
<link<% for (key in item) { %> <%= key %>="<%= item[key] %>"<% } %>>
|
||||||
|
<% } %>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
<% if (options.headHtmlSnippet) { %>
|
||||||
|
<%- options.headHtmlSnippet %>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<% if (options.window) { %>
|
||||||
|
<script>
|
||||||
|
<% for (key in options.window) { %>
|
||||||
|
window['<%= key %>'] = <%= JSON.stringify(options.window[key]) %>;
|
||||||
|
<% } %>
|
||||||
|
</script>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
<% if (options.bodyHtmlSnippet) { %>
|
||||||
|
<%- options.bodyHtmlSnippet %>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
<div id="root"></div>
|
||||||
|
|
||||||
|
<% if (options.scripts) { %>
|
||||||
|
<% for (item of options.scripts) { %>
|
||||||
|
<% if (typeof item === 'string' || item instanceof String) { item = { src: item } } %>
|
||||||
|
<script <% for (key in item) { %> <%= key %>="<%= item[key] %>"<% } %> defer></script>
|
||||||
|
<% } %>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
<% for (key in dlls) { %>
|
||||||
|
<script src="<%= compilation.outputOptions.publicPath %><%= dlls[key] %>" defer></script>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
<% for (index in chunks) { %>
|
||||||
|
<% for (key in chunks[index].files) { %>
|
||||||
|
<script src="<%= compilation.outputOptions.publicPath %><%= chunks[index].files[key] %>" defer></script>
|
||||||
|
<% } %>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
@ -27,22 +27,35 @@ export function getMiddleware(configDir) {
|
|||||||
return () => {};
|
return () => {};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getPreviewHeadHtml(configDirPath) {
|
const interpolate = (string, data = {}) =>
|
||||||
|
Object.entries(data).reduce((acc, [k, v]) => acc.replace(`%${k}%`, v), string);
|
||||||
|
|
||||||
|
export function getPreviewBodyHtml() {
|
||||||
|
return fs.readFileSync(path.resolve(__dirname, 'templates/base-preview-body.html'), 'utf8');
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getPreviewHeadHtml(configDirPath, interpolations) {
|
||||||
|
const base = fs.readFileSync(path.resolve(__dirname, 'templates/base-preview-head.html'), 'utf8');
|
||||||
const headHtmlPath = path.resolve(configDirPath, 'preview-head.html');
|
const headHtmlPath = path.resolve(configDirPath, 'preview-head.html');
|
||||||
|
|
||||||
|
let result = base;
|
||||||
|
|
||||||
if (fs.existsSync(headHtmlPath)) {
|
if (fs.existsSync(headHtmlPath)) {
|
||||||
return fs.readFileSync(headHtmlPath, 'utf8');
|
result += fs.readFileSync(headHtmlPath, 'utf8');
|
||||||
}
|
}
|
||||||
|
|
||||||
return '';
|
return interpolate(result, interpolations);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getManagerHeadHtml(configDirPath) {
|
export function getManagerHeadHtml(configDirPath, interpolations) {
|
||||||
|
const base = fs.readFileSync(path.resolve(__dirname, 'templates/base-manager-head.html'), 'utf8');
|
||||||
const scriptPath = path.resolve(configDirPath, 'manager-head.html');
|
const scriptPath = path.resolve(configDirPath, 'manager-head.html');
|
||||||
|
|
||||||
|
let result = base;
|
||||||
|
|
||||||
if (fs.existsSync(scriptPath)) {
|
if (fs.existsSync(scriptPath)) {
|
||||||
return fs.readFileSync(scriptPath, 'utf8');
|
result += fs.readFileSync(scriptPath, 'utf8');
|
||||||
}
|
}
|
||||||
|
|
||||||
return '';
|
return interpolate(result, interpolations);
|
||||||
}
|
}
|
||||||
|
@ -2,11 +2,13 @@ import mock from 'mock-fs';
|
|||||||
import { getPreviewHeadHtml } from './utils';
|
import { getPreviewHeadHtml } from './utils';
|
||||||
|
|
||||||
const HEAD_HTML_CONTENTS = '<script>console.log("custom script!");</script>';
|
const HEAD_HTML_CONTENTS = '<script>console.log("custom script!");</script>';
|
||||||
|
const BASE_HTML_CONTENTS = '<script>console.log("base script!");</script>';
|
||||||
|
|
||||||
describe('server.getPreviewHeadHtml', () => {
|
describe('server.getPreviewHeadHtml', () => {
|
||||||
describe('when .storybook/preview-head.html does not exist', () => {
|
describe('when .storybook/preview-head.html does not exist', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
mock({
|
mock({
|
||||||
|
[`${__dirname}/templates/base-preview-head.html`]: BASE_HTML_CONTENTS,
|
||||||
config: {},
|
config: {},
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@ -17,13 +19,14 @@ describe('server.getPreviewHeadHtml', () => {
|
|||||||
|
|
||||||
it('return an empty string', () => {
|
it('return an empty string', () => {
|
||||||
const result = getPreviewHeadHtml('./config');
|
const result = getPreviewHeadHtml('./config');
|
||||||
expect(result).toEqual('');
|
expect(result).toEqual(BASE_HTML_CONTENTS);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when .storybook/preview-head.html exists', () => {
|
describe('when .storybook/preview-head.html exists', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
mock({
|
mock({
|
||||||
|
[`${__dirname}/templates/base-preview-head.html`]: BASE_HTML_CONTENTS,
|
||||||
config: {
|
config: {
|
||||||
'preview-head.html': HEAD_HTML_CONTENTS,
|
'preview-head.html': HEAD_HTML_CONTENTS,
|
||||||
},
|
},
|
||||||
@ -36,7 +39,7 @@ describe('server.getPreviewHeadHtml', () => {
|
|||||||
|
|
||||||
it('return the contents of the file', () => {
|
it('return the contents of the file', () => {
|
||||||
const result = getPreviewHeadHtml('./config');
|
const result = getPreviewHeadHtml('./config');
|
||||||
expect(result).toEqual(HEAD_HTML_CONTENTS);
|
expect(result).toEqual(BASE_HTML_CONTENTS + HEAD_HTML_CONTENTS);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -5822,6 +5822,10 @@ ejs@^2.5.7:
|
|||||||
version "2.5.7"
|
version "2.5.7"
|
||||||
resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.5.7.tgz#cc872c168880ae3c7189762fd5ffc00896c9518a"
|
resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.5.7.tgz#cc872c168880ae3c7189762fd5ffc00896c9518a"
|
||||||
|
|
||||||
|
ejs@^2.6.1:
|
||||||
|
version "2.6.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.6.1.tgz#498ec0d495655abc6f23cd61868d926464071aa0"
|
||||||
|
|
||||||
ejson@^2.1.2:
|
ejson@^2.1.2:
|
||||||
version "2.1.2"
|
version "2.1.2"
|
||||||
resolved "https://registry.yarnpkg.com/ejson/-/ejson-2.1.2.tgz#0eed4055bc7e0e7561fe59e8c320edc3ff8ce7df"
|
resolved "https://registry.yarnpkg.com/ejson/-/ejson-2.1.2.tgz#0eed4055bc7e0e7561fe59e8c320edc3ff8ce7df"
|
||||||
@ -7576,6 +7580,10 @@ gaze@^1.0.0:
|
|||||||
dependencies:
|
dependencies:
|
||||||
globule "^1.0.0"
|
globule "^1.0.0"
|
||||||
|
|
||||||
|
generate-page-webpack-plugin@^1.0.0:
|
||||||
|
version "1.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/generate-page-webpack-plugin/-/generate-page-webpack-plugin-1.0.0.tgz#e6261efb7e6b9ef8a8136126b14fa26aedfb7f33"
|
||||||
|
|
||||||
genfun@^4.0.1:
|
genfun@^4.0.1:
|
||||||
version "4.0.1"
|
version "4.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/genfun/-/genfun-4.0.1.tgz#ed10041f2e4a7f1b0a38466d17a5c3e27df1dfc1"
|
resolved "https://registry.yarnpkg.com/genfun/-/genfun-4.0.1.tgz#ed10041f2e4a7f1b0a38466d17a5c3e27df1dfc1"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user