2022-09-27 15:41:01 +02:00

164 lines
3.8 KiB
TypeScript

import global from 'global';
import type { ReactElement } from 'react';
import { Channel } from '@storybook/channels';
import { SET_CONFIG } from '@storybook/core-events';
import type { API } from '@storybook/api';
import type { RenderData as RouterData } from '@storybook/router';
import { logger } from '@storybook/client-logger';
import type { ThemeVars } from '@storybook/theming';
import { mockChannel } from './storybook-channel-mock';
import { types, Types } from './types';
export { Channel };
export interface RenderOptions {
active?: boolean;
key?: string;
}
export interface Addon {
title: (() => string) | string;
type?: Types;
id?: string;
route?: (routeOptions: RouterData) => string;
match?: (matchOptions: RouterData) => boolean;
render: (renderOptions: RenderOptions) => ReactElement<any> | null;
paramKey?: string;
disabled?: boolean;
hidden?: boolean;
}
export type Loader = (api: API) => void;
interface Loaders {
[key: string]: Loader;
}
export interface Collection {
[key: string]: Addon;
}
interface Elements {
[key: string]: Collection;
}
interface ToolbarConfig {
hidden?: boolean;
}
export interface Config {
theme?: ThemeVars;
toolbar?: {
[id: string]: ToolbarConfig;
};
[key: string]: any;
}
export class AddonStore {
constructor() {
this.promise = new Promise((res) => {
this.resolve = () => res(this.getChannel());
}) as Promise<Channel>;
}
private loaders: Loaders = {};
private elements: Elements = {};
private config: Config = {};
private channel: Channel | undefined;
private serverChannel: Channel | undefined;
private promise: any;
private resolve: any;
getChannel = (): Channel => {
// this.channel should get overwritten by setChannel. If it wasn't called (e.g. in non-browser environment), set a mock instead.
if (!this.channel) {
this.setChannel(mockChannel());
}
return this.channel;
};
getServerChannel = (): Channel => {
if (!this.serverChannel) {
throw new Error('Accessing non-existent serverChannel');
}
return this.serverChannel;
};
ready = (): Promise<Channel> => this.promise;
hasChannel = (): boolean => !!this.channel;
hasServerChannel = (): boolean => !!this.serverChannel;
setChannel = (channel: Channel): void => {
this.channel = channel;
this.resolve();
};
setServerChannel = (channel: Channel): void => {
this.serverChannel = channel;
};
getElements = (type: Types): Collection => {
if (!this.elements[type]) {
this.elements[type] = {};
}
return this.elements[type];
};
addPanel = (name: string, options: Addon): void => {
this.add(name, {
type: types.PANEL,
...options,
});
};
add = (name: string, addon: Addon) => {
const { type } = addon;
const collection = this.getElements(type);
collection[name] = { id: name, ...addon };
};
setConfig = (value: Config) => {
Object.assign(this.config, value);
if (this.hasChannel()) {
this.getChannel().emit(SET_CONFIG, value);
}
};
getConfig = () => this.config;
register = (name: string, registerCallback: (api: API) => void): void => {
if (this.loaders[name]) {
logger.warn(`${name} was loaded twice, this could have bad side-effects`);
}
this.loaders[name] = registerCallback;
};
loadAddons = (api: any) => {
Object.values(this.loaders).forEach((value) => value(api));
};
}
// Enforce addons store to be a singleton
const KEY = '__STORYBOOK_ADDONS';
function getAddonsStore(): AddonStore {
if (!global[KEY]) {
global[KEY] = new AddonStore();
}
return global[KEY];
}
// Exporting this twice in order to to be able to import it like { addons } instead of 'addons'
// prefer import { addons } from '@storybook/addons' over import addons from '@storybook/addons'
//
// See public_api.ts
export const addons = getAddonsStore();