Merge pull request #14875 from robhil/single-story-mode

Core: Single story option in iframe view
This commit is contained in:
Michael Shilman 2021-05-19 08:51:13 +08:00 committed by GitHub
commit b88ab7c139
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 141 additions and 3 deletions

View File

@ -1613,4 +1613,78 @@ describe('preview.story_store', () => {
expect(onStorySpecified).not.toHaveBeenCalled();
});
});
describe('In Single Story mode', () => {
describe('when storySpecifier is story id', () => {
it('adds only one story specified in selection specifier when addStory is called', () => {
const store = new StoryStore({ channel });
store.setSelectionSpecifier({
storySpecifier: toId('kind-1', 'story-1.1'),
viewMode: 'story',
singleStory: true,
});
store.startConfiguring();
addStoryToStore(store, 'kind-1', 'story-1.1', () => 0);
addStoryToStore(store, 'kind-1', 'story-1.2', () => 0);
store.finishConfiguring();
expect(store.fromId(toId('kind-1', 'story-1.1'))).toBeTruthy();
expect(store.fromId(toId('kind-1', 'story-1.2'))).toBeFalsy();
});
it('adds only kind metadata specified in selection specifier when addKindMetadata is called', () => {
const store = new StoryStore({ channel });
store.setSelectionSpecifier({
storySpecifier: toId('kind-1', 'story-1.1'),
viewMode: 'story',
singleStory: true,
});
store.startConfiguring();
store.addKindMetadata('kind-1', {});
store.addKindMetadata('kind-2', {});
store.finishConfiguring();
expect(store._kinds['kind-1']).toBeDefined();
expect(store._kinds['kind-2']).not.toBeDefined();
});
});
describe('when storySpecifier is object', () => {
it('adds only one story specified in selection specifier when addStory is called', () => {
const store = new StoryStore({ channel });
store.setSelectionSpecifier({
storySpecifier: { kind: 'kind-1', name: 'story-1.1' },
viewMode: 'story',
singleStory: true,
});
store.startConfiguring();
addStoryToStore(store, 'kind-1', 'story-1.1', () => 0);
addStoryToStore(store, 'kind-1', 'story-1.2', () => 0);
store.finishConfiguring();
expect(store.fromId(toId('kind-1', 'story-1.1'))).toBeTruthy();
expect(store.fromId(toId('kind-1', 'story-1.2'))).toBeFalsy();
});
it('adds only kind metadata specified in selection specifier when addKindMetadata is called', () => {
const store = new StoryStore({ channel });
store.setSelectionSpecifier({
storySpecifier: { kind: 'kind-1', name: 'story-1.1' },
viewMode: 'story',
singleStory: true,
});
store.startConfiguring();
store.addKindMetadata('kind-1', {});
store.addKindMetadata('kind-2', {});
store.finishConfiguring();
expect(store._kinds['kind-1']).toBeDefined();
expect(store._kinds['kind-2']).not.toBeDefined();
});
});
});
});

View File

@ -10,6 +10,7 @@ import deprecate from 'util-deprecate';
import { Channel } from '@storybook/channels';
import Events from '@storybook/core-events';
import { logger } from '@storybook/client-logger';
import { sanitize, toId } from '@storybook/csf';
import {
Comparator,
Parameters,
@ -33,6 +34,7 @@ import {
ArgTypesEnhancer,
StoreSelectionSpecifier,
StoreSelection,
StorySpecifier,
} from './types';
import { combineArgs, mapArgsToTypes, validateOptions } from './args';
import { HooksContext } from './hooks';
@ -50,6 +52,22 @@ type KindMetadata = StoryMetadata & { order: number };
const STORAGE_KEY = '@storybook/preview/store';
function extractSanitizedKindNameFromStorySpecifier(storySpecifier: StorySpecifier): string {
if (typeof storySpecifier === 'string') {
return storySpecifier.split('--').shift();
}
return sanitize(storySpecifier.kind);
}
function extractIdFromStorySpecifier(storySpecifier: StorySpecifier): string {
if (typeof storySpecifier === 'string') {
return storySpecifier;
}
return toId(storySpecifier.kind, storySpecifier.name);
}
const isStoryDocsOnly = (parameters?: Parameters) => {
return parameters && parameters.docsOnly;
};
@ -313,6 +331,10 @@ export default class StoryStore {
}
addKindMetadata(kind: string, { parameters = {}, decorators = [], loaders = [] }: StoryMetadata) {
if (this.shouldBlockAddingKindMetadata(kind)) {
return;
}
this.ensureKind(kind);
if (parameters) {
checkGlobals(parameters);
@ -347,6 +369,21 @@ export default class StoryStore {
);
}
shouldBlockAddingStory(id: string): boolean {
return (
this.isSingleStoryMode() &&
id !== extractIdFromStorySpecifier(this._selectionSpecifier.storySpecifier)
);
}
shouldBlockAddingKindMetadata(kind: string): boolean {
return (
this.isSingleStoryMode() &&
sanitize(kind) !==
extractSanitizedKindNameFromStorySpecifier(this._selectionSpecifier.storySpecifier)
);
}
addStory(
{
id,
@ -369,6 +406,10 @@ export default class StoryStore {
'Cannot add a story when not configuring, see https://github.com/storybookjs/storybook/blob/next/MIGRATION.md#story-store-immutable-outside-of-configuration'
);
if (this.shouldBlockAddingStory(id)) {
return;
}
checkGlobals(storyParameters);
checkStorySort(storyParameters);
@ -694,6 +735,15 @@ export default class StoryStore {
}
}
isSingleStoryMode(): boolean {
if (!this._selectionSpecifier) {
return false;
}
const { singleStory, storySpecifier } = this._selectionSpecifier;
return storySpecifier && storySpecifier !== '*' && singleStory;
}
getSelection = (): StoreSelection => this._selection;
getDataForManager = () => {

View File

@ -32,11 +32,12 @@ export interface StoryMetadata {
export type ArgTypesEnhancer = (context: StoryContext) => ArgTypes;
export type ArgsEnhancer = (context: StoryContext) => Args;
type StorySpecifier = StoryId | { name: StoryName; kind: StoryKind } | '*';
export type StorySpecifier = StoryId | { name: StoryName; kind: StoryKind } | '*';
export interface StoreSelectionSpecifier {
storySpecifier: StorySpecifier;
viewMode: ViewMode;
singleStory?: boolean;
args?: Args;
}

View File

@ -73,6 +73,7 @@ describe('url', () => {
expect(getSelectionSpecifierFromPath()).toEqual({
storySpecifier: 'story--id',
viewMode: 'story',
singleStory: false,
});
});
it('should handle id queries with *', () => {
@ -80,6 +81,7 @@ describe('url', () => {
expect(getSelectionSpecifierFromPath()).toEqual({
storySpecifier: '*',
viewMode: 'story',
singleStory: false,
});
});
it('should redirect legacy queries', () => {
@ -87,6 +89,7 @@ describe('url', () => {
expect(getSelectionSpecifierFromPath()).toEqual({
storySpecifier: { kind: 'kind', name: 'story' },
viewMode: 'story',
singleStory: false,
});
});
it('should parse args', () => {
@ -94,8 +97,17 @@ describe('url', () => {
expect(getSelectionSpecifierFromPath()).toEqual({
storySpecifier: 'story--id',
viewMode: 'story',
singleStory: false,
args: { obj: { key: 'val' } },
});
});
it('should handle singleStory param', () => {
document.location.search = '?id=abc&singleStory=true';
expect(getSelectionSpecifierFromPath()).toEqual({
storySpecifier: 'abc',
viewMode: 'story',
singleStory: true,
});
});
});
});

View File

@ -72,11 +72,12 @@ export const getSelectionSpecifierFromPath: () => StoreSelectionSpecifier = () =
viewMode = 'story';
}
const singleStory = getFirstString(query.singleStory) === 'true';
const path = getFirstString(query.path);
const storyId = path ? pathToId(path) : getFirstString(query.id);
if (storyId) {
return { storySpecifier: storyId, args, viewMode };
return { storySpecifier: storyId, args, viewMode, singleStory };
}
// Legacy URL format
@ -85,7 +86,7 @@ export const getSelectionSpecifierFromPath: () => StoreSelectionSpecifier = () =
if (kind && name) {
deprecatedLegacyQuery();
return { storySpecifier: { kind, name }, args, viewMode };
return { storySpecifier: { kind, name }, args, viewMode, singleStory };
}
return null;
};