mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 16:41:08 +08:00
164 lines
3.8 KiB
TypeScript
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();
|