DE-DUPLICATE the code for switching between handling event from refs vs local iframe

This commit is contained in:
Norbert de Langen 2020-07-01 19:21:15 +02:00
parent 95b543ce51
commit bdcc586c28
No known key found for this signature in database
GPG Key ID: 976651DA156C2825

View File

@ -27,7 +27,7 @@ import {
} from '../lib/stories'; } from '../lib/stories';
import { Args, ModuleFn } from '../index'; import { Args, ModuleFn } from '../index';
import { getSourceType } from './refs'; import { getSourceType, ComposedRef } from './refs';
type Direction = -1 | 1; type Direction = -1 | 1;
type ParameterName = string; type ParameterName = string;
@ -72,6 +72,61 @@ export const init: ModuleFn = ({
storyId: initialStoryId, storyId: initialStoryId,
viewMode: initialViewMode, viewMode: initialViewMode,
}) => { }) => {
interface Meta {
ref?: ComposedRef;
source?: string;
sourceType?: 'local' | 'external';
refId?: string;
v?: number;
type: string;
}
function createEventHandlerWithRefSupport<T = { v?: number }>(
callback: (data: T, meta: Meta) => void
) {
function handle(data: T) {
const { source, refId, type }: Meta = this;
const [sourceType, sourceLocation] = getSourceType(source);
// @ts-ignore
const v = data.v || this.v;
if (v > 2) {
// eslint-disable-next-line no-console
console.warn(`Received ${this.type} event with version ${v}, we'll try and handle it`);
}
const ref =
refId && fullAPI.getRefs()[refId]
? fullAPI.getRefs()[refId]
: fullAPI.findRef(sourceLocation);
const meta = {
v,
source,
sourceLocation,
refId,
ref,
type,
};
switch (true) {
case typeof refId === 'string':
case sourceType === 'local':
case sourceType === 'external': {
callback(data, meta);
break;
}
// if we couldn't find the source, something risky happened, we ignore the input, and log a warning
default: {
logger.warn(`Received a ${this.type} frame that was not configured as a ref`);
break;
}
}
}
return handle;
}
const api: SubAPI = { const api: SubAPI = {
storyId: toId, storyId: toId,
getData: (storyId, refId) => { getData: (storyId, refId) => {
@ -268,105 +323,63 @@ export const init: ModuleFn = ({
// Later when we change story via the manager (or SELECT_STORY below), we'll already be at the // Later when we change story via the manager (or SELECT_STORY below), we'll already be at the
// correct path before CURRENT_STORY_WAS_SET is emitted, so this is less important (the navigate is a no-op) // correct path before CURRENT_STORY_WAS_SET is emitted, so this is less important (the navigate is a no-op)
// Note this is the case for refs also. // Note this is the case for refs also.
fullAPI.on(CURRENT_STORY_WAS_SET, function handleCurrentStoryWasSet({ storyId, viewMode }) { fullAPI.on(
const { source }: { source: string } = this; CURRENT_STORY_WAS_SET,
const [sourceType] = getSourceType(source); createEventHandlerWithRefSupport<{ storyId: string; viewMode: ViewMode; [k: string]: any }>(
({ storyId, viewMode }, { sourceType }) => {
if (sourceType === 'local' && storyId && viewMode) { if (sourceType === 'local' && storyId && viewMode) {
navigate(`/${viewMode}/${storyId}`); navigate(`/${viewMode}/${storyId}`);
} }
});
fullAPI.on(STORY_CHANGED, function handleStoryChange(storyId: string) {
const { source }: { source: string } = this;
const [sourceType] = getSourceType(source);
if (sourceType === 'local') {
const options = fullAPI.getCurrentParameter('options');
if (options) {
fullAPI.setOptions(options);
} }
} )
}); );
fullAPI.on(SET_STORIES, function handleSetStories(data: SetStoriesPayload) { fullAPI.on(
// the event originates from an iframe, event.source is the iframe's location origin + pathname STORY_CHANGED,
const { source }: { source: string } = this; createEventHandlerWithRefSupport((storyId, { sourceType }) => {
const [sourceType, sourceLocation] = getSourceType(source); if (sourceType === 'local') {
const options = fullAPI.getCurrentParameter('options');
// TODO: what is the mechanism where we warn here? if (options) {
if (data.v && data.v > 2) { fullAPI.setOptions(options);
// eslint-disable-next-line no-console }
console.warn(`Received SET_STORIES event with version ${data.v}, we'll try and handle it`); }
} })
);
const stories = data.v fullAPI.on(
? denormalizeStoryParameters(data as SetStoriesPayloadV2) SET_STORIES,
: data.stories; createEventHandlerWithRefSupport<SetStoriesPayloadV2>((data, { ref }) => {
const error = data.error || undefined;
const stories = data.v ? denormalizeStoryParameters(data) : data.stories;
// @ts-ignore if (!ref) {
const error = data.error || undefined;
switch (sourceType) {
// if it's a local source, we do nothing special
case 'local': {
if (!data.v) { if (!data.v) {
throw new Error('Unexpected legacy SET_STORIES event from local source'); throw new Error('Unexpected legacy SET_STORIES event from local source');
} }
fullAPI.setStories(stories, error); fullAPI.setStories(stories, error);
fullAPI.setOptions(data.globalParameters.options);
fullAPI.setOptions((data as SetStoriesPayloadV2).globalParameters.options); } else {
break; fullAPI.setRef(ref.id, { ...ref, ...data, stories }, true);
} }
})
);
// if it's a ref, we need to map the incoming stories to a prefixed version, so it cannot conflict with others fullAPI.on(
case 'external': { SELECT_STORY,
const ref = fullAPI.findRef(sourceLocation); createEventHandlerWithRefSupport<{
if (ref) { kind: string;
fullAPI.setRef(ref.id, { ...ref, ...data, stories }, true); story: string;
break; viewMode: ViewMode;
} }>(({ kind, story, ...rest }, { ref }) => {
} if (!ref) {
// if we couldn't find the source, something risky happened, we ignore the input, and log a warning
default: {
logger.warn('received a SET_STORIES frame that was not configured as a ref');
break;
}
}
});
fullAPI.on(SELECT_STORY, function selectStoryHandler({
kind,
story,
...rest
}: {
kind: string;
story: string;
[k: string]: any;
}) {
const { source }: { source: string } = this;
const [sourceType, sourceLocation] = getSourceType(source);
switch (sourceType) {
case 'local': {
fullAPI.selectStory(kind, story, rest); fullAPI.selectStory(kind, story, rest);
break; } else {
}
case 'external': {
const ref = fullAPI.findRef(sourceLocation);
fullAPI.selectStory(kind, story, { ...rest, ref: ref.id }); fullAPI.selectStory(kind, story, { ...rest, ref: ref.id });
break;
} }
default: { })
logger.warn('received a SET_STORIES frame that was not configured as a ref'); );
break;
}
}
});
fullAPI.on(STORY_ARGS_UPDATED, (id: StoryId, args: Args) => { fullAPI.on(STORY_ARGS_UPDATED, (id: StoryId, args: Args) => {
const { storiesHash } = store.getState(); const { storiesHash } = store.getState();