mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 06:41:17 +08:00
added ability to skip dispose call in clientApi
This commit is contained in:
parent
71aa9065dc
commit
23da639ffc
@ -5,11 +5,11 @@ import ClientApi from './client_api';
|
||||
import ConfigApi from './config_api';
|
||||
import StoryStore from './story_store';
|
||||
|
||||
const getContext = (() => decorateStory => {
|
||||
const getContext = ({ decorateStory = undefined, disableAddStoryDispose = false } = {}) => {
|
||||
const channel = mockChannel();
|
||||
addons.setChannel(channel);
|
||||
const storyStore = new StoryStore({ channel });
|
||||
const clientApi = new ClientApi({ storyStore, decorateStory });
|
||||
const clientApi = new ClientApi({ storyStore, decorateStory, disableAddStoryDispose });
|
||||
const { clearDecorators } = clientApi;
|
||||
const configApi = new ConfigApi({ clearDecorators, storyStore, channel, clientApi });
|
||||
|
||||
@ -19,7 +19,7 @@ const getContext = (() => decorateStory => {
|
||||
channel,
|
||||
clientApi,
|
||||
};
|
||||
})();
|
||||
};
|
||||
|
||||
jest.mock('@storybook/client-logger', () => ({
|
||||
logger: { warn: jest.fn(), log: jest.fn() },
|
||||
@ -27,13 +27,13 @@ jest.mock('@storybook/client-logger', () => ({
|
||||
|
||||
describe('preview.client_api', () => {
|
||||
afterEach(() => {
|
||||
const { clientApi } = getContext(undefined);
|
||||
const { clientApi } = getContext();
|
||||
clientApi.clearDecorators();
|
||||
clientApi.clearParameters();
|
||||
});
|
||||
describe('setAddon', () => {
|
||||
it('should register addons', () => {
|
||||
const { clientApi } = getContext(undefined);
|
||||
const { clientApi } = getContext();
|
||||
let data;
|
||||
|
||||
clientApi.setAddon({
|
||||
@ -47,7 +47,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should not remove previous addons', () => {
|
||||
const { clientApi } = getContext(undefined);
|
||||
const { clientApi } = getContext();
|
||||
const data = [];
|
||||
|
||||
clientApi.setAddon({
|
||||
@ -70,7 +70,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should call with the clientApi context', () => {
|
||||
const { clientApi } = getContext(undefined);
|
||||
const { clientApi } = getContext();
|
||||
let data;
|
||||
|
||||
clientApi.setAddon({
|
||||
@ -84,7 +84,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should be able to access addons added previously', () => {
|
||||
const { clientApi } = getContext(undefined);
|
||||
const { clientApi } = getContext();
|
||||
let data;
|
||||
|
||||
clientApi.setAddon({
|
||||
@ -104,7 +104,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should be able to access the current kind', () => {
|
||||
const { clientApi } = getContext(undefined);
|
||||
const { clientApi } = getContext();
|
||||
const kind = 'dfdwf3e3';
|
||||
let data;
|
||||
|
||||
@ -121,7 +121,7 @@ describe('preview.client_api', () => {
|
||||
|
||||
describe('addParameters', () => {
|
||||
it('should add parameters', () => {
|
||||
const { clientApi, storyStore } = getContext(undefined);
|
||||
const { clientApi, storyStore } = getContext();
|
||||
const { storiesOf } = clientApi;
|
||||
|
||||
clientApi.addParameters({ a: 1 });
|
||||
@ -135,7 +135,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should merge options', () => {
|
||||
const { clientApi, storyStore } = getContext(undefined);
|
||||
const { clientApi, storyStore } = getContext();
|
||||
const { storiesOf } = clientApi;
|
||||
|
||||
clientApi.addParameters({ options: { a: '1' } });
|
||||
@ -151,7 +151,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should override specific properties in options', () => {
|
||||
const { clientApi, storyStore } = getContext(undefined);
|
||||
const { clientApi, storyStore } = getContext();
|
||||
const { storiesOf } = clientApi;
|
||||
|
||||
clientApi.addParameters({ backgrounds: ['value'], options: { a: '1', b: '3' } });
|
||||
@ -169,7 +169,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should replace top level properties and override specific properties in options', () => {
|
||||
const { clientApi, storyStore } = getContext(undefined);
|
||||
const { clientApi, storyStore } = getContext();
|
||||
const { storiesOf } = clientApi;
|
||||
|
||||
clientApi.addParameters({ backgrounds: ['value'], options: { a: '1', b: '3' } });
|
||||
@ -187,7 +187,7 @@ describe('preview.client_api', () => {
|
||||
});
|
||||
|
||||
it('should deep merge in options', () => {
|
||||
const { clientApi, storyStore } = getContext(undefined);
|
||||
const { clientApi, storyStore } = getContext();
|
||||
const { storiesOf } = clientApi;
|
||||
|
||||
clientApi.addParameters({ options: { a: '1', b: '2', theming: { c: '3' } } });
|
||||
@ -208,7 +208,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
clientApi: { storiesOf },
|
||||
storyStore,
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
storiesOf('kind', module)
|
||||
.addDecorator(fn => `aa-${fn()}`)
|
||||
@ -221,7 +221,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
clientApi: { addDecorator, storiesOf },
|
||||
storyStore,
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
addDecorator(fn => `bb-${fn()}`);
|
||||
|
||||
@ -235,7 +235,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
clientApi: { addDecorator, storiesOf },
|
||||
storyStore,
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
addDecorator(fn => `aa-${fn()}`);
|
||||
|
||||
@ -250,7 +250,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
clientApi: { storiesOf },
|
||||
storyStore,
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
storiesOf('kind', module)
|
||||
.addDecorator(fn => `aa-${fn()}`)
|
||||
@ -264,7 +264,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
clientApi: { storiesOf },
|
||||
storyStore,
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
storiesOf('kind', module)
|
||||
.addDecorator((fn, { kind, name }) => `${kind}-${name}-${fn()}`)
|
||||
@ -277,7 +277,7 @@ describe('preview.client_api', () => {
|
||||
|
||||
describe('clearDecorators', () => {
|
||||
it('should remove all global decorators', () => {
|
||||
const { clientApi, storyStore } = getContext(undefined);
|
||||
const { clientApi, storyStore } = getContext();
|
||||
const { storiesOf } = clientApi;
|
||||
|
||||
clientApi.addDecorator(() => 'foo');
|
||||
@ -294,7 +294,7 @@ describe('preview.client_api', () => {
|
||||
it('should transform the storybook to an array with filenames', () => {
|
||||
const {
|
||||
clientApi: { getStorybook, storiesOf },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
let book;
|
||||
|
||||
@ -347,7 +347,7 @@ describe('preview.client_api', () => {
|
||||
it('returns values set via parameters', () => {
|
||||
const {
|
||||
clientApi: { getSeparators, storiesOf, addParameters },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
const options = { hierarchySeparator: /a/, hierarchyRootSeparator: 'b' };
|
||||
addParameters({ options });
|
||||
@ -358,7 +358,7 @@ describe('preview.client_api', () => {
|
||||
it('returns old defaults if kind uses old separators', () => {
|
||||
const {
|
||||
clientApi: { getSeparators, storiesOf },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
storiesOf('kind|1', module).add('name 1', () => '1');
|
||||
expect(getSeparators()).toEqual({
|
||||
@ -370,7 +370,7 @@ describe('preview.client_api', () => {
|
||||
it('returns new values if showRoots is set', () => {
|
||||
const {
|
||||
clientApi: { getSeparators, storiesOf, addParameters },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
addParameters({ options: { showRoots: false } });
|
||||
|
||||
storiesOf('kind|1', module).add('name 1', () => '1');
|
||||
@ -380,7 +380,7 @@ describe('preview.client_api', () => {
|
||||
it('returns new values if kind does not use old separators', () => {
|
||||
const {
|
||||
clientApi: { getSeparators, storiesOf },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
storiesOf('kind/1', module).add('name 1', () => '1');
|
||||
expect(getSeparators()).toEqual({ hierarchySeparator: '/' });
|
||||
@ -390,7 +390,7 @@ describe('preview.client_api', () => {
|
||||
it('reads filename from module', () => {
|
||||
const {
|
||||
clientApi: { getStorybook, storiesOf },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
const fn = jest.fn();
|
||||
storiesOf('kind', { id: 'foo.js' }).add('name', fn);
|
||||
@ -414,7 +414,7 @@ describe('preview.client_api', () => {
|
||||
it('should stringify ids from module', () => {
|
||||
const {
|
||||
clientApi: { getStorybook, storiesOf },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
const fn = jest.fn();
|
||||
storiesOf('kind', { id: 1211 }).add('name', fn);
|
||||
@ -455,7 +455,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
storyStore,
|
||||
clientApi: { storiesOf },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
const module = new MockModule();
|
||||
|
||||
expect(storyStore.getRevision()).toEqual(0);
|
||||
@ -470,7 +470,7 @@ describe('preview.client_api', () => {
|
||||
it('should replace a kind when the module reloads', () => {
|
||||
const {
|
||||
clientApi: { storiesOf, getStorybook },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
const module = new MockModule();
|
||||
|
||||
const stories = [jest.fn(), jest.fn()];
|
||||
@ -513,7 +513,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
clientApi: { storiesOf, getStorybook },
|
||||
channel,
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
const module0 = new MockModule();
|
||||
const module1 = new MockModule();
|
||||
const module2 = new MockModule();
|
||||
@ -548,6 +548,32 @@ describe('preview.client_api', () => {
|
||||
expect(Object.values(args.stories).map(v => v.kind)).toEqual(['kind0', 'kind1', 'kind2']);
|
||||
expect(getStorybook().map(story => story.kind)).toEqual(['kind1', 'kind2']);
|
||||
});
|
||||
|
||||
it('should bind dispose inside add and soriesOf by default', () => {
|
||||
const module = new MockModule();
|
||||
module.hot.dispose = jest.fn();
|
||||
|
||||
const {
|
||||
clientApi: { storiesOf, getStorybook },
|
||||
} = getContext();
|
||||
|
||||
storiesOf('kind', module).add('story', jest.fn());
|
||||
|
||||
expect(module.hot.dispose.calls.length).toEqual(2);
|
||||
});
|
||||
|
||||
it('should not bind dispose inside add when disableAddStoryDispose is true', () => {
|
||||
const module = new MockModule();
|
||||
module.hot.dispose = jest.fn();
|
||||
|
||||
const {
|
||||
clientApi: { storiesOf, getStorybook },
|
||||
} = getContext({ disableAddStoryDispose: true });
|
||||
|
||||
storiesOf('kind', module).add('story', jest.fn());
|
||||
|
||||
expect(module.hot.dispose.calls.length).toEqual(1);
|
||||
});
|
||||
});
|
||||
|
||||
describe('parameters', () => {
|
||||
@ -555,7 +581,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
storyStore,
|
||||
clientApi: { storiesOf, addParameters },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
addParameters({ a: 'global', b: 'global', c: 'global' });
|
||||
|
||||
@ -578,7 +604,7 @@ describe('preview.client_api', () => {
|
||||
const {
|
||||
storyStore,
|
||||
clientApi: { storiesOf, addParameters },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
addParameters({
|
||||
addon1: 'global string value',
|
||||
@ -638,7 +664,7 @@ describe('preview.client_api', () => {
|
||||
|
||||
const {
|
||||
clientApi: { storiesOf, getStorybook },
|
||||
} = getContext(undefined);
|
||||
} = getContext();
|
||||
|
||||
expect(getStorybook()).toEqual([]);
|
||||
|
||||
|
@ -86,10 +86,19 @@ export default class ClientApi {
|
||||
|
||||
private _decorateStory: (storyFn: StoryFn, decorators: DecoratorFunction[]) => any;
|
||||
|
||||
constructor({ storyStore, decorateStory = defaultDecorateStory }: ClientApiParams) {
|
||||
// React Native Fast refresh doesn't allow multiple dispose calls
|
||||
private _disableAddStoryDispose: boolean;
|
||||
|
||||
constructor({
|
||||
storyStore,
|
||||
decorateStory = defaultDecorateStory,
|
||||
disableAddStoryDispose,
|
||||
}: ClientApiParams) {
|
||||
this._storyStore = storyStore;
|
||||
this._addons = {};
|
||||
|
||||
this._disableAddStoryDispose = disableAddStoryDispose || false;
|
||||
|
||||
this._decorateStory = decorateStory;
|
||||
|
||||
if (!storyStore) {
|
||||
@ -206,7 +215,8 @@ export default class ClientApi {
|
||||
if (typeof storyName !== 'string') {
|
||||
throw new Error(`Invalid or missing storyName provided for a "${kind}" story.`);
|
||||
}
|
||||
if (m && m.hot && m.hot.dispose) {
|
||||
|
||||
if (!this._disableAddStoryDispose && m && m.hot && m.hot.dispose) {
|
||||
m.hot.dispose(() => {
|
||||
const { _storyStore } = this;
|
||||
_storyStore.remove(id);
|
||||
|
@ -29,6 +29,7 @@ export interface StoreData {
|
||||
export interface ClientApiParams {
|
||||
storyStore: StoryStore;
|
||||
decorateStory?: (storyFn: any, decorators: any) => any;
|
||||
disableAddStoryDispose?: boolean;
|
||||
}
|
||||
|
||||
export type ClientApiReturnFn<StoryFnReturnType> = (...args: any[]) => StoryApi<StoryFnReturnType>;
|
||||
|
Loading…
x
Reference in New Issue
Block a user