@@ -310,7 +312,7 @@ exports[`Storyshots Custom|Decorator for Vue render 1`] = `
`;
-exports[`Storyshots Custom|Decorator for Vue template 1`] = `
+exports[`Storyshots Custom|Decorator for Vue Template 1`] = `
@@ -327,7 +329,7 @@ exports[`Storyshots Custom|Decorator for Vue template 1`] = `
`;
-exports[`Storyshots Custom|Decorator for Vue withData 1`] = `
+exports[`Storyshots Custom|Decorator for Vue With Data 1`] = `
@@ -336,18 +338,18 @@ exports[`Storyshots Custom|Decorator for Vue withData 1`] = `
>
{
- "id": "custom-decorator-for-vue--withdata",
+ "id": "custom-decorator-for-vue--with-data",
"kind": "Custom|Decorator for Vue",
- "name": "withData",
- "story": "withData",
+ "name": "With Data",
+ "story": "With Data",
"customContext": 52,
"parameters": {
"options": {
"hierarchyRootSeparator": {},
- "hierarchySeparator": {},
- "docs": {
- "iframeHeight": "60px"
- }
+ "hierarchySeparator": {}
+ },
+ "docs": {
+ "iframeHeight": "60px"
},
"globalParameter": "globalParameter",
"framework": "vue",
@@ -368,7 +370,7 @@ exports[`Storyshots Custom|Method for rendering Vue JSX 1`] = `
`;
-exports[`Storyshots Custom|Method for rendering Vue pre-registered component 1`] = `
+exports[`Storyshots Custom|Method for rendering Vue Pre Registered Component 1`] = `
This component was pre-registered in .storybook/config.js
@@ -384,7 +386,13 @@ exports[`Storyshots Custom|Method for rendering Vue pre-registered component 1`]
`;
-exports[`Storyshots Custom|Method for rendering Vue render + component 1`] = `
+exports[`Storyshots Custom|Method for rendering Vue Render 1`] = `
+
+ renders a div with some text in it..
+
+`;
+
+exports[`Storyshots Custom|Method for rendering Vue Render Component 1`] = `
`;
-exports[`Storyshots Custom|Method for rendering Vue render 1`] = `
+exports[`Storyshots Custom|Method for rendering Vue Template 1`] = `
- renders a div with some text in it..
+
+ A template
+
+
+
+ rendered in vue in storybook
+
`;
-exports[`Storyshots Custom|Method for rendering Vue template + component 1`] = `
+exports[`Storyshots Custom|Method for rendering Vue Template Component 1`] = `
`;
-exports[`Storyshots Custom|Method for rendering Vue template + methods 1`] = `
+exports[`Storyshots Custom|Method for rendering Vue Template Methods 1`] = `
Clicking the button will navigate to another story using the 'addon-links'
@@ -424,19 +438,7 @@ exports[`Storyshots Custom|Method for rendering Vue template + methods 1`] = `
`;
-exports[`Storyshots Custom|Method for rendering Vue template 1`] = `
-
-
- A template
-
-
-
- rendered in vue in storybook
-
-
-`;
-
-exports[`Storyshots Custom|Method for rendering Vue vuex + actions 1`] = `
+exports[`Storyshots Custom|Method for rendering Vue Vuex Actions 1`] = `
`;
-exports[`Storyshots Custom|Method for rendering Vue whatever you want 1`] = `
+exports[`Storyshots Custom|Method for rendering Vue Whatever You Want 1`] = `
`;
-exports[`Storyshots Welcome welcome 1`] = `
+exports[`Storyshots Welcome Welcome 1`] = `
diff --git a/examples/vue-kitchen-sink/package.json b/examples/vue-kitchen-sink/package.json
index 002a1a0fec6..0655e865176 100644
--- a/examples/vue-kitchen-sink/package.json
+++ b/examples/vue-kitchen-sink/package.json
@@ -1,12 +1,11 @@
{
"name": "vue-example",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"private": true,
"scripts": {
"build": "cross-env NODE_ENV=production webpack --progress --hide-modules",
"build-storybook": "build-storybook -s public",
"dev": "cross-env NODE_ENV=development webpack-dev-server --open --hot",
- "now-build": "node ../../scripts/bootstrap --core && yarn run build-storybook --quiet",
"storybook": "start-storybook -p 9009 -s public"
},
"dependencies": {
@@ -15,22 +14,22 @@
},
"devDependencies": {
"@babel/core": "^7.3.4",
- "@storybook/addon-a11y": "5.2.0-beta.28",
- "@storybook/addon-actions": "5.2.0-beta.28",
- "@storybook/addon-backgrounds": "5.2.0-beta.28",
- "@storybook/addon-centered": "5.2.0-beta.28",
- "@storybook/addon-contexts": "5.2.0-beta.28",
- "@storybook/addon-docs": "5.2.0-beta.28",
- "@storybook/addon-knobs": "5.2.0-beta.28",
- "@storybook/addon-links": "5.2.0-beta.28",
- "@storybook/addon-notes": "5.2.0-beta.28",
- "@storybook/addon-options": "5.2.0-beta.28",
- "@storybook/addon-storyshots": "5.2.0-beta.28",
- "@storybook/addon-storysource": "5.2.0-beta.28",
- "@storybook/addon-viewport": "5.2.0-beta.28",
- "@storybook/addons": "5.2.0-beta.28",
- "@storybook/source-loader": "5.2.0-beta.28",
- "@storybook/vue": "5.2.0-beta.28",
+ "@storybook/addon-a11y": "5.3.0-alpha.0",
+ "@storybook/addon-actions": "5.3.0-alpha.0",
+ "@storybook/addon-backgrounds": "5.3.0-alpha.0",
+ "@storybook/addon-centered": "5.3.0-alpha.0",
+ "@storybook/addon-contexts": "5.3.0-alpha.0",
+ "@storybook/addon-docs": "5.3.0-alpha.0",
+ "@storybook/addon-knobs": "5.3.0-alpha.0",
+ "@storybook/addon-links": "5.3.0-alpha.0",
+ "@storybook/addon-notes": "5.3.0-alpha.0",
+ "@storybook/addon-options": "5.3.0-alpha.0",
+ "@storybook/addon-storyshots": "5.3.0-alpha.0",
+ "@storybook/addon-storysource": "5.3.0-alpha.0",
+ "@storybook/addon-viewport": "5.3.0-alpha.0",
+ "@storybook/addons": "5.3.0-alpha.0",
+ "@storybook/source-loader": "5.3.0-alpha.0",
+ "@storybook/vue": "5.3.0-alpha.0",
"babel-core": "^7.0.0-bridge.0",
"babel-loader": "^8.0.5",
"cross-env": "^5.2.0",
diff --git a/examples/vue-kitchen-sink/src/stories/addon-docs.stories.mdx b/examples/vue-kitchen-sink/src/stories/addon-docs.stories.mdx
index 445657fe2a1..cd2c3787a4c 100644
--- a/examples/vue-kitchen-sink/src/stories/addon-docs.stories.mdx
+++ b/examples/vue-kitchen-sink/src/stories/addon-docs.stories.mdx
@@ -73,6 +73,8 @@ Just like in React, we can easily reference other stories in our docs:
+
+
## More info
For more info, check out the [Storybook Docs Technical Preview](https://docs.google.com/document/d/1un6YX7xDKEKl5-MVb-egnOYN8dynb5Hf7mq0hipk8JE/edit?usp=sharing).
diff --git a/examples/vue-kitchen-sink/src/stories/components/welcome.stories.js b/examples/vue-kitchen-sink/src/stories/components/welcome.stories.js
index fd5f87471b0..d9eebaed48b 100644
--- a/examples/vue-kitchen-sink/src/stories/components/welcome.stories.js
+++ b/examples/vue-kitchen-sink/src/stories/components/welcome.stories.js
@@ -4,9 +4,7 @@ import Welcome from '../Welcome.vue';
export default {
title: 'Welcome',
- parameters: {
- component: Welcome,
- },
+ component: Welcome,
};
export const welcome = () => {
diff --git a/examples/vue-kitchen-sink/src/stories/core-errors.stories.js b/examples/vue-kitchen-sink/src/stories/core-errors.stories.js
new file mode 100644
index 00000000000..c74f8b07057
--- /dev/null
+++ b/examples/vue-kitchen-sink/src/stories/core-errors.stories.js
@@ -0,0 +1,10 @@
+export default {
+ title: 'Core|Errors',
+};
+
+export const throwsError = () => {
+ throw new Error('foo');
+};
+throwsError.story = { parameters: { storyshots: { disable: true } } };
+
+export const nullError = () => null;
diff --git a/examples/vue-kitchen-sink/src/stories/custom-decorators.stories.js b/examples/vue-kitchen-sink/src/stories/custom-decorators.stories.js
index c5bc0a2d2d1..208c02fda23 100644
--- a/examples/vue-kitchen-sink/src/stories/custom-decorators.stories.js
+++ b/examples/vue-kitchen-sink/src/stories/custom-decorators.stories.js
@@ -30,7 +30,7 @@ export const template = () => ({
template: '
MyButton with template ',
});
-export const withData = ({ parameters, ...rest }) => ({
+export const withData = ({ parameters, hooks, ...rest }) => ({
template: `
${JSON.stringify({ ...rest, parameters }, null, 2)} `,
});
diff --git a/lerna.json b/lerna.json
index a7dc9c3c697..1a74a0f0c43 100644
--- a/lerna.json
+++ b/lerna.json
@@ -2,5 +2,5 @@
"npmClient": "yarn",
"useWorkspaces": true,
"registry": "https://registry.npmjs.org",
- "version": "5.2.0-beta.28"
+ "version": "5.3.0-alpha.0"
}
diff --git a/lib/addons/package.json b/lib/addons/package.json
index 6feed12bbe2..bd090e0ebb2 100644
--- a/lib/addons/package.json
+++ b/lib/addons/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/addons",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Storybook addons store",
"keywords": [
"storybook"
@@ -15,15 +15,20 @@
"directory": "lib/addons"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/public_api.js",
"types": "dist/public_api.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/api": "5.2.0-beta.28",
- "@storybook/channels": "5.2.0-beta.28",
- "@storybook/client-logger": "5.2.0-beta.28",
+ "@storybook/api": "5.3.0-alpha.0",
+ "@storybook/channels": "5.3.0-alpha.0",
+ "@storybook/client-logger": "5.3.0-alpha.0",
+ "@storybook/core-events": "5.3.0-alpha.0",
"core-js": "^3.0.1",
"global": "^4.3.2",
"util-deprecate": "^1.0.2"
diff --git a/lib/addons/src/hooks.ts b/lib/addons/src/hooks.ts
new file mode 100644
index 00000000000..b6b8596c2f4
--- /dev/null
+++ b/lib/addons/src/hooks.ts
@@ -0,0 +1,406 @@
+import window from 'global';
+import { logger } from '@storybook/client-logger';
+import { FORCE_RE_RENDER, STORY_RENDERED, DOCS_RENDERED } from '@storybook/core-events';
+import addons, { StoryGetter, StoryContext } from './public_api';
+
+interface StoryStore {
+ fromId: (
+ id: string
+ ) => {
+ parameters: {
+ [parameterKey: string]: any;
+ };
+ };
+ getSelection: () => {
+ storyId: string;
+ viewMode: string;
+ };
+}
+
+interface Hook {
+ name: string;
+ memoizedState?: any;
+ deps?: any[] | undefined;
+}
+
+interface Effect {
+ create: () => (() => void) | void;
+ destroy?: (() => void) | void;
+}
+
+type Decorator = (getStory: StoryGetter, context: StoryContext) => any;
+type AbstractFunction = (...args: any[]) => any;
+
+const RenderEvents = [STORY_RENDERED, DOCS_RENDERED];
+
+export class HooksContext {
+ hookListsMap: WeakMap
;
+
+ mountedDecorators: Set;
+
+ prevMountedDecorators: Set;
+
+ currentHooks: Hook[];
+
+ nextHookIndex: number;
+
+ currentPhase: 'MOUNT' | 'UPDATE' | 'NONE';
+
+ currentEffects: Effect[];
+
+ prevEffects: Effect[];
+
+ currentDecoratorName: string | null;
+
+ hasUpdates: boolean;
+
+ currentContext: StoryContext | null;
+
+ renderListener = () => {
+ this.triggerEffects();
+ this.currentContext = null;
+ this.removeRenderListeners();
+ };
+
+ constructor() {
+ this.init();
+ }
+
+ init() {
+ this.hookListsMap = new WeakMap();
+ this.mountedDecorators = new Set();
+ this.prevMountedDecorators = this.mountedDecorators;
+ this.currentHooks = [];
+ this.nextHookIndex = 0;
+ this.currentPhase = 'NONE';
+ this.currentEffects = [];
+ this.prevEffects = [];
+ this.currentDecoratorName = null;
+ this.hasUpdates = false;
+ this.currentContext = null;
+ }
+
+ clean() {
+ this.prevEffects.forEach(effect => {
+ if (effect.destroy) {
+ effect.destroy();
+ }
+ });
+ this.init();
+ this.removeRenderListeners();
+ }
+
+ getNextHook() {
+ const hook = this.currentHooks[this.nextHookIndex];
+ this.nextHookIndex += 1;
+ return hook;
+ }
+
+ triggerEffects() {
+ // destroy removed effects
+ this.prevEffects.forEach(effect => {
+ if (!this.currentEffects.includes(effect) && effect.destroy) {
+ effect.destroy();
+ }
+ });
+ // trigger added effects
+ this.currentEffects.forEach(effect => {
+ if (!this.prevEffects.includes(effect)) {
+ // eslint-disable-next-line no-param-reassign
+ effect.destroy = effect.create();
+ }
+ });
+ this.prevEffects = this.currentEffects;
+ this.currentEffects = [];
+ }
+
+ addRenderListeners() {
+ const channel = addons.getChannel();
+ RenderEvents.forEach(e => channel.on(e, this.renderListener));
+ }
+
+ removeRenderListeners() {
+ const channel = addons.getChannel();
+ RenderEvents.forEach(e => channel.removeListener(e, this.renderListener));
+ }
+}
+
+const hookify = (fn: AbstractFunction) => (...args: any[]) => {
+ const { hooks }: StoryContext = typeof args[0] === 'function' ? args[1] : args[0];
+
+ const prevPhase = hooks.currentPhase;
+ const prevHooks = hooks.currentHooks;
+ const prevNextHookIndex = hooks.nextHookIndex;
+ const prevDecoratorName = hooks.currentDecoratorName;
+
+ hooks.currentDecoratorName = fn.name;
+ if (hooks.prevMountedDecorators.has(fn)) {
+ hooks.currentPhase = 'UPDATE';
+ hooks.currentHooks = hooks.hookListsMap.get(fn) || [];
+ } else {
+ hooks.currentPhase = 'MOUNT';
+ hooks.currentHooks = [];
+ hooks.hookListsMap.set(fn, hooks.currentHooks);
+ }
+ hooks.nextHookIndex = 0;
+
+ const prevContext = window.STORYBOOK_HOOKS_CONTEXT;
+ window.STORYBOOK_HOOKS_CONTEXT = hooks;
+ const result = fn(...args);
+ window.STORYBOOK_HOOKS_CONTEXT = prevContext;
+
+ if (hooks.currentPhase === 'UPDATE' && hooks.getNextHook() != null) {
+ throw new Error(
+ 'Rendered fewer hooks than expected. This may be caused by an accidental early return statement.'
+ );
+ }
+
+ hooks.currentPhase = prevPhase;
+ hooks.currentHooks = prevHooks;
+ hooks.nextHookIndex = prevNextHookIndex;
+ hooks.currentDecoratorName = prevDecoratorName;
+ return result;
+};
+
+// Counter to prevent infinite loops.
+let numberOfRenders = 0;
+const RENDER_LIMIT = 25;
+export const applyHooks = (
+ applyDecorators: (getStory: StoryGetter, decorators: Decorator[]) => StoryGetter
+) => (getStory: StoryGetter, decorators: Decorator[]) => {
+ const decorated = applyDecorators(hookify(getStory), decorators.map(hookify));
+ return (context: StoryContext) => {
+ const { hooks } = context;
+ hooks.prevMountedDecorators = hooks.mountedDecorators;
+ hooks.mountedDecorators = new Set([getStory, ...decorators]);
+ hooks.currentContext = context;
+ hooks.hasUpdates = false;
+ let result = decorated(context);
+ numberOfRenders = 1;
+ while (hooks.hasUpdates) {
+ hooks.hasUpdates = false;
+ hooks.currentEffects = [];
+ hooks.prevMountedDecorators = hooks.mountedDecorators;
+ result = decorated(context);
+ numberOfRenders += 1;
+ if (numberOfRenders > RENDER_LIMIT) {
+ throw new Error(
+ 'Too many re-renders. Storybook limits the number of renders to prevent an infinite loop.'
+ );
+ }
+ }
+ hooks.addRenderListeners();
+ return result;
+ };
+};
+
+const areDepsEqual = (deps: any[], nextDeps: any[]) =>
+ deps.length === nextDeps.length && deps.every((dep, i) => dep === nextDeps[i]);
+
+const invalidHooksError = () =>
+ new Error('Storybook preview hooks can only be called inside decorators and story functions.');
+
+function getHooksContextOrNull(): HooksContext | null {
+ return window.STORYBOOK_HOOKS_CONTEXT || null;
+}
+
+function getHooksContextOrThrow(): HooksContext {
+ const hooks = getHooksContextOrNull();
+ if (hooks == null) {
+ throw invalidHooksError();
+ }
+ return hooks;
+}
+
+function useHook(name: string, callback: (hook: Hook) => void, deps?: any[] | undefined): Hook {
+ const hooks = getHooksContextOrThrow();
+ if (hooks.currentPhase === 'MOUNT') {
+ if (deps != null && !Array.isArray(deps)) {
+ logger.warn(
+ `${name} received a final argument that is not an array (instead, received ${deps}). When specified, the final argument must be an array.`
+ );
+ }
+ const hook: Hook = { name, deps };
+ hooks.currentHooks.push(hook);
+ callback(hook);
+ return hook;
+ }
+
+ if (hooks.currentPhase === 'UPDATE') {
+ const hook = hooks.getNextHook();
+ if (hook == null) {
+ throw new Error('Rendered more hooks than during the previous render.');
+ }
+
+ if (hook.name !== name) {
+ logger.warn(
+ `Storybook has detected a change in the order of Hooks${
+ hooks.currentDecoratorName ? ` called by ${hooks.currentDecoratorName}` : ''
+ }. This will lead to bugs and errors if not fixed.`
+ );
+ }
+
+ if (deps != null && hook.deps == null) {
+ logger.warn(
+ `${name} received a final argument during this render, but not during the previous render. Even though the final argument is optional, its type cannot change between renders.`
+ );
+ }
+
+ if (deps != null && hook.deps != null && deps.length !== hook.deps.length) {
+ logger.warn(`The final argument passed to ${name} changed size between renders. The order and size of this array must remain constant.
+Previous: ${hook.deps}
+Incoming: ${deps}`);
+ }
+
+ if (deps == null || hook.deps == null || !areDepsEqual(deps, hook.deps)) {
+ callback(hook);
+ hook.deps = deps;
+ }
+ return hook;
+ }
+
+ throw invalidHooksError();
+}
+
+function useMemoLike(name: string, nextCreate: () => T, deps: any[] | undefined): T {
+ const { memoizedState } = useHook(
+ name,
+ hook => {
+ // eslint-disable-next-line no-param-reassign
+ hook.memoizedState = nextCreate();
+ },
+ deps
+ );
+ return memoizedState;
+}
+
+/* Returns a memoized value, see https://reactjs.org/docs/hooks-reference.html#usememo */
+export function useMemo(nextCreate: () => T, deps?: any[]): T {
+ return useMemoLike('useMemo', nextCreate, deps);
+}
+
+/* Returns a memoized callback, see https://reactjs.org/docs/hooks-reference.html#usecallback */
+export function useCallback(callback: T, deps?: any[]): T {
+ return useMemoLike('useCallback', () => callback, deps);
+}
+
+function useRefLike(name: string, initialValue: T): { current: T } {
+ return useMemoLike(name, () => ({ current: initialValue }), []);
+}
+
+/* Returns a mutable ref object, see https://reactjs.org/docs/hooks-reference.html#useref */
+export function useRef(initialValue: T): { current: T } {
+ return useRefLike('useRef', initialValue);
+}
+
+function triggerUpdate() {
+ const hooks = getHooksContextOrNull();
+ // Rerun getStory if updates were triggered synchronously, force rerender otherwise
+ if (hooks != null && hooks.currentPhase !== 'NONE') {
+ hooks.hasUpdates = true;
+ } else {
+ try {
+ addons.getChannel().emit(FORCE_RE_RENDER);
+ } catch (e) {
+ logger.warn('State updates of Storybook preview hooks work only in browser');
+ }
+ }
+}
+
+function useStateLike(
+ name: string,
+ initialState: (() => S) | S
+): [S, (update: ((prevState: S) => S) | S) => void] {
+ const stateRef = useRefLike(
+ name,
+ // @ts-ignore S type should never be function, but there's no way to tell that to TypeScript
+ typeof initialState === 'function' ? initialState() : initialState
+ );
+ const setState = (update: ((prevState: S) => S) | S) => {
+ // @ts-ignore S type should never be function, but there's no way to tell that to TypeScript
+ stateRef.current = typeof update === 'function' ? update(stateRef.current) : update;
+ triggerUpdate();
+ };
+ return [stateRef.current, setState];
+}
+
+/* Returns a stateful value, and a function to update it, see https://reactjs.org/docs/hooks-reference.html#usestate */
+export function useState(
+ initialState: (() => S) | S
+): [S, (update: ((prevState: S) => S) | S) => void] {
+ return useStateLike('useState', initialState);
+}
+
+/* A redux-like alternative to useState, see https://reactjs.org/docs/hooks-reference.html#usereducer */
+export function useReducer(
+ reducer: (state: S, action: A) => S,
+ initialState: S
+): [S, (action: A) => void];
+export function useReducer(
+ reducer: (state: S, action: A) => S,
+ initialArg: I,
+ init: (initialArg: I) => S
+): [S, (action: A) => void];
+export function useReducer(
+ reducer: (state: S, action: A) => S,
+ initialArg: any,
+ init?: any
+): [S, (action: A) => void] {
+ const initialState: (() => S) | S = init != null ? () => init(initialArg) : initialArg;
+ const [state, setState] = useStateLike('useReducer', initialState);
+ const dispatch = (action: A) => setState(prevState => reducer(prevState, action));
+ return [state, dispatch];
+}
+
+/*
+ Triggers a side effect, see https://reactjs.org/docs/hooks-reference.html#usestate
+ Effects are triggered synchronously after rendering the story
+*/
+export function useEffect(create: () => (() => void) | void, deps?: any[]): void {
+ const hooks = getHooksContextOrThrow();
+ const effect = useMemoLike('useEffect', () => ({ create }), deps);
+ hooks.currentEffects.push(effect);
+}
+
+export interface Listener {
+ (...args: any[]): void;
+ ignorePeer?: boolean;
+}
+
+export interface EventMap {
+ [eventId: string]: Listener;
+}
+
+/* Accepts a map of Storybook channel event listeners, returns an emit function */
+export function useChannel(eventMap: EventMap, deps: any[] = []) {
+ const channel = addons.getChannel();
+ useEffect(() => {
+ Object.entries(eventMap).forEach(([type, listener]) => channel.on(type, listener));
+ return () => {
+ Object.entries(eventMap).forEach(([type, listener]) =>
+ channel.removeListener(type, listener)
+ );
+ };
+ }, [...Object.keys(eventMap), ...deps]);
+
+ return channel.emit.bind(channel);
+}
+
+/* Returns current story context */
+export function useStoryContext(): StoryContext {
+ const { currentContext } = getHooksContextOrThrow();
+ if (currentContext == null) {
+ throw invalidHooksError();
+ }
+
+ return currentContext;
+}
+
+/* Returns current value of a story parameter */
+export function useParameter(parameterKey: string, defaultValue?: S): S | undefined {
+ const { parameters } = useStoryContext();
+ if (parameterKey) {
+ return parameters[parameterKey] || (defaultValue as S);
+ }
+ return undefined;
+}
diff --git a/lib/addons/src/public_api.ts b/lib/addons/src/public_api.ts
index 8a3b7347b61..7639be2ff7b 100644
--- a/lib/addons/src/public_api.ts
+++ b/lib/addons/src/public_api.ts
@@ -9,4 +9,6 @@ export * from './make-decorator';
export * from './index';
export * from './types';
export * from './storybook-channel-mock';
+export * from './hooks';
+
export default addons;
diff --git a/lib/addons/src/types.ts b/lib/addons/src/types.ts
index 2b9cddea818..35b7b2ba698 100644
--- a/lib/addons/src/types.ts
+++ b/lib/addons/src/types.ts
@@ -1,3 +1,4 @@
+import { HooksContext } from './hooks';
import { Addon } from './index';
export enum types {
@@ -26,6 +27,7 @@ export interface StoryContext {
kind: string;
[key: string]: any;
parameters: Parameters;
+ hooks?: HooksContext;
}
export interface WrapperSettings {
diff --git a/lib/api/package.json b/lib/api/package.json
index bcb18394d4d..dd8ce686678 100644
--- a/lib/api/package.json
+++ b/lib/api/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/api",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Core Storybook API & Context",
"keywords": [
"storybook"
@@ -14,17 +14,21 @@
"url": "https://github.com/storybookjs/storybook.git"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
"prepare": "node ./scripts/generateVersion.js && node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/channels": "5.2.0-beta.28",
- "@storybook/client-logger": "5.2.0-beta.28",
- "@storybook/core-events": "5.2.0-beta.28",
- "@storybook/router": "5.2.0-beta.28",
- "@storybook/theming": "5.2.0-beta.28",
+ "@storybook/channels": "5.3.0-alpha.0",
+ "@storybook/client-logger": "5.3.0-alpha.0",
+ "@storybook/core-events": "5.3.0-alpha.0",
+ "@storybook/router": "5.3.0-alpha.0",
+ "@storybook/theming": "5.3.0-alpha.0",
"core-js": "^3.0.1",
"fast-deep-equal": "^2.0.1",
"global": "^4.3.2",
diff --git a/lib/api/src/modules/shortcuts.ts b/lib/api/src/modules/shortcuts.ts
index c6f4ee46310..d78bebde5a6 100644
--- a/lib/api/src/modules/shortcuts.ts
+++ b/lib/api/src/modules/shortcuts.ts
@@ -118,8 +118,11 @@ export default function initShortcuts({ store }: Module) {
handleShortcutFeature(fullApi, feature) {
const {
layout: { isFullscreen, showNav, showPanel },
+ ui: { enableShortcuts },
} = store.getState();
-
+ if (!enableShortcuts) {
+ return;
+ }
switch (feature) {
case 'escape': {
if (isFullscreen) {
@@ -274,7 +277,6 @@ export default function initShortcuts({ store }: Module) {
event.target.getAttribute('contenteditable') !== null
);
}
-
// Listen for keydown events in the manager
document.addEventListener('keydown', (event: Event) => {
if (!focusInInput(event)) {
diff --git a/lib/api/src/version.ts b/lib/api/src/version.ts
index b4213934f7f..01130de2d2f 100644
--- a/lib/api/src/version.ts
+++ b/lib/api/src/version.ts
@@ -1 +1 @@
-export const version = '5.2.0-beta.28';
+export const version = '5.3.0-alpha.0';
diff --git a/lib/channel-postmessage/package.json b/lib/channel-postmessage/package.json
index 6725590aee5..b512795f212 100644
--- a/lib/channel-postmessage/package.json
+++ b/lib/channel-postmessage/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/channel-postmessage",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "",
"keywords": [
"storybook"
@@ -15,14 +15,18 @@
"directory": "lib/channel-postmessage"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/channels": "5.2.0-beta.28",
- "@storybook/client-logger": "5.2.0-beta.28",
+ "@storybook/channels": "5.3.0-alpha.0",
+ "@storybook/client-logger": "5.3.0-alpha.0",
"core-js": "^3.0.1",
"global": "^4.3.2",
"telejson": "^2.2.2"
diff --git a/lib/channel-websocket/package.json b/lib/channel-websocket/package.json
index 31aaf76446a..638701c3f3c 100644
--- a/lib/channel-websocket/package.json
+++ b/lib/channel-websocket/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/channel-websocket",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "",
"keywords": [
"storybook"
@@ -15,13 +15,17 @@
"directory": "lib/channel-websocket"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/channels": "5.2.0-beta.28",
+ "@storybook/channels": "5.3.0-alpha.0",
"core-js": "^3.0.1",
"global": "^4.3.2",
"json-fn": "^1.1.1"
diff --git a/lib/channels/package.json b/lib/channels/package.json
index f9c2841d419..b45ca5ae5c6 100644
--- a/lib/channels/package.json
+++ b/lib/channels/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/channels",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "",
"keywords": [
"storybook"
@@ -15,6 +15,10 @@
"directory": "lib/channels"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
diff --git a/lib/cli-sb/README.md b/lib/cli-sb/README.md
new file mode 100644
index 00000000000..3b6a3dc82ea
--- /dev/null
+++ b/lib/cli-sb/README.md
@@ -0,0 +1,3 @@
+# Storybook CLI
+
+This is a wrapper for https://www.npmjs.com/package/@storybook/cli
\ No newline at end of file
diff --git a/lib/cli-sb/index.js b/lib/cli-sb/index.js
new file mode 100755
index 00000000000..457ff863bb4
--- /dev/null
+++ b/lib/cli-sb/index.js
@@ -0,0 +1,3 @@
+#!/usr/bin/env node
+
+require('@storybook/cli/bin/index');
diff --git a/lib/cli-sb/package.json b/lib/cli-sb/package.json
new file mode 100644
index 00000000000..2159d628bb4
--- /dev/null
+++ b/lib/cli-sb/package.json
@@ -0,0 +1,31 @@
+{
+ "name": "sb",
+ "version": "5.3.0-alpha.0",
+ "private": true,
+ "description": "Storybook CLI",
+ "keywords": [
+ "storybook"
+ ],
+ "homepage": "https://github.com/storybookjs/storybook/tree/master/lib/cli",
+ "bugs": {
+ "url": "https://github.com/storybookjs/storybook/issues"
+ },
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/storybookjs/storybook.git",
+ "directory": "lib/cli"
+ },
+ "license": "MIT",
+ "bin": {
+ "sb": "./index.js"
+ },
+ "scripts": {
+ "prepare": "node ../../scripts/prepare.js"
+ },
+ "dependencies": {
+ "@storybook/cli": "5.3.0-alpha.0"
+ },
+ "publishConfig": {
+ "access": "public"
+ }
+}
diff --git a/lib/cli-storybook/README.md b/lib/cli-storybook/README.md
new file mode 100644
index 00000000000..3b6a3dc82ea
--- /dev/null
+++ b/lib/cli-storybook/README.md
@@ -0,0 +1,3 @@
+# Storybook CLI
+
+This is a wrapper for https://www.npmjs.com/package/@storybook/cli
\ No newline at end of file
diff --git a/lib/cli-storybook/index.js b/lib/cli-storybook/index.js
new file mode 100755
index 00000000000..457ff863bb4
--- /dev/null
+++ b/lib/cli-storybook/index.js
@@ -0,0 +1,3 @@
+#!/usr/bin/env node
+
+require('@storybook/cli/bin/index');
diff --git a/lib/cli-storybook/package.json b/lib/cli-storybook/package.json
new file mode 100644
index 00000000000..6a7af36abf4
--- /dev/null
+++ b/lib/cli-storybook/package.json
@@ -0,0 +1,32 @@
+{
+ "name": "storybook",
+ "version": "5.3.0-alpha.0",
+ "private": true,
+ "description": "Storybook CLI",
+ "keywords": [
+ "storybook"
+ ],
+ "homepage": "https://github.com/storybookjs/storybook/tree/master/lib/cli",
+ "bugs": {
+ "url": "https://github.com/storybookjs/storybook/issues"
+ },
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/storybookjs/storybook.git",
+ "directory": "lib/cli"
+ },
+ "license": "MIT",
+ "bin": {
+ "sb": "./index.js",
+ "storybook": "./index.js"
+ },
+ "scripts": {
+ "prepare": "node ../../scripts/prepare.js"
+ },
+ "dependencies": {
+ "@storybook/cli": "5.3.0-alpha.0"
+ },
+ "publishConfig": {
+ "access": "public"
+ }
+}
diff --git a/lib/cli/bin/index.js b/lib/cli/bin/index.js
index 306555fa5b0..136e6d73d60 100755
--- a/lib/cli/bin/index.js
+++ b/lib/cli/bin/index.js
@@ -1,9 +1,3 @@
#!/usr/bin/env node
-const path = require('path');
-require('@babel/register')({
- // see https://github.com/babel/babel/issues/7701#issuecomment-389720069
- cwd: path.resolve(__dirname, '..'),
- only: [new RegExp(`(@storybook|lib)${path.sep === '\\' ? '\\\\' : path.sep}cli`)],
-});
-require('./generate');
+require('esm')(module)('./generate.js');
diff --git a/lib/cli/generators/ANGULAR/template/src/stories/Welcome.stories.ts b/lib/cli/generators/ANGULAR/template/src/stories/0-Welcome.stories.ts
similarity index 100%
rename from lib/cli/generators/ANGULAR/template/src/stories/Welcome.stories.ts
rename to lib/cli/generators/ANGULAR/template/src/stories/0-Welcome.stories.ts
diff --git a/lib/cli/generators/ANGULAR/template/src/stories/Button.stories.ts b/lib/cli/generators/ANGULAR/template/src/stories/1-Button.stories.ts
similarity index 100%
rename from lib/cli/generators/ANGULAR/template/src/stories/Button.stories.ts
rename to lib/cli/generators/ANGULAR/template/src/stories/1-Button.stories.ts
diff --git a/lib/cli/generators/EMBER/template/stories/Welcome.stories.js b/lib/cli/generators/EMBER/template/stories/0-Welcome.stories.js
similarity index 90%
rename from lib/cli/generators/EMBER/template/stories/Welcome.stories.js
rename to lib/cli/generators/EMBER/template/stories/0-Welcome.stories.js
index b7db119a380..e3eaa5ba782 100644
--- a/lib/cli/generators/EMBER/template/stories/Welcome.stories.js
+++ b/lib/cli/generators/EMBER/template/stories/0-Welcome.stories.js
@@ -1,6 +1,5 @@
/* eslint-disable import/extensions */
import hbs from 'htmlbars-inline-precompile';
-import { action } from '@storybook/addon-actions';
import { linkTo } from '@storybook/addon-links';
export default {
diff --git a/lib/cli/generators/EMBER/template/stories/Button.stories.js b/lib/cli/generators/EMBER/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/EMBER/template/stories/Button.stories.js
rename to lib/cli/generators/EMBER/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/METEOR/template/stories/Welcome.stories.js b/lib/cli/generators/METEOR/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/METEOR/template/stories/Welcome.stories.js
rename to lib/cli/generators/METEOR/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/METEOR/template/stories/Button.stories.js b/lib/cli/generators/METEOR/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/METEOR/template/stories/Button.stories.js
rename to lib/cli/generators/METEOR/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/MITHRIL/template/stories/Welcome.stories.js b/lib/cli/generators/MITHRIL/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/MITHRIL/template/stories/Welcome.stories.js
rename to lib/cli/generators/MITHRIL/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/MITHRIL/template/stories/Button.stories.js b/lib/cli/generators/MITHRIL/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/MITHRIL/template/stories/Button.stories.js
rename to lib/cli/generators/MITHRIL/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/PREACT/template/stories/Welcome.stories.js b/lib/cli/generators/PREACT/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/PREACT/template/stories/Welcome.stories.js
rename to lib/cli/generators/PREACT/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/PREACT/template/stories/Button.stories.js b/lib/cli/generators/PREACT/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/PREACT/template/stories/Button.stories.js
rename to lib/cli/generators/PREACT/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/RAX/template/.storybook/config.js b/lib/cli/generators/RAX/template/.storybook/config.js
index 76f54a31448..d964ae12046 100644
--- a/lib/cli/generators/RAX/template/.storybook/config.js
+++ b/lib/cli/generators/RAX/template/.storybook/config.js
@@ -8,7 +8,6 @@ addParameters({
showAddonsPanel: true,
showSearchBox: false,
addonPanelInRight: true,
- sortStoriesByKind: false,
hierarchySeparator: /\./,
hierarchyRootSeparator: /\|/,
enableShortcuts: true,
diff --git a/lib/cli/generators/RAX/template/stories/Welcome.stories.js b/lib/cli/generators/RAX/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/RAX/template/stories/Welcome.stories.js
rename to lib/cli/generators/RAX/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/RAX/template/stories/Button.stories.js b/lib/cli/generators/RAX/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/RAX/template/stories/Button.stories.js
rename to lib/cli/generators/RAX/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/REACT/template/stories/Welcome.stories.js b/lib/cli/generators/REACT/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/REACT/template/stories/Welcome.stories.js
rename to lib/cli/generators/REACT/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/REACT/template/stories/Button.stories.js b/lib/cli/generators/REACT/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/REACT/template/stories/Button.stories.js
rename to lib/cli/generators/REACT/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/REACT_SCRIPTS/template/src/stories/Welcome.stories.js b/lib/cli/generators/REACT_SCRIPTS/template/src/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/REACT_SCRIPTS/template/src/stories/Welcome.stories.js
rename to lib/cli/generators/REACT_SCRIPTS/template/src/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/REACT_SCRIPTS/template/src/stories/Button.stories.js b/lib/cli/generators/REACT_SCRIPTS/template/src/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/REACT_SCRIPTS/template/src/stories/Button.stories.js
rename to lib/cli/generators/REACT_SCRIPTS/template/src/stories/1-Button.stories.js
diff --git a/lib/cli/generators/RIOT/template/stories/Welcome.stories.js b/lib/cli/generators/RIOT/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/RIOT/template/stories/Welcome.stories.js
rename to lib/cli/generators/RIOT/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/RIOT/template/stories/Button.stories.js b/lib/cli/generators/RIOT/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/RIOT/template/stories/Button.stories.js
rename to lib/cli/generators/RIOT/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/SFC_VUE/template/src/stories/Welcome.stories.js b/lib/cli/generators/SFC_VUE/template/src/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/SFC_VUE/template/src/stories/Welcome.stories.js
rename to lib/cli/generators/SFC_VUE/template/src/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/SFC_VUE/template/src/stories/Button.stories.js b/lib/cli/generators/SFC_VUE/template/src/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/SFC_VUE/template/src/stories/Button.stories.js
rename to lib/cli/generators/SFC_VUE/template/src/stories/1-Button.stories.js
diff --git a/lib/cli/generators/VUE/template/stories/Welcome.stories.js b/lib/cli/generators/VUE/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/VUE/template/stories/Welcome.stories.js
rename to lib/cli/generators/VUE/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/VUE/template/stories/Button.stories.js b/lib/cli/generators/VUE/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/VUE/template/stories/Button.stories.js
rename to lib/cli/generators/VUE/template/stories/1-Button.stories.js
diff --git a/lib/cli/generators/WEBPACK_REACT/template/stories/Welcome.stories.js b/lib/cli/generators/WEBPACK_REACT/template/stories/0-Welcome.stories.js
similarity index 100%
rename from lib/cli/generators/WEBPACK_REACT/template/stories/Welcome.stories.js
rename to lib/cli/generators/WEBPACK_REACT/template/stories/0-Welcome.stories.js
diff --git a/lib/cli/generators/WEBPACK_REACT/template/stories/Button.stories.js b/lib/cli/generators/WEBPACK_REACT/template/stories/1-Button.stories.js
similarity index 100%
rename from lib/cli/generators/WEBPACK_REACT/template/stories/Button.stories.js
rename to lib/cli/generators/WEBPACK_REACT/template/stories/1-Button.stories.js
diff --git a/lib/cli/package.json b/lib/cli/package.json
index ad9bb182df4..4d1c261b133 100644
--- a/lib/cli/package.json
+++ b/lib/cli/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/cli",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Storybook's CLI - easiest method of adding storybook to your projects",
"keywords": [
"cli",
@@ -18,6 +18,13 @@
},
"license": "MIT",
"author": "Storybook Team",
+ "files": [
+ "bin/**/*",
+ "docs/**/*",
+ "generators/**/*",
+ "lib/**/*",
+ "README.md"
+ ],
"bin": {
"getstorybook": "./bin/index.js",
"sb": "./bin/index.js"
@@ -29,14 +36,14 @@
"dependencies": {
"@babel/core": "^7.4.5",
"@babel/preset-env": "^7.4.5",
- "@babel/register": "^7.0.0",
- "@storybook/codemod": "5.2.0-beta.28",
+ "@storybook/codemod": "5.3.0-alpha.0",
"chalk": "^2.4.1",
"commander": "^2.19.0",
"core-js": "^3.0.1",
"cross-spawn": "^6.0.5",
"didyoumean": "^1.2.1",
"envinfo": "^7.3.1",
+ "esm": "3.2.25",
"fs-extra": "^8.0.1",
"inquirer": "^6.2.0",
"jscodeshift": "^0.6.3",
@@ -48,33 +55,33 @@
"update-notifier": "^3.0.0"
},
"devDependencies": {
- "@storybook/addon-actions": "5.2.0-beta.28",
- "@storybook/addon-centered": "5.2.0-beta.28",
- "@storybook/addon-graphql": "5.2.0-beta.28",
- "@storybook/addon-info": "5.2.0-beta.28",
- "@storybook/addon-knobs": "5.2.0-beta.28",
- "@storybook/addon-links": "5.2.0-beta.28",
- "@storybook/addon-notes": "5.2.0-beta.28",
- "@storybook/addon-options": "5.2.0-beta.28",
- "@storybook/addon-storyshots": "5.2.0-beta.28",
- "@storybook/addons": "5.2.0-beta.28",
- "@storybook/angular": "5.2.0-beta.28",
- "@storybook/channel-postmessage": "5.2.0-beta.28",
- "@storybook/channel-websocket": "5.2.0-beta.28",
- "@storybook/channels": "5.2.0-beta.28",
- "@storybook/ember": "5.2.0-beta.28",
- "@storybook/html": "5.2.0-beta.28",
- "@storybook/marko": "5.2.0-beta.28",
- "@storybook/mithril": "5.2.0-beta.28",
- "@storybook/polymer": "5.2.0-beta.28",
- "@storybook/preact": "5.2.0-beta.28",
- "@storybook/rax": "5.2.0-beta.28",
- "@storybook/react": "5.2.0-beta.28",
- "@storybook/react-native": "5.2.0-beta.28",
- "@storybook/riot": "5.2.0-beta.28",
- "@storybook/svelte": "5.2.0-beta.28",
- "@storybook/ui": "5.2.0-beta.28",
- "@storybook/vue": "5.2.0-beta.28"
+ "@storybook/addon-actions": "5.3.0-alpha.0",
+ "@storybook/addon-centered": "5.3.0-alpha.0",
+ "@storybook/addon-graphql": "5.3.0-alpha.0",
+ "@storybook/addon-info": "5.3.0-alpha.0",
+ "@storybook/addon-knobs": "5.3.0-alpha.0",
+ "@storybook/addon-links": "5.3.0-alpha.0",
+ "@storybook/addon-notes": "5.3.0-alpha.0",
+ "@storybook/addon-options": "5.3.0-alpha.0",
+ "@storybook/addon-storyshots": "5.3.0-alpha.0",
+ "@storybook/addons": "5.3.0-alpha.0",
+ "@storybook/angular": "5.3.0-alpha.0",
+ "@storybook/channel-postmessage": "5.3.0-alpha.0",
+ "@storybook/channel-websocket": "5.3.0-alpha.0",
+ "@storybook/channels": "5.3.0-alpha.0",
+ "@storybook/ember": "5.3.0-alpha.0",
+ "@storybook/html": "5.3.0-alpha.0",
+ "@storybook/marko": "5.3.0-alpha.0",
+ "@storybook/mithril": "5.3.0-alpha.0",
+ "@storybook/polymer": "5.3.0-alpha.0",
+ "@storybook/preact": "5.3.0-alpha.0",
+ "@storybook/rax": "5.3.0-alpha.0",
+ "@storybook/react": "5.3.0-alpha.0",
+ "@storybook/react-native": "5.3.0-alpha.0",
+ "@storybook/riot": "5.3.0-alpha.0",
+ "@storybook/svelte": "5.3.0-alpha.0",
+ "@storybook/ui": "5.3.0-alpha.0",
+ "@storybook/vue": "5.3.0-alpha.0"
},
"publishConfig": {
"access": "public"
diff --git a/lib/cli/story.js b/lib/cli/story.js
new file mode 100644
index 00000000000..2a64488eda1
--- /dev/null
+++ b/lib/cli/story.js
@@ -0,0 +1,10 @@
+export default {
+ title: 'Story',
+};
+
+export const startCase = () => 'foo';
+export const camelCase = () => 'foo';
+
+camelCase.story = {
+ name: 'camelCase',
+};
diff --git a/lib/cli/test/run_tests.sh b/lib/cli/test/run_tests.sh
index 34908300c65..8f7c0c7ad28 100755
--- a/lib/cli/test/run_tests.sh
+++ b/lib/cli/test/run_tests.sh
@@ -36,10 +36,10 @@ do
if [ $dir == *"native"* ]
then
# run @storybook/cli
- ../../../bin/index.js init --skip-install --yes --install-server
+ yarn sb init --skip-install --yes --install-server
else
# run @storybook/cli
- ../../../bin/index.js init --skip-install --yes
+ yarn sb init --skip-install --yes
fi
cd ..
diff --git a/lib/client-api/package.json b/lib/client-api/package.json
index e60f1bd7c90..fba6e58b494 100644
--- a/lib/client-api/package.json
+++ b/lib/client-api/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/client-api",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Storybook Client API",
"keywords": [
"storybook"
@@ -15,18 +15,22 @@
"directory": "lib/client-api"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "5.2.0-beta.28",
- "@storybook/channel-postmessage": "5.2.0-beta.28",
- "@storybook/channels": "5.2.0-beta.28",
- "@storybook/client-logger": "5.2.0-beta.28",
- "@storybook/core-events": "5.2.0-beta.28",
- "@storybook/router": "5.2.0-beta.28",
+ "@storybook/addons": "5.3.0-alpha.0",
+ "@storybook/channel-postmessage": "5.3.0-alpha.0",
+ "@storybook/channels": "5.3.0-alpha.0",
+ "@storybook/client-logger": "5.3.0-alpha.0",
+ "@storybook/core-events": "5.3.0-alpha.0",
+ "@storybook/router": "5.3.0-alpha.0",
"common-tags": "^1.8.0",
"core-js": "^3.0.1",
"eventemitter3": "^4.0.0",
diff --git a/lib/client-api/src/client_api.test.ts b/lib/client-api/src/client_api.test.ts
index 95c9e2c5858..a4cd692c184 100644
--- a/lib/client-api/src/client_api.test.ts
+++ b/lib/client-api/src/client_api.test.ts
@@ -1,12 +1,13 @@
/* eslint-disable no-underscore-dangle */
import { logger } from '@storybook/client-logger';
-import { mockChannel } from '@storybook/addons';
+import addons, { mockChannel } from '@storybook/addons';
import ClientApi from './client_api';
import ConfigApi from './config_api';
import StoryStore from './story_store';
export const getContext = (() => decorateStory => {
const channel = mockChannel();
+ addons.setChannel(channel);
const storyStore = new StoryStore({ channel });
const clientApi = new ClientApi({ storyStore, decorateStory });
const { clearDecorators } = clientApi;
@@ -120,7 +121,7 @@ describe('preview.client_api', () => {
clientApi.addParameters({ a: '1' });
// @ts-ignore
- expect(clientApi._globalParameters).toEqual({ a: '1', options: {} });
+ expect(clientApi._globalParameters).toEqual({ a: '1', options: {}, docs: {} });
});
it('should merge options', () => {
@@ -130,7 +131,7 @@ describe('preview.client_api', () => {
clientApi.addParameters({ options: { b: '2' } });
// @ts-ignore
- expect(clientApi._globalParameters).toEqual({ options: { a: '1', b: '2' } });
+ expect(clientApi._globalParameters).toEqual({ options: { a: '1', b: '2' }, docs: {} });
});
it('should override specific properties in options', () => {
@@ -143,6 +144,7 @@ describe('preview.client_api', () => {
expect(clientApi._globalParameters).toEqual({
backgrounds: ['value'],
options: { a: '2', b: '3' },
+ docs: {},
});
});
@@ -156,6 +158,7 @@ describe('preview.client_api', () => {
expect(clientApi._globalParameters).toEqual({
backgrounds: [],
options: { a: '2', b: '3' },
+ docs: {},
});
});
@@ -168,6 +171,7 @@ describe('preview.client_api', () => {
// @ts-ignore
expect(clientApi._globalParameters).toEqual({
options: { a: '1', b: '2', theming: { c: '4', d: '5' } },
+ docs: {},
});
});
});
@@ -453,6 +457,7 @@ describe('preview.client_api', () => {
c: 'story',
fileName: expect.any(String),
options: expect.any(Object),
+ docs: expect.any(Object),
});
});
@@ -470,6 +475,7 @@ describe('preview.client_api', () => {
sub: { global: true },
},
options: expect.any(Object),
+ docs: expect.any(Object),
});
storiesOf('kind', module)
@@ -507,6 +513,7 @@ describe('preview.client_api', () => {
},
fileName: expect.any(String),
options: expect.any(Object),
+ docs: expect.any(Object),
});
});
});
diff --git a/lib/client-api/src/client_api.ts b/lib/client-api/src/client_api.ts
index 84592f1c205..0050ead8477 100644
--- a/lib/client-api/src/client_api.ts
+++ b/lib/client-api/src/client_api.ts
@@ -134,6 +134,10 @@ export default class ClientApi {
options: {
...merge(get(this._globalParameters, 'options', {}), get(parameters, 'options', {})),
},
+ // FIXME: https://github.com/storybookjs/storybook/issues/7872
+ docs: {
+ ...merge(get(this._globalParameters, 'docs', {}), get(parameters, 'docs', {})),
+ },
};
};
diff --git a/lib/client-api/src/hooks.test.js b/lib/client-api/src/hooks.test.js
index 2705f56523e..b28b2f2aab3 100644
--- a/lib/client-api/src/hooks.test.js
+++ b/lib/client-api/src/hooks.test.js
@@ -1,4 +1,5 @@
-import { FORCE_RE_RENDER } from '@storybook/addon-contexts/src/shared/constants';
+import { FORCE_RE_RENDER, STORY_RENDERED } from '@storybook/core-events';
+import addons from '@storybook/addons';
import { defaultDecorateStory } from './client_api';
import {
applyHooks,
@@ -11,28 +12,48 @@ import {
useChannel,
useParameter,
useStoryContext,
+ HooksContext,
} from './hooks';
jest.mock('@storybook/client-logger', () => ({
logger: { warn: jest.fn(), log: jest.fn() },
}));
+const SOME_EVENT = 'someEvent';
let mockChannel;
+let hooks;
+let onSomeEvent;
+let removeSomeEventListener;
beforeEach(() => {
+ onSomeEvent = jest.fn();
+ removeSomeEventListener = jest.fn();
mockChannel = {
emit: jest.fn(),
- on: jest.fn(),
- removeListener: jest.fn(),
+ on(event, callback) {
+ switch (event) {
+ case STORY_RENDERED:
+ callback();
+ break;
+ case SOME_EVENT:
+ onSomeEvent(event, callback);
+ break;
+ default:
+ }
+ },
+ removeListener(event, callback) {
+ if (event === SOME_EVENT) {
+ removeSomeEventListener(event, callback);
+ }
+ },
};
+ hooks = new HooksContext();
+ addons.setChannel(mockChannel);
});
-jest.mock('@storybook/addons', () => ({
- getChannel: () => mockChannel,
-}));
-
const decorateStory = applyHooks(defaultDecorateStory);
-const run = (storyFn, decorators = [], context) => decorateStory(storyFn, decorators)(context);
+const run = (storyFn, decorators = [], context) =>
+ decorateStory(storyFn, decorators)({ ...context, hooks });
describe('Preview hooks', () => {
describe('useEffect', () => {
@@ -159,28 +180,28 @@ describe('Preview hooks', () => {
run(() => {}, [
storyFn => {
useChannel({
- SOME_EVENT: handler,
+ [SOME_EVENT]: handler,
});
return storyFn();
},
]);
- expect(mockChannel.on).toHaveBeenCalledTimes(1);
- expect(mockChannel.removeListener).toHaveBeenCalledTimes(0);
+ expect(onSomeEvent).toHaveBeenCalledTimes(1);
+ expect(removeSomeEventListener).toHaveBeenCalledTimes(0);
});
it('calls .removeListener when removing the decorator', () => {
const handler = () => {};
run(() => {}, [
storyFn => {
useChannel({
- SOME_EVENT: handler,
+ [SOME_EVENT]: handler,
});
return storyFn();
},
]);
- expect(mockChannel.on).toHaveBeenCalledTimes(1);
- expect(mockChannel.removeListener).toHaveBeenCalledTimes(0);
+ expect(onSomeEvent).toHaveBeenCalledTimes(1);
+ expect(removeSomeEventListener).toHaveBeenCalledTimes(0);
run(() => {});
- expect(mockChannel.removeListener).toHaveBeenCalledTimes(1);
+ expect(removeSomeEventListener).toHaveBeenCalledTimes(1);
});
});
describe('useStoryContext', () => {
@@ -188,7 +209,7 @@ describe('Preview hooks', () => {
const context = {};
run(
() => {
- expect(useStoryContext()).toBe(context);
+ expect(useStoryContext()).toEqual({ ...context, hooks });
},
[],
context
@@ -359,6 +380,19 @@ describe('Preview hooks', () => {
expect(storyFn).toHaveBeenCalledTimes(2);
expect(state).toBe('bar');
});
+ it('triggers only the last effect when updating state synchronously', () => {
+ const effects = [jest.fn(), jest.fn()];
+ const storyFn = jest.fn(() => {
+ const [effectIndex, setEffectIndex] = useState(0);
+ useEffect(effects[effectIndex], [effectIndex]);
+ if (effectIndex === 0) {
+ setEffectIndex(1);
+ }
+ });
+ run(storyFn);
+ expect(effects[0]).not.toHaveBeenCalled();
+ expect(effects[1]).toHaveBeenCalledTimes(1);
+ });
it('performs synchronous state updates with updater function', () => {
let state;
let setState;
diff --git a/lib/client-api/src/hooks.ts b/lib/client-api/src/hooks.ts
index c2ddad32082..3017db5d2af 100644
--- a/lib/client-api/src/hooks.ts
+++ b/lib/client-api/src/hooks.ts
@@ -1,323 +1,27 @@
-import { logger } from '@storybook/client-logger';
-import addons, { StoryGetter, StoryContext } from '@storybook/addons';
-import { FORCE_RE_RENDER } from '@storybook/core-events';
+import {
+ HooksContext,
+ applyHooks,
+ useMemo,
+ useCallback,
+ useRef,
+ useState,
+ useReducer,
+ useEffect,
+ useChannel,
+ useStoryContext,
+ useParameter,
+} from '@storybook/addons';
-interface StoryStore {
- fromId: (
- id: string
- ) => {
- parameters: {
- [parameterKey: string]: any;
- };
- };
- getSelection: () => {
- storyId: string;
- viewMode: string;
- };
-}
-
-interface Hook {
- name: string;
- memoizedState?: any;
- deps?: any[] | undefined;
-}
-
-interface Effect {
- create: () => (() => void) | void;
- destroy?: (() => void) | void;
-}
-
-type Decorator = (getStory: StoryGetter, context: StoryContext) => any;
-type AbstractFunction = (...args: any[]) => any;
-
-const hookListsMap = new WeakMap();
-let mountedDecorators = new Set();
-
-let currentHooks: Hook[] = [];
-let nextHookIndex = 0;
-const getNextHook = () => {
- const hook = currentHooks[nextHookIndex];
- nextHookIndex += 1;
- return hook;
+export {
+ HooksContext,
+ applyHooks,
+ useMemo,
+ useCallback,
+ useRef,
+ useState,
+ useReducer,
+ useEffect,
+ useChannel,
+ useStoryContext,
+ useParameter,
};
-let currentPhase: 'MOUNT' | 'UPDATE' | 'NONE' = 'NONE';
-let currentEffects: Effect[] = [];
-let prevEffects: Effect[] = [];
-let currentDecoratorName: string | null = null;
-let hasUpdates = false;
-let currentContext: StoryContext | null = null;
-
-const triggerEffects = () => {
- // destroy removed effects
- prevEffects.forEach(effect => {
- if (!currentEffects.includes(effect) && effect.destroy) {
- effect.destroy();
- }
- });
- // trigger added effects
- currentEffects.forEach(effect => {
- if (!prevEffects.includes(effect)) {
- // eslint-disable-next-line no-param-reassign
- effect.destroy = effect.create();
- }
- });
- prevEffects = currentEffects;
- currentEffects = [];
-};
-
-const hookify = (fn: AbstractFunction) => (...args: any[]) => {
- const prevPhase = currentPhase;
- const prevHooks = currentHooks;
- const prevNextHookIndex = nextHookIndex;
- const prevDecoratorName = currentDecoratorName;
-
- currentDecoratorName = fn.name;
- if (mountedDecorators.has(fn)) {
- currentPhase = 'UPDATE';
- currentHooks = hookListsMap.get(fn) || [];
- } else {
- currentPhase = 'MOUNT';
- currentHooks = [];
- hookListsMap.set(fn, currentHooks);
- }
- nextHookIndex = 0;
-
- const result = fn(...args);
-
- if (currentPhase === 'UPDATE' && getNextHook() != null) {
- throw new Error(
- 'Rendered fewer hooks than expected. This may be caused by an accidental early return statement.'
- );
- }
-
- currentPhase = prevPhase;
- currentHooks = prevHooks;
- nextHookIndex = prevNextHookIndex;
- currentDecoratorName = prevDecoratorName;
- return result;
-};
-
-// Counter to prevent infinite loops.
-let numberOfRenders = 0;
-const RENDER_LIMIT = 25;
-export const applyHooks = (
- applyDecorators: (getStory: StoryGetter, decorators: Decorator[]) => StoryGetter
-) => (getStory: StoryGetter, decorators: Decorator[]) => {
- const decorated = applyDecorators(hookify(getStory), decorators.map(hookify));
- return (context: StoryContext) => {
- currentContext = context;
- hasUpdates = false;
- let result = decorated(context);
- mountedDecorators = new Set([getStory, ...decorators]);
- numberOfRenders = 1;
- while (hasUpdates) {
- hasUpdates = false;
- result = decorated(context);
- numberOfRenders += 1;
- if (numberOfRenders > RENDER_LIMIT) {
- throw new Error(
- 'Too many re-renders. Storybook limits the number of renders to prevent an infinite loop.'
- );
- }
- }
- triggerEffects();
- currentContext = null;
- return result;
- };
-};
-
-const areDepsEqual = (deps: any[], nextDeps: any[]) =>
- deps.length === nextDeps.length && deps.every((dep, i) => dep === nextDeps[i]);
-
-function useHook(name: string, callback: (hook: Hook) => void, deps?: any[] | undefined): Hook {
- if (currentPhase === 'MOUNT') {
- if (deps != null && !Array.isArray(deps)) {
- logger.warn(
- `${name} received a final argument that is not an array (instead, received ${deps}). When specified, the final argument must be an array.`
- );
- }
- const hook: Hook = { name, deps };
- currentHooks.push(hook);
- callback(hook);
- return hook;
- }
-
- if (currentPhase === 'UPDATE') {
- const hook = getNextHook();
- if (hook == null) {
- throw new Error('Rendered more hooks than during the previous render.');
- }
-
- if (hook.name !== name) {
- logger.warn(
- `Storybook has detected a change in the order of Hooks${
- currentDecoratorName ? ` called by ${currentDecoratorName}` : ''
- }. This will lead to bugs and errors if not fixed.`
- );
- }
-
- if (deps != null && hook.deps == null) {
- logger.warn(
- `${name} received a final argument during this render, but not during the previous render. Even though the final argument is optional, its type cannot change between renders.`
- );
- }
-
- if (deps != null && hook.deps != null && deps.length !== hook.deps.length) {
- logger.warn(`The final argument passed to ${name} changed size between renders. The order and size of this array must remain constant.
-Previous: ${hook.deps}
-Incoming: ${deps}`);
- }
-
- if (deps == null || hook.deps == null || !areDepsEqual(deps, hook.deps)) {
- callback(hook);
- hook.deps = deps;
- }
- return hook;
- }
-
- throw new Error(
- 'Storybook preview hooks can only be called inside decorators and story functions.'
- );
-}
-
-function useMemoLike(name: string, nextCreate: () => T, deps: any[] | undefined): T {
- const { memoizedState } = useHook(
- name,
- hook => {
- // eslint-disable-next-line no-param-reassign
- hook.memoizedState = nextCreate();
- },
- deps
- );
- return memoizedState;
-}
-
-/* Returns a memoized value, see https://reactjs.org/docs/hooks-reference.html#usememo */
-export function useMemo(nextCreate: () => T, deps?: any[]): T {
- return useMemoLike('useMemo', nextCreate, deps);
-}
-
-/* Returns a memoized callback, see https://reactjs.org/docs/hooks-reference.html#usecallback */
-export function useCallback(callback: T, deps?: any[]): T {
- return useMemoLike('useCallback', () => callback, deps);
-}
-
-function useRefLike(name: string, initialValue: T): { current: T } {
- return useMemoLike(name, () => ({ current: initialValue }), []);
-}
-
-/* Returns a mutable ref object, see https://reactjs.org/docs/hooks-reference.html#useref */
-export function useRef(initialValue: T): { current: T } {
- return useRefLike('useRef', initialValue);
-}
-
-function triggerUpdate() {
- // Rerun getStory if updates were triggered synchronously, force rerender otherwise
- if (currentPhase !== 'NONE') {
- hasUpdates = true;
- } else {
- try {
- addons.getChannel().emit(FORCE_RE_RENDER);
- } catch (e) {
- logger.warn('State updates of Storybook preview hooks work only in browser');
- }
- }
-}
-
-function useStateLike(
- name: string,
- initialState: (() => S) | S
-): [S, (update: ((prevState: S) => S) | S) => void] {
- const stateRef = useRefLike(
- name,
- // @ts-ignore S type should never be function, but there's no way to tell that to TypeScript
- typeof initialState === 'function' ? initialState() : initialState
- );
- const setState = (update: ((prevState: S) => S) | S) => {
- // @ts-ignore S type should never be function, but there's no way to tell that to TypeScript
- stateRef.current = typeof update === 'function' ? update(stateRef.current) : update;
- triggerUpdate();
- };
- return [stateRef.current, setState];
-}
-
-/* Returns a stateful value, and a function to update it, see https://reactjs.org/docs/hooks-reference.html#usestate */
-export function useState(
- initialState: (() => S) | S
-): [S, (update: ((prevState: S) => S) | S) => void] {
- return useStateLike('useState', initialState);
-}
-
-/* A redux-like alternative to useState, see https://reactjs.org/docs/hooks-reference.html#usereducer */
-export function useReducer(
- reducer: (state: S, action: A) => S,
- initialState: S
-): [S, (action: A) => void];
-export function useReducer(
- reducer: (state: S, action: A) => S,
- initialArg: I,
- init: (initialArg: I) => S
-): [S, (action: A) => void];
-export function useReducer(
- reducer: (state: S, action: A) => S,
- initialArg: any,
- init?: any
-): [S, (action: A) => void] {
- const initialState: (() => S) | S = init != null ? () => init(initialArg) : initialArg;
- const [state, setState] = useStateLike('useReducer', initialState);
- const dispatch = (action: A) => setState(prevState => reducer(prevState, action));
- return [state, dispatch];
-}
-
-/*
- Triggers a side effect, see https://reactjs.org/docs/hooks-reference.html#usestate
- Effects are triggered synchronously after calling the decorated story function
-*/
-export function useEffect(create: () => (() => void) | void, deps?: any[]): void {
- const effect = useMemoLike('useEffect', () => ({ create }), deps);
- currentEffects.push(effect);
-}
-
-export interface Listener {
- (...args: any[]): void;
- ignorePeer?: boolean;
-}
-
-export interface EventMap {
- [eventId: string]: Listener;
-}
-
-/* Accepts a map of Storybook channel event listeners, returns an emit function */
-export function useChannel(eventMap: EventMap, deps: any[] = []) {
- const channel = addons.getChannel();
- useEffect(() => {
- Object.entries(eventMap).forEach(([type, listener]) => channel.on(type, listener));
- return () => {
- Object.entries(eventMap).forEach(([type, listener]) =>
- channel.removeListener(type, listener)
- );
- };
- }, [...Object.keys(eventMap), ...deps]);
-
- return channel.emit.bind(channel);
-}
-
-/* Returns current story context */
-export function useStoryContext(): StoryContext {
- if (currentContext == null) {
- throw new Error(
- 'Storybook preview hooks can only be called inside decorators and story functions.'
- );
- }
-
- return currentContext;
-}
-
-/* Returns current value of a story parameter */
-export function useParameter(parameterKey: string, defaultValue?: S): S | undefined {
- const { parameters } = useStoryContext();
- if (parameterKey) {
- return parameters[parameterKey] || (defaultValue as S);
- }
- return undefined;
-}
diff --git a/lib/client-api/src/story_store.test.ts b/lib/client-api/src/story_store.test.ts
index 118acee1d9d..51b913b8e7d 100644
--- a/lib/client-api/src/story_store.test.ts
+++ b/lib/client-api/src/story_store.test.ts
@@ -158,12 +158,14 @@ describe('preview.story_store', () => {
const storyWithContext = store.getStoryWithContext('kind', 'name');
storyWithContext();
+ const { hooks } = store.fromId(toId('kind', 'name'));
expect(storyFn).toHaveBeenCalledWith({
id: 'kind--name',
name: 'name',
kind: 'kind',
story: 'name',
parameters,
+ hooks,
});
});
});
diff --git a/lib/client-api/src/story_store.ts b/lib/client-api/src/story_store.ts
index b00994d0ab3..80aaa37d5ae 100644
--- a/lib/client-api/src/story_store.ts
+++ b/lib/client-api/src/story_store.ts
@@ -17,6 +17,7 @@ import {
StoreItem,
ErrorLike,
} from './types';
+import { HooksContext } from './hooks';
// TODO: these are copies from components/nav/lib
// refactor to DRY
@@ -205,16 +206,20 @@ export default class StoryStore extends EventEmitter {
applyDecorators(getOriginal(), getDecorators())
);
+ const hooks = new HooksContext();
+
const storyFn: StoryFn = p =>
getDecorated()({
...identification,
...p,
+ hooks,
parameters: { ...parameters, ...(p && p.parameters) },
});
_data[id] = {
...identification,
+ hooks,
getDecorated,
getOriginal,
storyFn,
@@ -299,6 +304,10 @@ export default class StoryStore extends EventEmitter {
.map(info => info.name);
}
+ getStoriesForKind(kind: string) {
+ return this.raw().filter(story => story.kind === kind);
+ }
+
getStoryFileName(kind: string) {
const key = toKey(kind);
const storiesKind = this._legacydata[key as string];
@@ -383,4 +392,12 @@ export default class StoryStore extends EventEmitter {
clean() {
this.getStoryKinds().forEach(kind => delete this._legacydata[toKey(kind) as string]);
}
+
+ cleanHooks(id: string) {
+ this._data[id].hooks.clean();
+ }
+
+ cleanHooksForKind(kind: string) {
+ this.getStoriesForKind(kind).map(story => this.cleanHooks(story.id));
+ }
}
diff --git a/lib/client-api/src/types.ts b/lib/client-api/src/types.ts
index 06d405628c1..d509750fb6c 100644
--- a/lib/client-api/src/types.ts
+++ b/lib/client-api/src/types.ts
@@ -7,6 +7,7 @@ import {
DecoratorFunction,
} from '@storybook/addons';
import StoryStore from './story_store';
+import { HooksContext } from './hooks';
export interface ErrorLike {
message: string;
@@ -18,6 +19,7 @@ export interface StoreItem extends StoryContext {
getOriginal: () => StoryFn;
story: string;
storyFn: StoryFn;
+ hooks: HooksContext;
}
export interface StoreData {
diff --git a/lib/client-logger/package.json b/lib/client-logger/package.json
index 754771e43a4..c813e49aa7a 100644
--- a/lib/client-logger/package.json
+++ b/lib/client-logger/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/client-logger",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "",
"keywords": [
"storybook"
@@ -15,6 +15,10 @@
"directory": "lib/client-logger"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
diff --git a/lib/codemod/README.md b/lib/codemod/README.md
index be3e3f4bd05..d3cd22b840d 100644
--- a/lib/codemod/README.md
+++ b/lib/codemod/README.md
@@ -10,13 +10,13 @@ The preferred way to run these codemods is via the CLI's `migrate` command.
To get a list of available codemods:
```
-npx -p @storybook/cli sb migrate --list
+npx -p @storybook/cli@next sb migrate --list
```
To run a codemod ``:
```
-npx -p @storybook/cli sb migrate --glob "**/*.stories.js"
+npx -p @storybook/cli@next sb migrate --glob "**/*.stories.js"
```
## Installation
diff --git a/lib/codemod/package.json b/lib/codemod/package.json
index e9154511c9e..e0fb0164abe 100644
--- a/lib/codemod/package.json
+++ b/lib/codemod/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/codemod",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "A collection of codemod scripts written with JSCodeshift",
"keywords": [
"storybook"
@@ -15,6 +15,10 @@
"directory": "lib/codemod"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"jsnext:main": "src/index.js",
"scripts": {
@@ -22,7 +26,8 @@
},
"dependencies": {
"@mdx-js/mdx": "^1.0.0",
- "@storybook/node-logger": "5.2.0-beta.28",
+ "@storybook/node-logger": "5.3.0-alpha.0",
+ "@storybook/router": "5.3.0-alpha.0",
"core-js": "^3.0.1",
"cross-spawn": "^6.0.5",
"globby": "^10.0.1",
diff --git a/lib/codemod/src/lib/utils.js b/lib/codemod/src/lib/utils.js
index 822774440a2..4d782569a21 100644
--- a/lib/codemod/src/lib/utils.js
+++ b/lib/codemod/src/lib/utils.js
@@ -9,5 +9,9 @@ export const sanitizeName = name => {
if (isReserved(key)) {
key = `${key}Story`;
}
+ // prepend _ if name starts with a digit
+ if (/^\d/.test(key)) {
+ key = `_${key}`;
+ }
return key;
};
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.input.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.input.js
index 459203a0429..17c8f6abc1a 100644
--- a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.input.js
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.input.js
@@ -14,4 +14,5 @@ storiesOf('Button', module)
))
- .add('w/punctuation', () => );
+ .add('w/punctuation', () => )
+ .add('Start Case', () => );
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.output.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.output.js
index f5e86149cc1..a678481c047 100644
--- a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.output.js
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/basic.output.js
@@ -10,6 +10,11 @@ export default {
};
export const story1 = () => ;
+
+story1.story = {
+ name: 'story1',
+};
+
export const secondStory = () => ;
secondStory.story = {
@@ -32,3 +37,5 @@ export const wPunctuation = () => ;
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.input.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.input.js
index 72e64decea8..232cce44a42 100644
--- a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.input.js
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.input.js
@@ -1,2 +1,11 @@
export const foo = 1;
-storiesOf('bar', module).add('foo', () => );
+const bar = 1;
+const barStory = 1;
+const baz = 1;
+const bazStory1 = 1;
+
+storiesOf('foo', module)
+ .add('foo', () => )
+ .add('bar', () => )
+ .add('bazStory', () => )
+ .add('baz', () => );
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.output.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.output.js
index d1300c5bd14..86f83d12732 100644
--- a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.output.js
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/collision.output.js
@@ -1,12 +1,34 @@
export const foo = 1;
+const bar = 1;
+const barStory = 1;
+const baz = 1;
+const bazStory1 = 1;
export default {
- title: 'bar',
+ title: 'foo',
excludeStories: ['foo'],
};
-export const story0 = () => ;
+export const fooStory = () => ;
-story0.story = {
+fooStory.story = {
name: 'foo',
};
+
+export const barStory1 = () => ;
+
+barStory1.story = {
+ name: 'bar',
+};
+
+export const bazStory = () => ;
+
+bazStory.story = {
+ name: 'bazStory',
+};
+
+export const bazStory2 = () => ;
+
+bazStory2.story = {
+ name: 'baz',
+};
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/digit.input.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/digit.input.js
new file mode 100644
index 00000000000..9931f13df1f
--- /dev/null
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/digit.input.js
@@ -0,0 +1 @@
+storiesOf('bar', module).add('1', () => );
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/digit.output.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/digit.output.js
new file mode 100644
index 00000000000..0dfb48fb260
--- /dev/null
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/digit.output.js
@@ -0,0 +1,5 @@
+export default {
+ title: 'bar',
+};
+
+export const _1 = () => ;
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/exports.output.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/exports.output.js
index 8cf839c793e..57f84925325 100644
--- a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/exports.output.js
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/exports.output.js
@@ -6,3 +6,7 @@ export default {
};
export const baz = () => ;
+
+baz.story = {
+ name: 'baz',
+};
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/multi.output.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/multi.output.js
index ff1fefb3d2b..588710a8a59 100644
--- a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/multi.output.js
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/multi.output.js
@@ -7,11 +7,29 @@ export default {
};
export const story1 = () => ;
+
+story1.story = {
+ name: 'story1',
+};
+
export const story2 = () => ;
+story2.story = {
+ name: 'story2',
+};
+
export default {
title: 'Button2',
};
-export const story1 = () => ;
-export const story2 = () => ;
+export const story1Story = () => ;
+
+story1Story.story = {
+ name: 'story1',
+};
+
+export const story2Story = () => ;
+
+story2Story.story = {
+ name: 'story2',
+};
diff --git a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/story-parameters.output.js b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/story-parameters.output.js
index 6d0b9a5afe4..9aedd776c4f 100644
--- a/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/story-parameters.output.js
+++ b/lib/codemod/src/transforms/__testfixtures__/storiesof-to-csf/story-parameters.output.js
@@ -20,6 +20,8 @@ withStoryParameters.story = {
export const foo = () => ;
foo.story = {
+ name: 'foo',
+
parameters: {
bar: 1,
},
diff --git a/lib/codemod/src/transforms/__tests__/storiesof-to-csf.test.js b/lib/codemod/src/transforms/__tests__/storiesof-to-csf.test.js
index 73c8b648689..05f60dd9e76 100644
--- a/lib/codemod/src/transforms/__tests__/storiesof-to-csf.test.js
+++ b/lib/codemod/src/transforms/__tests__/storiesof-to-csf.test.js
@@ -1,21 +1,14 @@
+import fs from 'fs';
+import path from 'path';
+
import { defineTest } from 'jscodeshift/dist/testUtils';
jest.mock('@storybook/node-logger');
-const testNames = [
- 'basic',
- 'decorators',
- 'parameters',
- 'story-parameters',
- 'module',
- 'multi',
- 'default',
- 'exports',
- 'collision',
- 'const',
- 'story-decorators',
-];
-
-testNames.forEach(testName => {
- defineTest(__dirname, `storiesof-to-csf`, null, `storiesof-to-csf/${testName}`);
-});
+fs.readdirSync(path.resolve(__dirname, '../__testfixtures__/storiesof-to-csf'))
+ .map(filename => filename.match(/^(.*)\.input\.js$/))
+ .filter(Boolean)
+ .map(match => match[1])
+ .forEach(testName => {
+ defineTest(__dirname, `storiesof-to-csf`, null, `storiesof-to-csf/${testName}`);
+ });
diff --git a/lib/codemod/src/transforms/storiesof-to-csf.js b/lib/codemod/src/transforms/storiesof-to-csf.js
index 1fe45368471..fca733b0a40 100644
--- a/lib/codemod/src/transforms/storiesof-to-csf.js
+++ b/lib/codemod/src/transforms/storiesof-to-csf.js
@@ -1,5 +1,6 @@
import prettier from 'prettier';
import { logger } from '@storybook/node-logger';
+import { storyNameFromExport } from '@storybook/router';
import { sanitizeName } from '../lib/utils';
/**
@@ -44,7 +45,7 @@ export default function transformer(file, api, options) {
return { storyParams, storyDecorators };
}
- function convertToModuleExports(path, originalExports, counter) {
+ function convertToModuleExports(path, originalExports) {
const base = j(path);
const statements = [];
@@ -127,13 +128,19 @@ export default function transformer(file, api, options) {
adds.reverse();
adds.push(path);
+ const identifiers = new Set();
+ root.find(j.Identifier).forEach(({ value }) => identifiers.add(value.name));
adds.forEach(add => {
let name = add.node.arguments[0].value;
- let key = sanitizeName(name);
- if (originalExports.includes(key)) {
- key = `story${counter}`;
+ const sanitized = sanitizeName(name);
+ let key = sanitized;
+ let counter = 0;
+ while (identifiers.has(key)) {
+ key = `${sanitized}Story${counter || ''}`;
+ counter += 1;
}
- if (key === name) {
+ identifiers.add(key);
+ if (storyNameFromExport(key) === name) {
name = null;
}
@@ -209,7 +216,7 @@ export default function transformer(file, api, options) {
.filter(add => add.node.callee.property && add.node.callee.property.name === 'add')
.filter(add => add.node.arguments.length >= 2 && add.node.arguments[0].type === 'Literal')
.filter(add => add.parentPath.node.type === 'ExpressionStatement')
- .forEach((path, i) => convertToModuleExports(path, originalExports, i));
+ .forEach(path => convertToModuleExports(path, originalExports));
// remove storiesOf import
root
diff --git a/lib/components/html.d.ts b/lib/components/html.d.ts
new file mode 100644
index 00000000000..7d31127d99f
--- /dev/null
+++ b/lib/components/html.d.ts
@@ -0,0 +1 @@
+export * from './dist/html.d';
diff --git a/lib/components/html.js b/lib/components/html.js
new file mode 100644
index 00000000000..4f7edc6bedb
--- /dev/null
+++ b/lib/components/html.js
@@ -0,0 +1 @@
+module.exports = require('./dist/html');
diff --git a/lib/components/package.json b/lib/components/package.json
index d96c54870fc..80629c61051 100644
--- a/lib/components/package.json
+++ b/lib/components/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/components",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Core Storybook Components",
"keywords": [
"storybook"
@@ -15,14 +15,20 @@
"directory": "lib/components"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md",
+ "html.d.ts",
+ "html.js"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/client-logger": "5.2.0-beta.28",
- "@storybook/theming": "5.2.0-beta.28",
+ "@storybook/client-logger": "5.3.0-alpha.0",
+ "@storybook/theming": "5.3.0-alpha.0",
"@types/react-syntax-highlighter": "10.1.0",
"core-js": "^3.0.1",
"global": "^4.3.2",
diff --git a/lib/components/src/ActionBar/ActionBar.tsx b/lib/components/src/ActionBar/ActionBar.tsx
index 3b566348aff..d0e233d299f 100644
--- a/lib/components/src/ActionBar/ActionBar.tsx
+++ b/lib/components/src/ActionBar/ActionBar.tsx
@@ -12,42 +12,51 @@ const Container = styled.div<{}>(({ theme }) => ({
zIndex: 1,
}));
-export const ActionButton = styled.button<{}>(({ theme }) => ({
- border: '0 none',
- padding: '4px 10px',
- cursor: 'pointer',
- display: 'flex',
- alignItems: 'center',
+export const ActionButton = styled.button<{ disabled: boolean }>(
+ ({ theme }) => ({
+ border: '0 none',
+ padding: '4px 10px',
+ cursor: 'pointer',
+ display: 'flex',
+ alignItems: 'center',
- color: theme.color.defaultText,
- background: theme.background.content,
+ color: theme.color.defaultText,
+ background: theme.background.content,
- fontSize: 12,
- lineHeight: '16px',
- fontWeight: theme.typography.weight.bold,
+ fontSize: 12,
+ lineHeight: '16px',
+ fontFamily: theme.typography.fonts.base,
+ fontWeight: theme.typography.weight.bold,
- borderTop: `1px solid ${theme.appBorderColor}`,
- borderLeft: `1px solid ${theme.appBorderColor}`,
- marginLeft: -1,
-
- borderRadius: `4px 0 0 0`,
-
- '&:not(:last-child)': { borderRight: `1px solid ${theme.appBorderColor}` },
- '& + *': {
+ borderTop: `1px solid ${theme.appBorderColor}`,
borderLeft: `1px solid ${theme.appBorderColor}`,
- borderRadius: 0,
- },
+ marginLeft: -1,
- '&:focus': {
- boxShadow: `${theme.color.secondary} 0 -3px 0 0 inset`,
- outline: '0 none',
- },
-}));
+ borderRadius: `4px 0 0 0`,
+
+ '&:not(:last-child)': { borderRight: `1px solid ${theme.appBorderColor}` },
+ '& + *': {
+ borderLeft: `1px solid ${theme.appBorderColor}`,
+ borderRadius: 0,
+ },
+
+ '&:focus': {
+ boxShadow: `${theme.color.secondary} 0 -3px 0 0 inset`,
+ outline: '0 none',
+ },
+ }),
+ ({ disabled }) =>
+ disabled && {
+ cursor: 'not-allowed',
+ opacity: 0.5,
+ }
+);
ActionButton.displayName = 'ActionButton';
export interface ActionItem {
title: string | JSX.Element;
onClick: (e: React.MouseEvent) => void;
+ disabled?: boolean;
}
export interface ActionBarProps {
@@ -56,9 +65,9 @@ export interface ActionBarProps {
export const ActionBar: FunctionComponent = ({ actionItems, ...props }) => (
- {actionItems.map(({ title, onClick }, index: number) => (
+ {actionItems.map(({ title, onClick, disabled }, index: number) => (
// eslint-disable-next-line react/no-array-index-key
-
+
{title}
))}
diff --git a/lib/components/src/blocks/ColorPalette.stories.tsx b/lib/components/src/blocks/ColorPalette.stories.tsx
index 6f619847de8..876f48f9477 100644
--- a/lib/components/src/blocks/ColorPalette.stories.tsx
+++ b/lib/components/src/blocks/ColorPalette.stories.tsx
@@ -1,12 +1,9 @@
import React from 'react';
import { ColorItem, ColorPalette } from './ColorPalette';
-import { DocsPageWrapper } from './DocsPage';
-
export default {
title: 'Docs|ColorPalette',
component: ColorPalette,
- decorators: [getStory => {getStory()} ],
};
export const defaultStyle = () => (
diff --git a/lib/components/src/blocks/ColorPalette.tsx b/lib/components/src/blocks/ColorPalette.tsx
index 8796a29fd69..ec9580e7663 100644
--- a/lib/components/src/blocks/ColorPalette.tsx
+++ b/lib/components/src/blocks/ColorPalette.tsx
@@ -3,9 +3,11 @@ import { styled } from '@storybook/theming';
import { transparentize } from 'polished';
import { getBlockBackgroundStyle } from './BlockBackgroundStyles';
+import { ResetWrapper } from '../typography/DocumentFormatting';
const ItemTitle = styled.div<{}>(({ theme }) => ({
fontWeight: theme.typography.weight.bold,
+ color: theme.color.defaultText,
}));
const ItemSubtitle = styled.div<{}>(({ theme }) => ({
@@ -91,17 +93,19 @@ const ListHeading = styled.div<{}>(({ theme }) => ({
alignItems: 'center',
paddingBottom: 20,
fontWeight: theme.typography.weight.bold,
-
color:
theme.base === 'light'
? transparentize(0.4, theme.color.defaultText)
: transparentize(0.6, theme.color.defaultText),
}));
-const List = styled.div({
+const List = styled.div<{}>(({ theme }) => ({
+ fontSize: theme.typography.size.s2,
+ lineHeight: `20px`,
+
display: 'flex',
flexDirection: 'column',
-});
+}));
interface ColorProps {
title: string;
@@ -151,14 +155,14 @@ export const ColorItem: React.FunctionComponent = ({ title, subtitle
* Styleguide documentation for colors, including names, captions, and color swatches,
* all specified as `ColorItem` children of this wrapper component.
*/
-export const ColorPalette: React.FunctionComponent = ({ children, ...props }) => {
- return (
-
+export const ColorPalette: React.FunctionComponent = ({ children, ...props }) => (
+
+
Name
Swatches
{children}
- );
-};
+
+);
diff --git a/lib/components/src/blocks/Description.stories.tsx b/lib/components/src/blocks/Description.stories.tsx
index 7f45da999b8..60eafb77a72 100644
--- a/lib/components/src/blocks/Description.stories.tsx
+++ b/lib/components/src/blocks/Description.stories.tsx
@@ -1,13 +1,11 @@
import React from 'react';
import { Description } from './Description';
-import { DocsPageWrapper } from './DocsPage';
import markdownCaption from './DocsPageExampleCaption.md';
export default {
title: 'Docs|Description',
component: Description,
- decorators: [getStory => {getStory()} ],
};
const textCaption = `That was Wintermute, manipulating the lock the way it had manipulated the drone micro and the amplified breathing of the room where Case waited. The semiotics of the bright void beyond the chain link. The tug Marcus Garvey, a steel drum nine meters long and two in diameter, creaked and shuddered as Maelcum punched for a California gambling cartel, then as a paid killer in the dark, curled in his capsule in some coffin hotel, his hands clawed into the nearest door and watched the other passengers as he rode. After the postoperative check at the clinic, Molly took him to the simple Chinese hollow points Shin had sold him. Still it was a handgun and nine rounds of ammunition, and as he made his way down Shiga from the missionaries, the train reached Case’s station. Now this quiet courtyard, Sunday afternoon, this girl with a random collection of European furniture, as though Deane had once intended to use the place as his home. Case felt the edge of the Flatline as a construct, a hardwired ROM cassette replicating a dead man’s skills, obsessions, kneejerk responses. They were dropping, losing altitude in a canyon of rainbow foliage, a lurid communal mural that completely covered the hull of the console in faded pinks and yellows.`;
diff --git a/lib/components/src/blocks/Description.tsx b/lib/components/src/blocks/Description.tsx
index bfc51d88eaf..90d1c5994a8 100644
--- a/lib/components/src/blocks/Description.tsx
+++ b/lib/components/src/blocks/Description.tsx
@@ -1,5 +1,7 @@
import React from 'react';
import Markdown from 'markdown-to-jsx';
+import { ResetWrapper } from '../typography/DocumentFormatting';
+import { components } from '../html';
export interface DescriptionProps {
markdown: string;
@@ -10,5 +12,7 @@ export interface DescriptionProps {
* components docgen docs.
*/
export const Description: React.FunctionComponent = ({ markdown }) => (
- {markdown}
+
+ {markdown}
+
);
diff --git a/lib/components/src/blocks/DocsPage.stories.tsx b/lib/components/src/blocks/DocsPage.stories.tsx
index 30c4c5b2c07..efcfc2554d9 100644
--- a/lib/components/src/blocks/DocsPage.stories.tsx
+++ b/lib/components/src/blocks/DocsPage.stories.tsx
@@ -18,6 +18,19 @@ export default {
],
};
+export const withSubtitle = () => (
+
+ {descriptionStories.text()}
+ {previewStories.single()}
+ {propsTableStories.normal()}
+ {sourceStories.jsx()}
+
+);
+withSubtitle.story = { name: 'with subtitle' };
+
export const empty = () => (
{storyStories.error()}
@@ -44,19 +57,6 @@ export const text = () => (
);
-export const withSubtitle = () => (
-
- {descriptionStories.text()}
- {previewStories.single()}
- {propsTableStories.normal()}
- {sourceStories.jsx()}
-
-);
-withSubtitle.story = { name: 'with subtitle' };
-
export const markdown = () => (
{descriptionStories.markdown()}
diff --git a/lib/components/src/blocks/DocsPage.tsx b/lib/components/src/blocks/DocsPage.tsx
index 3d4907467ab..cfc789770ca 100644
--- a/lib/components/src/blocks/DocsPage.tsx
+++ b/lib/components/src/blocks/DocsPage.tsx
@@ -2,7 +2,7 @@ import React from 'react';
import { styled } from '@storybook/theming';
import { transparentize } from 'polished';
-import { DocumentFormatting } from '../typography/DocumentFormatting';
+import { withReset } from '../typography/withReset';
const breakpoint = 600;
const pageMargin = '5.55555';
@@ -12,33 +12,30 @@ export interface DocsPageProps {
subtitle?: string;
}
-const Title = styled.h1<{}>(({ theme }) => ({
- // overrides h1 in DocumentFormatting
- '&&': {
- fontSize: theme.typography.size.m3,
- lineHeight: '32px',
+const Title = styled.h1<{}>(withReset, ({ theme }) => ({
+ color: theme.color.defaultText,
+ fontSize: theme.typography.size.m3,
+ fontWeight: theme.typography.weight.black,
+ lineHeight: '32px',
- [`@media (min-width: ${breakpoint * 1}px)`]: {
- fontSize: theme.typography.size.l1,
- lineHeight: '36px',
- },
+ [`@media (min-width: ${breakpoint * 1}px)`]: {
+ fontSize: theme.typography.size.l1,
+ lineHeight: '36px',
+ marginBottom: '.5rem', // 8px
},
}));
-const Subtitle = styled.h2<{}>(({ theme }) => ({
- // overrides h2 in DocumentFormatting
- '&&': {
- fontWeight: theme.typography.weight.regular,
- fontSize: theme.typography.size.s3,
- lineHeight: '20px',
- borderBottom: 'none',
- marginBottom: '15px',
+const Subtitle = styled.h2<{}>(withReset, ({ theme }) => ({
+ fontWeight: theme.typography.weight.regular,
+ fontSize: theme.typography.size.s3,
+ lineHeight: '20px',
+ borderBottom: 'none',
+ marginBottom: '15px',
- [`@media (min-width: ${breakpoint * 1}px)`]: {
- fontSize: theme.typography.size.m1,
- lineHeight: '28px',
- marginBottom: '25px',
- },
+ [`@media (min-width: ${breakpoint * 1}px)`]: {
+ fontSize: theme.typography.size.m1,
+ lineHeight: '28px',
+ marginBottom: '24px',
},
color:
@@ -47,7 +44,7 @@ const Subtitle = styled.h2<{}>(({ theme }) => ({
: transparentize(0.25, theme.color.defaultText),
}));
-export const DocsContent = styled(DocumentFormatting)({
+export const DocsContent = styled.div({
maxWidth: 800,
width: '100%',
});
@@ -75,8 +72,8 @@ export const DocsPageWrapper: React.FunctionComponent = ({ children }) => (
*/
const DocsPage: React.FunctionComponent = ({ title, subtitle, children }) => (
<>
- {title && {title} }
- {subtitle && {subtitle} }
+ {title && {title} }
+ {subtitle && {subtitle} }
{children}
>
);
diff --git a/lib/components/src/blocks/EmptyBlock.stories.tsx b/lib/components/src/blocks/EmptyBlock.stories.tsx
index 4ea61e3071f..8af0ad71c12 100644
--- a/lib/components/src/blocks/EmptyBlock.stories.tsx
+++ b/lib/components/src/blocks/EmptyBlock.stories.tsx
@@ -1,12 +1,9 @@
import React from 'react';
import { EmptyBlock } from './EmptyBlock';
-import { DocsPageWrapper } from './DocsPage';
-
export default {
title: 'Docs|EmptyBlock',
component: EmptyBlock,
- decorators: [getStory => {getStory()} ],
};
export const error = () => Generic error message ;
diff --git a/lib/components/src/blocks/EmptyBlock.tsx b/lib/components/src/blocks/EmptyBlock.tsx
index db4ecf43ec6..e5ed7bbafe6 100644
--- a/lib/components/src/blocks/EmptyBlock.tsx
+++ b/lib/components/src/blocks/EmptyBlock.tsx
@@ -1,12 +1,14 @@
import React, { FunctionComponent } from 'react';
import { styled } from '@storybook/theming';
import { transparentize } from 'polished';
+import { withReset } from '../typography/withReset';
-const Wrapper = styled.div<{}>(({ theme }) => ({
+const Wrapper = styled.div<{}>(withReset, ({ theme }) => ({
backgroundColor: theme.base === 'light' ? 'rgba(0,0,0,.01)' : 'rgba(255,255,255,.01)',
borderRadius: theme.appBorderRadius,
border: `1px dashed ${theme.appBorderColor}`,
display: 'flex',
+ fontSize: `${theme.typography.size.s2}px`,
alignItems: 'center',
justifyContent: 'center',
padding: '20px',
@@ -18,4 +20,4 @@ const Wrapper = styled.div<{}>(({ theme }) => ({
: transparentize(0.6, theme.color.defaultText),
}));
-export const EmptyBlock = (props: any) => ;
+export const EmptyBlock = (props: any) => ;
diff --git a/lib/components/src/blocks/IconGallery.stories.tsx b/lib/components/src/blocks/IconGallery.stories.tsx
index 07960847264..4758be85598 100644
--- a/lib/components/src/blocks/IconGallery.stories.tsx
+++ b/lib/components/src/blocks/IconGallery.stories.tsx
@@ -1,13 +1,11 @@
import React from 'react';
import { IconItem, IconGallery } from './IconGallery';
-import { DocsPageWrapper } from './DocsPage';
import { Icons as ExampleIcon } from '../icon/icon';
export default {
title: 'Docs|IconGallery',
component: IconGallery,
- decorators: [getStory => {getStory()} ],
};
export const defaultStyle = () => (
diff --git a/lib/components/src/blocks/IconGallery.tsx b/lib/components/src/blocks/IconGallery.tsx
index 22ceb329c48..6d0e604265d 100644
--- a/lib/components/src/blocks/IconGallery.tsx
+++ b/lib/components/src/blocks/IconGallery.tsx
@@ -1,12 +1,15 @@
import React from 'react';
import { styled } from '@storybook/theming';
+import { ResetWrapper } from '../typography/DocumentFormatting';
import { getBlockBackgroundStyle } from './BlockBackgroundStyles';
-const ItemLabel = styled.div({
+const ItemLabel = styled.div<{}>(({ theme }) => ({
+ fontFamily: theme.typography.fonts.base,
+ fontSize: theme.typography.size.s2,
marginLeft: 10,
lineHeight: 1.2,
-});
+}));
const ItemSpecimen = styled.div<{}>(({ theme }) => ({
...getBlockBackgroundStyle(theme),
@@ -59,5 +62,11 @@ export const IconItem: React.FunctionComponent = ({ name, childre
* Show a grid of icons, as specified by `IconItem`.
*/
export const IconGallery: React.FunctionComponent = ({ children, ...props }) => {
- return {children}
;
+ return (
+
+
+ {children}
+
+
+ );
};
diff --git a/lib/components/src/blocks/Preview.stories.tsx b/lib/components/src/blocks/Preview.stories.tsx
index 1513daa2741..2b89cf16186 100644
--- a/lib/components/src/blocks/Preview.stories.tsx
+++ b/lib/components/src/blocks/Preview.stories.tsx
@@ -1,15 +1,14 @@
import React from 'react';
-import { action } from '@storybook/addon-actions';
import { Preview } from './Preview';
-import { DocsPageWrapper } from './DocsPage';
+
+import { Story } from './Story';
import { Button } from '../Button/Button';
import * as sourceStories from './Source.stories';
export default {
title: 'Docs|Preview',
component: Preview,
- decorators: [getStory => {getStory()} ],
};
export const codeCollapsed = () => (
@@ -80,3 +79,18 @@ export const gridWith3Columns = () => (
Button 20
);
+
+const buttonFn = () => Hello Button ;
+
+export const withToolbar = () => (
+
+
+
+);
+
+export const withToolbarMulti = () => (
+
+
+
+
+);
diff --git a/lib/components/src/blocks/Preview.tsx b/lib/components/src/blocks/Preview.tsx
index 49e275a4162..e91bd9c6ddf 100644
--- a/lib/components/src/blocks/Preview.tsx
+++ b/lib/components/src/blocks/Preview.tsx
@@ -1,16 +1,20 @@
import React from 'react';
import { styled } from '@storybook/theming';
import { darken } from 'polished';
+import { logger } from '@storybook/client-logger';
import { getBlockBackgroundStyle } from './BlockBackgroundStyles';
import { Source, SourceProps } from './Source';
-import { ActionBar } from '../ActionBar/ActionBar';
+import { ActionBar, ActionItem } from '../ActionBar/ActionBar';
+import { Toolbar } from './Toolbar';
+import { ZoomContext } from './ZoomContext';
export interface PreviewProps {
isColumn?: boolean;
columns?: number;
withSource?: SourceProps;
isExpanded?: boolean;
+ withToolbar?: boolean;
}
const ChildrenContainer = styled.div(({ isColumn, columns }) => ({
@@ -30,6 +34,8 @@ const StyledSource = styled(Source)<{}>(({ theme }) => ({
margin: 0,
borderTopLeftRadius: 0,
borderTopRightRadius: 0,
+ borderBottomLeftRadius: theme.appBorderRadius,
+ borderBottomRightRadius: theme.appBorderRadius,
border: 'none',
background:
@@ -41,21 +47,71 @@ const StyledSource = styled(Source)<{}>(({ theme }) => ({
},
}));
-const PreviewWrapper = styled.div(({ theme, withSource }) => ({
- ...getBlockBackgroundStyle(theme),
- padding: '30px 20px',
- position: 'relative',
- borderBottomLeftRadius: withSource && 0,
- borderBottomRightRadius: withSource && 0,
-}));
+const PreviewWrapper = styled.div(
+ ({ theme, withSource, isExpanded }) => ({
+ ...getBlockBackgroundStyle(theme),
+ padding: '30px 20px',
+ position: 'relative',
+ overflow: 'hidden',
+ borderBottomLeftRadius: withSource && isExpanded && 0,
+ borderBottomRightRadius: withSource && isExpanded && 0,
+ borderBottomWidth: isExpanded && 0,
+ }),
+ ({ withToolbar }) => withToolbar && { paddingTop: 64 }
+);
const PreviewContainer = styled.div({
margin: '25px 0 40px',
});
+interface SourceItem {
+ source?: React.ReactElement;
+ actionItem: ActionItem;
+}
+
+const getSource = (
+ withSource: SourceProps,
+ expanded: boolean,
+ setExpanded: Function
+): SourceItem => {
+ switch (true) {
+ case !!(withSource && withSource.error): {
+ return {
+ source: null,
+ actionItem: {
+ title: 'No code available',
+ disabled: true,
+ onClick: () => setExpanded(false),
+ },
+ };
+ }
+ case expanded: {
+ return {
+ source: ,
+ actionItem: { title: 'Hide code', onClick: () => setExpanded(false) },
+ };
+ }
+ default: {
+ return {
+ source: null,
+ actionItem: { title: 'Show code', onClick: () => setExpanded(true) },
+ };
+ }
+ }
+};
+function getStoryId(children: React.ReactNode) {
+ if (React.Children.count(children) === 1) {
+ const elt = children as React.ReactElement;
+ if (elt.props) {
+ return elt.props.id;
+ }
+ }
+ return null;
+}
+
/**
* A preview component for showing one or more component `Story`
- * items. The preview also shows the source for the componnent
+ * items. The preview also shows the source for the component
* as a drop-down.
*/
const Preview: React.FunctionComponent = ({
@@ -63,29 +119,39 @@ const Preview: React.FunctionComponent = ({
columns,
children,
withSource,
+ withToolbar = false,
isExpanded = false,
...props
}) => {
const [expanded, setExpanded] = React.useState(isExpanded);
- const { source, actionItem } = expanded
- ? {
- source: ,
- actionItem: { title: 'Hide code', onClick: () => setExpanded(false) },
- }
- : {
- source: null,
- actionItem: { title: 'Show code', onClick: () => setExpanded(true) },
- };
+ const { source, actionItem } = getSource(withSource, expanded, setExpanded);
+ const [scale, setScale] = React.useState(1);
+
+ if (withToolbar && Array.isArray(children)) {
+ logger.warn('Cannot use toolbar with multiple preview children, disabling');
+ }
+ const showToolbar = withToolbar && !Array.isArray(children);
return (
-
-
- {Array.isArray(children) ? (
- children.map((child, i) => {child}
)
- ) : (
- {children}
- )}
-
+
+ {showToolbar && (
+ setScale(scale * z)}
+ resetZoom={() => setScale(1)}
+ storyId={getStoryId(children)}
+ baseUrl="./iframe.html"
+ />
+ )}
+
+
+ {Array.isArray(children) ? (
+ children.map((child, i) => {child}
)
+ ) : (
+ {children}
+ )}
+
+
{withSource && }
{withSource && source}
diff --git a/lib/components/src/blocks/PropsTable/PropRow.stories.tsx b/lib/components/src/blocks/PropsTable/PropRow.stories.tsx
index e244e0b77b5..04b0e6f98b5 100644
--- a/lib/components/src/blocks/PropsTable/PropRow.stories.tsx
+++ b/lib/components/src/blocks/PropsTable/PropRow.stories.tsx
@@ -1,8 +1,7 @@
import React from 'react';
import { PropRow } from './PropRow';
-
import { Table } from './PropsTable';
-import { DocsPageWrapper } from '../DocsPage';
+import { ResetWrapper } from '../../typography/DocumentFormatting';
export default {
component: PropRow,
@@ -10,11 +9,11 @@ export default {
excludeStories: /.*Def$/,
decorators: [
getStory => (
-
+
-
+
),
],
};
diff --git a/lib/components/src/blocks/PropsTable/PropsTable.stories.tsx b/lib/components/src/blocks/PropsTable/PropsTable.stories.tsx
index f62a7a975b1..3a43436ae02 100644
--- a/lib/components/src/blocks/PropsTable/PropsTable.stories.tsx
+++ b/lib/components/src/blocks/PropsTable/PropsTable.stories.tsx
@@ -1,16 +1,14 @@
import React from 'react';
import { PropsTable, PropsTableError } from './PropsTable';
-import { DocsPageWrapper } from '../DocsPage';
import { stringDef, numberDef } from './PropRow.stories';
export default {
component: PropsTable,
title: 'Docs|PropTable',
- decorators: [storyFn => {storyFn()} ],
};
+export const normal = () => ;
+
export const error = () => ;
export const empty = () => ;
-
-export const normal = () => ;
diff --git a/lib/components/src/blocks/PropsTable/PropsTable.tsx b/lib/components/src/blocks/PropsTable/PropsTable.tsx
index b23ec02ef82..e462a57de7f 100644
--- a/lib/components/src/blocks/PropsTable/PropsTable.tsx
+++ b/lib/components/src/blocks/PropsTable/PropsTable.tsx
@@ -4,13 +4,14 @@ import { opacify, transparentize } from 'polished';
import { PropRow } from './PropRow';
import { PropDef } from './PropDef';
import { EmptyBlock } from '../EmptyBlock';
+import { ResetWrapper } from '../../typography/DocumentFormatting';
export const Table = styled.table<{}>(({ theme }) => ({
'&&': {
// Resets for cascading/system styles
borderCollapse: 'collapse',
borderSpacing: 0,
-
+ color: theme.color.defaultText,
tr: {
border: 'none',
background: 'none',
@@ -146,20 +147,22 @@ const PropsTable: React.FunctionComponent = props => {
return No props found for this component ;
}
return (
-
-
-
- Name
- Description
- Default
-
-
-
- {rows.map(row => (
-
- ))}
-
-
+
+
+
+
+ Name
+ Description
+ Default
+
+
+
+ {rows.map(row => (
+
+ ))}
+
+
+
);
};
diff --git a/lib/components/src/blocks/Source.stories.tsx b/lib/components/src/blocks/Source.stories.tsx
index 2166ce10ef3..bad3ea4acae 100644
--- a/lib/components/src/blocks/Source.stories.tsx
+++ b/lib/components/src/blocks/Source.stories.tsx
@@ -1,19 +1,11 @@
import React from 'react';
import { Source, SourceError } from './Source';
-import { DocsPageWrapper } from './DocsPage';
export default {
title: 'Docs|Source',
component: Source,
- decorators: [getStory => {getStory()} ],
};
-export const noStory = () => ;
-noStory.story = { name: 'no story' };
-
-export const sourceUnavailable = () => ;
-sourceUnavailable.story = { name: 'source unavailable' };
-
const jsxCode = `
a.id} />
@@ -38,3 +30,7 @@ const cssCode = `
`.trim();
export const css = () => ;
+
+export const noStory = () => ;
+
+export const sourceUnavailable = () => ;
diff --git a/lib/components/src/blocks/Source.tsx b/lib/components/src/blocks/Source.tsx
index dadd3d97364..34faf42c845 100644
--- a/lib/components/src/blocks/Source.tsx
+++ b/lib/components/src/blocks/Source.tsx
@@ -6,7 +6,10 @@ import { SyntaxHighlighter } from '../syntaxhighlighter/syntaxhighlighter';
const StyledSyntaxHighlighter = styled(SyntaxHighlighter)<{}>(({ theme }) => ({
// DocBlocks-specific styling and overrides
+ fontSize: `${theme.typography.size.s2 - 1}px`,
+ lineHeight: '19px',
margin: '25px 0 40px',
+ borderRadius: theme.appBorderRadius,
boxShadow:
theme.base === 'light' ? 'rgba(0, 0, 0, 0.10) 0 1px 3px 0' : 'rgba(0, 0, 0, 0.20) 0 2px 5px 0',
@@ -28,6 +31,7 @@ interface SourceErrorProps {
interface SourceCodeProps {
language?: string;
code?: string;
+ format?: boolean;
dark?: boolean;
}
@@ -44,12 +48,13 @@ const Source: React.FunctionComponent = props => {
return {error} ;
}
- const { language, code, dark, ...rest } = props as SourceCodeProps;
+ const { language, code, dark, format, ...rest } = props as SourceCodeProps;
const syntaxHighlighter = (
{getStory()} ],
};
+const buttonFn = () => Inline story ;
+
+const buttonHookFn = () => {
+ const [count, setCount] = React.useState(0);
+ return (
+ setCount(count + 1)}>
+ {`count: ${count}`}
+
+ );
+};
+
+export const inline = () => ;
+
export const error = () => ;
-const buttonFn = () => Hello Button ;
-
-export const inline = () => ;
+export const reactHook = () => ;
diff --git a/lib/components/src/blocks/Story.tsx b/lib/components/src/blocks/Story.tsx
index 0cdd022ab67..c098d043554 100644
--- a/lib/components/src/blocks/Story.tsx
+++ b/lib/components/src/blocks/Story.tsx
@@ -1,7 +1,7 @@
import React from 'react';
-
import { IFrame } from './IFrame';
import { EmptyBlock } from './EmptyBlock';
+import { ZoomContext } from './ZoomContext';
const BASE_URL = 'iframe.html';
@@ -9,18 +9,23 @@ export enum StoryError {
NO_STORY = 'No component or story to display',
}
+/** error message for Story with null storyFn
+ * if the story id exists, it must be pointing to a non-existing story
+ * if there is assigned story id, the story must be empty
+ */
+const MISSING_STORY = (id?: string) => (id ? `Story "${id}" doesn't exist.` : StoryError.NO_STORY);
+
interface CommonProps {
title: string;
height?: string;
+ id: string;
}
type InlineStoryProps = {
- storyFn: () => React.ElementType;
+ storyFn: React.ElementType;
} & CommonProps;
-type IFrameStoryProps = {
- id: string;
-} & CommonProps;
+type IFrameStoryProps = CommonProps;
type ErrorProps = {
error?: StoryError;
@@ -31,8 +36,33 @@ export type StoryProps = (InlineStoryProps | IFrameStoryProps | ErrorProps) & {
inline: boolean;
};
-const InlineStory: React.FunctionComponent = ({ storyFn, height }) => (
- {storyFn()}
+const InlineZoomWrapper: React.FC<{ scale: number }> = ({ scale, children }) => {
+ return scale === 1 ? (
+ <>{children}>
+ ) : (
+
+ );
+};
+
+const InlineStory: React.FunctionComponent = ({ storyFn, height, id }) => (
+
+
+ {({ scale }) => (
+
+ {storyFn ? React.createElement(storyFn) : {MISSING_STORY(id)} }
+
+ )}
+
+
);
const IFrameStory: React.FunctionComponent = ({
@@ -41,19 +71,25 @@ const IFrameStory: React.FunctionComponent = ({
height = '500px',
}) => (
-
);
@@ -64,14 +100,13 @@ const IFrameStory: React.FunctionComponent = ({
const Story: React.FunctionComponent = props => {
const { error } = props as ErrorProps;
const { storyFn } = props as InlineStoryProps;
- const { id } = props as IFrameStoryProps;
- const { inline, title, height } = props;
+ const { id, inline, title, height } = props;
if (error) {
return {error} ;
}
return inline ? (
-
+
) : (
);
diff --git a/lib/components/src/blocks/Toolbar.tsx b/lib/components/src/blocks/Toolbar.tsx
new file mode 100644
index 00000000000..30277f4fac8
--- /dev/null
+++ b/lib/components/src/blocks/Toolbar.tsx
@@ -0,0 +1,85 @@
+import React, { Fragment } from 'react';
+import { styled } from '@storybook/theming';
+
+import { window } from 'global';
+import { FlexBar } from '../bar/bar';
+import { Icons } from '../icon/icon';
+import { IconButton } from '../bar/button';
+
+interface ZoomProps {
+ zoom: (val: number) => void;
+ resetZoom: () => void;
+}
+
+interface EjectProps {
+ storyId?: string;
+ baseUrl?: string;
+}
+
+interface BarProps {
+ border?: boolean;
+}
+
+export type ToolbarProps = BarProps & ZoomProps & EjectProps;
+
+const Zoom: React.FC = ({ zoom, resetZoom }) => (
+ <>
+ {
+ e.preventDefault();
+ zoom(0.8);
+ }}
+ title="Zoom in"
+ >
+
+
+ {
+ e.preventDefault();
+ zoom(1.25);
+ }}
+ title="Zoom out"
+ >
+
+
+ {
+ e.preventDefault();
+ resetZoom();
+ }}
+ title="Reset zoom"
+ >
+
+
+ >
+);
+
+const Eject: React.FC = ({ baseUrl, storyId }) => (
+ window.open(`${baseUrl}?id=${storyId}`)}
+ title="Open canvas in new tab"
+ >
+
+
+);
+
+const Bar = styled(props => )({
+ position: 'absolute',
+ left: 0,
+ right: 0,
+ top: 0,
+ transition: 'transform .2s linear',
+});
+
+export const Toolbar: React.FC = ({ storyId, baseUrl, zoom, resetZoom, ...rest }) => (
+
+
+
+
+ {storyId && }
+
+);
diff --git a/lib/components/src/blocks/Typeset.stories.tsx b/lib/components/src/blocks/Typeset.stories.tsx
index 3383bf7ac20..31c48023044 100644
--- a/lib/components/src/blocks/Typeset.stories.tsx
+++ b/lib/components/src/blocks/Typeset.stories.tsx
@@ -1,12 +1,9 @@
import React from 'react';
import { Typeset } from './Typeset';
-import { DocsPageWrapper } from './DocsPage';
-
export default {
title: 'Docs|Typeset',
component: Typeset,
- decorators: [getStory => {getStory()} ],
};
const fontSizes = [12, 14, 16, 20, 24, 32, 40, 48];
diff --git a/lib/components/src/blocks/Typeset.tsx b/lib/components/src/blocks/Typeset.tsx
index 6cd034fd5f9..053263bac90 100644
--- a/lib/components/src/blocks/Typeset.tsx
+++ b/lib/components/src/blocks/Typeset.tsx
@@ -2,6 +2,7 @@ import React from 'react';
import { styled } from '@storybook/theming';
import { transparentize } from 'polished';
+import { withReset } from '../typography/withReset';
import { getBlockBackgroundStyle } from './BlockBackgroundStyles';
const Label = styled.div<{}>(({ theme }) => ({
@@ -27,7 +28,7 @@ const TypeSpecimen = styled.div({
'&:not(:last-child)': { marginBottom: '1rem' },
});
-const Wrapper = styled.div<{}>(({ theme }) => ({
+const Wrapper = styled.div<{}>(withReset, ({ theme }) => ({
...getBlockBackgroundStyle(theme),
margin: '25px 0 40px',
padding: '30px 20px',
@@ -49,7 +50,7 @@ export const Typeset: React.FunctionComponent = ({
sampleText,
...props
}) => (
-
+
{fontSizes.map(num => (
{num}px
diff --git a/lib/components/src/blocks/ZoomContext.tsx b/lib/components/src/blocks/ZoomContext.tsx
new file mode 100644
index 00000000000..46f35eb3039
--- /dev/null
+++ b/lib/components/src/blocks/ZoomContext.tsx
@@ -0,0 +1,5 @@
+import React from 'react';
+
+export const ZoomContext: React.Context<{ scale: number }> = React.createContext({
+ scale: 1,
+});
diff --git a/lib/components/src/html.tsx b/lib/components/src/html.tsx
new file mode 100644
index 00000000000..65e3a4af0be
--- /dev/null
+++ b/lib/components/src/html.tsx
@@ -0,0 +1,14 @@
+import React from 'react';
+import * as rawComponents from './typography/DocumentFormatting';
+
+export * from './typography/DocumentFormatting';
+
+export const components = Object.entries(rawComponents).reduce(
+ (acc, [k, V]) => ({
+ ...acc,
+ [k.toLowerCase()]: (props: object) => (
+
+ ),
+ }),
+ {}
+);
diff --git a/lib/components/src/index.ts b/lib/components/src/index.ts
index 9d59301baed..8768052d31c 100644
--- a/lib/components/src/index.ts
+++ b/lib/components/src/index.ts
@@ -2,7 +2,7 @@ export { Badge } from './Badge/Badge';
// Typography
export { Link } from './typography/link/link';
-export { DocumentFormatting } from './typography/DocumentFormatting';
+export { DocumentWrapper } from './typography/DocumentWrapper';
export { SyntaxHighlighter } from './syntaxhighlighter/syntaxhighlighter';
// UI
diff --git a/lib/components/src/tabs/tabs.tsx b/lib/components/src/tabs/tabs.tsx
index 21a3ef6cd09..6423a603a76 100644
--- a/lib/components/src/tabs/tabs.tsx
+++ b/lib/components/src/tabs/tabs.tsx
@@ -13,6 +13,9 @@ import { Placeholder } from '../placeholder/placeholder';
import { FlexBar } from '../bar/bar';
import { TabButton } from '../bar/button';
+const ignoreSsrWarning =
+ '/* emotion-disable-server-rendering-unsafe-selector-warning-please-do-not-use-this-the-warning-exists-for-a-reason */';
+
export interface WrapperProps {
bordered?: boolean;
absolute?: boolean;
@@ -73,7 +76,7 @@ const Content = styled.div(
bottom: 0,
top: 40,
overflow: 'auto',
- '& > *:first-child': {
+ [`& > *:first-child${ignoreSsrWarning}`]: {
position: 'absolute',
left: 0,
right: 0,
diff --git a/lib/components/src/tooltip/TooltipLinkList.tsx b/lib/components/src/tooltip/TooltipLinkList.tsx
index b1e38fe0848..439d972795a 100644
--- a/lib/components/src/tooltip/TooltipLinkList.tsx
+++ b/lib/components/src/tooltip/TooltipLinkList.tsx
@@ -7,6 +7,8 @@ const List = styled.div<{}>(
{
minWidth: 180,
overflow: 'hidden',
+ overflowY: 'auto',
+ maxHeight: 10.5 * 32, // 10.5 items
},
({ theme }) => ({
borderRadius: theme.appBorderRadius * 2,
diff --git a/lib/components/src/typography/DocumentFormatting.tsx b/lib/components/src/typography/DocumentFormatting.tsx
index 1e5a0f56f08..675036b365f 100644
--- a/lib/components/src/typography/DocumentFormatting.tsx
+++ b/lib/components/src/typography/DocumentFormatting.tsx
@@ -1,444 +1,355 @@
-import React from 'react';
-import { styled, css } from '@storybook/theming';
-
-const Wrapper = styled.div(
- props => css`
- /* Custom styles atop GitHub base theme (see below) */
- font-size: ${props.theme.typography.size.s2}px;
- line-height: 1.6;
-
- h1 {
- font-size: ${props.theme.typography.size.l1}px;
- font-weight: ${props.theme.typography.weight.black};
- }
-
- h2 {
- font-size: ${props.theme.typography.size.m2}px;
- border-bottom: 1px solid ${props.theme.appBorderColor};
- }
-
- h3 {
- font-size: ${props.theme.typography.size.m1}px;
- }
-
- h4 {
- font-size: ${props.theme.typography.size.s3}px;
- }
-
- h5 {
- font-size: ${props.theme.typography.size.s2}px;
- }
-
- h6 {
- font-size: ${props.theme.typography.size.s2}px;
- color: ${props.theme.color.dark};
- }
-
- /* Custom for SB SyntaxHighlighter */
-
- pre:not(.hljs) {
- background: transparent;
- border: none;
- border-radius: 0;
- padding: 0;
- margin: 0;
- }
-
- pre pre,
- pre.hljs {
- padding: 15px;
- margin: 0;
-
- white-space: pre-wrap;
- color: inherit;
-
- font-size: 13px;
- line-height: 19px;
-
- code {
- color: inherit;
- font-size: inherit;
- }
- }
-
- pre code {
- margin: 0;
- padding: 0;
- white-space: pre;
- border: none;
- background: transparent;
- }
-
- pre code,
- pre tt {
- background-color: transparent;
- border: none;
- }
-
- /* GitHub inspired Markdown styles loosely from https://gist.github.com/tuzz/3331384 */
-
- body > *:first-of-type {
- margin-top: 0 !important;
- }
-
- body > *:last-child {
- margin-bottom: 0 !important;
- }
-
- a {
- color: ${props.theme.color.secondary};
- text-decoration: none;
- }
-
- a.absent {
- color: #cc0000;
- }
-
- a.anchor {
- display: block;
- padding-left: 30px;
- margin-left: -30px;
- cursor: pointer;
- position: absolute;
- top: 0;
- left: 0;
- bottom: 0;
- }
-
- h1,
- h2,
- h3,
- h4,
- h5,
- h6 {
- margin: 20px 0 10px;
- padding: 0;
- cursor: text;
- position: relative;
- }
-
- h2:first-of-type,
- h1:first-of-type,
- h1:first-of-type + h2,
- h3:first-of-type,
- h4:first-of-type,
- h5:first-of-type,
- h6:first-of-type {
- margin-top: 0;
- padding-top: 0;
- }
-
- h1:hover a.anchor,
- h2:hover a.anchor,
- h3:hover a.anchor,
- h4:hover a.anchor,
- h5:hover a.anchor,
- h6:hover a.anchor {
- text-decoration: none;
- }
-
- h1 tt,
- h1 code {
- font-size: inherit;
- }
-
- h2 tt,
- h2 code {
- font-size: inherit;
- }
-
- h3 tt,
- h3 code {
- font-size: inherit;
- }
-
- h4 tt,
- h4 code {
- font-size: inherit;
- }
-
- h5 tt,
- h5 code {
- font-size: inherit;
- }
-
- h6 tt,
- h6 code {
- font-size: inherit;
- }
-
- p,
- blockquote,
- ul,
- ol,
- dl,
- li,
- table,
- pre {
- margin: 15px 0;
- }
-
- hr {
- border: 0 none;
- color: ${props.theme.appBorderColor};
- height: 4px;
- padding: 0;
- }
-
- body > h2:first-of-type {
- margin-top: 0;
- padding-top: 0;
- }
-
- body > h1:first-of-type {
- margin-top: 0;
- padding-top: 0;
- }
-
- body > h1:first-of-type + h2 {
- margin-top: 0;
- padding-top: 0;
- }
-
- body > h3:first-of-type,
- body > h4:first-of-type,
- body > h5:first-of-type,
- body > h6:first-of-type {
- margin-top: 0;
- padding-top: 0;
- }
-
- a:first-of-type h1,
- a:first-of-type h2,
- a:first-of-type h3,
- a:first-of-type h4,
- a:first-of-type h5,
- a:first-of-type h6 {
- margin-top: 0;
- padding-top: 0;
- }
-
- h1 p,
- h2 p,
- h3 p,
- h4 p,
- h5 p,
- h6 p {
- margin-top: 0;
- }
-
- li p.first {
- display: inline-block;
- }
-
- ul,
- ol {
- padding-left: 30px;
- }
-
- ul :first-of-type,
- ol :first-of-type {
- margin-top: 0;
- }
-
- ul :last-child,
- ol :last-child {
- margin-bottom: 0;
- }
-
- dl {
- padding: 0;
- }
-
- dl dt {
- font-size: 14px;
- font-weight: bold;
- font-style: italic;
- padding: 0;
- margin: 15px 0 5px;
- }
-
- dl dt:first-of-type {
- padding: 0;
- }
-
- dl dt > :first-of-type {
- margin-top: 0;
- }
-
- dl dt > :last-child {
- margin-bottom: 0;
- }
-
- dl dd {
- margin: 0 0 15px;
- padding: 0 15px;
- }
-
- dl dd > :first-of-type {
- margin-top: 0;
- }
-
- dl dd > :last-child {
- margin-bottom: 0;
- }
-
- blockquote {
- border-left: 4px solid ${props.theme.color.medium};
- padding: 0 15px;
- color: ${props.theme.color.dark};
- }
-
- blockquote > :first-of-type {
- margin-top: 0;
- }
-
- blockquote > :last-child {
- margin-bottom: 0;
- }
-
- table {
- padding: 0;
- border-collapse: collapse;
- }
- table tr {
- border-top: 1px solid ${props.theme.appBorderColor};
- background-color: white;
- margin: 0;
- padding: 0;
- }
-
- table tr:nth-of-type(2n) {
- background-color: ${props.theme.color.lighter};
- }
-
- table tr th {
- font-weight: bold;
- border: 1px solid ${props.theme.appBorderColor};
- text-align: left;
- margin: 0;
- padding: 6px 13px;
- }
-
- table tr td {
- border: 1px solid ${props.theme.appBorderColor};
- text-align: left;
- margin: 0;
- padding: 6px 13px;
- }
-
- table tr th :first-of-type,
- table tr td :first-of-type {
- margin-top: 0;
- }
-
- table tr th :last-child,
- table tr td :last-child {
- margin-bottom: 0;
- }
-
- img {
- max-width: 100%;
- }
-
- span.frame {
- display: block;
- overflow: hidden;
- }
-
- span.frame > span {
- border: 1px solid ${props.theme.color.medium};
- display: block;
- float: left;
- overflow: hidden;
- margin: 13px 0 0;
- padding: 7px;
- width: auto;
- }
-
- span.frame span img {
- display: block;
- float: left;
- }
-
- span.frame span span {
- clear: both;
- color: ${props.theme.color.darkest};
- display: block;
- padding: 5px 0 0;
- }
-
- span.align-center {
- display: block;
- overflow: hidden;
- clear: both;
- }
-
- span.align-center > span {
- display: block;
- overflow: hidden;
- margin: 13px auto 0;
- text-align: center;
- }
-
- span.align-center span img {
- margin: 0 auto;
- text-align: center;
- }
-
- span.align-right {
- display: block;
- overflow: hidden;
- clear: both;
- }
-
- span.align-right > span {
- display: block;
- overflow: hidden;
- margin: 13px 0 0;
- text-align: right;
- }
-
- span.align-right span img {
- margin: 0;
- text-align: right;
- }
-
- span.float-left {
- display: block;
- margin-right: 13px;
- overflow: hidden;
- float: left;
- }
-
- span.float-left span {
- margin: 13px 0 0;
- }
-
- span.float-right {
- display: block;
- margin-left: 13px;
- overflow: hidden;
- float: right;
- }
-
- span.float-right > span {
- display: block;
- overflow: hidden;
- margin: 13px auto 0;
- text-align: right;
- }
-
- code,
- tt {
- margin: 0 2px;
- padding: 0 5px;
- white-space: nowrap;
- border: 1px solid ${props.theme.color.mediumlight};
- background-color: ${props.theme.color.lighter};
- border-radius: 3px;
- }
- `
+import { styled, CSSObject, Theme } from '@storybook/theming';
+import { withReset } from './withReset';
+
+const headerCommon = ({ theme }: { theme: Theme }): CSSObject => ({
+ margin: '20px 0 8px',
+ padding: 0,
+ cursor: 'text',
+ position: 'relative',
+ color: theme.color.defaultText,
+ '&:first-of-type': {
+ marginTop: 0,
+ paddingTop: 0,
+ },
+ '&:hover a.anchor': {
+ textDecoration: 'none',
+ },
+ '& tt, & code': {
+ fontSize: 'inherit',
+ },
+});
+
+const withMargin: CSSObject = {
+ margin: '16px 0',
+};
+
+export const H1 = styled.h1<{}>(withReset, headerCommon, ({ theme }) => ({
+ fontSize: `${theme.typography.size.l1}px`,
+ fontWeight: theme.typography.weight.black,
+}));
+
+export const H2 = styled.h2<{}>(withReset, headerCommon, ({ theme }) => ({
+ fontSize: `${theme.typography.size.m2}px`,
+ paddingBottom: '4px',
+ borderBottom: `1px solid ${theme.appBorderColor}`,
+}));
+
+export const H3 = styled.h3<{}>(withReset, headerCommon, ({ theme }) => ({
+ fontSize: `${theme.typography.size.m1}px`,
+}));
+
+export const H4 = styled.h4<{}>(withReset, headerCommon, ({ theme }) => ({
+ fontSize: `${theme.typography.size.s3}px`,
+}));
+
+export const H5 = styled.h5<{}>(withReset, headerCommon, ({ theme }) => ({
+ fontSize: `${theme.typography.size.s2}px`,
+}));
+
+export const H6 = styled.h6<{}>(withReset, headerCommon, ({ theme }) => ({
+ fontSize: `${theme.typography.size.s2}px`,
+ color: theme.color.dark,
+}));
+
+export const Pre = styled.pre<{}>(withReset, withMargin, ({ theme }) => ({
+ // reset
+ fontFamily: theme.typography.fonts.mono,
+ WebkitFontSmoothing: 'antialiased',
+ MozOsxFontSmoothing: 'grayscale',
+ lineHeight: '18px',
+ padding: '11px 1rem',
+ whiteSpace: 'pre-wrap',
+ color: 'inherit',
+ borderRadius: 3,
+ margin: '1rem 0',
+
+ '&:not(.hljs)': {
+ background: 'transparent',
+ border: 'none',
+ borderRadius: 0,
+ padding: 0,
+ margin: 0,
+ },
+ '& pre, &.hljs': {
+ padding: 15,
+ margin: 0,
+ whiteSpace: 'pre-wrap',
+ color: 'inherit',
+ fontSize: '13px',
+ lineHeight: '19px',
+ code: {
+ color: 'inherit',
+ fontSize: 'inherit',
+ },
+ },
+ '& code': {
+ whiteSpace: 'pre',
+ },
+ '& code, & tt': {
+ border: 'none',
+ },
+}));
+
+export const A = styled.a<{}>(withReset, ({ theme }) => ({
+ fontSize: theme.typography.size.s2,
+ lineHeight: '24px',
+
+ color: `${theme.color.secondary}`,
+ textDecoration: 'none',
+ '&.absent': {
+ color: '#cc0000',
+ },
+ '&.anchor': {
+ display: 'block',
+ paddingLeft: '30px',
+ marginLeft: '-30px',
+ cursor: 'pointer',
+ position: 'absolute',
+ top: '0',
+ left: '0',
+ bottom: '0',
+ },
+}));
+
+export const HR = styled.hr<{}>(({ theme }) => ({
+ border: '0 none',
+ color: theme.appBorderColor,
+ height: '4px',
+ padding: '0',
+}));
+
+export const DL = styled.dl<{}>(withReset, {
+ ...withMargin,
+ padding: 0,
+ '& dt': {
+ fontSize: '14px',
+ fontWeight: 'bold',
+ fontStyle: 'italic',
+ padding: 0,
+ margin: '16px 0 4px',
+ },
+ '& dt:first-of-type': {
+ padding: 0,
+ },
+ '& dt > :first-of-type': {
+ marginTop: 0,
+ },
+
+ '& dt > :last-child': {
+ marginBottom: 0,
+ },
+
+ '& dd': {
+ margin: '0 0 16px',
+ padding: '0 15px',
+ },
+
+ '& dd > :first-of-type': {
+ marginTop: 0,
+ },
+
+ '& dd > :last-child': {
+ marginBottom: 0,
+ },
+});
+
+export const Blockquote = styled.blockquote<{}>(withReset, withMargin, ({ theme }) => ({
+ borderLeft: `4px solid ${theme.color.medium}`,
+ padding: '0 15px',
+ color: theme.color.dark,
+ '& > :first-of-type': {
+ marginTop: '0',
+ },
+ '& > :last-child': {
+ marginBottom: 0,
+ },
+}));
+
+export const Table = styled.table<{}>(withReset, withMargin, ({ theme }) => ({
+ fontSize: theme.typography.size.s2,
+ lineHeight: '24px',
+ padding: 0,
+ borderCollapse: 'collapse',
+ '& tr': {
+ borderTop: `1px solid ${theme.appBorderColor}`,
+ backgroundColor: 'white',
+ margin: '0',
+ padding: '0',
+ },
+ '& tr:nth-of-type(2n)': {
+ backgroundColor: `${theme.color.lighter}`,
+ },
+ '& tr th': {
+ fontWeight: 'bold',
+ border: `1px solid ${theme.appBorderColor}`,
+ textAlign: 'left',
+ margin: '0',
+ padding: '6px 13px',
+ },
+ '& tr td': {
+ border: `1px solid ${theme.appBorderColor}`,
+ textAlign: 'left',
+ margin: '0',
+ padding: '6px 13px',
+ },
+ '& tr th :first-of-type, & tr td :first-of-type': {
+ marginTop: '0',
+ },
+ '& tr th :last-child, & tr td :last-child': {
+ marginBottom: '0',
+ },
+}));
+
+export const Img = styled.img<{}>({
+ maxWidth: '100%',
+});
+
+export const Div = styled.div<{}>(withReset);
+
+export const Span = styled.span<{}>(withReset, ({ theme }) => ({
+ '&.frame': {
+ display: 'block',
+ overflow: 'hidden',
+ },
+ '&.frame > span': {
+ border: `1px solid ${theme.color.medium}`,
+ display: 'block',
+ float: 'left',
+ overflow: 'hidden',
+ margin: '13px 0 0',
+ padding: '7px',
+ width: 'auto',
+ },
+ '&.frame span img': {
+ display: 'block',
+ float: 'left',
+ },
+ '&.frame span span': {
+ clear: 'both',
+ color: theme.color.darkest,
+ display: 'block',
+ padding: '5px 0 0',
+ },
+ '&.align-center': {
+ display: 'block',
+ overflow: 'hidden',
+ clear: 'both',
+ },
+ '&.align-center > span': {
+ display: 'block',
+ overflow: 'hidden',
+ margin: '13px auto 0',
+ textAlign: 'center',
+ },
+ '&.align-center span img': {
+ margin: '0 auto',
+ textAlign: 'center',
+ },
+ '&.align-right': {
+ display: 'block',
+ overflow: 'hidden',
+ clear: 'both',
+ },
+ '&.align-right > span': {
+ display: 'block',
+ overflow: 'hidden',
+ margin: '13px 0 0',
+ textAlign: 'right',
+ },
+ '&.align-right span img': {
+ margin: '0',
+ textAlign: 'right',
+ },
+ '&.float-left': {
+ display: 'block',
+ marginRight: '13px',
+ overflow: 'hidden',
+ float: 'left',
+ },
+ '&.float-left span': {
+ margin: '13px 0 0',
+ },
+ '&.float-right': {
+ display: 'block',
+ marginLeft: '13px',
+ overflow: 'hidden',
+ float: 'right',
+ },
+ '&.float-right > span': {
+ display: 'block',
+ overflow: 'hidden',
+ margin: '13px auto 0',
+ textAlign: 'right',
+ },
+}));
+
+const listCommon: CSSObject = {
+ paddingLeft: '30px',
+ '& :first-of-type': {
+ marginTop: 0,
+ },
+ '& :last-child': {
+ marginBottom: 0,
+ },
+};
+
+export const LI = styled.li<{}>(withReset, ({ theme }) => ({
+ fontSize: theme.typography.size.s2,
+ color: theme.color.defaultText,
+ lineHeight: '24px',
+ '& + li': {
+ marginTop: '.25em',
+ },
+ '& ul, & ol': {
+ marginTop: '.25em',
+ marginBottom: 0,
+ },
+}));
+
+export const UL = styled.ul<{}>(withReset, withMargin, listCommon, {});
+
+export const OL = styled.ol<{}>(withReset, withMargin, listCommon);
+
+const codeCommon = ({ theme }: { theme: Theme }): CSSObject => ({
+ margin: '0 2px',
+ padding: '0 5px',
+ whiteSpace: 'nowrap',
+ border: `1px solid ${theme.color.mediumlight}`,
+ backgroundColor: theme.color.lighter,
+ borderRadius: '3px',
+ fontSize: theme.typography.size.s2 - 1,
+});
+
+export const P = styled.p<{}>(withReset, withMargin, ({ theme }) => ({
+ fontSize: theme.typography.size.s2,
+ lineHeight: '24px',
+ color: theme.color.defaultText,
+ '& code': codeCommon({ theme }),
+}));
+
+export const Code = styled.code<{}>(
+ ({ theme }) => ({
+ // from reset
+ fontFamily: theme.typography.fonts.mono,
+ WebkitFontSmoothing: 'antialiased',
+ MozOsxFontSmoothing: 'grayscale',
+ display: 'inline-block',
+ paddingLeft: 2,
+ paddingRight: 2,
+ verticalAlign: 'baseline',
+ color: 'inherit',
+ }),
+ codeCommon
);
-export const DocumentFormatting = (props: any) => ;
+export const TT = styled.title<{}>(codeCommon);
+
+/**
+ * This is a "local" reset to style subtrees with Storybook styles
+ *
+ * We can't style individual elements (e.g. h1, h2, etc.) in here
+ * because the CSS specificity is too high, so those styles can too
+ * easily override child elements that are not expecting it.
+ */
+
+export const ResetWrapper = styled.div<{}>(withReset);
diff --git a/lib/components/src/typography/DocumentFormatting.stories.tsx b/lib/components/src/typography/DocumentWrapper.stories.tsx
similarity index 96%
rename from lib/components/src/typography/DocumentFormatting.stories.tsx
rename to lib/components/src/typography/DocumentWrapper.stories.tsx
index 283b78b6845..fc44d5984ab 100644
--- a/lib/components/src/typography/DocumentFormatting.stories.tsx
+++ b/lib/components/src/typography/DocumentWrapper.stories.tsx
@@ -1,11 +1,11 @@
import React from 'react';
import Markdown from 'markdown-to-jsx';
-import { DocumentFormatting } from './DocumentFormatting';
+import { DocumentWrapper } from './DocumentWrapper';
import markdownSample from './DocumentFormattingSample.md';
export default {
- component: DocumentFormatting,
+ component: DocumentWrapper,
title: 'Basics|DocumentFormatting',
decorators: [
(storyFn: any) => (
@@ -19,13 +19,13 @@ export default {
};
export const withMarkdown = () => (
-
+
{markdownSample}
-
+
);
export const withDOM = () => (
-
+
h1 Heading
h2 Heading
h3 Heading
@@ -186,5 +186,5 @@ export const withDOM = () => (
title="The Stormtroopocat"
/>
-
+
);
diff --git a/lib/components/src/typography/DocumentWrapper.tsx b/lib/components/src/typography/DocumentWrapper.tsx
new file mode 100644
index 00000000000..d125542f370
--- /dev/null
+++ b/lib/components/src/typography/DocumentWrapper.tsx
@@ -0,0 +1,441 @@
+import { styled, css } from '@storybook/theming';
+
+export const DocumentWrapper = styled.div(
+ ({ theme }) => css`
+ /* Custom styles atop GitHub base theme (see below) */
+ font-size: ${theme.typography.size.s2}px;
+ line-height: 1.6;
+
+ h1 {
+ font-size: ${theme.typography.size.l1}px;
+ font-weight: ${theme.typography.weight.black};
+ }
+
+ h2 {
+ font-size: ${theme.typography.size.m2}px;
+ border-bottom: 1px solid ${theme.appBorderColor};
+ }
+
+ h3 {
+ font-size: ${theme.typography.size.m1}px;
+ }
+
+ h4 {
+ font-size: ${theme.typography.size.s3}px;
+ }
+
+ h5 {
+ font-size: ${theme.typography.size.s2}px;
+ }
+
+ h6 {
+ font-size: ${theme.typography.size.s2}px;
+ color: ${theme.color.dark};
+ }
+
+ /* Custom for SB SyntaxHighlighter */
+
+ pre:not(.hljs) {
+ background: transparent;
+ border: none;
+ border-radius: 0;
+ padding: 0;
+ margin: 0;
+ }
+
+ pre pre,
+ pre.hljs {
+ padding: 15px;
+ margin: 0;
+
+ white-space: pre-wrap;
+ color: inherit;
+
+ font-size: 13px;
+ line-height: 19px;
+
+ code {
+ color: inherit;
+ font-size: inherit;
+ }
+ }
+
+ pre code {
+ margin: 0;
+ padding: 0;
+ white-space: pre;
+ border: none;
+ background: transparent;
+ }
+
+ pre code,
+ pre tt {
+ background-color: transparent;
+ border: none;
+ }
+
+ /* GitHub inspired Markdown styles loosely from https://gist.github.com/tuzz/3331384 */
+
+ body > *:first-of-type {
+ margin-top: 0 !important;
+ }
+
+ body > *:last-child {
+ margin-bottom: 0 !important;
+ }
+
+ a {
+ color: ${theme.color.secondary};
+ text-decoration: none;
+ }
+
+ a.absent {
+ color: #cc0000;
+ }
+
+ a.anchor {
+ display: block;
+ padding-left: 30px;
+ margin-left: -30px;
+ cursor: pointer;
+ position: absolute;
+ top: 0;
+ left: 0;
+ bottom: 0;
+ }
+
+ h1,
+ h2,
+ h3,
+ h4,
+ h5,
+ h6 {
+ margin: 20px 0 10px;
+ padding: 0;
+ cursor: text;
+ position: relative;
+ }
+
+ h2:first-of-type,
+ h1:first-of-type,
+ h1:first-of-type + h2,
+ h3:first-of-type,
+ h4:first-of-type,
+ h5:first-of-type,
+ h6:first-of-type {
+ margin-top: 0;
+ padding-top: 0;
+ }
+
+ h1:hover a.anchor,
+ h2:hover a.anchor,
+ h3:hover a.anchor,
+ h4:hover a.anchor,
+ h5:hover a.anchor,
+ h6:hover a.anchor {
+ text-decoration: none;
+ }
+
+ h1 tt,
+ h1 code {
+ font-size: inherit;
+ }
+
+ h2 tt,
+ h2 code {
+ font-size: inherit;
+ }
+
+ h3 tt,
+ h3 code {
+ font-size: inherit;
+ }
+
+ h4 tt,
+ h4 code {
+ font-size: inherit;
+ }
+
+ h5 tt,
+ h5 code {
+ font-size: inherit;
+ }
+
+ h6 tt,
+ h6 code {
+ font-size: inherit;
+ }
+
+ p,
+ blockquote,
+ ul,
+ ol,
+ dl,
+ li,
+ table,
+ pre {
+ margin: 15px 0;
+ }
+
+ hr {
+ border: 0 none;
+ color: ${theme.appBorderColor};
+ height: 4px;
+ padding: 0;
+ }
+
+ body > h2:first-of-type {
+ margin-top: 0;
+ padding-top: 0;
+ }
+
+ body > h1:first-of-type {
+ margin-top: 0;
+ padding-top: 0;
+ }
+
+ body > h1:first-of-type + h2 {
+ margin-top: 0;
+ padding-top: 0;
+ }
+
+ body > h3:first-of-type,
+ body > h4:first-of-type,
+ body > h5:first-of-type,
+ body > h6:first-of-type {
+ margin-top: 0;
+ padding-top: 0;
+ }
+
+ a:first-of-type h1,
+ a:first-of-type h2,
+ a:first-of-type h3,
+ a:first-of-type h4,
+ a:first-of-type h5,
+ a:first-of-type h6 {
+ margin-top: 0;
+ padding-top: 0;
+ }
+
+ h1 p,
+ h2 p,
+ h3 p,
+ h4 p,
+ h5 p,
+ h6 p {
+ margin-top: 0;
+ }
+
+ li p.first {
+ display: inline-block;
+ }
+
+ ul,
+ ol {
+ padding-left: 30px;
+ }
+
+ ul :first-of-type,
+ ol :first-of-type {
+ margin-top: 0;
+ }
+
+ ul :last-child,
+ ol :last-child {
+ margin-bottom: 0;
+ }
+
+ dl {
+ padding: 0;
+ }
+
+ dl dt {
+ font-size: 14px;
+ font-weight: bold;
+ font-style: italic;
+ padding: 0;
+ margin: 15px 0 5px;
+ }
+
+ dl dt:first-of-type {
+ padding: 0;
+ }
+
+ dl dt > :first-of-type {
+ margin-top: 0;
+ }
+
+ dl dt > :last-child {
+ margin-bottom: 0;
+ }
+
+ dl dd {
+ margin: 0 0 15px;
+ padding: 0 15px;
+ }
+
+ dl dd > :first-of-type {
+ margin-top: 0;
+ }
+
+ dl dd > :last-child {
+ margin-bottom: 0;
+ }
+
+ blockquote {
+ border-left: 4px solid ${theme.color.medium};
+ padding: 0 15px;
+ color: ${theme.color.dark};
+ }
+
+ blockquote > :first-of-type {
+ margin-top: 0;
+ }
+
+ blockquote > :last-child {
+ margin-bottom: 0;
+ }
+
+ table {
+ padding: 0;
+ border-collapse: collapse;
+ }
+ table tr {
+ border-top: 1px solid ${theme.appBorderColor};
+ background-color: white;
+ margin: 0;
+ padding: 0;
+ }
+
+ table tr:nth-of-type(2n) {
+ background-color: ${theme.color.lighter};
+ }
+
+ table tr th {
+ font-weight: bold;
+ border: 1px solid ${theme.appBorderColor};
+ text-align: left;
+ margin: 0;
+ padding: 6px 13px;
+ }
+
+ table tr td {
+ border: 1px solid ${theme.appBorderColor};
+ text-align: left;
+ margin: 0;
+ padding: 6px 13px;
+ }
+
+ table tr th :first-of-type,
+ table tr td :first-of-type {
+ margin-top: 0;
+ }
+
+ table tr th :last-child,
+ table tr td :last-child {
+ margin-bottom: 0;
+ }
+
+ img {
+ max-width: 100%;
+ }
+
+ span.frame {
+ display: block;
+ overflow: hidden;
+ }
+
+ span.frame > span {
+ border: 1px solid ${theme.color.medium};
+ display: block;
+ float: left;
+ overflow: hidden;
+ margin: 13px 0 0;
+ padding: 7px;
+ width: auto;
+ }
+
+ span.frame span img {
+ display: block;
+ float: left;
+ }
+
+ span.frame span span {
+ clear: both;
+ color: ${theme.color.darkest};
+ display: block;
+ padding: 5px 0 0;
+ }
+
+ span.align-center {
+ display: block;
+ overflow: hidden;
+ clear: both;
+ }
+
+ span.align-center > span {
+ display: block;
+ overflow: hidden;
+ margin: 13px auto 0;
+ text-align: center;
+ }
+
+ span.align-center span img {
+ margin: 0 auto;
+ text-align: center;
+ }
+
+ span.align-right {
+ display: block;
+ overflow: hidden;
+ clear: both;
+ }
+
+ span.align-right > span {
+ display: block;
+ overflow: hidden;
+ margin: 13px 0 0;
+ text-align: right;
+ }
+
+ span.align-right span img {
+ margin: 0;
+ text-align: right;
+ }
+
+ span.float-left {
+ display: block;
+ margin-right: 13px;
+ overflow: hidden;
+ float: left;
+ }
+
+ span.float-left span {
+ margin: 13px 0 0;
+ }
+
+ span.float-right {
+ display: block;
+ margin-left: 13px;
+ overflow: hidden;
+ float: right;
+ }
+
+ span.float-right > span {
+ display: block;
+ overflow: hidden;
+ margin: 13px auto 0;
+ text-align: right;
+ }
+
+ code,
+ tt {
+ margin: 0 2px;
+ padding: 0 5px;
+ white-space: nowrap;
+ border: 1px solid ${theme.color.mediumlight};
+ background-color: ${theme.color.lighter};
+ border-radius: 3px;
+ }
+ `
+);
diff --git a/lib/components/src/typography/withReset.tsx b/lib/components/src/typography/withReset.tsx
new file mode 100644
index 00000000000..24d68e8cac9
--- /dev/null
+++ b/lib/components/src/typography/withReset.tsx
@@ -0,0 +1,12 @@
+import { CSSObject, Theme } from '@storybook/theming';
+
+export const withReset = ({ theme }: { theme: Theme }): CSSObject => ({
+ fontFamily: theme.typography.fonts.base,
+ fontSize: theme.typography.size.s3,
+ margin: 0,
+
+ WebkitFontSmoothing: 'antialiased',
+ MozOsxFontSmoothing: 'grayscale',
+ WebkitTapHighlightColor: 'rgba(0, 0, 0, 0)',
+ WebkitOverflowScrolling: 'touch',
+});
diff --git a/lib/core-events/package.json b/lib/core-events/package.json
index 98b1e883f46..f69988f5b12 100644
--- a/lib/core-events/package.json
+++ b/lib/core-events/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/core-events",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Event names used in storybook core",
"keywords": [
"storybook"
@@ -15,6 +15,10 @@
"directory": "lib/core-events"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
diff --git a/lib/core-events/src/index.ts b/lib/core-events/src/index.ts
index e5015c3b4ac..dfcea9f6489 100644
--- a/lib/core-events/src/index.ts
+++ b/lib/core-events/src/index.ts
@@ -18,6 +18,7 @@ enum events {
STORY_MISSING = 'storyMissing',
STORY_ERRORED = 'storyErrored',
STORY_THREW_EXCEPTION = 'storyThrewException',
+ DOCS_RENDERED = 'docsRendered',
}
// Enables: `import Events from ...`
@@ -25,21 +26,24 @@ export default events;
// Enables: `import * as Events from ...` or `import { CHANNEL_CREATED } as Events from ...`
// This is the preferred method
-export const { CHANNEL_CREATED } = events;
-export const { GET_CURRENT_STORY } = events;
-export const { SET_CURRENT_STORY } = events;
-export const { GET_STORIES } = events;
-export const { SET_STORIES } = events;
-export const { STORIES_CONFIGURED } = events;
-export const { SELECT_STORY } = events;
-export const { PREVIEW_KEYDOWN } = events;
-export const { FORCE_RE_RENDER } = events;
-export const { REGISTER_SUBSCRIPTION } = events;
-export const { STORY_INIT } = events;
-export const { STORY_ADDED } = events;
-export const { STORY_RENDER } = events;
-export const { STORY_RENDERED } = events;
-export const { STORY_MISSING } = events;
-export const { STORY_ERRORED } = events;
-export const { STORY_CHANGED } = events;
-export const { STORY_THREW_EXCEPTION } = events;
+export const {
+ CHANNEL_CREATED,
+ GET_CURRENT_STORY,
+ SET_CURRENT_STORY,
+ GET_STORIES,
+ SET_STORIES,
+ STORIES_CONFIGURED,
+ SELECT_STORY,
+ PREVIEW_KEYDOWN,
+ FORCE_RE_RENDER,
+ REGISTER_SUBSCRIPTION,
+ STORY_INIT,
+ STORY_ADDED,
+ STORY_RENDER,
+ STORY_RENDERED,
+ STORY_MISSING,
+ STORY_ERRORED,
+ STORY_CHANGED,
+ STORY_THREW_EXCEPTION,
+ DOCS_RENDERED,
+} = events;
diff --git a/lib/core/package.json b/lib/core/package.json
index f0a82a9e111..a194a7b16b2 100644
--- a/lib/core/package.json
+++ b/lib/core/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/core",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Storybook framework-agnostic API",
"keywords": [
"storybook"
@@ -15,6 +15,14 @@
"directory": "lib/core"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "dll/**/*",
+ "README.md",
+ "server.js",
+ "standalone.ts",
+ "client.ts"
+ ],
"main": "dist/client/index.js",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
@@ -25,15 +33,15 @@
"@babel/plugin-syntax-dynamic-import": "^7.2.0",
"@babel/plugin-transform-react-constant-elements": "^7.2.0",
"@babel/preset-env": "^7.4.5",
- "@storybook/addons": "5.2.0-beta.28",
- "@storybook/channel-postmessage": "5.2.0-beta.28",
- "@storybook/client-api": "5.2.0-beta.28",
- "@storybook/client-logger": "5.2.0-beta.28",
- "@storybook/core-events": "5.2.0-beta.28",
- "@storybook/node-logger": "5.2.0-beta.28",
- "@storybook/router": "5.2.0-beta.28",
- "@storybook/theming": "5.2.0-beta.28",
- "@storybook/ui": "5.2.0-beta.28",
+ "@storybook/addons": "5.3.0-alpha.0",
+ "@storybook/channel-postmessage": "5.3.0-alpha.0",
+ "@storybook/client-api": "5.3.0-alpha.0",
+ "@storybook/client-logger": "5.3.0-alpha.0",
+ "@storybook/core-events": "5.3.0-alpha.0",
+ "@storybook/node-logger": "5.3.0-alpha.0",
+ "@storybook/router": "5.3.0-alpha.0",
+ "@storybook/theming": "5.3.0-alpha.0",
+ "@storybook/ui": "5.3.0-alpha.0",
"airbnb-js-shims": "^1 || ^2",
"ansi-to-html": "^0.6.11",
"autoprefixer": "^9.4.9",
diff --git a/lib/core/src/client/preview/NoDocs.js b/lib/core/src/client/preview/NoDocs.js
new file mode 100644
index 00000000000..6fd48feb708
--- /dev/null
+++ b/lib/core/src/client/preview/NoDocs.js
@@ -0,0 +1,40 @@
+import React from 'react';
+import { styled } from '@storybook/theming';
+
+const Wrapper = styled.div({
+ fontSize: '14px',
+ letterSpacing: '0.2px',
+ margin: '10px 0',
+});
+
+const Main = styled.div({
+ margin: 'auto',
+ padding: '30px',
+ borderRadius: '10px',
+ background: 'rgba(0,0,0,0.03)',
+});
+
+const Heading = styled.h1({
+ textAlign: 'center',
+});
+
+export const NoDocs = () => (
+
+
+ No Docs
+
+ Sorry, but there are no docs for the selected story. To add them, set the story's{' '}
+ docs
parameter. If you think this is an error:
+
+
+ Please check the story definition.
+ Please check the Storybook config.
+ Try reloading the page.
+
+
+ If the problem persists, check the browser console, or the terminal you've run Storybook
+ from.
+
+
+
+);
diff --git a/lib/core/src/client/preview/start.js b/lib/core/src/client/preview/start.js
index 36d0bfe9b58..6a5e53563a6 100644
--- a/lib/core/src/client/preview/start.js
+++ b/lib/core/src/client/preview/start.js
@@ -7,11 +7,12 @@ import AnsiToHtml from 'ansi-to-html';
import addons from '@storybook/addons';
import createChannel from '@storybook/channel-postmessage';
import { ClientApi, StoryStore, ConfigApi } from '@storybook/client-api';
-import { toId } from '@storybook/router/utils';
+import { toId, storyNameFromExport } from '@storybook/router/utils';
import { logger } from '@storybook/client-logger';
import Events from '@storybook/core-events';
import { initializePath, setPath } from './url';
+import { NoDocs } from './NoDocs';
const ansiConverter = new AnsiToHtml();
@@ -142,6 +143,7 @@ export default function start(render, { decorateStory } = {}) {
let previousStory = '';
let previousRevision = -1;
let previousViewMode = '';
+ let previousId = null;
const renderMain = forceRender => {
const revision = storyStore.getRevision();
@@ -192,12 +194,26 @@ export default function start(render, { decorateStory } = {}) {
addons.getChannel().emit(Events.STORY_CHANGED, id);
}
+ switch (previousViewMode) {
+ case 'docs':
+ if (previousKind != null && (kind !== previousKind || viewMode !== previousViewMode)) {
+ storyStore.cleanHooksForKind(previousKind);
+ }
+ break;
+ case 'story':
+ default:
+ if (previousId != null && (id !== previousId || viewMode !== previousViewMode)) {
+ storyStore.cleanHooks(previousId);
+ }
+ }
+
// Docs view renders into a different root ID to avoid conflicts
// with the user's view layer. Therefore we need to clean up whenever
// we transition between view modes
if (viewMode !== previousViewMode) {
switch (viewMode) {
case 'docs': {
+ showMain();
document.getElementById('root').setAttribute('hidden', true);
document.getElementById('docs-root').removeAttribute('hidden');
break;
@@ -215,19 +231,30 @@ export default function start(render, { decorateStory } = {}) {
// Given a cleaned up state, render the appropriate view mode
switch (viewMode) {
case 'docs': {
- const NoDocs = () => No docs found
;
- const StoryDocs = (parameters && parameters.docs) || NoDocs;
+ const docs = (parameters && parameters.docs) || {};
+ // eslint-disable-next-line react/prop-types
+ const DocsContainer = docs.container || (({ children }) => <>{children}>);
+ const Page = docs.page || NoDocs;
ReactDOM.render(
- ,
- document.getElementById('docs-root')
+
+
+ ,
+ document.getElementById('docs-root'),
+ () => addons.getChannel().emit(Events.DOCS_RENDERED, kind)
);
break;
}
case 'story':
default: {
if (getDecorated) {
- render(renderContext);
- addons.getChannel().emit(Events.STORY_RENDERED, id);
+ (async () => {
+ try {
+ await render(renderContext);
+ addons.getChannel().emit(Events.STORY_RENDERED, id);
+ } catch (ex) {
+ showException(ex);
+ }
+ })();
} else {
showNopreview();
addons.getChannel().emit(Events.STORY_MISSING, id);
@@ -240,6 +267,7 @@ export default function start(render, { decorateStory } = {}) {
previousKind = kind;
previousStory = name;
previousViewMode = viewMode;
+ previousId = id;
if (!forceRender) {
document.documentElement.scrollTop = 0;
@@ -391,7 +419,12 @@ export default function start(render, { decorateStory } = {}) {
`${kindName} => ${name || key}: story.parameters.decorators is deprecated; use story.decorators instead.`)();
}
const decoratorParams = decorators ? { decorators } : null;
- kind.add(name || key, storyFn, { ...parameters, ...decoratorParams });
+ const displayNameParams = name ? { displayName: name } : {};
+ kind.add(storyNameFromExport(key), storyFn, {
+ ...parameters,
+ ...decoratorParams,
+ ...displayNameParams,
+ });
}
});
});
diff --git a/lib/core/src/server/build-dev.js b/lib/core/src/server/build-dev.js
index ba35a22a609..9657655a98e 100644
--- a/lib/core/src/server/build-dev.js
+++ b/lib/core/src/server/build-dev.js
@@ -298,15 +298,6 @@ export async function buildDevStandalone(options) {
const address = `${proto}://${options.host || 'localhost'}:${port}/`;
const networkAddress = `${proto}://${ip.address()}:${port}/`;
- outputStartupInformation({
- updateInfo,
- version,
- address,
- networkAddress,
- managerTotalTime,
- previewTotalTime,
- });
-
if (options.smokeTest) {
await outputStats(previewStats, managerStats);
@@ -319,7 +310,19 @@ export async function buildDevStandalone(options) {
warning += managerStats.toJson().warnings.length;
process.exit(warning ? 1 : 0);
- } else if (!options.ci) {
+ return;
+ }
+
+ outputStartupInformation({
+ updateInfo,
+ version,
+ address,
+ networkAddress,
+ managerTotalTime,
+ previewTotalTime,
+ });
+
+ if (!options.ci) {
openInBrowser(address);
}
} catch (error) {
diff --git a/lib/core/src/server/cli/dev.js b/lib/core/src/server/cli/dev.js
index 71136e3c6b1..8bf7f4c9102 100644
--- a/lib/core/src/server/cli/dev.js
+++ b/lib/core/src/server/cli/dev.js
@@ -25,7 +25,7 @@ async function getCLI(packageJson) {
.option('--ssl-cert ', 'Provide an SSL certificate. (Required with --https)')
.option('--ssl-key ', 'Provide an SSL key. (Required with --https)')
.option('--smoke-test', 'Exit after successful start')
- .option('--ci', "CI mode (skip interactive prompts, don't open browser")
+ .option('--ci', "CI mode (skip interactive prompts, don't open browser)")
.option('--quiet', 'Suppress verbose build output')
.option('--no-dll', 'Do not use dll reference')
.option('--debug-webpack', 'Display final webpack configurations for debugging purposes')
diff --git a/lib/core/src/server/manager/manager-webpack.config.js b/lib/core/src/server/manager/manager-webpack.config.js
index 69897a77b5e..9ec831cb9df 100644
--- a/lib/core/src/server/manager/manager-webpack.config.js
+++ b/lib/core/src/server/manager/manager-webpack.config.js
@@ -98,7 +98,7 @@ export default ({
],
},
{
- test: /\.(svg|ico|jpg|jpeg|png|gif|eot|otf|webp|ttf|woff|woff2|cur|ani)(\?.*)?$/,
+ test: /\.(svg|ico|jpg|jpeg|png|gif|eot|otf|webp|ttf|woff|woff2|cur|ani|pdf)(\?.*)?$/,
loader: require.resolve('file-loader'),
query: {
name: 'static/media/[name].[hash:8].[ext]',
diff --git a/lib/core/src/server/preview/base-webpack.config.js b/lib/core/src/server/preview/base-webpack.config.js
index d6122ec2c7a..9b64c146981 100644
--- a/lib/core/src/server/preview/base-webpack.config.js
+++ b/lib/core/src/server/preview/base-webpack.config.js
@@ -34,7 +34,7 @@ export function createDefaultWebpackConfig(storybookBaseConfig) {
],
},
{
- test: /\.(svg|ico|jpg|jpeg|png|gif|eot|otf|webp|ttf|woff|woff2|cur|ani)(\?.*)?$/,
+ test: /\.(svg|ico|jpg|jpeg|png|gif|eot|otf|webp|ttf|woff|woff2|cur|ani|pdf)(\?.*)?$/,
loader: require.resolve('file-loader'),
query: {
name: 'static/media/[name].[hash:8].[ext]',
diff --git a/lib/core/src/server/public/favicon.ico b/lib/core/src/server/public/favicon.ico
index 8c2246af8bf..428500fde18 100644
Binary files a/lib/core/src/server/public/favicon.ico and b/lib/core/src/server/public/favicon.ico differ
diff --git a/lib/core/src/server/templates/base-preview-head.html b/lib/core/src/server/templates/base-preview-head.html
index f63899f0f37..d671b0f1c31 100644
--- a/lib/core/src/server/templates/base-preview-head.html
+++ b/lib/core/src/server/templates/base-preview-head.html
@@ -1,15 +1,6 @@
<% if (typeof bodyHtmlSnippet !== 'undefined') { %>
diff --git a/lib/node-logger/package.json b/lib/node-logger/package.json
index db2f9c54947..9f46b0840a6 100644
--- a/lib/node-logger/package.json
+++ b/lib/node-logger/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/node-logger",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "",
"keywords": [
"storybook"
@@ -15,6 +15,10 @@
"directory": "lib/node-logger"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
diff --git a/lib/router/package.json b/lib/router/package.json
index f4c46255eb7..d0516ca194c 100644
--- a/lib/router/package.json
+++ b/lib/router/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/router",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Core Storybook Router",
"keywords": [
"storybook"
@@ -15,6 +15,12 @@
"directory": "lib/router"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md",
+ "utils.js",
+ "utils.d.ts"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
@@ -25,6 +31,7 @@
"@types/reach__router": "^1.2.3",
"core-js": "^3.0.1",
"global": "^4.3.2",
+ "lodash": "^4.17.11",
"memoizerific": "^1.11.3",
"qs": "^6.6.0"
},
diff --git a/lib/router/src/tests/id.test.js b/lib/router/src/utils.test.js
similarity index 74%
rename from lib/router/src/tests/id.test.js
rename to lib/router/src/utils.test.js
index 7f87b9015e4..57a9bf40edf 100644
--- a/lib/router/src/tests/id.test.js
+++ b/lib/router/src/utils.test.js
@@ -1,4 +1,4 @@
-import { toId } from '../utils';
+import { toId, storyNameFromExport } from './utils';
describe('toId', () => {
[
@@ -37,3 +37,17 @@ describe('toId', () => {
expect(() => toId('kind', '')).toThrow(`Invalid name '', must include alphanumeric characters`);
});
});
+
+describe('storyNameFromExport', () => {
+ it('should format CSF exports with sensible defaults', () => {
+ const testCases = {
+ name: 'Name',
+ someName: 'Some Name',
+ someNAME: 'Some NAME',
+ some_custom_NAME: 'Some Custom NAME',
+ someName1234: 'Some Name 1234',
+ someName1_2_3_4: 'Some Name 1 2 3 4',
+ };
+ Object.entries(testCases).forEach(([key, val]) => expect(storyNameFromExport(key)).toBe(val));
+ });
+});
diff --git a/lib/router/src/utils.ts b/lib/router/src/utils.ts
index 5a03d8f24eb..c8cd8570667 100644
--- a/lib/router/src/utils.ts
+++ b/lib/router/src/utils.ts
@@ -1,5 +1,6 @@
import qs from 'qs';
import memoize from 'memoizerific';
+import startCase from 'lodash/startCase';
interface StoryData {
viewMode?: string;
@@ -91,3 +92,6 @@ export const parseKind = (kind: string, { rootSeparator, groupSeparator }: Separ
groups,
};
};
+
+// Transform the CSF named export into a readable story name
+export const storyNameFromExport = (key: string) => startCase(key);
diff --git a/lib/source-loader/package.json b/lib/source-loader/package.json
index 49b8556d651..3c2257c0c73 100644
--- a/lib/source-loader/package.json
+++ b/lib/source-loader/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/source-loader",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Source loader",
"keywords": [
"lib",
@@ -16,14 +16,19 @@
"directory": "lib/source-loader"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md",
+ "preview.js"
+ ],
"main": "dist/server/index.js",
"jsnext:main": "src/server/index.js",
"scripts": {
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addons": "5.2.0-beta.28",
- "@storybook/router": "5.2.0-beta.28",
+ "@storybook/addons": "5.3.0-alpha.0",
+ "@storybook/router": "5.3.0-alpha.0",
"core-js": "^3.0.1",
"estraverse": "^4.2.0",
"global": "^4.3.2",
diff --git a/lib/source-loader/src/server/abstract-syntax-tree/traverse-helpers.js b/lib/source-loader/src/server/abstract-syntax-tree/traverse-helpers.js
index 4edda95f55f..751bc5162c9 100644
--- a/lib/source-loader/src/server/abstract-syntax-tree/traverse-helpers.js
+++ b/lib/source-loader/src/server/abstract-syntax-tree/traverse-helpers.js
@@ -1,3 +1,4 @@
+import { storyNameFromExport } from '@storybook/router/utils';
import { handleADD, handleSTORYOF, patchNode, handleExportedName } from './parse-helpers';
const estraverse = require('estraverse');
@@ -73,37 +74,6 @@ export function findAddsMap(ast, storiesOfIdentifiers) {
return { addsMap, idsToFrameworks };
}
-// Handle cases like:
-// export const withText = () => ;
-// withText.story = { name: 'with text' };
-function findStoryTitle(storyVar, ast) {
- const titleAssignment = ast.program.body.find(
- d =>
- d.type === 'ExpressionStatement' &&
- d.expression &&
- d.expression.type === 'AssignmentExpression' &&
- d.expression.left &&
- d.expression.left.type === 'MemberExpression' &&
- d.expression.left.object &&
- d.expression.left.object.type === 'Identifier' &&
- d.expression.left.object.name === storyVar &&
- d.expression.left.property.type === 'Identifier' &&
- d.expression.left.property.name === 'story' &&
- d.expression.right &&
- d.expression.right.type === 'ObjectExpression'
- );
- const nameProp =
- titleAssignment &&
- titleAssignment.expression.right.properties.find(
- prop =>
- prop.key.type === 'Identifier' &&
- prop.key.name === 'name' &&
- prop.value.type === 'StringLiteral'
- );
-
- return nameProp && nameProp.value.value;
-}
-
export function findExportsMap(ast) {
const addsMap = {};
const idsToFrameworks = {};
@@ -145,8 +115,7 @@ export function findExportsMap(ast) {
node.declaration.declarations[0].init.type
)
) {
- let storyName = node.declaration.declarations[0].id.name;
- storyName = findStoryTitle(storyName, ast) || storyName;
+ const storyName = storyNameFromExport(node.declaration.declarations[0].id.name);
const toAdd = handleExportedName(title, storyName, node.declaration.declarations[0].init);
Object.assign(addsMap, toAdd);
}
diff --git a/lib/source-loader/src/server/build.js b/lib/source-loader/src/server/build.js
index 1b0816c825c..8ed9758727b 100644
--- a/lib/source-loader/src/server/build.js
+++ b/lib/source-loader/src/server/build.js
@@ -15,13 +15,21 @@ export function transform(inputSource) {
localDependencies,
idsToFrameworks,
}) => `
+ // @ts-ignore
var withSourceLoader = require('@storybook/source-loader/preview').withSource;
+ // @ts-ignore
var __SOURCE_PREFIX__ = "${prefix.replace(/\\([^\\ ])/g, '\\\\$1')}";
+ // @ts-ignore
var __STORY__ = ${sourceJson};
+ // @ts-ignore
var __ADDS_MAP__ = ${JSON.stringify(addsMap)};
+ // @ts-ignore
var __MAIN_FILE_LOCATION__ = ${JSON.stringify(resource)};
+ // @ts-ignore
var __MODULE_DEPENDENCIES__ = ${JSON.stringify(dependencies)};
+ // @ts-ignore
var __LOCAL_DEPENDENCIES__ = ${JSON.stringify(localDependencies)};
+ // @ts-ignore
var __IDS_TO_FRAMEWORKS__ = ${JSON.stringify(idsToFrameworks)};
${source}
diff --git a/lib/theming/package.json b/lib/theming/package.json
index faedb347aab..8aa354f3886 100644
--- a/lib/theming/package.json
+++ b/lib/theming/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/theming",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Core Storybook Components",
"keywords": [
"storybook"
@@ -15,6 +15,12 @@
"directory": "lib/theming"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md",
+ "paths.js",
+ "create.js"
+ ],
"main": "dist/index.js",
"types": "dist/index.d.ts",
"scripts": {
@@ -23,7 +29,7 @@
"dependencies": {
"@emotion/core": "^10.0.14",
"@emotion/styled": "^10.0.14",
- "@storybook/client-logger": "5.2.0-beta.28",
+ "@storybook/client-logger": "5.3.0-alpha.0",
"common-tags": "^1.8.0",
"core-js": "^3.0.1",
"deep-object-diff": "^1.1.0",
diff --git a/lib/ui/.npmignore b/lib/ui/.npmignore
deleted file mode 100644
index 02ba1793bbe..00000000000
--- a/lib/ui/.npmignore
+++ /dev/null
@@ -1 +0,0 @@
-.babelrc
diff --git a/lib/ui/package.json b/lib/ui/package.json
index b8e4b3482d2..028104713c7 100644
--- a/lib/ui/package.json
+++ b/lib/ui/package.json
@@ -1,6 +1,6 @@
{
"name": "@storybook/ui",
- "version": "5.2.0-beta.28",
+ "version": "5.3.0-alpha.0",
"description": "Core Storybook UI",
"keywords": [
"storybook"
@@ -15,23 +15,27 @@
"directory": "lib/ui"
},
"license": "MIT",
+ "files": [
+ "dist/**/*",
+ "README.md",
+ "paths.js"
+ ],
"main": "dist/index.js",
- "jsnext:main": "src/index.js",
"scripts": {
"createDlls": "node -r esm ./scripts/createDlls.js",
"prepare": "node ../../scripts/prepare.js"
},
"dependencies": {
- "@storybook/addon-actions": "5.2.0-beta.28",
- "@storybook/addon-knobs": "5.2.0-beta.28",
- "@storybook/addons": "5.2.0-beta.28",
- "@storybook/api": "5.2.0-beta.28",
- "@storybook/channels": "5.2.0-beta.28",
- "@storybook/client-logger": "5.2.0-beta.28",
- "@storybook/components": "5.2.0-beta.28",
- "@storybook/core-events": "5.2.0-beta.28",
- "@storybook/router": "5.2.0-beta.28",
- "@storybook/theming": "5.2.0-beta.28",
+ "@storybook/addon-actions": "5.3.0-alpha.0",
+ "@storybook/addon-knobs": "5.3.0-alpha.0",
+ "@storybook/addons": "5.3.0-alpha.0",
+ "@storybook/api": "5.3.0-alpha.0",
+ "@storybook/channels": "5.3.0-alpha.0",
+ "@storybook/client-logger": "5.3.0-alpha.0",
+ "@storybook/components": "5.3.0-alpha.0",
+ "@storybook/core-events": "5.3.0-alpha.0",
+ "@storybook/router": "5.3.0-alpha.0",
+ "@storybook/theming": "5.3.0-alpha.0",
"copy-to-clipboard": "^3.0.8",
"core-js": "^3.0.1",
"core-js-pure": "^3.0.1",
@@ -47,7 +51,7 @@
"qs": "^6.6.0",
"react": "^16.8.3",
"react-dom": "^16.8.3",
- "react-draggable": "^3.1.1",
+ "react-draggable": "^4.0.3",
"react-helmet-async": "^1.0.2",
"react-hotkeys": "2.0.0-pre4",
"react-sizeme": "^2.6.7",
diff --git a/lib/ui/src/components/preview/iframe.js b/lib/ui/src/components/preview/iframe.js
index 383cbce1840..2f934f04c62 100644
--- a/lib/ui/src/components/preview/iframe.js
+++ b/lib/ui/src/components/preview/iframe.js
@@ -28,7 +28,7 @@ export class IFrame extends Component {
if (scale !== nextProps.scale) {
this.setIframeBodyStyle({
- width: `calc(${nextProps.scale * 100}% - 16px)`,
+ width: `${nextProps.scale * 100}%`,
height: `${nextProps.scale * 100}%`,
transform: `scale(${1 / nextProps.scale})`,
transformOrigin: 'top left',
diff --git a/lib/ui/src/components/sidebar/Sidebar.stories.mdx b/lib/ui/src/components/sidebar/Sidebar.stories.mdx
new file mode 100644
index 00000000000..22b569b006e
--- /dev/null
+++ b/lib/ui/src/components/sidebar/Sidebar.stories.mdx
@@ -0,0 +1,27 @@
+import { Meta, Story, Preview } from '@storybook/addon-docs/blocks';
+
+import Sidebar from './Sidebar';
+import { standardData } from './SidebarHeading.stories';
+import { withRootData } from './SidebarStories.stories';
+
+
+
+# Sidebar
+
+Here's some custom documentation for Storybook's `Sidebar` component.
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/lib/ui/src/components/sidebar/SidebarItem.tsx b/lib/ui/src/components/sidebar/SidebarItem.tsx
index 7673a82f007..36767a03062 100644
--- a/lib/ui/src/components/sidebar/SidebarItem.tsx
+++ b/lib/ui/src/components/sidebar/SidebarItem.tsx
@@ -135,6 +135,9 @@ const SidebarItem = ({
iconName = 'folder';
}
+ // eslint-disable-next-line react/destructuring-assignment
+ const displayName = (props.parameters && props.parameters.displayName) || name;
+
return (
-
- {name}
+ {displayName}
);
};
diff --git a/lib/ui/src/components/sidebar/treeview/treeview.js b/lib/ui/src/components/sidebar/treeview/treeview.js
index 2ab3b768548..0ecddcd1203 100644
--- a/lib/ui/src/components/sidebar/treeview/treeview.js
+++ b/lib/ui/src/components/sidebar/treeview/treeview.js
@@ -2,6 +2,7 @@ import { document } from 'global';
import React, { PureComponent, Fragment } from 'react';
import PropTypes from 'prop-types';
import memoize from 'memoizerific';
+import debounce from 'lodash/debounce';
import {
createId,
@@ -150,14 +151,10 @@ const calculateTreeState = memoize(50)(
// If a new selection is made, we need to ensure it is part of the expanded set
const selectedAncestorIds = selectedId ? getParents(selectedId, dataset).map(i => i.id) : [];
-
- const newExpanded = Object.keys(dataset).reduce(
- (acc, key) => ({
- ...acc,
- [key]: selectedAncestorIds.includes(key) || unfilteredExpanded[key],
- }),
- {}
- );
+ const newExpanded = Object.keys(dataset).reduce((acc, key) => {
+ acc[key] = selectedAncestorIds.includes(key) || unfilteredExpanded[key];
+ return acc;
+ }, {});
return {
lastSelectedId: selectedId,
@@ -224,7 +221,7 @@ class TreeState extends PureComponent {
}))
);
},
- onFilter: inputFilter => {
+ onFilter: debounce(inputFilter => {
const { dataset } = this.props;
const filter = inputFilter.length >= 2 ? inputFilter : '';
const filteredDataset = getFilteredDataset({ dataset, filter });
@@ -236,7 +233,7 @@ class TreeState extends PureComponent {
!!filter &&
Object.keys(filteredDataset).reduce((acc, k) => Object.assign(acc, { [k]: true }), {}),
});
- },
+ }, 100),
onKeyUp: (e, item) => {
const { prefix, dataset } = this.props;
const { filter } = this.state;
diff --git a/lib/ui/src/components/sidebar/treeview/utils.js b/lib/ui/src/components/sidebar/treeview/utils.js
index b313bb569bc..84a77494a3f 100644
--- a/lib/ui/src/components/sidebar/treeview/utils.js
+++ b/lib/ui/src/components/sidebar/treeview/utils.js
@@ -172,20 +172,17 @@ export const toFiltered = (dataset, filter) => {
// get all parents for all results
const result = found.reduce((acc, item) => {
- const parents = getParents(item.id, dataset).reduce(
- (pacc, pitem) => ({ ...pacc, [pitem.id]: { ...pitem } }),
- {}
- );
+ getParents(item.id, dataset).forEach(pitem => {
+ acc[pitem.id] = pitem;
+ });
- return { ...acc, [item.id]: item, ...parents };
+ acc[item.id] = item;
+ return acc;
}, {});
// filter the children of the found items (and their parents) so only found entries are present
- return Object.entries(result).reduce(
- (acc, [k, v]) => ({
- ...acc,
- [k]: v.children ? { ...v, children: v.children.filter(c => !!result[c]) } : v,
- }),
- {}
- );
+ return Object.entries(result).reduce((acc, [k, v]) => {
+ acc[k] = v.children ? { ...v, children: v.children.filter(c => !!result[c]) } : v;
+ return acc;
+ }, {});
};
diff --git a/lib/ui/src/containers/nav.js b/lib/ui/src/containers/nav.js
index d460cc2e455..2c80936eb66 100755
--- a/lib/ui/src/containers/nav.js
+++ b/lib/ui/src/containers/nav.js
@@ -15,85 +15,90 @@ const focusableUIElements = {
storyPanelRoot: 'storybook-panel-root',
};
-const createMenu = memoize(1)((api, shortcutKeys, isFullscreen, showPanel, showNav) => [
- {
- id: 'S',
- title: 'Show sidebar',
- onClick: () => api.toggleNav(),
- right: shortcutToHumanString(shortcutKeys.toggleNav),
- left: showNav ? : ,
- },
- {
- id: 'A',
- title: 'Show addons',
- onClick: () => api.togglePanel(),
- right: shortcutToHumanString(shortcutKeys.togglePanel),
- left: showPanel ? : ,
- },
- {
- id: 'D',
- title: 'Change addons orientation',
- onClick: () => api.togglePanelPosition(),
- right: shortcutToHumanString(shortcutKeys.panelPosition),
- left: ,
- },
- {
- id: 'F',
- title: 'Go full screen',
- onClick: api.toggleFullscreen,
- right: shortcutToHumanString(shortcutKeys.fullScreen),
- left: isFullscreen ? 'check' : ,
- },
- {
- id: '/',
- title: 'Search',
- onClick: () => api.focusOnUIElement(focusableUIElements.storySearchField),
- right: shortcutToHumanString(shortcutKeys.search),
- left: ,
- },
- {
- id: 'up',
- title: 'Previous component',
- onClick: () => api.jumpToComponent(-1),
- right: shortcutToHumanString(shortcutKeys.prevComponent),
- left: ,
- },
- {
- id: 'down',
- title: 'Next component',
- onClick: () => api.jumpToComponent(1),
- right: shortcutToHumanString(shortcutKeys.nextComponent),
- left: ,
- },
- {
- id: 'prev',
- title: 'Previous story',
- onClick: () => api.jumpToStory(-1),
- right: shortcutToHumanString(shortcutKeys.prevStory),
- left: ,
- },
- {
- id: 'next',
- title: 'Next story',
- onClick: () => api.jumpToStory(1),
- right: shortcutToHumanString(shortcutKeys.nextStory),
- left: ,
- },
- {
- id: 'about',
- title: 'About your Storybook',
- onClick: () => api.navigate('/settings/about'),
- right: api.versionUpdateAvailable() && Update ,
- left: ,
- },
- {
- id: 'shortcuts',
- title: 'Keyboard shortcuts',
- onClick: () => api.navigate('/settings/shortcuts'),
- right: shortcutToHumanString(shortcutKeys.shortcutsPage),
- left: ,
- },
-]);
+const shortcutToHumanStringIfEnabled = (shortcuts, enableShortcuts) =>
+ enableShortcuts ? shortcutToHumanString(shortcuts) : null;
+
+const createMenu = memoize(1)(
+ (api, shortcutKeys, isFullscreen, showPanel, showNav, enableShortcuts) => [
+ {
+ id: 'S',
+ title: 'Show sidebar',
+ onClick: () => api.toggleNav(),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.toggleNav, enableShortcuts),
+ left: showNav ? : ,
+ },
+ {
+ id: 'A',
+ title: 'Show addons',
+ onClick: () => api.togglePanel(),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.togglePanel, enableShortcuts),
+ left: showPanel ? : ,
+ },
+ {
+ id: 'D',
+ title: 'Change addons orientation',
+ onClick: () => api.togglePanelPosition(),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.panelPosition, enableShortcuts),
+ left: ,
+ },
+ {
+ id: 'F',
+ title: 'Go full screen',
+ onClick: api.toggleFullscreen,
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.fullScreen, enableShortcuts),
+ left: isFullscreen ? 'check' : ,
+ },
+ {
+ id: '/',
+ title: 'Search',
+ onClick: () => api.focusOnUIElement(focusableUIElements.storySearchField),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.search, enableShortcuts),
+ left: ,
+ },
+ {
+ id: 'up',
+ title: 'Previous component',
+ onClick: () => api.jumpToComponent(-1),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.prevComponent, enableShortcuts),
+ left: ,
+ },
+ {
+ id: 'down',
+ title: 'Next component',
+ onClick: () => api.jumpToComponent(1),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.nextComponent, enableShortcuts),
+ left: ,
+ },
+ {
+ id: 'prev',
+ title: 'Previous story',
+ onClick: () => api.jumpToStory(-1),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.prevStory, enableShortcuts),
+ left: ,
+ },
+ {
+ id: 'next',
+ title: 'Next story',
+ onClick: () => api.jumpToStory(1),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.nextStory, enableShortcuts),
+ left: ,
+ },
+ {
+ id: 'about',
+ title: 'About your Storybook',
+ onClick: () => api.navigate('/settings/about'),
+ right: api.versionUpdateAvailable() && Update ,
+ left: ,
+ },
+ {
+ id: 'shortcuts',
+ title: 'Keyboard shortcuts',
+ onClick: () => api.navigate('/settings/shortcuts'),
+ right: shortcutToHumanStringIfEnabled(shortcutKeys.shortcutsPage, enableShortcuts),
+ left: ,
+ },
+ ]
+);
const collapseDocsOnlyStories = storiesHash => {
// keep track of component IDs that have been rewritten to the ID of their first leaf child
@@ -140,16 +145,14 @@ const collapseDocsOnlyStories = storiesHash => {
export const mapper = ({ state, api }) => {
const {
- ui: { name, url },
+ ui: { name, url, enableShortcuts },
viewMode,
storyId,
- layout: { isFullscreen, showPanel, showNav, panelPosition },
+ layout: { isFullscreen, showPanel, showNav },
storiesHash,
storiesConfigured,
} = state;
-
const stories = collapseDocsOnlyStories(storiesHash);
-
const shortcutKeys = api.getShortcutKeys();
return {
loading: !storiesConfigured,
@@ -158,7 +161,7 @@ export const mapper = ({ state, api }) => {
stories,
storyId,
viewMode,
- menu: createMenu(api, shortcutKeys, isFullscreen, showPanel, showNav, panelPosition),
+ menu: createMenu(api, shortcutKeys, isFullscreen, showPanel, showNav, enableShortcuts),
menuHighlighted: api.versionUpdateAvailable(),
};
};
diff --git a/lib/ui/src/settings/about.js b/lib/ui/src/settings/about.js
index ea37e090ca3..96762c69714 100644
--- a/lib/ui/src/settings/about.js
+++ b/lib/ui/src/settings/about.js
@@ -12,7 +12,7 @@ import {
Icons,
Tabs,
Link,
- DocumentFormatting,
+ DocumentWrapper,
} from '@storybook/components';
import SettingsFooter from './SettingsFooter';
@@ -159,9 +159,9 @@ const AboutScreen = ({ latest, current, onClose }) => {
Read full changelog
-
+
{latest.info.plain}
-
+
) : (
@@ -179,7 +179,7 @@ const AboutScreen = ({ latest, current, onClose }) => {
{canUpdate && (
-
+
Upgrade all Storybook packages to latest:
@@ -193,7 +193,7 @@ const AboutScreen = ({ latest, current, onClose }) => {
yarn upgrade-interactive --latest
-
+
)}
diff --git a/now.json b/now.json
index 17dcab3f682..26ca24c59d4 100644
--- a/now.json
+++ b/now.json
@@ -4,78 +4,20 @@
"public": true,
"builds": [
{
- "src": "examples/angular-cli/package.json",
+ "src": "package.json",
"use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/cra-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/cra-ts-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/ember-cli/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/html-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/marko-cli/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/mithril-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/official-storybook/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/polymer-cli/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/preact-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/riot-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/svelte-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/vue-kitchen-sink/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
- },
- {
- "src": "examples/cra-react15/package.json",
- "use": "@now/static-build",
- "config": { "distDir": "storybook-static" }
+ "config": {
+ "distDir": "built-storybooks"
+ }
}
],
+ "build": {
+ "env": {
+ "DOTENV_DISPLAY_WARNING": "none",
+ "STORYBOOK_EXAMPLE_APP":"true"
+ }
+ },
"routes": [
- { "src": "/docs", "dest": "https://storybook.js.org" },
- { "src": "/(.*)", "dest": "/examples/$1" }
+ { "src": "/(.*)", "dest": "/built-storybooks/$1" }
]
}
diff --git a/package.json b/package.json
index 91f08a3925c..02a0a269589 100644
--- a/package.json
+++ b/package.json
@@ -63,6 +63,7 @@
"lint:js": "cross-env NODE_ENV=production eslint --cache --cache-location=.cache/eslint --ext .js,.jsx,.json,.html,.ts,.tsx,.mjs --report-unused-disable-directives",
"lint:md": "remark -q",
"lint:package": "sort-package-json",
+ "now-build": "./scripts/now.js",
"publish:crna": "yarn --cwd examples-native/crna-kitchen-sink expo publish",
"publish:debug": "npm run publish:latest -- --npm-tag=debug --no-push",
"publish:latest": "lerna publish --exact --concurrency 1 --force-publish",
@@ -210,6 +211,7 @@
"react-dom": "^16.8.3",
"react-test-renderer": "^16.8.3",
"react-testing-library": "^6.0.0",
+ "recursive-copy": "^2.0.10",
"regenerator-runtime": "^0.12.1",
"remark-cli": "^6.0.1",
"remark-lint": "^6.0.4",
diff --git a/scripts/now.js b/scripts/now.js
new file mode 100755
index 00000000000..9d8af1993b9
--- /dev/null
+++ b/scripts/now.js
@@ -0,0 +1,187 @@
+#!/usr/bin/env node
+
+/* eslint-disable global-require */
+
+const { spawn } = require('child_process');
+const { promisify } = require('util');
+const {
+ readdir: readdirRaw,
+ readFile: readFileRaw,
+ writeFile: writeFileRaw,
+ statSync,
+ cop,
+} = require('fs');
+const { join } = require('path');
+
+const readdir = promisify(readdirRaw);
+const readFile = promisify(readFileRaw);
+const writeFile = promisify(writeFileRaw);
+
+const p = l => join(__dirname, '..', ...l);
+const logger = console;
+
+const exec = async (command, args = [], options = {}) =>
+ new Promise((resolve, reject) => {
+ const child = spawn(command, args, { ...options, stdio: 'inherit' });
+
+ child.on('close', code => {
+ if (code) {
+ reject();
+ } else {
+ resolve();
+ }
+ });
+ });
+
+const hasBuildScript = async l => {
+ const text = await readFile(l, 'utf8');
+ const json = JSON.parse(text);
+
+ return !!json.scripts['build-storybook'];
+};
+
+const handleExamples = async files => {
+ const deployables = files.filter(f => {
+ const packageJsonLocation = p(['examples', f, 'package.json']);
+ const stats = statSync(packageJsonLocation);
+
+ return stats.isFile() && hasBuildScript(packageJsonLocation);
+ });
+
+ await deployables.reduce(async (acc, d) => {
+ await acc;
+
+ logger.log('');
+ logger.log(
+ `-----------------${Array(d.length)
+ .fill('-')
+ .join('')}`
+ );
+ logger.log(`▶️ building: ${d}`);
+ logger.log(
+ `-----------------${Array(d.length)
+ .fill('-')
+ .join('')}`
+ );
+ const out = p(['built-storybooks', d]);
+ const cwd = p(['examples', d]);
+
+ await exec(`yarn`, [`build-storybook`, `--output-dir=${out}`, '--quiet'], { cwd });
+
+ logger.log('-------');
+ logger.log('✅ done');
+ logger.log('-------');
+ }, Promise.resolve());
+
+ const copy = require('recursive-copy');
+ const target = 'official-storybook';
+ const copyables = deployables.filter(f => f !== target);
+
+ await copyables.reduce(async (acc, d) => {
+ await acc;
+
+ logger.log(`💿 copy ${d} to built-storybooks`);
+ const to = p(['built-storybooks', target, d]);
+ const from = p(['built-storybooks', d]);
+
+ await copy(from, to, {
+ overwrite: true,
+ });
+ }, Promise.resolve());
+
+ logger.log('-------');
+ logger.log('✅ done');
+ logger.log('-------');
+ logger.log('');
+
+ logger.log(`📑 creating index`);
+
+ const indexLocation = p(['built-storybooks', 'index.html']);
+ const indexContent = `
+
+
+
+
+ open
+
+
+ ${deployables.map(i => `${i} `).join('\n')}
+
+
+
+ `;
+
+ await writeFile(indexLocation, indexContent);
+
+ logger.log('-------');
+ logger.log('✅ done');
+ logger.log('-------');
+};
+
+const run = async () => {
+ await exec('yarn', ['bootstrap', '--core']);
+
+ const examples = await readdir(p(['examples']));
+
+ await handleExamples(examples);
+};
+
+run();
diff --git a/scripts/verdaccio.yaml b/scripts/verdaccio.yaml
index 38c5db930e2..e00f8c8dbf4 100644
--- a/scripts/verdaccio.yaml
+++ b/scripts/verdaccio.yaml
@@ -26,6 +26,14 @@ packages:
access: $all
publish: $all
+ 'sb':
+ access: $all
+ publish: $all
+
+ 'storybook':
+ access: $all
+ publish: $all
+
'@*/*':
access: $all
publish: $all
diff --git a/yarn.lock b/yarn.lock
index 79b3acee04e..21fa494da5d 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -3787,9 +3787,9 @@
"@types/lodash" "*"
"@types/lodash@*", "@types/lodash@^4.14.129":
- version "4.14.135"
- resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.135.tgz#d2607c35dd68f70c2b35ba020c667493dedd8447"
- integrity sha512-Ed+tSZ9qM1oYpi5kzdsBuOzcAIn1wDW+e8TFJ50IMJMlSopGdJgKAbhHzN6h1E1OfjlGOr2JepzEWtg9NIfoNg==
+ version "4.14.136"
+ resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.136.tgz#413e85089046b865d960c9ff1d400e04c31ab60f"
+ integrity sha512-0GJhzBdvsW2RUccNHOBkabI8HZVdOXmXbXhuKlDEd5Vv12P7oAVGfomGp3Ne21o5D/qu1WmthlNKFaoZJJeErA==
"@types/mini-css-extract-plugin@^0.2.1":
version "0.2.1"
@@ -5438,10 +5438,10 @@ aws4@^1.8.0:
resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.8.0.tgz#f0e003d9ca9e7f59c7a508945d7b2ef9a04a542f"
integrity sha512-ReZxvNHIOv88FlT7rxcXIIC0fPt4KZqZbOlivyWtXLt8ESx84zd3kMC6iK5jVeS2qt+g7ftS7ye4fi06X5rtRQ==
-axe-core@^3.2.2:
- version "3.2.2"
- resolved "https://registry.yarnpkg.com/axe-core/-/axe-core-3.2.2.tgz#b06d6e9ae4636d706068843272bfaeed3fe97362"
- integrity sha512-gAy4kMSPpuRJV3mwictJqlg5LhE84Vw2CydKdC4tvrLhR6+G3KW51zbL/vYujcLA2jvWOq3HMHrVeNuw+mrLVA==
+axe-core@^3.3.2:
+ version "3.3.2"
+ resolved "https://registry.yarnpkg.com/axe-core/-/axe-core-3.3.2.tgz#7baf3c55a5cf1621534a2c38735f5a1bf2f7e1a8"
+ integrity sha512-lRdxsRt7yNhqpcXQk1ao1BL73OZDzmFCWOG0mC4tGR/r14ohH2payjHwCMQjHGbBKm924eDlmG7utAGHiX/A6g==
axios-retry@^3.0.2:
version "3.1.2"
@@ -11121,6 +11121,11 @@ ember-test-waiters@^1.0.0:
dependencies:
ember-cli-babel "^7.1.2"
+emitter-mixin@0.0.3:
+ version "0.0.3"
+ resolved "https://registry.yarnpkg.com/emitter-mixin/-/emitter-mixin-0.0.3.tgz#5948cb286f2e48edc3b251a7cfc1f7883396d65c"
+ integrity sha1-WUjLKG8uSO3DslGnz8H3iDOW1lw=
+
emoji-regex@^7.0.1, emoji-regex@^7.0.2:
version "7.0.3"
resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-7.0.3.tgz#933a04052860c85e83c122479c4748a8e4c72156"
@@ -11329,7 +11334,7 @@ err-code@^1.0.0:
resolved "https://registry.yarnpkg.com/err-code/-/err-code-1.1.2.tgz#06e0116d3028f6aef4806849eb0ea6a748ae6960"
integrity sha1-BuARbTAo9q70gGhJ6w6mp0iuaWA=
-errno@^0.1.1, errno@^0.1.3, errno@~0.1.7:
+errno@^0.1.1, errno@^0.1.2, errno@^0.1.3, errno@~0.1.7:
version "0.1.7"
resolved "https://registry.yarnpkg.com/errno/-/errno-0.1.7.tgz#4684d71779ad39af177e3f007996f7c67c852618"
integrity sha512-MfrRBDWzIWifgq6tJj60gkAwtLNb6sQPlcFrSOflcP1aFmmruKQ2wRnze/8V6kgyz7H3FF8Npzv78mZ7XLLflg==
@@ -11717,14 +11722,16 @@ eslint-scope@^4.0.0, eslint-scope@^4.0.3:
estraverse "^4.1.1"
eslint-utils@^1.3.1:
- version "1.3.1"
- resolved "https://registry.yarnpkg.com/eslint-utils/-/eslint-utils-1.3.1.tgz#9a851ba89ee7c460346f97cf8939c7298827e512"
- integrity sha512-Z7YjnIldX+2XMcjr7ZkgEsOj/bREONV60qYeB/bjMAqqqZ4zxKyWX+BOUkdmRmA9riiIPVvo5x86m5elviOk0Q==
+ version "1.4.2"
+ resolved "https://registry.yarnpkg.com/eslint-utils/-/eslint-utils-1.4.2.tgz#166a5180ef6ab7eb462f162fd0e6f2463d7309ab"
+ integrity sha512-eAZS2sEUMlIeCjBeubdj45dmBHQwPHWyBcT1VSYB7o9x9WRRqKxyUoiXlRjyAwzN7YEzHJlYg0NmzDRWx6GP4Q==
+ dependencies:
+ eslint-visitor-keys "^1.0.0"
eslint-visitor-keys@^1.0.0:
- version "1.0.0"
- resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz#3f3180fb2e291017716acb4c9d6d5b5c34a6a81d"
- integrity sha512-qzm/XxIbxm/FHyH341ZrbnMUpe+5Bocte9xkmFMzPMjRaZMcXww+MpBptFvtU+79L362nqiLhekCxCxDPaUMBQ==
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-1.1.0.tgz#e2a82cea84ff246ad6fb57f9bde5b46621459ec2"
+ integrity sha512-8y9YjtM1JBJU/A9Kc+SbaOV4y29sSWckBwMHa+FGtVj5gN/sbnKDf6xJUl+8g7FAij9LVaP8C24DUiH/f/2Z9A==
eslint@^5.10.0, eslint@^5.14.1, eslint@^5.16.0:
version "5.16.0"
@@ -11768,7 +11775,7 @@ eslint@^5.10.0, eslint@^5.14.1, eslint@^5.16.0:
table "^5.2.3"
text-table "^0.2.0"
-esm@^3.2.25, esm@^3.2.4:
+esm@3.2.25, esm@^3.2.25, esm@^3.2.4:
version "3.2.25"
resolved "https://registry.yarnpkg.com/esm/-/esm-3.2.25.tgz#342c18c29d56157688ba5ce31f8431fbb795cc10"
integrity sha512-U1suiZ2oDVWv4zPO56S0NcR5QriEahGtdN2OR6FiOG4WJvcjBVFB0qI4+eKoWFH483PKGuLuu6V8Z4T5g63UVA==
@@ -14164,6 +14171,11 @@ graceful-fs@^4.0.0, graceful-fs@^4.1.11, graceful-fs@^4.1.15, graceful-fs@^4.1.2
resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.0.tgz#8d8fdc73977cb04104721cb53666c1ca64cd328b"
integrity sha512-jpSvDPV4Cq/bgtpndIWbI5hmYxhQGHPC4d4cqBPb4DLniCfhJokdXhwhaDuLBGLQdvvRum/UiX6ECVIPvDXqdg==
+graceful-fs@^4.1.4:
+ version "4.2.2"
+ resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.2.tgz#6f0952605d0140c1cfdb138ed005775b92d67b02"
+ integrity sha512-IItsdsea19BoLC7ELy13q1iJFNmd7ofZH5+X/pJr90/nRoPEX0DJo1dHDbgtYWOhJhcCgMDTOw84RZ72q6lB+Q==
+
"graceful-readlink@>= 1.0.0":
version "1.0.1"
resolved "https://registry.yarnpkg.com/graceful-readlink/-/graceful-readlink-1.0.1.tgz#4cafad76bc62f02fa039b2f94e9a3dd3a391a725"
@@ -17340,6 +17352,11 @@ jszip@^3.1.3:
readable-stream "~2.3.6"
set-immediate-shim "~1.0.1"
+junk@^1.0.1:
+ version "1.0.3"
+ resolved "https://registry.yarnpkg.com/junk/-/junk-1.0.3.tgz#87be63488649cbdca6f53ab39bec9ccd2347f592"
+ integrity sha1-h75jSIZJy9ym9Tqzm+yczSNH9ZI=
+
just-debounce@^1.0.0:
version "1.0.0"
resolved "https://registry.yarnpkg.com/just-debounce/-/just-debounce-1.0.0.tgz#87fccfaeffc0b68cd19d55f6722943f929ea35ea"
@@ -19079,6 +19096,16 @@ math-random@^1.0.1:
resolved "https://registry.yarnpkg.com/math-random/-/math-random-1.0.4.tgz#5dd6943c938548267016d4e34f057583080c514c"
integrity sha512-rUxjysqif/BZQH2yhd5Aaq7vXMSx9NdEsQcyA07uEzIvxgI7zIr33gGsh+RU0/XjmQpCW7RsVof1vlkvQVCK5A==
+maximatch@^0.1.0:
+ version "0.1.0"
+ resolved "https://registry.yarnpkg.com/maximatch/-/maximatch-0.1.0.tgz#86cd8d6b04c9f307c05a6b9419906d0360fb13a2"
+ integrity sha1-hs2NawTJ8wfAWmuUGZBtA2D7E6I=
+ dependencies:
+ array-differ "^1.0.0"
+ array-union "^1.0.1"
+ arrify "^1.0.0"
+ minimatch "^3.0.0"
+
md5-file@^3.2.3:
version "3.2.3"
resolved "https://registry.yarnpkg.com/md5-file/-/md5-file-3.2.3.tgz#f9bceb941eca2214a4c0727f5e700314e770f06f"
@@ -20492,7 +20519,7 @@ node-releases@^1.1.13, node-releases@^1.1.23, node-releases@^1.1.3:
dependencies:
semver "^5.3.0"
-node-sass@4.12.0:
+node-sass@4.12.0, node-sass@^4.12.0:
version "4.12.0"
resolved "https://registry.yarnpkg.com/node-sass/-/node-sass-4.12.0.tgz#0914f531932380114a30cc5fa4fa63233a25f017"
integrity sha512-A1Iv4oN+Iel6EPv77/HddXErL2a+gZ4uBeZUy+a8O35CFYTXhgA8MgLCWBtwpGZdCvTvQ9d+bQxX/QC36GDPpQ==
@@ -24217,10 +24244,10 @@ react-dom@^16.8.3, react-dom@^16.8.4:
prop-types "^15.6.2"
scheduler "^0.13.6"
-react-draggable@^3.1.1:
- version "3.3.0"
- resolved "https://registry.yarnpkg.com/react-draggable/-/react-draggable-3.3.0.tgz#2ed7ea3f92e7d742d747f9e6324860606cd4d997"
- integrity sha512-U7/jD0tAW4T0S7DCPK0kkKLyL0z61sC/eqU+NUfDjnq+JtBKaYKDHpsK2wazctiA4alEzCXUnzkREoxppOySVw==
+react-draggable@^4.0.3:
+ version "4.0.3"
+ resolved "https://registry.yarnpkg.com/react-draggable/-/react-draggable-4.0.3.tgz#6b9f76f66431c47b9070e9b805bbc520df8ca481"
+ integrity sha512-4vD6zms+9QGeZ2RQXzlUBw8PBYUXy+dzYX5r22idjp9YwQKIIvD/EojL0rbjS1GK4C3P0rAJnmKa8gDQYWUDyA==
dependencies:
classnames "^2.2.5"
prop-types "^15.6.0"
@@ -25044,6 +25071,22 @@ recompose@^0.26.0:
hoist-non-react-statics "^2.3.1"
symbol-observable "^1.0.4"
+recursive-copy@^2.0.10:
+ version "2.0.10"
+ resolved "https://registry.yarnpkg.com/recursive-copy/-/recursive-copy-2.0.10.tgz#a39402f2270c5f8b562b48d438a42e2e6e5c644c"
+ integrity sha512-S9J9XJUnfZ2NUS3lK6lx6HWLl2nWui+f7AKuu+qoFs4ikEPYgZ3qKk1T6tmBnr7PzhtKnawE+6TREy9XQKmxCA==
+ dependencies:
+ del "^2.2.0"
+ emitter-mixin "0.0.3"
+ errno "^0.1.2"
+ graceful-fs "^4.1.4"
+ junk "^1.0.1"
+ maximatch "^0.1.0"
+ mkdirp "^0.5.1"
+ pify "^2.3.0"
+ promise "^7.0.1"
+ slash "^1.0.0"
+
recursive-readdir@2.2.2:
version "2.2.2"
resolved "https://registry.yarnpkg.com/recursive-readdir/-/recursive-readdir-2.2.2.tgz#9946fb3274e1628de6e36b2f6714953b4845094f"
@@ -27370,9 +27413,9 @@ store2@^2.7.1:
integrity sha512-zzzP5ZY6QWumnAFV6kBRbS44pUMcpZBNER5DWUe1HETlaKXqLcCQxbNu6IHaKr1pUsjuhUGBdOy8sWKmMkL6pQ==
storybook-addon-vue-info@^1.2.1:
- version "1.2.1"
- resolved "https://registry.yarnpkg.com/storybook-addon-vue-info/-/storybook-addon-vue-info-1.2.1.tgz#41dc5b16392640866878441abb8c060414803c78"
- integrity sha512-B4l93SBHjVilrbzZ7sFMyfBlwm5nCuecU+ZOhpeci5zOK0ixTgCAwg/zIWBpCT9yI0iUAZv70xaIO9ICxm3vOQ==
+ version "1.2.2"
+ resolved "https://registry.yarnpkg.com/storybook-addon-vue-info/-/storybook-addon-vue-info-1.2.2.tgz#3479ed65fe2cfc87bafff4f711716851986b8101"
+ integrity sha512-DkT2SjwAJxuv/G5f1NZ8DH8Q+1u2FIfutbdgJTbah+ZAQz5PeLS5pkY9/zL4oETs6DA7hB8NXBeFWiZoPGbEfw==
dependencies:
clone "^2.1.2"
dedent-tabs "^0.8.0"
@@ -27870,9 +27913,9 @@ svelte-loader@^2.13.4:
svelte-dev-helper "^1.1.9"
svelte@^3.4.1:
- version "3.6.4"
- resolved "https://registry.yarnpkg.com/svelte/-/svelte-3.6.4.tgz#6b04e9b261af17e2ee9b3bf92104e8a0dbcd35ff"
- integrity sha512-oI2FAZK4iLUKwwf/ltn9Y9185+03LX3uMzKlVnLrkW5P1xazJnCOHBX5mI5FeyhSIPIyueTQtqJ9cQaEUMMXgA==
+ version "3.9.1"
+ resolved "https://registry.yarnpkg.com/svelte/-/svelte-3.9.1.tgz#f0dc0993d80e69da46f7b21d858895c37c2fd419"
+ integrity sha512-oy6FjffuCzPueBnzs8SxkV5VaOS+VX/eUywgE9LCA+skKdXt7wFlpKH3DNi1Uy+K23sb9bzlWxn0AaDQOnEUEw==
sver-compat@^1.5.0:
version "1.5.0"
@@ -29989,7 +30032,7 @@ vue-template-es2015-compiler@^1.9.0:
resolved "https://registry.yarnpkg.com/vue-template-es2015-compiler/-/vue-template-es2015-compiler-1.9.1.tgz#1ee3bc9a16ecbf5118be334bb15f9c46f82f5825"
integrity sha512-4gDntzrifFnCEvyoO8PqyJDmguXgVPxKiIxrBKjIowvL9l+N66196+72XVYR8BBf1Uv1Fgt3bGevJ+sEmxfZzw==
-vue@*, vue@^2.6.8:
+vue@^2.6.8:
version "2.6.10"
resolved "https://registry.yarnpkg.com/vue/-/vue-2.6.10.tgz#a72b1a42a4d82a721ea438d1b6bf55e66195c637"
integrity sha512-ImThpeNU9HbdZL3utgMCq0oiMzAkt1mcgy3/E6zWC/G6AaQoeuFdsl9nDhTDU3X1R6FK7nsIUuRACVcjI+A2GQ==