From c1e359dedd19fd522e0c2ea383dd8bdfcaeef589 Mon Sep 17 00:00:00 2001 From: Tom Coleman Date: Thu, 22 Mar 2018 21:16:27 +1100 Subject: [PATCH] Refactor all startup code into `@storybook/core` also --- app/angular/package.json | 1 - app/angular/src/client/preview/index.js | 63 ++--------------------- app/polymer/package.json | 1 - app/polymer/src/client/preview/index.js | 63 ++--------------------- app/react/package.json | 1 - app/react/src/client/preview/index.js | 63 ++--------------------- app/vue/package.json | 1 - app/vue/src/client/preview/index.js | 66 ++----------------------- lib/core/package.json | 1 + lib/core/src/client/preview/index.js | 3 +- lib/core/src/client/preview/start.js | 65 ++++++++++++++++++++++++ yarn.lock | 2 +- 12 files changed, 84 insertions(+), 246 deletions(-) create mode 100644 lib/core/src/client/preview/start.js diff --git a/app/angular/package.json b/app/angular/package.json index 2eac21ae8e7..03ea8b24e07 100644 --- a/app/angular/package.json +++ b/app/angular/package.json @@ -49,7 +49,6 @@ "react": "^16.0.0", "react-dev-utils": "^5.0.0", "react-dom": "^16.0.0", - "redux": "^3.6.0", "sass-loader": "^6.0.7", "ts-loader": "^3.5.0", "uglifyjs-webpack-plugin": "^1.2.4", diff --git a/app/angular/src/client/preview/index.js b/app/angular/src/client/preview/index.js index 5f4e9bb8d16..b04f8439b60 100644 --- a/app/angular/src/client/preview/index.js +++ b/app/angular/src/client/preview/index.js @@ -1,30 +1,9 @@ -import { window, navigator } from 'global'; -import { createStore } from 'redux'; -import addons from '@storybook/addons'; -import createChannel from '@storybook/channel-postmessage'; -import { handleKeyboardShortcuts } from '@storybook/ui/dist/libs/key_events'; -import { - StoryStore, - ClientApi, - ConfigApi, - Actions, - reducer, - syncUrlWithStore, -} from '@storybook/core/client'; +import { start } from '@storybook/core/client'; + import render from './render'; -// check whether we're running on node/browser -const isBrowser = - navigator && - navigator.userAgent && - navigator.userAgent !== 'storyshots' && - !(navigator.userAgent.indexOf('Node.js') > -1) && - !(navigator.userAgent.indexOf('jsdom') > -1); +const { clientApi, configApi, forceReRender } = start(render); -const storyStore = new StoryStore(); -const reduxStore = createStore(reducer); -const context = { storyStore, reduxStore }; -const clientApi = new ClientApi(context); export const { storiesOf, setAddon, @@ -34,39 +13,5 @@ export const { getStorybook, } = clientApi; -let channel; -if (isBrowser) { - // create preview channel - channel = createChannel({ page: 'preview' }); - channel.on('setCurrentStory', data => { - reduxStore.dispatch(Actions.selectStory(data.kind, data.story)); - }); - addons.setChannel(channel); - Object.assign(context, { channel }); - - syncUrlWithStore(reduxStore); - - // Handle keyboard shortcuts - window.onkeydown = handleKeyboardShortcuts(channel); -} - -// Provide access to external scripts if `window` is defined. -// NOTE this is different to isBrowser, primarily for the JSDOM use case -if (typeof window !== 'undefined') { - window.__STORYBOOK_CLIENT_API__ = clientApi; - window.__STORYBOOK_ADDONS_CHANNEL__ = channel; // may not be defined -} - -const configApi = new ConfigApi({ ...context, clearDecorators }); export const { configure } = configApi; - -// initialize the UI -const renderUI = () => { - if (isBrowser) { - render(context); - } -}; - -reduxStore.subscribe(renderUI); - -export const forceReRender = () => render(context, true); +export { forceReRender }; diff --git a/app/polymer/package.json b/app/polymer/package.json index 4e3e10a74fb..51c0f1c1dab 100644 --- a/app/polymer/package.json +++ b/app/polymer/package.json @@ -51,7 +51,6 @@ "react": "^16.0.0", "react-dev-utils": "^5.0.0", "react-dom": "^16.0.0", - "redux": "^3.7.2", "uglifyjs-webpack-plugin": "^1.2.4", "util-deprecate": "^1.0.2", "webpack": "^3.11.0", diff --git a/app/polymer/src/client/preview/index.js b/app/polymer/src/client/preview/index.js index 063e2edb583..b04f8439b60 100644 --- a/app/polymer/src/client/preview/index.js +++ b/app/polymer/src/client/preview/index.js @@ -1,32 +1,9 @@ -import { createStore } from 'redux'; -import addons from '@storybook/addons'; -import { navigator, window } from 'global'; -import createChannel from '@storybook/channel-postmessage'; -import { handleKeyboardShortcuts } from '@storybook/ui/dist/libs/key_events'; -import { - StoryStore, - ClientApi, - ConfigApi, - Actions, - reducer, - syncUrlWithStore, -} from '@storybook/core/client'; +import { start } from '@storybook/core/client'; import render from './render'; -// check whether we're running on node/browser -const isBrowser = - navigator && - navigator.userAgent && - navigator.userAgent !== 'storyshots' && - !(navigator.userAgent.indexOf('Node.js') > -1) && - !(navigator.userAgent.indexOf('jsdom') > -1); +const { clientApi, configApi, forceReRender } = start(render); -const storyStore = new StoryStore(); -const reduxStore = createStore(reducer); -const context = { storyStore, reduxStore }; - -const clientApi = new ClientApi(context); export const { storiesOf, setAddon, @@ -36,39 +13,5 @@ export const { getStorybook, } = clientApi; -let channel; -if (isBrowser) { - // setup preview channel - channel = createChannel({ page: 'preview' }); - channel.on('setCurrentStory', data => { - reduxStore.dispatch(Actions.selectStory(data.kind, data.story)); - }); - addons.setChannel(channel); - Object.assign(context, { channel }); - - syncUrlWithStore(reduxStore); - - // Handle keyboard shortcuts - window.onkeydown = handleKeyboardShortcuts(channel); -} - -// Provide access to external scripts if `window` is defined. -// NOTE this is different to isBrowser, primarily for the JSDOM use case -if (typeof window !== 'undefined') { - window.__STORYBOOK_CLIENT_API__ = clientApi; - window.__STORYBOOK_ADDONS_CHANNEL__ = channel; // may not be defined -} - -const configApi = new ConfigApi({ clearDecorators, ...context }); export const { configure } = configApi; - -// initialize the UI -const renderUI = () => { - if (isBrowser) { - render(context); - } -}; - -reduxStore.subscribe(renderUI); - -export const forceReRender = () => render(context, true); +export { forceReRender }; diff --git a/app/react/package.json b/app/react/package.json index 0fdcf085682..ec33b5f4e67 100644 --- a/app/react/package.json +++ b/app/react/package.json @@ -57,7 +57,6 @@ "markdown-loader": "^2.0.2", "prop-types": "^15.6.1", "react-dev-utils": "^5.0.0", - "redux": "^3.7.2", "uglifyjs-webpack-plugin": "^1.2.4", "util-deprecate": "^1.0.2", "webpack": "^3.11.0", diff --git a/app/react/src/client/preview/index.js b/app/react/src/client/preview/index.js index 063e2edb583..b04f8439b60 100644 --- a/app/react/src/client/preview/index.js +++ b/app/react/src/client/preview/index.js @@ -1,32 +1,9 @@ -import { createStore } from 'redux'; -import addons from '@storybook/addons'; -import { navigator, window } from 'global'; -import createChannel from '@storybook/channel-postmessage'; -import { handleKeyboardShortcuts } from '@storybook/ui/dist/libs/key_events'; -import { - StoryStore, - ClientApi, - ConfigApi, - Actions, - reducer, - syncUrlWithStore, -} from '@storybook/core/client'; +import { start } from '@storybook/core/client'; import render from './render'; -// check whether we're running on node/browser -const isBrowser = - navigator && - navigator.userAgent && - navigator.userAgent !== 'storyshots' && - !(navigator.userAgent.indexOf('Node.js') > -1) && - !(navigator.userAgent.indexOf('jsdom') > -1); +const { clientApi, configApi, forceReRender } = start(render); -const storyStore = new StoryStore(); -const reduxStore = createStore(reducer); -const context = { storyStore, reduxStore }; - -const clientApi = new ClientApi(context); export const { storiesOf, setAddon, @@ -36,39 +13,5 @@ export const { getStorybook, } = clientApi; -let channel; -if (isBrowser) { - // setup preview channel - channel = createChannel({ page: 'preview' }); - channel.on('setCurrentStory', data => { - reduxStore.dispatch(Actions.selectStory(data.kind, data.story)); - }); - addons.setChannel(channel); - Object.assign(context, { channel }); - - syncUrlWithStore(reduxStore); - - // Handle keyboard shortcuts - window.onkeydown = handleKeyboardShortcuts(channel); -} - -// Provide access to external scripts if `window` is defined. -// NOTE this is different to isBrowser, primarily for the JSDOM use case -if (typeof window !== 'undefined') { - window.__STORYBOOK_CLIENT_API__ = clientApi; - window.__STORYBOOK_ADDONS_CHANNEL__ = channel; // may not be defined -} - -const configApi = new ConfigApi({ clearDecorators, ...context }); export const { configure } = configApi; - -// initialize the UI -const renderUI = () => { - if (isBrowser) { - render(context); - } -}; - -reduxStore.subscribe(renderUI); - -export const forceReRender = () => render(context, true); +export { forceReRender }; diff --git a/app/vue/package.json b/app/vue/package.json index ff14a4eab5a..cccabe61939 100644 --- a/app/vue/package.json +++ b/app/vue/package.json @@ -49,7 +49,6 @@ "react": "^16.2.0", "react-dev-utils": "^5.0.0", "react-dom": "^16.2.0", - "redux": "^3.7.2", "uglifyjs-webpack-plugin": "^1.2.4", "util-deprecate": "^1.0.2", "webpack": "^3.11.0", diff --git a/app/vue/src/client/preview/index.js b/app/vue/src/client/preview/index.js index c916f494d7a..b082c2545dd 100644 --- a/app/vue/src/client/preview/index.js +++ b/app/vue/src/client/preview/index.js @@ -1,30 +1,7 @@ -import { createStore } from 'redux'; -import addons from '@storybook/addons'; -import createChannel from '@storybook/channel-postmessage'; -import { navigator, window } from 'global'; -import { handleKeyboardShortcuts } from '@storybook/ui/dist/libs/key_events'; -import { - StoryStore, - ClientApi, - ConfigApi, - Actions, - reducer, - syncUrlWithStore, -} from '@storybook/core/client'; +import { start } from '@storybook/core/client'; import render from './render'; -// check whether we're running on node/browser -const isBrowser = - navigator && - navigator.userAgent && - navigator.userAgent !== 'storyshots' && - !(navigator.userAgent.indexOf('Node.js') > -1) && - !(navigator.userAgent.indexOf('jsdom') > -1); - -const storyStore = new StoryStore(); -const reduxStore = createStore(reducer); - const createWrapperComponent = Target => ({ functional: true, render(h, c) { @@ -42,8 +19,9 @@ const decorateStory = (getStory, decorators) => }, getStory ); -const context = { storyStore, reduxStore, decorateStory }; -const clientApi = new ClientApi(context); + +const { clientApi, configApi, forceReRender } = start(render, { decorateStory }); + export const { storiesOf, setAddon, @@ -53,39 +31,5 @@ export const { getStorybook, } = clientApi; -let channel; -if (isBrowser) { - // create preview channel - channel = createChannel({ page: 'preview' }); - channel.on('setCurrentStory', data => { - reduxStore.dispatch(Actions.selectStory(data.kind, data.story)); - }); - addons.setChannel(channel); - Object.assign(context, { channel }); - - syncUrlWithStore(reduxStore); - - // Handle keyboard shortcuts - window.onkeydown = handleKeyboardShortcuts(channel); -} - -// Provide access to external scripts if `window` is defined. -// NOTE this is different to isBrowser, primarily for the JSDOM use case -if (typeof window !== 'undefined') { - window.__STORYBOOK_CLIENT_API__ = clientApi; - window.__STORYBOOK_ADDONS_CHANNEL__ = channel; // may not be defined -} - -const configApi = new ConfigApi({ ...context, clearDecorators }); export const { configure } = configApi; - -// initialize the UI -const renderUI = () => { - if (isBrowser) { - render(context); - } -}; - -reduxStore.subscribe(renderUI); - -export const forceReRender = () => render(context, true); +export { forceReRender }; diff --git a/lib/core/package.json b/lib/core/package.json index 81a0181eb4a..79194b5daab 100644 --- a/lib/core/package.json +++ b/lib/core/package.json @@ -39,6 +39,7 @@ "qs": "^6.5.1", "react": "^16.0.0", "react-dom": "^16.0.0", + "redux": "^3.7.2", "serve-favicon": "^2.4.5", "shelljs": "^0.8.1", "style-loader": "^0.20.3", diff --git a/lib/core/src/client/preview/index.js b/lib/core/src/client/preview/index.js index f7597eec562..394707f5a58 100644 --- a/lib/core/src/client/preview/index.js +++ b/lib/core/src/client/preview/index.js @@ -1,3 +1,4 @@ +import start from './start'; import * as Actions from './actions'; import ClientApi from './client_api'; import ConfigApi from './config_api'; @@ -5,4 +6,4 @@ import StoryStore from './story_store'; import reducer from './reducer'; import syncUrlWithStore from './syncUrlWithStore'; -export default { Actions, ClientApi, ConfigApi, StoryStore, reducer, syncUrlWithStore }; +export default { start, Actions, ClientApi, ConfigApi, StoryStore, reducer, syncUrlWithStore }; diff --git a/lib/core/src/client/preview/start.js b/lib/core/src/client/preview/start.js new file mode 100644 index 00000000000..426f887ad95 --- /dev/null +++ b/lib/core/src/client/preview/start.js @@ -0,0 +1,65 @@ +import { createStore } from 'redux'; +import addons from '@storybook/addons'; +import { navigator, window } from 'global'; +import createChannel from '@storybook/channel-postmessage'; +import { handleKeyboardShortcuts } from '@storybook/ui/dist/libs/key_events'; + +import StoryStore from './story_store'; +import ClientApi from './client_api'; +import ConfigApi from './config_api'; +import reducer from './reducer'; +import * as Actions from './actions'; +import syncUrlWithStore from './syncUrlWithStore'; + +export default function start(render, { decorateStory } = {}) { + // check whether we're running on node/browser + const isBrowser = + navigator && + navigator.userAgent && + navigator.userAgent !== 'storyshots' && + !(navigator.userAgent.indexOf('Node.js') > -1) && + !(navigator.userAgent.indexOf('jsdom') > -1); + + const storyStore = new StoryStore(); + const reduxStore = createStore(reducer); + const context = { storyStore, reduxStore, decorateStory }; + + const clientApi = new ClientApi(context); + + let channel; + if (isBrowser) { + // setup preview channel + channel = createChannel({ page: 'preview' }); + channel.on('setCurrentStory', data => { + reduxStore.dispatch(Actions.selectStory(data.kind, data.story)); + }); + addons.setChannel(channel); + Object.assign(context, { channel }); + + syncUrlWithStore(reduxStore); + + // Handle keyboard shortcuts + window.onkeydown = handleKeyboardShortcuts(channel); + } + + // Provide access to external scripts if `window` is defined. + // NOTE this is different to isBrowser, primarily for the JSDOM use case + if (typeof window !== 'undefined') { + window.__STORYBOOK_CLIENT_API__ = clientApi; + window.__STORYBOOK_ADDONS_CHANNEL__ = channel; // may not be defined + } + + const { clearDecorators } = clientApi; + const configApi = new ConfigApi({ clearDecorators, ...context }); + + // initialize the UI + const renderUI = () => { + if (isBrowser) { + render(context); + } + }; + + reduxStore.subscribe(renderUI); + + return { context, clientApi, configApi, forceReRender: () => render(context, true) }; +} diff --git a/yarn.lock b/yarn.lock index 7526c56f04f..d8095f937ba 100644 --- a/yarn.lock +++ b/yarn.lock @@ -13395,7 +13395,7 @@ reduce-function-call@^1.0.1: dependencies: balanced-match "^0.4.2" -redux@^3.6.0, redux@^3.7.2: +redux@^3.7.2: version "3.7.2" resolved "https://registry.yarnpkg.com/redux/-/redux-3.7.2.tgz#06b73123215901d25d065be342eb026bc1c8537b" dependencies: