added ability to skip dispose call in clientApi

This commit is contained in:
Gytis Vinclovas 2020-02-16 00:30:47 +02:00
parent 71aa9065dc
commit 23da639ffc
3 changed files with 72 additions and 35 deletions

View File

@ -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([]);

View File

@ -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);

View File

@ -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>;