storybook/addons/knobs/src/registerKnobs.js

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