mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 01:51:05 +08:00
88 lines
2.2 KiB
JavaScript
88 lines
2.2 KiB
JavaScript
import addons from '@storybook/addons';
|
|
import { STORY_CHANGED, FORCE_RE_RENDER, REGISTER_SUBSCRIPTION } from '@storybook/core-events';
|
|
import debounce from 'lodash/debounce';
|
|
|
|
import KnobManager from './KnobManager';
|
|
import { CHANGE, CLICK, RESET, SET } from './shared';
|
|
|
|
export const manager = new KnobManager();
|
|
const { knobStore } = manager;
|
|
const COMPONENT_FORCE_RENDER_DEBOUNCE_DELAY_MS = 325;
|
|
|
|
function forceReRender() {
|
|
addons.getChannel().emit(FORCE_RE_RENDER);
|
|
}
|
|
|
|
function setPaneKnobs(timestamp = +new Date()) {
|
|
const channel = addons.getChannel();
|
|
channel.emit(SET, { knobs: knobStore.getAll(), timestamp });
|
|
}
|
|
|
|
const resetAndForceUpdate = () => {
|
|
knobStore.markAllUnused();
|
|
forceReRender();
|
|
};
|
|
|
|
// Increase performance by reducing how frequently the story is recreated during knob changes
|
|
const debouncedResetAndForceUpdate = debounce(
|
|
resetAndForceUpdate,
|
|
COMPONENT_FORCE_RENDER_DEBOUNCE_DELAY_MS
|
|
);
|
|
|
|
function knobChanged(change) {
|
|
const { name } = change;
|
|
const { value } = change; // Update the related knob and it's value.
|
|
const knobOptions = knobStore.get(name);
|
|
knobOptions.value = value;
|
|
|
|
if (!manager.options.disableDebounce) {
|
|
debouncedResetAndForceUpdate();
|
|
} else {
|
|
resetAndForceUpdate();
|
|
}
|
|
}
|
|
|
|
function knobClicked(clicked) {
|
|
const knobOptions = knobStore.get(clicked.name);
|
|
knobOptions.callback();
|
|
forceReRender();
|
|
}
|
|
|
|
function resetKnobs() {
|
|
knobStore.reset();
|
|
|
|
setPaneKnobs(false);
|
|
}
|
|
|
|
function resetKnobsAndForceReRender() {
|
|
knobStore.reset();
|
|
|
|
forceReRender();
|
|
|
|
setPaneKnobs(false);
|
|
}
|
|
|
|
function disconnectCallbacks() {
|
|
const channel = addons.getChannel();
|
|
channel.removeListener(CHANGE, knobChanged);
|
|
channel.removeListener(CLICK, knobClicked);
|
|
channel.removeListener(STORY_CHANGED, resetKnobs);
|
|
channel.removeListener(RESET, resetKnobsAndForceReRender);
|
|
knobStore.unsubscribe(setPaneKnobs);
|
|
}
|
|
|
|
function connectCallbacks() {
|
|
const channel = addons.getChannel();
|
|
channel.on(CHANGE, knobChanged);
|
|
channel.on(CLICK, knobClicked);
|
|
channel.on(STORY_CHANGED, resetKnobs);
|
|
channel.on(RESET, resetKnobsAndForceReRender);
|
|
knobStore.subscribe(setPaneKnobs);
|
|
|
|
return disconnectCallbacks;
|
|
}
|
|
|
|
export function registerKnobs() {
|
|
addons.getChannel().emit(REGISTER_SUBSCRIPTION, connectCallbacks);
|
|
}
|