storybook/lib/client-api/src/client_api.ts

207 lines
6.0 KiB
TypeScript

/* eslint no-underscore-dangle: 0 */
import { logger } from '@storybook/client-logger';
import { StoryFn, Parameters } from '@storybook/addons';
import { toId } from '@storybook/csf';
import { ClientApiParams, DecoratorFunction, ClientApiAddons, StoryApi } from './types';
import { applyHooks } from './hooks';
import StoryStore from './story_store';
import { defaultDecorateStory } from './decorators';
// ClientApi (and StoreStore) are really singletons. However they are not created until the
// relevant framework instanciates them via `start.js`. The good news is this happens right away.
let singleton: ClientApi;
export const addDecorator = (decorator: DecoratorFunction) => {
if (!singleton)
throw new Error(`Singleton client API not yet initialized, cannot call addDecorator`);
singleton.addDecorator(decorator);
};
export const addParameters = (parameters: Parameters) => {
if (!singleton)
throw new Error(`Singleton client API not yet initialized, cannot call addParameters`);
singleton.addParameters(parameters);
};
export default class ClientApi {
private _storyStore: StoryStore;
private _addons: ClientApiAddons<unknown>;
private _decorateStory: (storyFn: StoryFn, decorators: DecoratorFunction[]) => any;
constructor({ storyStore, decorateStory = defaultDecorateStory }: ClientApiParams) {
this._storyStore = storyStore;
this._addons = {};
this._decorateStory = decorateStory;
if (!storyStore) throw new Error('storyStore is required');
singleton = this;
}
setAddon = (addon: any) => {
this._addons = {
...this._addons,
...addon,
};
};
getSeparators = () => {
const { hierarchySeparator, hierarchyRootSeparator, showRoots } =
this._storyStore._globalMetadata.parameters.options || {};
// Note these checks will be removed in 6.0, leaving this much simpler
if (
typeof hierarchySeparator !== 'undefined' ||
typeof hierarchyRootSeparator !== 'undefined'
) {
return { hierarchySeparator, hierarchyRootSeparator };
}
if (
typeof showRoots === 'undefined' &&
this.store()
.getStoryKinds()
.some(kind => kind.match(/\.|\|/))
) {
return {
hierarchyRootSeparator: '|',
hierarchySeparator: /\/|\./,
};
}
return { hierarchySeparator: '/' };
};
addDecorator = (decorator: DecoratorFunction) => {
this._storyStore.addGlobalMetadata({ decorators: [decorator], parameters: {} });
};
addParameters = (parameters: Parameters) => {
this._storyStore.addGlobalMetadata({ decorators: [], parameters });
};
clearDecorators = () => {
this._storyStore.clearGlobalDecorators();
};
// what are the occasions that "m" is a boolean vs an obj
storiesOf = <StoryFnReturnType = unknown>(
kind: string,
m: NodeModule
): StoryApi<StoryFnReturnType> => {
if (!kind && typeof kind !== 'string') {
throw new Error('Invalid or missing kind provided for stories, should be a string');
}
if (!m) {
logger.warn(
`Missing 'module' parameter for story with a kind of '${kind}'. It will break your HMR`
);
}
if (m) {
const proto = Object.getPrototypeOf(m);
if (proto.exports && proto.exports.default) {
// FIXME: throw an error in SB6.0
logger.error(
`Illegal mix of CSF default export and storiesOf calls in a single file: ${proto.i}`
);
}
}
if (m && m.hot && m.hot.dispose) {
m.hot.dispose(() => {
const { _storyStore } = this;
_storyStore.removeStoryKind(kind);
_storyStore.incrementRevision();
});
}
let hasAdded = false;
const api: StoryApi<StoryFnReturnType> = {
kind: kind.toString(),
add: () => api,
addDecorator: () => api,
addParameters: () => api,
};
// apply addons
Object.keys(this._addons).forEach(name => {
const addon = this._addons[name];
api[name] = (...args: any[]) => {
addon.apply(api, args);
return api;
};
});
api.add = (
storyName: string,
storyFn: StoryFn<StoryFnReturnType>,
parameters: Parameters = {}
) => {
hasAdded = true;
const id = parameters.__id || toId(kind, storyName);
if (typeof storyName !== 'string') {
throw new Error(`Invalid or missing storyName provided for a "${kind}" story.`);
}
if (m && m.hot && m.hot.dispose) {
m.hot.dispose(() => {
const { _storyStore } = this;
_storyStore.remove(id);
});
}
const fileName = m && m.id ? `${m.id}` : undefined;
const { decorators, ...storyParameters } = parameters;
this._storyStore.addStory(
{
id,
kind,
name: storyName,
storyFn,
parameters: { fileName, ...storyParameters },
decorators,
},
{
applyDecorators: applyHooks(this._decorateStory),
}
);
return api;
};
api.addDecorator = (decorator: DecoratorFunction<StoryFnReturnType>) => {
if (hasAdded)
throw new Error(`You cannot add a decorator after the first story for a kind.
Read more here: https://github.com/storybookjs/storybook/blob/master/MIGRATION.md#can-no-longer-add-decorators-parameters-after-stories`);
this._storyStore.addKindMetadata(kind, { decorators: [decorator], parameters: [] });
return api;
};
api.addParameters = (parameters: Parameters) => {
if (hasAdded)
throw new Error(`You cannot add parameters after the first story for a kind.
Read more here: https://github.com/storybookjs/storybook/blob/master/MIGRATION.md#can-no-longer-add-decorators-parameters-after-stories`);
this._storyStore.addKindMetadata(kind, { decorators: [], parameters });
return api;
};
return api;
};
getStorybook = () => this._storyStore.getStorybook();
raw = () => this._storyStore.raw();
// FIXME: temporary expose the store for react-native
// Longer term react-native should use the Provider/Consumer api
store = () => this._storyStore;
}