mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 08:01:20 +08:00
Merge pull request #3913 from storybooks/support-different-config-files-exts
Support different extensions for "config" and "addons" files
This commit is contained in:
commit
989aa6c9b3
@ -123,7 +123,9 @@ exports[`addon Info should render <Info /> and external markdown 1`] = `
|
|||||||
>
|
>
|
||||||
<div>
|
<div>
|
||||||
<h1>
|
<h1>
|
||||||
function func(x) {return x + 1;}
|
function func(x) {
|
||||||
|
return x + 1;
|
||||||
|
}
|
||||||
</h1>
|
</h1>
|
||||||
<h2>
|
<h2>
|
||||||
[object Object]
|
[object Object]
|
||||||
@ -1430,7 +1432,9 @@ containing **bold**, *cursive* text, \`code\` and [a link](https://github.com)"
|
|||||||
>
|
>
|
||||||
<div>
|
<div>
|
||||||
<h1>
|
<h1>
|
||||||
function func(x) {return x + 1;}
|
function func(x) {
|
||||||
|
return x + 1;
|
||||||
|
}
|
||||||
</h1>
|
</h1>
|
||||||
<h2>
|
<h2>
|
||||||
[object Object]
|
[object Object]
|
||||||
|
@ -5,7 +5,7 @@ function getConfigPathParts(configPath) {
|
|||||||
const resolvedConfigPath = path.resolve(configPath);
|
const resolvedConfigPath = path.resolve(configPath);
|
||||||
|
|
||||||
if (fs.lstatSync(resolvedConfigPath).isDirectory()) {
|
if (fs.lstatSync(resolvedConfigPath).isDirectory()) {
|
||||||
return path.join(resolvedConfigPath, 'config.js');
|
return path.join(resolvedConfigPath, 'config');
|
||||||
}
|
}
|
||||||
|
|
||||||
return resolvedConfigPath;
|
return resolvedConfigPath;
|
||||||
|
2
app/angular/index.d.ts
vendored
2
app/angular/index.d.ts
vendored
@ -39,6 +39,6 @@ declare module '@storybook/angular' {
|
|||||||
export function setAddon(addon: any): void;
|
export function setAddon(addon: any): void;
|
||||||
export function addDecorator(decorator: any): IApi;
|
export function addDecorator(decorator: any): IApi;
|
||||||
export function addParameters(parameters: any): IApi;
|
export function addParameters(parameters: any): IApi;
|
||||||
export function configure(loaders: () => NodeRequire, module: NodeModule): void;
|
export function configure(loaders: () => void, module: NodeModule): void;
|
||||||
export function getStorybook(): IStoribookSection[];
|
export function getStorybook(): IStoribookSection[];
|
||||||
}
|
}
|
||||||
|
2
app/angular/src/client/preview/index.d.ts
vendored
2
app/angular/src/client/preview/index.d.ts
vendored
@ -1,6 +1,6 @@
|
|||||||
export function storiesOf(kind: string, module: NodeModule): IApi;
|
export function storiesOf(kind: string, module: NodeModule): IApi;
|
||||||
export function setAddon(addon: any): void;
|
export function setAddon(addon: any): void;
|
||||||
export function addDecorator(decorator: any): IApi;
|
export function addDecorator(decorator: any): IApi;
|
||||||
export function configure(loaders: () => NodeRequire, module: NodeModule): void;
|
export function configure(loaders: () => void, module: NodeModule): void;
|
||||||
export function getStorybook(): IStoribookSection[];
|
export function getStorybook(): IStoribookSection[];
|
||||||
export function clearDecorators(): void;
|
export function clearDecorators(): void;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
const path = require('path');
|
const path = require('path');
|
||||||
|
|
||||||
module.exports = baseConfig => {
|
module.exports = (baseConfig: any) => {
|
||||||
baseConfig.module.rules.push({
|
baseConfig.module.rules.push({
|
||||||
test: [/\.stories\.tsx?$/, /index\.ts$/],
|
test: [/\.stories\.tsx?$/, /index\.ts$/],
|
||||||
loaders: [
|
loaders: [
|
@ -70,6 +70,8 @@
|
|||||||
"@types/jasminewd2": "^2.0.3",
|
"@types/jasminewd2": "^2.0.3",
|
||||||
"@types/jest": "^22.2.3",
|
"@types/jest": "^22.2.3",
|
||||||
"@types/node": "~9.6.20",
|
"@types/node": "~9.6.20",
|
||||||
|
"@types/storybook__addon-options": "^3.2.2",
|
||||||
|
"@types/webpack-env": "^1.13.6",
|
||||||
"babel-core": "^6.26.3",
|
"babel-core": "^6.26.3",
|
||||||
"global": "^4.3.2",
|
"global": "^4.3.2",
|
||||||
"jasmine-core": "~3.1.0",
|
"jasmine-core": "~3.1.0",
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
"experimentalDecorators": true,
|
"experimentalDecorators": true,
|
||||||
"target": "es5",
|
"target": "es5",
|
||||||
"typeRoots": [
|
"typeRoots": [
|
||||||
|
"../../node_modules/@types",
|
||||||
"node_modules/@types"
|
"node_modules/@types"
|
||||||
],
|
],
|
||||||
"lib": [
|
"lib": [
|
||||||
|
@ -22,6 +22,7 @@ module.exports = {
|
|||||||
],
|
],
|
||||||
transform: {
|
transform: {
|
||||||
'^.+\\.jsx?$': 'babel-jest',
|
'^.+\\.jsx?$': 'babel-jest',
|
||||||
|
'^.+[/\\\\].storybook[/\\\\]config\\.ts$': '<rootDir>/scripts/jest-ts-babel.js',
|
||||||
'^.+\\.(ts|html)$': '<rootDir>/node_modules/jest-preset-angular/preprocessor.js',
|
'^.+\\.(ts|html)$': '<rootDir>/node_modules/jest-preset-angular/preprocessor.js',
|
||||||
'.*\\.(vue)$': '<rootDir>/node_modules/jest-vue-preprocessor',
|
'.*\\.(vue)$': '<rootDir>/node_modules/jest-vue-preprocessor',
|
||||||
},
|
},
|
||||||
|
@ -2,8 +2,8 @@ import findCacheDir from 'find-cache-dir';
|
|||||||
import { logger } from '@storybook/node-logger';
|
import { logger } from '@storybook/node-logger';
|
||||||
import { createDefaultWebpackConfig } from './config/defaults/webpack.config';
|
import { createDefaultWebpackConfig } from './config/defaults/webpack.config';
|
||||||
import devBabelConfig from './config/babel';
|
import devBabelConfig from './config/babel';
|
||||||
import loadBabelConfig from './babel_config';
|
import loadCustomBabelConfig from './loadCustomBabelConfig';
|
||||||
import loadCustomConfig from './loadCustomWebpackConfig';
|
import loadCustomWebpackConfig from './loadCustomWebpackConfig';
|
||||||
|
|
||||||
const noopWrapper = config => config;
|
const noopWrapper = config => config;
|
||||||
|
|
||||||
@ -14,7 +14,7 @@ function getBabelConfig({
|
|||||||
wrapBabelConfig = noopWrapper,
|
wrapBabelConfig = noopWrapper,
|
||||||
}) {
|
}) {
|
||||||
const defaultConfig = wrapDefaultBabelConfig(defaultBabelConfig);
|
const defaultConfig = wrapDefaultBabelConfig(defaultBabelConfig);
|
||||||
return wrapBabelConfig(loadBabelConfig(configDir, defaultConfig));
|
return wrapBabelConfig(loadCustomBabelConfig(configDir, defaultConfig));
|
||||||
}
|
}
|
||||||
|
|
||||||
function mergeConfigs(config, customConfig) {
|
function mergeConfigs(config, customConfig) {
|
||||||
@ -84,7 +84,7 @@ export default options => {
|
|||||||
|
|
||||||
// Check whether user has a custom webpack config file and
|
// Check whether user has a custom webpack config file and
|
||||||
// return the (extended) base configuration if it's not available.
|
// return the (extended) base configuration if it's not available.
|
||||||
const customConfig = loadCustomConfig(configDir);
|
const customConfig = loadCustomWebpackConfig(configDir);
|
||||||
|
|
||||||
if (customConfig === null) {
|
if (customConfig === null) {
|
||||||
informAboutCustomConfig(defaultConfigName);
|
informAboutCustomConfig(defaultConfigName);
|
||||||
|
21
lib/core/src/server/config/interpret-files.js
Normal file
21
lib/core/src/server/config/interpret-files.js
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import fs from 'fs';
|
||||||
|
import { extensions } from 'interpret';
|
||||||
|
|
||||||
|
const boost = new Set(['.js', '.jsx', '.ts', '.tsx']);
|
||||||
|
|
||||||
|
function sortExtensions() {
|
||||||
|
return [
|
||||||
|
...Array.from(boost),
|
||||||
|
...Object.keys(extensions)
|
||||||
|
.filter(ext => !boost.has(ext))
|
||||||
|
.sort((a, b) => a.length - b.length),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
const possibleExtensions = sortExtensions();
|
||||||
|
|
||||||
|
export function getInterpretedFile(pathToFile) {
|
||||||
|
return possibleExtensions
|
||||||
|
.map(ext => `${pathToFile}${ext}`)
|
||||||
|
.find(candidate => fs.existsSync(candidate));
|
||||||
|
}
|
37
lib/core/src/server/config/interpret-files.test.js
Normal file
37
lib/core/src/server/config/interpret-files.test.js
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
import mock from 'mock-fs';
|
||||||
|
import { getInterpretedFile } from './interpret-files';
|
||||||
|
|
||||||
|
describe('interpret-files', () => {
|
||||||
|
it('will interpret file as file.ts when it exists in fs', () => {
|
||||||
|
mock({
|
||||||
|
'path/to/file.ts': 'ts file contents',
|
||||||
|
});
|
||||||
|
|
||||||
|
const file = getInterpretedFile('path/to/file');
|
||||||
|
|
||||||
|
expect(file).toEqual('path/to/file.ts');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('will interpret file as file.js when both are in fs', () => {
|
||||||
|
mock({
|
||||||
|
'path/to/file.js': 'js file contents',
|
||||||
|
'path/to/file.ts': 'ts file contents',
|
||||||
|
});
|
||||||
|
|
||||||
|
const file = getInterpretedFile('path/to/file');
|
||||||
|
|
||||||
|
expect(file).toEqual('path/to/file.js');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('will return undefined if there is no candidate of a file in fs', () => {
|
||||||
|
mock({
|
||||||
|
'path/to/file.js': 'js file contents',
|
||||||
|
});
|
||||||
|
|
||||||
|
const file = getInterpretedFile('path/to/file2');
|
||||||
|
|
||||||
|
expect(file).toBeUndefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(mock.restore);
|
||||||
|
});
|
@ -1,7 +1,6 @@
|
|||||||
import fs from 'fs';
|
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
|
|
||||||
import { logger } from '@storybook/node-logger';
|
import { logger } from '@storybook/node-logger';
|
||||||
|
import { getInterpretedFile } from './interpret-files';
|
||||||
|
|
||||||
export const includePaths = [path.resolve('./')];
|
export const includePaths = [path.resolve('./')];
|
||||||
|
|
||||||
@ -41,18 +40,18 @@ export function getEntries(configDir) {
|
|||||||
const preview = [require.resolve('./polyfills'), require.resolve('./globals')];
|
const preview = [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.js file exists inside the storybook
|
// Check whether a config.{ext} file exists inside the storybook
|
||||||
// config directory and throw an error if it's not.
|
// config directory and throw an error if it's not.
|
||||||
const storybookConfigPath = path.resolve(configDir, 'config.js');
|
const storybookConfigPath = getInterpretedFile(path.resolve(configDir, 'config'));
|
||||||
if (!fs.existsSync(storybookConfigPath)) {
|
if (!storybookConfigPath) {
|
||||||
throw new Error(`=> Create a storybook config file in "${configDir}/config.js".`);
|
throw new Error(`=> Create a storybook config file in "${configDir}/config.{ext}".`);
|
||||||
}
|
}
|
||||||
|
|
||||||
preview.push(require.resolve(storybookConfigPath));
|
preview.push(require.resolve(storybookConfigPath));
|
||||||
|
|
||||||
// Check whether addons.js file exists inside the storybook.
|
// Check whether addons.{ext} file exists inside the storybook.
|
||||||
const storybookCustomAddonsPath = path.resolve(configDir, 'addons.js');
|
const storybookCustomAddonsPath = getInterpretedFile(path.resolve(configDir, 'addons'));
|
||||||
if (fs.existsSync(storybookCustomAddonsPath)) {
|
if (storybookCustomAddonsPath) {
|
||||||
logger.info('=> Loading custom addons config.');
|
logger.info('=> Loading custom addons config.');
|
||||||
manager.unshift(storybookCustomAddonsPath);
|
manager.unshift(storybookCustomAddonsPath);
|
||||||
}
|
}
|
||||||
|
@ -58,6 +58,7 @@ export default configDir => {
|
|||||||
}
|
}
|
||||||
return a.length - b.length;
|
return a.length - b.length;
|
||||||
});
|
});
|
||||||
|
|
||||||
const customConfigCandidates = ['webpack.config', 'webpackfile']
|
const customConfigCandidates = ['webpack.config', 'webpackfile']
|
||||||
.map(filename =>
|
.map(filename =>
|
||||||
extensions.map(ext => ({
|
extensions.map(ext => ({
|
||||||
|
@ -50,6 +50,7 @@
|
|||||||
"babel-cli": "^6.26.0",
|
"babel-cli": "^6.26.0",
|
||||||
"babel-core": "^6.26.3",
|
"babel-core": "^6.26.3",
|
||||||
"babel-eslint": "^8.2.3",
|
"babel-eslint": "^8.2.3",
|
||||||
|
"babel-jest": "^22.4.4",
|
||||||
"babel-plugin-emotion": "^9.1.2",
|
"babel-plugin-emotion": "^9.1.2",
|
||||||
"babel-plugin-macros": "^2.2.2",
|
"babel-plugin-macros": "^2.2.2",
|
||||||
"babel-plugin-require-context-hook": "^1.0.0",
|
"babel-plugin-require-context-hook": "^1.0.0",
|
||||||
|
9
scripts/jest-ts-babel.js
Normal file
9
scripts/jest-ts-babel.js
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
const tsTransformer = require('jest-preset-angular/preprocessor');
|
||||||
|
const babelTransformer = require('babel-jest');
|
||||||
|
|
||||||
|
module.exports.process = function transform(src, path, config, transformOptions) {
|
||||||
|
const tsResult = tsTransformer.process(src, path, config, transformOptions);
|
||||||
|
const jsPath = path.replace('.ts', '.js');
|
||||||
|
|
||||||
|
return babelTransformer.process(tsResult, jsPath, config, transformOptions);
|
||||||
|
};
|
@ -476,6 +476,14 @@
|
|||||||
version "2.53.43"
|
version "2.53.43"
|
||||||
resolved "https://registry.yarnpkg.com/@types/selenium-webdriver/-/selenium-webdriver-2.53.43.tgz#2de3d718819bc20165754c4a59afb7e9833f6707"
|
resolved "https://registry.yarnpkg.com/@types/selenium-webdriver/-/selenium-webdriver-2.53.43.tgz#2de3d718819bc20165754c4a59afb7e9833f6707"
|
||||||
|
|
||||||
|
"@types/storybook__addon-options@^3.2.2":
|
||||||
|
version "3.2.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/@types/storybook__addon-options/-/storybook__addon-options-3.2.2.tgz#f42f81414fa9692cf20d947e9b49c60c4bdfbc4d"
|
||||||
|
|
||||||
|
"@types/webpack-env@^1.13.6":
|
||||||
|
version "1.13.6"
|
||||||
|
resolved "https://registry.yarnpkg.com/@types/webpack-env/-/webpack-env-1.13.6.tgz#128d1685a7c34d31ed17010fc87d6a12c1de6976"
|
||||||
|
|
||||||
"@vue/component-compiler-utils@^2.0.0":
|
"@vue/component-compiler-utils@^2.0.0":
|
||||||
version "2.1.0"
|
version "2.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/@vue/component-compiler-utils/-/component-compiler-utils-2.1.0.tgz#8331eadc8acdbc35aace5a61d2153e6f6434bfe2"
|
resolved "https://registry.yarnpkg.com/@vue/component-compiler-utils/-/component-compiler-utils-2.1.0.tgz#8331eadc8acdbc35aace5a61d2153e6f6434bfe2"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user