storybook/lib/api/src/modules/stories.ts
Michael Shilman 16a1756084 Fix linting
2020-07-06 22:08:33 +08:00

420 lines
12 KiB
TypeScript

import { DOCS_MODE } from 'global';
import { toId, sanitize } from '@storybook/csf';
import {
UPDATE_STORY_ARGS,
STORY_ARGS_UPDATED,
STORY_CHANGED,
SELECT_STORY,
SET_STORIES,
CURRENT_STORY_WAS_SET,
} from '@storybook/core-events';
import deprecate from 'util-deprecate';
import { logger } from '@storybook/client-logger';
import {
denormalizeStoryParameters,
transformStoriesRawToStoriesHash,
StoriesHash,
Story,
Group,
StoryId,
isStory,
Root,
isRoot,
StoriesRaw,
SetStoriesPayload,
} from '../lib/stories';
import { Args, ModuleFn } from '../index';
import { getSourceType, ComposedRef } from './refs';
type Direction = -1 | 1;
type ParameterName = string;
type ViewMode = 'story' | 'info' | 'settings' | string | undefined;
export interface SubState {
storiesHash: StoriesHash;
storyId: StoryId;
viewMode: ViewMode;
storiesConfigured: boolean;
storiesFailed?: Error;
}
export interface SubAPI {
storyId: typeof toId;
resolveStory: (storyId: StoryId, refsId?: string) => Story | Group | Root;
selectStory: (
kindOrId: string,
story?: string,
obj?: { ref?: string; viewMode?: ViewMode }
) => void;
getCurrentStoryData: () => Story | Group;
setStories: (stories: StoriesRaw, failed?: Error) => Promise<void>;
jumpToComponent: (direction: Direction) => void;
jumpToStory: (direction: Direction) => void;
getData: (storyId: StoryId, refId?: string) => Story | Group;
getParameters: (
storyId: StoryId | { storyId: StoryId; refId: string },
parameterName?: ParameterName
) => Story['parameters'] | any;
getCurrentParameter<S>(parameterName?: ParameterName): S;
updateStoryArgs(id: StoryId, newArgs: Args): void;
findLeafStoryId(StoriesHash: StoriesHash, storyId: StoryId): StoryId;
}
interface Meta {
ref?: ComposedRef;
source?: string;
sourceType?: 'local' | 'external';
sourceLocation?: string;
refId?: string;
v?: number;
type: string;
}
const deprecatedOptionsParameterWarnings: Record<string, () => void> = [
'sidebarAnimations',
'enableShortcuts',
'theme',
'showRoots',
].reduce((acc, option: string) => {
acc[option] = deprecate(
() => {},
`parameters.options.${option} is deprecated and will be removed in Storybook 7.0.
To change this setting, use \`addons.setConfig\`. See https://github.com/storybookjs/storybook/MIGRATION.md#deprecated-immutable-options-parameters
`
);
return acc;
}, {} as Record<string, () => void>);
function checkDeprecatedOptionParameters(options?: Record<string, any>) {
if (!options) {
return;
}
Object.keys(options).forEach((option: string) => {
if (deprecatedOptionsParameterWarnings[option]) {
deprecatedOptionsParameterWarnings[option]();
}
});
}
export const init: ModuleFn = ({
fullAPI,
store,
navigate,
provider,
storyId: initialStoryId,
viewMode: initialViewMode,
}) => {
const api: SubAPI = {
storyId: toId,
getData: (storyId, refId) => {
const result = api.resolveStory(storyId, refId);
return isRoot(result) ? undefined : result;
},
resolveStory: (storyId, refId) => {
const { refs, storiesHash } = store.getState();
if (refId) {
return refs[refId].stories ? refs[refId].stories[storyId] : undefined;
}
return storiesHash ? storiesHash[storyId] : undefined;
},
getCurrentStoryData: () => {
const { storyId, refId } = store.getState();
return api.getData(storyId, refId);
},
getParameters: (storyIdOrCombo, parameterName) => {
const { storyId, refId } =
typeof storyIdOrCombo === 'string'
? { storyId: storyIdOrCombo, refId: undefined }
: storyIdOrCombo;
const data = api.getData(storyId, refId);
if (isStory(data)) {
const { parameters } = data;
return parameterName ? parameters[parameterName] : parameters;
}
return null;
},
getCurrentParameter: (parameterName) => {
const { storyId, refId } = store.getState();
const parameters = api.getParameters({ storyId, refId }, parameterName);
if (parameters) {
return parameters;
}
return undefined;
},
jumpToComponent: (direction) => {
const { storiesHash, storyId, refs, refId } = store.getState();
const story = api.getData(storyId, refId);
// cannot navigate when there's no current selection
if (!story) {
return;
}
const hash = refId ? refs[refId].stories || {} : storiesHash;
const lookupList = Object.entries(hash).reduce((acc, i) => {
const value = i[1];
if (value.isComponent) {
acc.push([...i[1].children]);
}
return acc;
}, []);
const index = lookupList.findIndex((i) => i.includes(storyId));
// cannot navigate beyond fist or last
if (index === lookupList.length - 1 && direction > 0) {
return;
}
if (index === 0 && direction < 0) {
return;
}
const result = lookupList[index + direction][0];
if (result) {
api.selectStory(result, undefined, { ref: refId });
}
},
jumpToStory: (direction) => {
const { storiesHash, storyId, refs, refId } = store.getState();
const story = api.getData(storyId, refId);
if (DOCS_MODE) {
api.jumpToComponent(direction);
return;
}
// cannot navigate when there's no current selection
if (!story) {
return;
}
const hash = story.refId ? refs[story.refId].stories : storiesHash;
const lookupList = Object.keys(hash).filter(
(k) => !(hash[k].children || Array.isArray(hash[k]))
);
const index = lookupList.indexOf(storyId);
// cannot navigate beyond fist or last
if (index === lookupList.length - 1 && direction > 0) {
return;
}
if (index === 0 && direction < 0) {
return;
}
const result = lookupList[index + direction];
if (result) {
api.selectStory(result, undefined, { ref: refId });
}
},
setStories: async (input, error) => {
// Now create storiesHash by reordering the above by group
const hash = transformStoriesRawToStoriesHash(input, {
provider,
});
await store.setState({
storiesHash: hash,
storiesConfigured: true,
storiesFailed: error,
});
},
selectStory: (kindOrId, story = undefined, options = {}) => {
const { ref, viewMode: viewModeFromArgs } = options;
const {
viewMode: viewModeFromState = 'story',
storyId,
storiesHash,
refs,
} = store.getState();
const hash = ref ? refs[ref].stories : storiesHash;
if (!story) {
const s = hash[kindOrId] || hash[sanitize(kindOrId)];
// eslint-disable-next-line no-nested-ternary
const id = s ? (s.children ? s.children[0] : s.id) : kindOrId;
let viewMode =
viewModeFromArgs || (s && s.parameters.viewMode)
? s.parameters.viewMode
: viewModeFromState;
// In some cases, the viewMode could be something other than docs/story
// ('settings', for example) and therefore we should make sure we go back
// to the 'story' viewMode when navigating away from those pages.
if (!viewMode.match(/docs|story/)) {
viewMode = 'story';
}
const p = s && s.refId ? `/${viewMode}/${s.refId}_${id}` : `/${viewMode}/${id}`;
navigate(p);
} else if (!kindOrId) {
// This is a slugified version of the kind, but that's OK, our toId function is idempotent
const kind = storyId.split('--', 2)[0];
const id = toId(kind, story);
api.selectStory(id, undefined, options);
} else {
const id = ref ? `${ref}_${toId(kindOrId, story)}` : toId(kindOrId, story);
if (hash[id]) {
api.selectStory(id, undefined, options);
} else {
// Support legacy API with component permalinks, where kind is `x/y` but permalink is 'z'
const k = hash[sanitize(kindOrId)];
if (k && k.children) {
const foundId = k.children.find((childId) => hash[childId].name === story);
if (foundId) {
api.selectStory(foundId, undefined, options);
}
}
}
}
},
findLeafStoryId(storiesHash, storyId) {
if (storiesHash[storyId].isLeaf) {
return storyId;
}
const childStoryId = storiesHash[storyId].children[0];
return api.findLeafStoryId(storiesHash, childStoryId);
},
updateStoryArgs: (id, newArgs) => {
fullAPI.emit(UPDATE_STORY_ARGS, id, newArgs);
},
};
const getEventMetadata = (context: Meta) => {
const { source, refId, type } = context;
const [sourceType, sourceLocation] = getSourceType(source, refId);
const ref =
refId && fullAPI.getRefs()[refId]
? fullAPI.getRefs()[refId]
: fullAPI.findRef(sourceLocation);
const meta = {
source,
sourceType,
sourceLocation,
refId,
ref,
type,
};
switch (true) {
case typeof refId === 'string':
case sourceType === 'local':
case sourceType === 'external': {
return meta;
}
// if we couldn't find the source, something risky happened, we ignore the input, and log a warning
default: {
logger.warn(`Received a ${type} frame that was not configured as a ref`);
return null;
}
}
};
const initModule = () => {
// On initial load, the local iframe will select the first story (or other "selection specifier")
// and emit CURRENT_STORY_WAS_SET with the id. We need to ensure we respond to this change.
// 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)
// Note this is the case for refs also.
fullAPI.on(CURRENT_STORY_WAS_SET, function handler({
storyId,
viewMode,
}: {
storyId: string;
viewMode: ViewMode;
[k: string]: any;
}) {
const { sourceType } = getEventMetadata(this);
if (sourceType === 'local' && storyId && viewMode) {
navigate(`/${viewMode}/${storyId}`);
}
});
fullAPI.on(STORY_CHANGED, function handler() {
const { sourceType } = getEventMetadata(this);
if (sourceType === 'local') {
const options = fullAPI.getCurrentParameter('options');
if (options) {
checkDeprecatedOptionParameters(options);
fullAPI.setOptions(options);
}
}
});
fullAPI.on(SET_STORIES, function handler(data: SetStoriesPayload) {
const { ref } = getEventMetadata(this);
const error = data.error || undefined;
const stories = data.v ? denormalizeStoryParameters(data) : data.stories;
if (!ref) {
if (!data.v) {
throw new Error('Unexpected legacy SET_STORIES event from local source');
}
fullAPI.setStories(stories, error);
const { options } = data.globalParameters;
checkDeprecatedOptionParameters(options);
fullAPI.setOptions(options);
} else {
fullAPI.setRef(ref.id, { ...ref, ...data, stories }, true);
}
});
fullAPI.on(SELECT_STORY, function handler({
kind,
story,
...rest
}: {
kind: string;
story: string;
viewMode: ViewMode;
}) {
const { ref } = getEventMetadata(this);
if (!ref) {
fullAPI.selectStory(kind, story, rest);
} else {
fullAPI.selectStory(kind, story, { ...rest, ref: ref.id });
}
});
fullAPI.on(STORY_ARGS_UPDATED, (id: StoryId, args: Args) => {
const { storiesHash } = store.getState();
(storiesHash[id] as Story).args = args;
store.setState({ storiesHash });
});
};
return {
api,
state: {
storiesHash: {},
storyId: initialStoryId,
viewMode: initialViewMode,
storiesConfigured: false,
},
init: initModule,
};
};