mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-07 03:21:05 +08:00
Ignore globals from non-local refs
This commit is contained in:
parent
b18ef1c371
commit
c55a62d2fd
@ -8,6 +8,9 @@ module.exports = {
|
|||||||
'./stories/**/*.stories.@(js|ts|tsx|mdx)',
|
'./stories/**/*.stories.@(js|ts|tsx|mdx)',
|
||||||
'./../../addons/docs/**/*.stories.tsx',
|
'./../../addons/docs/**/*.stories.tsx',
|
||||||
],
|
],
|
||||||
|
refs: {
|
||||||
|
'cra-ts-essentials': 'http://localhost:9009',
|
||||||
|
},
|
||||||
addons: [
|
addons: [
|
||||||
{
|
{
|
||||||
name: '@storybook/addon-docs',
|
name: '@storybook/addon-docs',
|
||||||
|
@ -420,7 +420,7 @@ export function useGlobals(): [Args, (newGlobals: Args) => void] {
|
|||||||
const { globals } = useStoryContext();
|
const { globals } = useStoryContext();
|
||||||
|
|
||||||
const updateGlobals = useCallback(
|
const updateGlobals = useCallback(
|
||||||
(newGlobals: Args) => channel.emit(UPDATE_GLOBALS, newGlobals),
|
(newGlobals: Args) => channel.emit(UPDATE_GLOBALS, { globals: newGlobals }),
|
||||||
[channel]
|
[channel]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -1,6 +1,9 @@
|
|||||||
import { SET_STORIES, UPDATE_GLOBALS, GLOBALS_UPDATED } from '@storybook/core-events';
|
import { SET_STORIES, UPDATE_GLOBALS, GLOBALS_UPDATED } from '@storybook/core-events';
|
||||||
|
import { logger } from '@storybook/client-logger';
|
||||||
|
|
||||||
import { Args, ModuleFn } from '../index';
|
import { Args, ModuleFn } from '../index';
|
||||||
import { SetStoriesPayloadV2 } from '../lib/stories';
|
import { SetStoriesPayloadV2 } from '../lib/stories';
|
||||||
|
import { getSourceType } from './refs';
|
||||||
|
|
||||||
export interface SubState {
|
export interface SubState {
|
||||||
globals: Args;
|
globals: Args;
|
||||||
@ -13,7 +16,13 @@ export interface SubAPI {
|
|||||||
export const init: ModuleFn = ({ store, fullAPI }) => {
|
export const init: ModuleFn = ({ store, fullAPI }) => {
|
||||||
const api: SubAPI = {
|
const api: SubAPI = {
|
||||||
updateGlobals(newGlobals) {
|
updateGlobals(newGlobals) {
|
||||||
fullAPI.emit(UPDATE_GLOBALS, newGlobals);
|
// Only emit the message to the local ref
|
||||||
|
fullAPI.emit(UPDATE_GLOBALS, {
|
||||||
|
globals: newGlobals,
|
||||||
|
options: {
|
||||||
|
target: 'storybook-preview-iframe',
|
||||||
|
},
|
||||||
|
});
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -23,8 +32,46 @@ export const init: ModuleFn = ({ store, fullAPI }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const initModule = () => {
|
const initModule = () => {
|
||||||
fullAPI.on(GLOBALS_UPDATED, (globals: Args) => store.setState({ globals }));
|
fullAPI.on(GLOBALS_UPDATED, function handleGlobalsUpdated({ globals }: { globals: Args }) {
|
||||||
fullAPI.on(SET_STORIES, ({ globals }: SetStoriesPayloadV2) => store.setState({ globals }));
|
const { source }: { source: string } = this;
|
||||||
|
const [sourceType] = getSourceType(source);
|
||||||
|
|
||||||
|
switch (sourceType) {
|
||||||
|
case 'local': {
|
||||||
|
store.setState({ globals });
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'external': {
|
||||||
|
logger.warn(
|
||||||
|
'received a GLOBALS_UPDATED from a non-local ref. This is not currently supported.'
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
logger.warn('received a SET_STORIES frame that was not configured as a ref');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
fullAPI.on(SET_STORIES, function handleSetStories({ globals }: SetStoriesPayloadV2) {
|
||||||
|
const { source }: { source: string } = this;
|
||||||
|
const [sourceType] = getSourceType(source);
|
||||||
|
|
||||||
|
switch (sourceType) {
|
||||||
|
case 'local': {
|
||||||
|
store.setState({ globals });
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'external': {
|
||||||
|
if (Object.keys(globals).length > 0) {
|
||||||
|
logger.warn('received globals from a non-local ref. This is not currently supported.');
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
// This is already going to be warned about in stories.ts
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
@ -1,9 +1,24 @@
|
|||||||
import EventEmitter from 'event-emitter';
|
import EventEmitter from 'events';
|
||||||
import { SET_STORIES, UPDATE_GLOBALS, GLOBALS_UPDATED } from '@storybook/core-events';
|
import { SET_STORIES, UPDATE_GLOBALS, GLOBALS_UPDATED } from '@storybook/core-events';
|
||||||
|
import { location } from 'global';
|
||||||
|
import { logger } from '@storybook/client-logger';
|
||||||
|
|
||||||
import { ModuleArgs, API } from '../index';
|
import { ModuleArgs, API } from '../index';
|
||||||
import { init as initModule, SubAPI } from '../modules/globals';
|
import { init as initModule, SubAPI } from '../modules/globals';
|
||||||
|
|
||||||
|
jest.mock('@storybook/client-logger');
|
||||||
|
|
||||||
|
const mockLocation = jest.fn();
|
||||||
|
class LocalEventEmitter extends EventEmitter {
|
||||||
|
on(event, callback) {
|
||||||
|
return super.on(event, (...args) => callback.apply({ source: mockLocation() }, args));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
mockLocation.mockReturnValue(location.toString());
|
||||||
|
});
|
||||||
|
|
||||||
function createMockStore() {
|
function createMockStore() {
|
||||||
let state = {};
|
let state = {};
|
||||||
return {
|
return {
|
||||||
@ -25,7 +40,7 @@ describe('stories API', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('set global args on SET_STORIES', () => {
|
it('set global args on SET_STORIES', () => {
|
||||||
const api = EventEmitter();
|
const api = new LocalEventEmitter();
|
||||||
const store = createMockStore();
|
const store = createMockStore();
|
||||||
const { state, init } = initModule(({ store, fullAPI: api } as unknown) as ModuleArgs);
|
const { state, init } = initModule(({ store, fullAPI: api } as unknown) as ModuleArgs);
|
||||||
store.setState(state);
|
store.setState(state);
|
||||||
@ -39,25 +54,52 @@ describe('stories API', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('ignores SET_STORIES from other refs', () => {
|
||||||
|
const api = new LocalEventEmitter();
|
||||||
|
const store = createMockStore();
|
||||||
|
const { state, init } = initModule(({ store, fullAPI: api } as unknown) as ModuleArgs);
|
||||||
|
store.setState(state);
|
||||||
|
init();
|
||||||
|
|
||||||
|
mockLocation.mockReturnValueOnce('https://ref');
|
||||||
|
api.emit(SET_STORIES, { globals: { a: 'b' } });
|
||||||
|
expect(store.getState()).toEqual({ globals: {} });
|
||||||
|
});
|
||||||
|
|
||||||
it('updates the state when the preview emits GLOBALS_UPDATED', () => {
|
it('updates the state when the preview emits GLOBALS_UPDATED', () => {
|
||||||
const api = EventEmitter();
|
const api = new LocalEventEmitter();
|
||||||
const store = createMockStore();
|
const store = createMockStore();
|
||||||
const { state, init } = initModule(({ store, fullAPI: api } as unknown) as ModuleArgs);
|
const { state, init } = initModule(({ store, fullAPI: api } as unknown) as ModuleArgs);
|
||||||
store.setState(state);
|
store.setState(state);
|
||||||
|
|
||||||
init();
|
init();
|
||||||
|
|
||||||
api.emit(GLOBALS_UPDATED, { a: 'b' });
|
api.emit(GLOBALS_UPDATED, { globals: { a: 'b' } });
|
||||||
expect(store.getState()).toEqual({ globals: { a: 'b' } });
|
expect(store.getState()).toEqual({ globals: { a: 'b' } });
|
||||||
|
|
||||||
api.emit(GLOBALS_UPDATED, { a: 'c' });
|
api.emit(GLOBALS_UPDATED, { globals: { a: 'c' } });
|
||||||
expect(store.getState()).toEqual({ globals: { a: 'c' } });
|
expect(store.getState()).toEqual({ globals: { a: 'c' } });
|
||||||
|
|
||||||
// SHOULD NOT merge global args
|
// SHOULD NOT merge global args
|
||||||
api.emit(GLOBALS_UPDATED, { d: 'e' });
|
api.emit(GLOBALS_UPDATED, { globals: { d: 'e' } });
|
||||||
expect(store.getState()).toEqual({ globals: { d: 'e' } });
|
expect(store.getState()).toEqual({ globals: { d: 'e' } });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('ignores GLOBALS_UPDATED from other refs', () => {
|
||||||
|
const api = new LocalEventEmitter();
|
||||||
|
const store = createMockStore();
|
||||||
|
const { state, init } = initModule(({ store, fullAPI: api } as unknown) as ModuleArgs);
|
||||||
|
store.setState(state);
|
||||||
|
|
||||||
|
init();
|
||||||
|
|
||||||
|
mockLocation.mockReturnValueOnce('https://ref');
|
||||||
|
logger.warn.mockClear();
|
||||||
|
api.emit(GLOBALS_UPDATED, { globals: { a: 'b' } });
|
||||||
|
expect(store.getState()).toEqual({ globals: {} });
|
||||||
|
expect(logger.warn).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
it('emits UPDATE_GLOBALS when updateGlobals is called', () => {
|
it('emits UPDATE_GLOBALS when updateGlobals is called', () => {
|
||||||
const fullAPI = ({ emit: jest.fn(), on: jest.fn() } as unknown) as API;
|
const fullAPI = ({ emit: jest.fn(), on: jest.fn() } as unknown) as API;
|
||||||
const store = createMockStore();
|
const store = createMockStore();
|
||||||
@ -66,6 +108,9 @@ describe('stories API', () => {
|
|||||||
init();
|
init();
|
||||||
|
|
||||||
(api as SubAPI).updateGlobals({ a: 'b' });
|
(api as SubAPI).updateGlobals({ a: 'b' });
|
||||||
expect(fullAPI.emit).toHaveBeenCalledWith(UPDATE_GLOBALS, { a: 'b' });
|
expect(fullAPI.emit).toHaveBeenCalledWith(UPDATE_GLOBALS, {
|
||||||
|
globals: { a: 'b' },
|
||||||
|
options: { target: 'storybook-preview-iframe' },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -546,7 +546,7 @@ describe('Preview hooks', () => {
|
|||||||
[
|
[
|
||||||
(storyFn) => {
|
(storyFn) => {
|
||||||
useGlobals()[1]({ a: 'b' });
|
useGlobals()[1]({ a: 'b' });
|
||||||
expect(mockChannel.emit).toHaveBeenCalledWith(UPDATE_GLOBALS, { a: 'b' });
|
expect(mockChannel.emit).toHaveBeenCalledWith(UPDATE_GLOBALS, { globals: { a: 'b' } });
|
||||||
return storyFn();
|
return storyFn();
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -456,10 +456,12 @@ describe('preview.story_store', () => {
|
|||||||
addStoryToStore(store, 'a', '1', () => 0);
|
addStoryToStore(store, 'a', '1', () => 0);
|
||||||
|
|
||||||
store.updateGlobals({ foo: 'bar' });
|
store.updateGlobals({ foo: 'bar' });
|
||||||
expect(onGlobalsChangedChannel).toHaveBeenCalledWith({ foo: 'bar' });
|
expect(onGlobalsChangedChannel).toHaveBeenCalledWith({ globals: { foo: 'bar' } });
|
||||||
|
|
||||||
store.updateGlobals({ baz: 'bing' });
|
store.updateGlobals({ baz: 'bing' });
|
||||||
expect(onGlobalsChangedChannel).toHaveBeenCalledWith({ foo: 'bar', baz: 'bing' });
|
expect(onGlobalsChangedChannel).toHaveBeenCalledWith({
|
||||||
|
globals: { foo: 'bar', baz: 'bing' },
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should update if the UPDATE_GLOBALS event is received', () => {
|
it('should update if the UPDATE_GLOBALS event is received', () => {
|
||||||
@ -467,7 +469,7 @@ describe('preview.story_store', () => {
|
|||||||
const store = new StoryStore({ channel: testChannel });
|
const store = new StoryStore({ channel: testChannel });
|
||||||
addStoryToStore(store, 'a', '1', () => 0);
|
addStoryToStore(store, 'a', '1', () => 0);
|
||||||
|
|
||||||
testChannel.emit(Events.UPDATE_GLOBALS, { foo: 'bar' });
|
testChannel.emit(Events.UPDATE_GLOBALS, { globals: { foo: 'bar' } });
|
||||||
|
|
||||||
expect(store.getRawStory('a', '1').globals).toEqual({ foo: 'bar' });
|
expect(store.getRawStory('a', '1').globals).toEqual({ foo: 'bar' });
|
||||||
});
|
});
|
||||||
|
@ -159,7 +159,9 @@ export default class StoryStore {
|
|||||||
this.updateStoryArgs(id, newArgs)
|
this.updateStoryArgs(id, newArgs)
|
||||||
);
|
);
|
||||||
|
|
||||||
this._channel.on(Events.UPDATE_GLOBALS, (newGlobals: Args) => this.updateGlobals(newGlobals));
|
this._channel.on(Events.UPDATE_GLOBALS, ({ globals }: { globals: Args }) =>
|
||||||
|
this.updateGlobals(globals)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
startConfiguring() {
|
startConfiguring() {
|
||||||
@ -441,7 +443,7 @@ export default class StoryStore {
|
|||||||
updateGlobals(newGlobals: Args) {
|
updateGlobals(newGlobals: Args) {
|
||||||
this._globals = { ...this._globals, ...newGlobals };
|
this._globals = { ...this._globals, ...newGlobals };
|
||||||
this.storeGlobals();
|
this.storeGlobals();
|
||||||
this._channel.emit(Events.GLOBALS_UPDATED, this._globals);
|
this._channel.emit(Events.GLOBALS_UPDATED, { globals: this._globals });
|
||||||
}
|
}
|
||||||
|
|
||||||
updateStoryArgs(id: string, newArgs: Args) {
|
updateStoryArgs(id: string, newArgs: Args) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user