mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 23:21:15 +08:00
69 lines
2.2 KiB
TypeScript
69 lines
2.2 KiB
TypeScript
import uuidv4 from 'uuid-browser/v4';
|
|
import { addons } from '@storybook/addons';
|
|
import { EVENT_ID } from '../constants';
|
|
import { ActionDisplay, ActionOptions, HandlerFunction } from '../models';
|
|
import { config } from './configureActions';
|
|
|
|
type SyntheticEvent = any; // import('react').SyntheticEvent;
|
|
const findProto = (obj: unknown, callback: (proto: any) => boolean): Function | null => {
|
|
const proto = Object.getPrototypeOf(obj);
|
|
if (!proto || callback(proto)) return proto;
|
|
return findProto(proto, callback);
|
|
};
|
|
const isReactSyntheticEvent = (e: unknown): e is SyntheticEvent =>
|
|
Boolean(
|
|
typeof e === 'object' &&
|
|
e &&
|
|
findProto(e, (proto) => /^Synthetic(?:Base)?Event$/.test(proto.constructor.name)) &&
|
|
typeof (e as SyntheticEvent).persist === 'function'
|
|
);
|
|
const serializeArg = <T>(a: T) => {
|
|
if (isReactSyntheticEvent(a)) {
|
|
const e: SyntheticEvent = Object.create(
|
|
a.constructor.prototype,
|
|
Object.getOwnPropertyDescriptors(a)
|
|
);
|
|
e.persist();
|
|
const viewDescriptor = Object.getOwnPropertyDescriptor(e, 'view');
|
|
// don't send the entire window object over.
|
|
const view: unknown = viewDescriptor?.value;
|
|
if (typeof view === 'object' && view?.constructor.name === 'Window') {
|
|
Object.defineProperty(e, 'view', {
|
|
...viewDescriptor,
|
|
value: Object.create(view.constructor.prototype),
|
|
});
|
|
}
|
|
return e;
|
|
}
|
|
return a;
|
|
};
|
|
|
|
export function action(name: string, options: ActionOptions = {}): HandlerFunction {
|
|
const actionOptions = {
|
|
...config,
|
|
...options,
|
|
};
|
|
|
|
const handler = function actionHandler(...args: any[]) {
|
|
const channel = addons.getChannel();
|
|
const id = uuidv4();
|
|
const minDepth = 5; // anything less is really just storybook internals
|
|
const serializedArgs = args.map(serializeArg);
|
|
const normalizedArgs = args.length > 1 ? serializedArgs : serializedArgs[0];
|
|
|
|
const actionDisplayToEmit: ActionDisplay = {
|
|
id,
|
|
count: 0,
|
|
data: { name, args: normalizedArgs },
|
|
options: {
|
|
...actionOptions,
|
|
maxDepth: minDepth + (actionOptions.depth || 3),
|
|
allowFunction: actionOptions.allowFunction || false,
|
|
},
|
|
};
|
|
channel.emit(EVENT_ID, actionDisplayToEmit);
|
|
};
|
|
|
|
return handler;
|
|
}
|