diff --git a/CHANGELOG.md b/CHANGELOG.md index 136bedd23de..13873380eda 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,50 @@ +# 4.0.0-alpha.14 + +2018-July-11 + +#### Bug Fixes + +- Upgrade universal-dotenv to fix core-js dependency [#3874](https://github.com/storybooks/storybook/pull/3874) + +# 4.0.0-alpha.13 + +2018-July-09 + +#### Features + +- Refactor addon-jest to use a parameter-based pattern [#3678](https://github.com/storybooks/storybook/pull/3678) + +#### Bug Fixes + +- Upgrade universal-dotenv to fix babel-runtime [#3863](https://github.com/storybooks/storybook/pull/3863) + +#### Maintenance + +- Added a test for parameter combination [#3844](https://github.com/storybooks/storybook/pull/3844) + +# 4.0.0-alpha.12 + +2018-July-03 + +#### Bug Fixes + +- Fix non-polyfilled themed UI components [#3829](https://github.com/storybooks/storybook/pull/3829) + +# 4.0.0-alpha.11 + +2018-July-02 + +#### Features + +- Storybook UI theming [#3628](https://github.com/storybooks/storybook/pull/3628) +- Replaced 'dotenv-webpack' with 'universal-dotenv' to support multiple dot env files (like CRA) [#3744](https://github.com/storybooks/storybook/pull/3744) +- Support other type of webpack configs [#3785](https://github.com/storybooks/storybook/pull/3785) + +#### Bug Fixes + +- Marko: fix welcome component [#3796](https://github.com/storybooks/storybook/pull/3796) +- Addon-a11y: Run analysis on demand [#3690](https://github.com/storybooks/storybook/pull/3690) + # 4.0.0-alpha.10 2018-June-21 diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 975d3d50656..267258d62b8 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -71,8 +71,15 @@ Before these tests are ran, the project must be bootstrapped with the React Nati `yarn test --image` -This option executes tests from `/examples/cra-kitchen-sink` -In order for the image snapshots to be correctly generated, you must have static build of the storybook up-to-date. +This option executes tests from `/examples/official-storybook` +In order for the image snapshots to be correctly generated, you must have static build of the storybook up-to-date : + +```javascript +cd examples/official-storybook +yarn build-storybook +cd ../.. +yarn test --image +``` Puppeteer is used to launch and grab screenshots of example pages, while jest is used to assert matching images. (just like integration tests) @@ -331,7 +338,7 @@ yarn bootstrap --reset --core ```sh # publish and tag the release -npm run publish -- --concurrency 1 --npm-tag=alpha --force-publish=* +npm run publish:alpha # update the release page open https://github.com/storybooks/storybook/releases @@ -355,7 +362,7 @@ git commit -m "Changelog for vX.Y" yarn bootstrap --reset --core # publish and tag the release -npm run publish -- --concurrency 1 --force-publish=* +npm run publish # update the release page open https://github.com/storybooks/storybook/releases diff --git a/addons/a11y/README.md b/addons/a11y/README.md index c150afd5264..df2c841e0f4 100755 --- a/addons/a11y/README.md +++ b/addons/a11y/README.md @@ -67,6 +67,24 @@ storiesOf('button', module) )); ``` +If you want to add a11y globally to your stories, you can use the global Storybook decorator in your *.storybook/config.js* file: + +```js +import { configure, addDecorator } from '@storybook/react'; +import { checkA11y } from '@storybook/addon-a11y'; + +// pick all stories.js files within the src/ folder +const req = require.context('../src', true, /stories\.js$/); + +addDecorator(checkA11y); + +function loadStories() { + req.keys().forEach(filename => req(filename)); +} + +configure(loadStories, module); +``` + ## Roadmap * Make UI accessibile diff --git a/addons/a11y/package.json b/addons/a11y/package.json index e7d37166fa2..5e2c350895f 100644 --- a/addons/a11y/package.json +++ b/addons/a11y/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-a11y", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "a11y addon for storybook", "keywords": [ "a11y", @@ -25,16 +25,16 @@ "prepare": "node ../../scripts/prepare.js" }, "dependencies": { - "@storybook/addons": "4.0.0-alpha.10", - "@storybook/client-logger": "4.0.0-alpha.10", - "@storybook/components": "4.0.0-alpha.10", - "@storybook/core-events": "4.0.0-alpha.10", + "@storybook/addons": "4.0.0-alpha.14", + "@storybook/client-logger": "4.0.0-alpha.14", + "@storybook/components": "4.0.0-alpha.14", + "@storybook/core-events": "4.0.0-alpha.14", "axe-core": "^3.0.3", "babel-runtime": "^6.26.0", - "emotion": "^9.1.3", "global": "^4.3.2", "prop-types": "^15.6.1", - "react-emotion": "^9.1.3" + "react-emotion": "^9.1.3", + "react-icons": "^2.2.7" }, "peerDependencies": { "react": "*", diff --git a/addons/a11y/src/components/Panel.js b/addons/a11y/src/components/Panel.js index 8e3617c81b1..d371ebc1035 100644 --- a/addons/a11y/src/components/Panel.js +++ b/addons/a11y/src/components/Panel.js @@ -1,65 +1,93 @@ import React, { Component } from 'react'; -import addons from '@storybook/addons'; +import PropTypes from 'prop-types'; import styled from 'react-emotion'; -import { CHECK_EVENT_ID } from '../shared'; +import { STORY_RENDERED } from '@storybook/core-events'; +import { ActionBar, ActionButton } from '@storybook/components'; + +import { CHECK_EVENT_ID, RERUN_EVENT_ID, REQUEST_CHECK_EVENT_ID } from '../shared'; import Tabs from './Tabs'; import Report from './Report'; -const Passes = styled('span')({ - color: '#0D6731', -}); +const Passes = styled('span')(({ theme }) => ({ + color: theme.successColor, +})); -const Violations = styled('span')({ - color: '#AC2300', -}); +const Violations = styled('span')(({ theme }) => ({ + color: theme.failColor, +})); class Panel extends Component { - constructor(props, ...args) { - super(props, ...args); - this.state = { - passes: [], - violations: [], - }; - this.channel = addons.getChannel(); + static propTypes = { + active: PropTypes.bool.isRequired, + channel: PropTypes.shape({ + on: PropTypes.func, + emit: PropTypes.func, + removeListener: PropTypes.func, + }).isRequired, + }; - this.onUpdate = this.onUpdate.bind(this); - } + state = { + passes: [], + violations: [], + }; componentDidMount() { - this.channel.on(CHECK_EVENT_ID, this.onUpdate); + this.props.channel.on(CHECK_EVENT_ID, this.onUpdate); + this.props.channel.on(STORY_RENDERED, this.requestCheck); + this.props.channel.on(RERUN_EVENT_ID, this.requestCheck); + } + + componentDidUpdate(prevProps) { + if (!prevProps.active && this.props.active) { + this.requestCheck(); + } } componentWillUnmount() { - this.channel.removeListener(CHECK_EVENT_ID, this.onUpdate); + this.props.channel.removeListener(CHECK_EVENT_ID, this.onUpdate); + this.props.channel.removeListener(STORY_RENDERED, this.requestCheck); + this.props.channel.removeListener(RERUN_EVENT_ID, this.requestCheck); } - onUpdate({ passes, violations }) { + onUpdate = ({ passes, violations }) => { this.setState({ passes, violations, }); - } + }; + + requestCheck = () => { + if (this.props.active) { + this.props.channel.emit(REQUEST_CHECK_EVENT_ID); + } + }; render() { const { passes, violations } = this.state; + const { active } = this.props; - return ( - {violations.length} Violations, - panel: , - }, - { - label: {passes.length} Passes, - panel: , - }, - ]} - /> - ); + return active ? ( +
+ {violations.length} Violations, + panel: , + }, + { + label: {passes.length} Passes, + panel: , + }, + ]} + /> + + RERUN TEST + +
+ ) : null; } } diff --git a/addons/a11y/src/components/Report/Info.js b/addons/a11y/src/components/Report/Info.js index 8b2679424d7..5c1a9b6cfaa 100644 --- a/addons/a11y/src/components/Report/Info.js +++ b/addons/a11y/src/components/Report/Info.js @@ -3,18 +3,18 @@ import PropTypes from 'prop-types'; import styled from 'react-emotion'; -const Wrapper = styled('div')({ - backgroundColor: 'rgb(234, 234, 234)', +const Wrapper = styled('div')(({ theme }) => ({ + backgroundColor: theme.barFill, padding: '12px', marginBottom: '10px', -}); +})); const Help = styled('p')({ margin: '0 0 12px', }); const Link = styled('a')({ marginTop: '12px', textDecoration: 'underline', - color: 'rgb(130, 130, 130)', + color: 'inherit', display: 'block', }); diff --git a/addons/a11y/src/components/Report/Item.js b/addons/a11y/src/components/Report/Item.js index cae1eeb03e5..b865b7ffaf2 100644 --- a/addons/a11y/src/components/Report/Item.js +++ b/addons/a11y/src/components/Report/Item.js @@ -2,24 +2,34 @@ import React, { Component } from 'react'; import PropTypes from 'prop-types'; import styled from 'react-emotion'; +import IoChevronRight from 'react-icons/lib/io/chevron-right'; import Info from './Info'; import Tags from './Tags'; import Elements from './Elements'; -const Wrapper = styled('div')({ +const Wrapper = styled('div')(({ theme }) => ({ padding: '0 14px', cursor: 'pointer', - borderBottom: '1px solid rgb(234, 234, 234)', -}); + borderBottom: theme.mainBorder, +})); -const HeaderBar = styled('button')({ +const HeaderBar = styled('button')(({ theme }) => ({ padding: '12px 0px', display: 'block', width: '100%', border: 0, background: 'none', -}); + color: 'inherit', + + borderTop: '3px solid transparent', + borderBottom: '3px solid transparent', + + '&:focus': { + outline: '0 none', + borderBottom: `3px solid ${theme.highlightColor}`, + }, +})); class Item extends Component { static propTypes = { @@ -46,7 +56,19 @@ class Item extends Component { return ( - {item.description} + + + {item.description} + {open && } {open && } {open && } diff --git a/addons/a11y/src/components/Report/RerunButton.js b/addons/a11y/src/components/Report/RerunButton.js deleted file mode 100644 index f419e3b197b..00000000000 --- a/addons/a11y/src/components/Report/RerunButton.js +++ /dev/null @@ -1,17 +0,0 @@ -import styled from 'react-emotion'; - -const RerunButton = styled('button')({ - position: 'absolute', - bottom: 0, - right: 0, - border: 'none', - borderTop: 'solid 1px rgba(0, 0, 0, 0.2)', - borderLeft: 'solid 1px rgba(0, 0, 0, 0.2)', - background: 'rgba(255, 255, 255, 0.5)', - padding: '5px 10px', - borderRadius: '4px 0 0 0', - color: 'rgba(0, 0, 0, 0.5)', - textTransform: 'uppercase', -}); - -export default RerunButton; diff --git a/addons/a11y/src/components/Report/Tags.js b/addons/a11y/src/components/Report/Tags.js index 2ad147f64cf..0800dbb6c29 100644 --- a/addons/a11y/src/components/Report/Tags.js +++ b/addons/a11y/src/components/Report/Tags.js @@ -9,14 +9,12 @@ const Wrapper = styled('div')({ margin: '12px 0', }); -const Item = styled('div')({ +const Item = styled('div')(({ theme }) => ({ margin: '0 6px', padding: '5px', - border: '1px solid rgb(234, 234, 234)', - borderRadius: '2px', - color: 'rgb(130, 130, 130)', - fontSize: '12px', -}); + border: theme.mainBorder, + borderRadius: theme.mainBorderRadius, +})); function Tags({ tags }) { return {tags.map(tag => {tag})}; diff --git a/addons/a11y/src/components/Report/index.js b/addons/a11y/src/components/Report/index.js index 7a6b3e75eaa..d9897dfa681 100644 --- a/addons/a11y/src/components/Report/index.js +++ b/addons/a11y/src/components/Report/index.js @@ -1,18 +1,9 @@ import React from 'react'; import PropTypes from 'prop-types'; -import addons from '@storybook/addons'; import { Placeholder } from '@storybook/components'; -import { RERUN_EVENT_ID } from '../../shared'; - -import RerunButton from './RerunButton'; import Item from './Item'; -function onRerunClick() { - const channel = addons.getChannel(); - channel.emit(RERUN_EVENT_ID); -} - const Report = ({ items, empty, passes }) => (
{items.length ? ( @@ -20,7 +11,6 @@ const Report = ({ items, empty, passes }) => ( ) : ( {empty} )} - Re-run tests
); diff --git a/addons/a11y/src/components/Tabs.js b/addons/a11y/src/components/Tabs.js index 2bdd3efcfff..970b287249c 100644 --- a/addons/a11y/src/components/Tabs.js +++ b/addons/a11y/src/components/Tabs.js @@ -6,13 +6,14 @@ import styled from 'react-emotion'; const Container = styled('div')({ width: '100%', position: 'relative', + minHeight: '100%', }); -const List = styled('div')({ - borderBottom: '1px solid rgb(234, 234, 234)', +const List = styled('div')(({ theme }) => ({ + borderBottom: theme.mainBorder, flexWrap: 'wrap', display: 'flex', -}); +})); const Item = styled('button')( ({ active }) => @@ -22,9 +23,7 @@ const Item = styled('button')( fontWeight: 600, } : {}, - { - color: 'rgb(68, 68, 68)', - fontSize: '11px', + ({ theme }) => ({ textDecoration: 'none', textTransform: 'uppercase', padding: '10px 15px', @@ -33,9 +32,16 @@ const Item = styled('button')( fontWeight: 500, opacity: 0.7, border: 'none', + borderTop: '3px solid transparent', + borderBottom: '3px solid transparent', background: 'none', flex: 1, - } + + '&:focus': { + outline: '0 none', + borderBottom: `3px solid ${theme.highlightColor}`, + }, + }) ); class Tabs extends Component { diff --git a/addons/a11y/src/index.js b/addons/a11y/src/index.js index e330e75f4a9..7d579a9b66d 100644 --- a/addons/a11y/src/index.js +++ b/addons/a11y/src/index.js @@ -4,7 +4,7 @@ import addons from '@storybook/addons'; import Events from '@storybook/core-events'; import { logger } from '@storybook/client-logger'; -import { CHECK_EVENT_ID, RERUN_EVENT_ID } from './shared'; +import { CHECK_EVENT_ID, REQUEST_CHECK_EVENT_ID } from './shared'; let axeOptions = {}; @@ -23,11 +23,10 @@ const runA11yCheck = () => { const a11ySubscription = () => { const channel = addons.getChannel(); - channel.on(Events.STORY_RENDERED, runA11yCheck); - channel.on(RERUN_EVENT_ID, runA11yCheck); + channel.on(REQUEST_CHECK_EVENT_ID, runA11yCheck); + return () => { - channel.removeListener(Events.STORY_RENDERED, runA11yCheck); - channel.removeListener(RERUN_EVENT_ID, runA11yCheck); + channel.removeListener(REQUEST_CHECK_EVENT_ID, runA11yCheck); }; }; diff --git a/addons/a11y/src/register.js b/addons/a11y/src/register.js index 043797800f6..7074287dc6f 100644 --- a/addons/a11y/src/register.js +++ b/addons/a11y/src/register.js @@ -5,12 +5,12 @@ import Panel from './components/Panel'; import { ADDON_ID, PANEL_ID } from './shared'; function init() { - addons.register(ADDON_ID, () => { + addons.register(ADDON_ID, api => { + const channel = addons.getChannel(); addons.addPanel(PANEL_ID, { title: 'Accessibility', - render() { - return ; - }, + // eslint-disable-next-line react/prop-types + render: ({ active }) => , }); }); } diff --git a/addons/a11y/src/shared/index.js b/addons/a11y/src/shared/index.js index 2bcaf9d1968..5c810f2a67f 100755 --- a/addons/a11y/src/shared/index.js +++ b/addons/a11y/src/shared/index.js @@ -3,5 +3,6 @@ const ADDON_ID = '@storybook/addon-a11y'; const PANEL_ID = `${ADDON_ID}/panel`; const CHECK_EVENT_ID = `${ADDON_ID}/check`; const RERUN_EVENT_ID = `${ADDON_ID}/rerun`; +const REQUEST_CHECK_EVENT_ID = `${ADDON_ID}/request-check`; -export { ADDON_ID, PANEL_ID, CHECK_EVENT_ID, RERUN_EVENT_ID }; +export { ADDON_ID, PANEL_ID, CHECK_EVENT_ID, RERUN_EVENT_ID, REQUEST_CHECK_EVENT_ID }; diff --git a/addons/actions/package.json b/addons/actions/package.json index 49a13d60b71..ccc3c7214d7 100644 --- a/addons/actions/package.json +++ b/addons/actions/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-actions", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "Action Logger addon for storybook", "keywords": [ "storybook" @@ -20,12 +20,12 @@ "prepare": "node ../../scripts/prepare.js" }, "dependencies": { - "@storybook/addons": "4.0.0-alpha.10", - "@storybook/components": "4.0.0-alpha.10", - "@storybook/core-events": "4.0.0-alpha.10", + "@storybook/addons": "4.0.0-alpha.14", + "@storybook/components": "4.0.0-alpha.14", + "@storybook/core-events": "4.0.0-alpha.14", "babel-runtime": "^6.26.0", "deep-equal": "^1.0.1", - "emotion": "^9.1.3", + "emotion-theming": "^9.1.2", "global": "^4.3.2", "lodash.isequal": "^4.5.0", "make-error": "^1.3.4", diff --git a/addons/actions/src/components/ActionLogger/index.js b/addons/actions/src/components/ActionLogger/index.js index d565ce46331..2a31e78831a 100644 --- a/addons/actions/src/components/ActionLogger/index.js +++ b/addons/actions/src/components/ActionLogger/index.js @@ -1,48 +1,48 @@ import PropTypes from 'prop-types'; -import React, { Component } from 'react'; +import React from 'react'; import Inspector from 'react-inspector'; -import { Actions, Action, Button, Wrapper, InspectorContainer, Countwrap, Counter } from './style'; +import { withTheme } from 'emotion-theming'; -class ActionLogger extends Component { - getActionData() { - return this.props.actions.map(action => this.renderAction(action)); - } +import { ActionBar, ActionButton } from '@storybook/components'; - renderAction(action) { - const counter = {action.count}; - return ( - - {action.count > 1 && counter} - - - - - ); - } +import { Actions, Action, Wrapper, InspectorContainer, Countwrap, Counter } from './style'; - render() { - return ( - - {this.getActionData()} - - - ); - } -} +const ActionLogger = withTheme(({ actions, onClear, theme }) => ( + + + {actions.map(action => ( + + {action.count > 1 && {action.count}} + + + + + ))} + + + + CLEAR + + +)); ActionLogger.propTypes = { - onClear: PropTypes.func, - // eslint-disable-next-line react/forbid-prop-types - actions: PropTypes.array, -}; -ActionLogger.defaultProps = { - onClear: () => {}, - actions: [], + onClear: PropTypes.func.isRequired, + actions: PropTypes.arrayOf( + PropTypes.shape({ + count: PropTypes.node, + data: PropTypes.shape({ + name: PropTypes.node.isRequired, + args: PropTypes.any, + }), + }) + ).isRequired, }; export default ActionLogger; diff --git a/addons/actions/src/components/ActionLogger/style.js b/addons/actions/src/components/ActionLogger/style.js index e2d4b67081f..202a547b82b 100644 --- a/addons/actions/src/components/ActionLogger/style.js +++ b/addons/actions/src/components/ActionLogger/style.js @@ -1,5 +1,4 @@ import styled from 'react-emotion'; -import { Button as BaseButton } from '@storybook/components'; export const Actions = styled('pre')({ flex: 1, @@ -12,24 +11,12 @@ export const Actions = styled('pre')({ export const Action = styled('div')({ display: 'flex', padding: '3px 3px 3px 0', - borderLeft: '5px solid white', - borderBottom: '1px solid #fafafa', + borderLeft: '5px solid transparent', + borderBottom: '1px solid transparent', transition: 'all 0.1s', alignItems: 'start', }); -export const Button = styled(BaseButton)({ - position: 'absolute', - bottom: 0, - right: 0, - borderRadius: '4px 0 0 0', - textTransform: 'uppercase', - letterSpacing: 1, - paddingTop: 5, - paddingBottom: 5, - border: '0 none', -}); - export const Counter = styled('div')({ margin: '0 5px 0 5px', backgroundColor: '#777777', @@ -51,4 +38,5 @@ export const Wrapper = styled('div')({ flex: 1, display: 'flex', position: 'relative', + minHeight: '100%', }); diff --git a/addons/actions/src/containers/ActionLogger/index.js b/addons/actions/src/containers/ActionLogger/index.js index df814b21ad8..6d78106dfb9 100644 --- a/addons/actions/src/containers/ActionLogger/index.js +++ b/addons/actions/src/containers/ActionLogger/index.js @@ -56,21 +56,25 @@ export default class ActionLogger extends React.Component { } render() { + const { active } = this.props; const props = { actions: this.state.actions, onClear: () => this.clearActions(), }; - return ; + return active ? : null; } } ActionLogger.propTypes = { - // eslint-disable-next-line react/forbid-prop-types - channel: PropTypes.object, + active: PropTypes.bool.isRequired, + channel: PropTypes.shape({ + emit: PropTypes.func, + on: PropTypes.func, + removeListener: PropTypes.func, + }).isRequired, api: PropTypes.shape({ - onStory: PropTypes.func.isRequired, + onStory: PropTypes.func, + getQueryParam: PropTypes.func, + setQueryParams: PropTypes.func, }).isRequired, }; -ActionLogger.defaultProps = { - channel: {}, -}; diff --git a/addons/actions/src/manager.js b/addons/actions/src/manager.js index 0fc82a627c7..43eea1709d9 100644 --- a/addons/actions/src/manager.js +++ b/addons/actions/src/manager.js @@ -8,7 +8,8 @@ export function register() { const channel = addons.getChannel(); addons.addPanel(PANEL_ID, { title: 'Action Logger', - render: () => , + // eslint-disable-next-line react/prop-types + render: ({ active }) => , }); }); } diff --git a/addons/backgrounds/package.json b/addons/backgrounds/package.json index f6aa46398aa..37e842d7a6d 100644 --- a/addons/backgrounds/package.json +++ b/addons/backgrounds/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-backgrounds", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "A storybook addon to show different backgrounds for your preview", "keywords": [ "addon", @@ -24,10 +24,9 @@ "prepare": "node ../../scripts/prepare.js" }, "dependencies": { - "@storybook/addons": "4.0.0-alpha.10", - "@storybook/core-events": "4.0.0-alpha.10", + "@storybook/addons": "4.0.0-alpha.14", + "@storybook/core-events": "4.0.0-alpha.14", "babel-runtime": "^6.26.0", - "emotion": "^9.1.3", "global": "^4.3.2", "prop-types": "^15.6.1", "react-emotion": "^9.1.3", diff --git a/addons/backgrounds/src/BackgroundPanel.js b/addons/backgrounds/src/BackgroundPanel.js index 4752d3c5d29..5d571fd1f08 100644 --- a/addons/backgrounds/src/BackgroundPanel.js +++ b/addons/backgrounds/src/BackgroundPanel.js @@ -16,14 +16,13 @@ const Title = styled('h5')({ fontSize: 16, }); -const Pre = styled('pre')({ +const Pre = styled('pre')(({ theme }) => ({ padding: '30px', display: 'block', - background: 'rgba(19,19,19,0.9)', - color: 'rgba(255,255,255,0.95)', + background: theme.fillColor, marginTop: '15px', lineHeight: '1.75em', -}); +})); const List = styled('div')({ display: 'inline-block', @@ -47,13 +46,13 @@ const defaultBackground = { }; const instructionsHtml = ` -import { storiesOf } from "@storybook/react"; -import { withBackgrounds } from "@storybook/addon-backgrounds"; +import { storiesOf } from '@storybook/react'; +import { withBackgrounds } from '@storybook/addon-backgrounds'; -storiesOf("First Component", module) +storiesOf('First Component', module) .addDecorator(withBackgrounds([ - { name: "twitter", value: "#00aced" }, - { name: "facebook", value: "#3b5998" }, + { name: 'twitter', value: '#00aced' }, + { name: 'facebook', value: '#3b5998" }, ])) .add("First Button", () => ); `.trim(); @@ -130,8 +129,11 @@ export default class BackgroundPanel extends Component { } render() { + const { active } = this.props; const backgrounds = [...this.state.backgrounds]; - + if (!active) { + return null; + } if (!backgrounds.length) return ; const hasDefault = backgrounds.filter(x => x.default).length; @@ -149,6 +151,7 @@ export default class BackgroundPanel extends Component { } } BackgroundPanel.propTypes = { + active: PropTypes.bool.isRequired, api: PropTypes.shape({ getQueryParam: PropTypes.func, setQueryParams: PropTypes.func, diff --git a/addons/backgrounds/src/Swatch.js b/addons/backgrounds/src/Swatch.js index 02a7a7c0b7e..e518da10bfe 100644 --- a/addons/backgrounds/src/Swatch.js +++ b/addons/backgrounds/src/Swatch.js @@ -3,24 +3,26 @@ import PropTypes from 'prop-types'; import styled from 'react-emotion'; -const Button = styled('button')({ +const Button = styled('button')(({ theme }) => ({ listStyle: 'none', - backgroundColor: '#fff', + backgroundColor: theme.barFill, textAlign: 'center', - border: '1px solid rgba(0,0,0,0.1)', - borderRadius: 4, + border: theme.mainBorder, + borderRadius: theme.mainBorderRadius, + color: 'inherit', cursor: 'pointer', display: 'inline-block', width: 175, verticalAlign: 'top', wordWrap: 'break-word', padding: 0, -}); -const Block = styled('div')(({ bg }) => ({ + overflow: 'hidden', +})); + +const Block = styled('div')(({ bg, theme }) => ({ height: 80, - borderRadius: '4px 4px 0 0', transition: 'opacity 0.25s ease-in-out', - borderBottom: '1px solid rgba(0,0,0,0.1)', + borderBottom: theme.mainBorder, background: bg, backgroundSize: 'cover', backgroundPosition: 'center', diff --git a/addons/backgrounds/src/__tests__/BackgroundPanel.js b/addons/backgrounds/src/__tests__/BackgroundPanel.js index 59eb89db51f..8c9f59aac76 100644 --- a/addons/backgrounds/src/__tests__/BackgroundPanel.js +++ b/addons/backgrounds/src/__tests__/BackgroundPanel.js @@ -30,26 +30,26 @@ jest.mock('global', () => ({ describe('Background Panel', () => { it('should exist', () => { - const backgroundPanel = shallow(); + const backgroundPanel = shallow(); expect(backgroundPanel).toBeDefined(); }); it('should have a default background value of transparent', () => { - const backgroundPanel = shallow(); + const backgroundPanel = shallow(); expect(backgroundPanel.state().backgrounds).toHaveLength(0); }); it('should show setup instructions if no colors provided', () => { - const backgroundPanel = shallow(); + const backgroundPanel = shallow(); expect(backgroundPanel.html().match(/Setup Instructions/gim).length).toBeGreaterThan(0); }); it('should set the query string', () => { const SpiedChannel = new EventEmitter(); - mount(); + mount(); SpiedChannel.emit(Events.SET, backgrounds); expect(mockedApi.getQueryParam).toBeCalledWith('background'); @@ -57,7 +57,7 @@ describe('Background Panel', () => { it('should not unset the query string', () => { const SpiedChannel = new EventEmitter(); - mount(); + mount(); SpiedChannel.emit(Events.UNSET, []); expect(mockedApi.setQueryParams).not.toHaveBeenCalled(); @@ -65,7 +65,9 @@ describe('Background Panel', () => { it('should accept colors through channel and render the correct swatches with a default swatch', () => { const SpiedChannel = new EventEmitter(); - const backgroundPanel = mount(); + const backgroundPanel = mount( + + ); SpiedChannel.emit(Events.SET, backgrounds); expect(backgroundPanel.state('backgrounds')).toEqual(backgrounds); @@ -73,7 +75,9 @@ describe('Background Panel', () => { it('should allow setting a default swatch', () => { const SpiedChannel = new EventEmitter(); - const backgroundPanel = mount(); + const backgroundPanel = mount( + + ); const [head, ...tail] = backgrounds; const localBgs = [{ ...head, default: true }, ...tail]; SpiedChannel.emit(Events.SET, localBgs); @@ -88,7 +92,9 @@ describe('Background Panel', () => { it('should allow the default swatch become the background color', () => { const SpiedChannel = new EventEmitter(); - const backgroundPanel = mount(); + const backgroundPanel = mount( + + ); const [head, second, ...tail] = backgrounds; const localBgs = [head, { ...second, default: true }, ...tail]; SpiedChannel.on('background', bg => { @@ -106,7 +112,9 @@ describe('Background Panel', () => { it('should unset all swatches on receiving the background-unset message', () => { const SpiedChannel = new EventEmitter(); - const backgroundPanel = mount(); + const backgroundPanel = mount( + + ); SpiedChannel.emit(Events.SET, backgrounds); expect(backgroundPanel.state('backgrounds')).toEqual(backgrounds); @@ -118,7 +126,9 @@ describe('Background Panel', () => { it('should set iframe background', () => { const SpiedChannel = new EventEmitter(); - const backgroundPanel = mount(); + const backgroundPanel = mount( + + ); backgroundPanel.setState({ backgrounds }); // force re-render backgroundPanel diff --git a/addons/backgrounds/src/index.js b/addons/backgrounds/src/index.js index 293d82d6106..859375d6607 100644 --- a/addons/backgrounds/src/index.js +++ b/addons/backgrounds/src/index.js @@ -15,6 +15,7 @@ export const withBackgrounds = makeDecorator({ name: 'backgrounds', parameterName: 'backgrounds', skipIfNoParametersOrOptions: true, + allowDeprecatedUsage: true, wrapper: (getStory, context, { options, parameters }) => { const backgrounds = parameters || options; diff --git a/addons/backgrounds/src/register.js b/addons/backgrounds/src/register.js index 50d22ddbba6..c1a380e2544 100644 --- a/addons/backgrounds/src/register.js +++ b/addons/backgrounds/src/register.js @@ -10,6 +10,7 @@ addons.register(ADDON_ID, api => { const channel = addons.getChannel(); addons.addPanel(PANEL_ID, { title: 'Backgrounds', - render: () => , + // eslint-disable-next-line react/prop-types + render: ({ active }) => , }); }); diff --git a/addons/centered/package.json b/addons/centered/package.json index db6de5f9f4c..61f6cda8924 100644 --- a/addons/centered/package.json +++ b/addons/centered/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-centered", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "Storybook decorator to center components", "license": "MIT", "author": "Muhammed Thanish ", diff --git a/addons/events/package.json b/addons/events/package.json index 0813545c8e4..f7058dea590 100644 --- a/addons/events/package.json +++ b/addons/events/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-events", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "Add events to your Storybook stories.", "keywords": [ "addon", @@ -19,10 +19,9 @@ "prepare": "node ../../scripts/prepare.js" }, "dependencies": { - "@storybook/addons": "4.0.0-alpha.10", - "@storybook/core-events": "4.0.0-alpha.10", + "@storybook/addons": "4.0.0-alpha.14", + "@storybook/core-events": "4.0.0-alpha.14", "babel-runtime": "^6.26.0", - "emotion": "^9.1.3", "format-json": "^1.0.3", "prop-types": "^15.6.1", "react-emotion": "^9.1.3", diff --git a/addons/events/src/components/Panel.js b/addons/events/src/components/Panel.js index adcab42e4b0..7061171ac8d 100644 --- a/addons/events/src/components/Panel.js +++ b/addons/events/src/components/Panel.js @@ -10,10 +10,12 @@ const Wrapper = styled('div')({ width: '100%', boxSizing: 'border-box', padding: '10px', + minHeight: '100%', }); export default class Events extends Component { static propTypes = { + active: PropTypes.bool.isRequired, channel: PropTypes.shape({ on: PropTypes.func, emit: PropTypes.func, @@ -43,10 +45,11 @@ export default class Events extends Component { render() { const { events } = this.state; - return ( + const { active } = this.props; + return active ? ( {events.map(event => )} - ); + ) : null; } } diff --git a/addons/events/src/manager.js b/addons/events/src/manager.js index c5929d7d525..842bf3177d1 100644 --- a/addons/events/src/manager.js +++ b/addons/events/src/manager.js @@ -6,9 +6,11 @@ import { ADDON_ID, PANEL_ID } from './constants'; export function register() { addons.register(ADDON_ID, () => { + const channel = addons.getChannel(); addons.addPanel(PANEL_ID, { title: 'Events', - render: () => , + // eslint-disable-next-line react/prop-types + render: ({ active }) => , }); }); } diff --git a/addons/graphql/package.json b/addons/graphql/package.json index c5a8a7e64e8..39a89cf924f 100644 --- a/addons/graphql/package.json +++ b/addons/graphql/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-graphql", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "Storybook addon to display the GraphiQL IDE", "keywords": [ "storybook" diff --git a/addons/info/README.md b/addons/info/README.md index 1c7e29f7bcc..f391e4de9f4 100644 --- a/addons/info/README.md +++ b/addons/info/README.md @@ -7,7 +7,7 @@ [![Storybook Slack](https://now-examples-slackin-rrirkqohko.now.sh/badge.svg)](https://now-examples-slackin-rrirkqohko.now.sh/) [![Backers on Open Collective](https://opencollective.com/storybook/backers/badge.svg)](#backers) [![Sponsors on Open Collective](https://opencollective.com/storybook/sponsors/badge.svg)](#sponsors) -* * * +--- Storybook Info Addon will show additional information for your stories in [Storybook](https://storybook.js.org). Useful when you want to display usage or other types of documentation alongside your story. @@ -25,7 +25,8 @@ npm i -D @storybook/addon-info ``` ## Basic usage -Then, add `withInfo` as a decarator to your book of stories. + +Then, add `withInfo` as a decarator to your book of stories. It is possible to add `info` by default to all or a subsection of stories by using a global or story decorator. It is important to declare this decorator as **the first decorator**, otherwise it won't work well. @@ -33,7 +34,9 @@ It is important to declare this decorator as **the first decorator**, otherwise ```js addDecorator(withInfo); // Globally in your .storybook/config.js. ``` + or + ```js storiesOf('Component', module) .addDecorator(withInfo) // At your stories directly. @@ -53,12 +56,13 @@ storiesOf('Component', module) .addParameters({ info: { // Your settings - } + }, }) - .add('with some emoji', () => ); + .add('with some emoji', () => ); ``` ...or for each story individually: + ```js import { storiesOf } from '@storybook/react'; @@ -67,12 +71,12 @@ import Component from './Component'; storiesOf('Component', module) .add( 'with some emoji', - () => , - { info : { inline: false, header: false } } // Make your component render inline with the additional info + () => , + { info: { inline: true, header: false } } // Make your component render inline with the additional info ) .add( 'with no emoji', - () => , + () => , { info: '☹️ no emojis' } // Add additional info text directly ); ``` @@ -86,41 +90,36 @@ import Component from './Component'; storiesOf('Component', module) .addParameters({ - info: { // Make a default for all stories in this book, - inline: true, // where the components are inlined + info: { + // Make a default for all stories in this book, + inline: true, // where the components are inlined styles: { header: { h1: { - color: 'red' // and the headers of the sections are red. - } - } + color: 'red', // and the headers of the sections are red. + }, + }, }, - } + }, }) - .add( - 'green version', - () => , - { - info: { - styles: stylesheet => ({ // Setting the style with a function - ...stylesheet, - header: { - ...stylesheet.header, - h1: { - ...stylesheet.header.h1, - color: 'green' // Still inlined but with green headers! - } - } - }) - } - }) - .add( - 'something else', - () => , - { - info: "This story has additional text added to the info!" // Still inlined and with red headers! - } - ); + .add('green version', () => , { + info: { + styles: stylesheet => ({ + // Setting the style with a function + ...stylesheet, + header: { + ...stylesheet.header, + h1: { + ...stylesheet.header.h1, + color: 'green', // Still inlined but with green headers! + }, + }, + }), + }, + }) + .add('something else', () => , { + info: 'This story has additional text added to the info!', // Still inlined and with red headers! + }); ``` It is also possible to disable the `info` addon entirely. @@ -135,27 +134,22 @@ Depending on the scope at which you want to disable the addon, pass the followin ``` ## Markdown + The `info` addon also supports markdown. To use markdown as additional textual documentation for your stories, either pass it directly as a String to the `info` parameters, or use the `text` option. - ```js -storiesOf('Button', module) - .add( - 'Button Component', - () => ~~~ - ` - } - } - ); + `, + }, +}); ``` ## Setting Global Options @@ -166,9 +160,11 @@ To configure default options for all usage of the info addon, pass a option obje // config.js import { withInfo } from '@storybook/addon-info'; -addDecorator(withInfo({ - header: false, // Global configuration for the info addon across all of your stories. -})); +addDecorator( + withInfo({ + header: false, // Global configuration for the info addon across all of your stories. + }) +); ``` Configuration parameters can be set at 3 different locations: passed as default options along the `addDecorator` call, passed as an object of parameters to a book of stories to the `addParameters` call, and passed as direct parameters to each individual story. @@ -274,40 +270,40 @@ Example: ```js // button.js // @flow -import React from 'react' +import React from 'react'; const paddingStyles = { small: '4px 8px', - medium: '8px 16px' -} + medium: '8px 16px', +}; const Button = ({ size, ...rest }: { /** The size of the button */ - size: 'small' | 'medium' + size: 'small' | 'medium', }) => { const style = { - padding: paddingStyles[size] || '' - } - return ) -); +storiesOf('Button', module).add('with text', () => , { + info: { + TableComponent, + }, +}); ``` ### React Docgen Integration @@ -359,10 +354,11 @@ import React from 'react'; import PropTypes from 'prop-types'; /** Button component description */ -const DocgenButton = ({ disabled, label, style, onClick }) => +const DocgenButton = ({ disabled, label, style, onClick }) => ( ; + +); DocgenButton.defaultProps = { disabled: false, diff --git a/addons/info/package.json b/addons/info/package.json index ec5df366c29..27b7067ccb4 100644 --- a/addons/info/package.json +++ b/addons/info/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-info", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "A Storybook addon to show additional information for your stories.", "repository": { "type": "git", @@ -13,12 +13,11 @@ "prepare": "node ../../scripts/prepare.js" }, "dependencies": { - "@storybook/addons": "4.0.0-alpha.10", - "@storybook/client-logger": "4.0.0-alpha.10", - "@storybook/components": "4.0.0-alpha.10", + "@storybook/addons": "4.0.0-alpha.14", + "@storybook/client-logger": "4.0.0-alpha.14", + "@storybook/components": "4.0.0-alpha.14", "babel-runtime": "^6.26.0", "core-js": "2.5.7", - "emotion": "^9.1.3", "global": "^4.3.2", "marksy": "^6.0.3", "nested-object-assign": "^1.0.1", diff --git a/addons/info/src/__snapshots__/index.test.js.snap b/addons/info/src/__snapshots__/index.test.js.snap index e935cd48a7a..cd708d7a0a6 100644 --- a/addons/info/src/__snapshots__/index.test.js.snap +++ b/addons/info/src/__snapshots__/index.test.js.snap @@ -33,6 +33,9 @@ exports[`addon Info should render and external markdown 1`] = ` -webkit-align-self: flex-start; -ms-flex-item-align: start; align-self: flex-start; + -webkit-flex-shrink: 0; + -ms-flex-negative: 0; + flex-shrink: 0; } .emotion-1:hover { @@ -1335,6 +1338,9 @@ exports[`addon Info should render and markdown 1`] = ` -webkit-align-self: flex-start; -ms-flex-item-align: start; align-self: flex-start; + -webkit-flex-shrink: 0; + -ms-flex-negative: 0; + flex-shrink: 0; } .emotion-1:hover { diff --git a/addons/info/src/components/markdown/pre/copyButton.js b/addons/info/src/components/markdown/pre/copyButton.js index fa84c05a0ce..ff11d7a60d0 100644 --- a/addons/info/src/components/markdown/pre/copyButton.js +++ b/addons/info/src/components/markdown/pre/copyButton.js @@ -12,6 +12,7 @@ const Button = styled('button')( fontSize: 13, padding: '3px 10px', alignSelf: 'flex-start', + flexShrink: 0, ':hover': { backgroundColor: '#f4f7fa', diff --git a/addons/info/src/index.js b/addons/info/src/index.js index 3813949ab15..b6abfae8ad9 100644 --- a/addons/info/src/index.js +++ b/addons/info/src/index.js @@ -87,6 +87,7 @@ function addInfo(storyFn, context, infoOptions) { export const withInfo = makeDecorator({ name: 'withInfo', parameterName: 'info', + allowDeprecatedUsage: true, wrapper: (getStory, context, { options, parameters }) => { const storyOptions = parameters || options; const infoOptions = typeof storyOptions === 'string' ? { text: storyOptions } : storyOptions; diff --git a/addons/jest/README.md b/addons/jest/README.md index a2180eaa571..9441c9499dc 100644 --- a/addons/jest/README.md +++ b/addons/jest/README.md @@ -31,17 +31,19 @@ When running **Jest**, be sure to save the results in a json file: ``` You may want to add it the result file to `.gitignore`, since it's a generated file: + ``` jest-test-results.json ``` + But much like lockfiles and snapshots checking-in generated files can have certain advantages as well. It's up to you. -We recommend to **do** check in the test results file so starting storybook from an clean git clone doesn't require running all tests first, -but this can mean you'll experience merge conflicts on this file in the future. (*re-generating this file is super easy though, just like lockfiles and snapshots*) +We recommend to **do** check in the test results file so starting storybook from an clean git clone doesn't require running all tests first, +but this can mean you'll experience merge conflicts on this file in the future. (_re-generating this file is super easy though, just like lockfiles and snapshots_) ## Generating the test results -You need to make sure the generated test-results file exists before you start storybook. -During development you will likely start jest in watch-mode +You need to make sure the generated test-restuls file exists before you start storybook. +During development you will likely start jest in watch-mode and so the json file will be re-generated every time code or tests change. ```sh @@ -50,9 +52,10 @@ npm run test:generate-output -- --watch This change will then be HMR (hot module reloaded) using webpack and displayed by this addon. -If you want to pre-run jest automaticly during development or a static build, +If you want to pre-run jest automaticly during development or a static build, you may need to consider that if your tests fail, the script receives a non-0 exit code and will exit. You could create a `prebuild:storybook` npm script, which will never fail by appending `|| true`: + ```json "scripts": { "test:generate-output": "jest --json --outputFile=.jest-test-results.json || true", @@ -83,40 +86,57 @@ import results from '../.jest-test-results.json'; import { withTests } from '@storybook/addon-jest'; storiesOf('MyComponent', module) - .addDecorator(withTests({ results })('MyComponent', 'MyOtherComponent')) - .add('This story shows test results from MyComponent.test.js and MyOtherComponent.test.js', () => ( -
Jest results in storybook
- )); + .addDecorator(withTests({ results })) + .add( + 'This story shows test results from MyComponent.test.js and MyOtherComponent.test.js', + () =>
Jest results in storybook
, + { + jest: ['MyComponent.test.js', 'MyOtherComponent.test.js'], + } + ); ``` -Or in order to avoid importing `.jest-test-results.json` in each story, you can create a simple file `withTests.js`: +Or in order to avoid importing `.jest-test-results.json` in each story, simply add the decorator in your `.storybook/config.js` and results will display for stories that you have set the `jest` parameter on: ```js -import results from '../.jest-test-results.json'; +import { addDecorator } from '@storybook/react'; // <- or your view layer import { withTests } from '@storybook/addon-jest'; -export default withTests({ - results, -}); +import results from '../.jest-test-results.json'; + +addDecorator( + withTests({ + results, + }) +); ``` Then in your story: ```js -// import your file -import withTests from '.withTests'; - storiesOf('MyComponent', module) - .addDecorator(withTests('MyComponent', 'MyOtherComponent')) - .add('This story shows test results from MyComponent.test.js and MyOtherComponent.test.js', () => ( -
Jest results in storybook
- )); + // Use .addParameters if you want the same tests displayed for all stories of the component + .addParameters({ jest: ['MyComponent', 'MyOtherComponent'] }) + .add( + 'This story shows test results from MyComponent.test.js and MyOtherComponent.test.js', + () =>
Jest results in storybook
+ ); +``` + +### Disabling + +You can disable the addon for a single story by setting the `jest` parameter to `{disable: true}`: + +```js +storiesOf('MyComponent', module).add('Story', () =>
Jest results disabled here
, { + jest: { disable: true }, +}); ``` ### withTests(options) -- **options.results**: OBJECT jest output results. *mandatory* -- **filesExt**: STRING test file extention. *optional*. This allow you to write "MyComponent" and not "MyComponent.test.js". It will be used as regex to find your file results. Default value is `((\\.specs?)|(\\.tests?))?(\\.js)?$`. That mean it will match: MyComponent.js, MyComponent.test.js, MyComponent.tests.js, MyComponent.spec.js, MyComponent.specs.js... +- **options.results**: OBJECT jest output results. _mandatory_ +- **filesExt**: STRING test file extention. _optional_. This allow you to write "MyComponent" and not "MyComponent.test.js". It will be used as regex to find your file results. Default value is `((\\.specs?)|(\\.tests?))?(\\.js)?$`. That mean it will match: MyComponent.js, MyComponent.test.js, MyComponent.tests.js, MyComponent.spec.js, MyComponent.specs.js... ## Usage with Angular @@ -124,7 +144,7 @@ Assuming that you have created a test files `my.component.spec.ts` and `my-other Configure Jest with [jest-preset-angular](https://www.npmjs.com/package/jest-preset-angular) -In project`s `typings.d.ts` add +In project`s`typings.d.ts` add ```ts declare module '*.json' { @@ -133,29 +153,31 @@ declare module '*.json' { } ``` -Create a simple file `withTests.ts`: +In your `.storybook/config.ts`: ```ts -import * as results from '../.jest-test-results.json'; +import { addDecorator } from '@storybook/angular'; import { withTests } from '@storybook/addon-jest'; -export const wTests = withTests({ - results, - filesExt: '((\\.specs?)|(\\.tests?))?(\\.ts)?$' -}); +import * as results from '../.jest-test-results.json'; + +addDecorator( + withTests({ + results, + filesExt: '((\\.specs?)|(\\.tests?))?(\\.ts)?$', + }) +); ``` Then in your story: ```js -// import your file -import wTests from '.withTests'; - storiesOf('MyComponent', module) - .addDecorator(wTests('my.component', 'my-other.component')) - .add('This story shows test results from my.component.spec.ts and my-other.component.spec.ts', () => ( -
Jest results in storybook
- )); + .addParameters({ jest: ['my.component', 'my-other.component'] }) + .add( + 'This story shows test results from my.component.spec.ts and my-other.component.spec.ts', + () =>
Jest results in storybook
+ ); ``` ##### Example [here](https://github.com/storybooks/storybook/tree/master/examples/angular-cli) diff --git a/addons/jest/package.json b/addons/jest/package.json index 9ce0cea93ef..48158577808 100644 --- a/addons/jest/package.json +++ b/addons/jest/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-jest", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "React storybook addon that show component jest report", "keywords": [ "addon", @@ -25,13 +25,13 @@ "prepare": "node ../../scripts/prepare.js" }, "dependencies": { - "@storybook/addons": "4.0.0-alpha.10", - "@storybook/components": "4.0.0-alpha.10", + "@storybook/addons": "4.0.0-alpha.14", + "@storybook/components": "4.0.0-alpha.14", "babel-runtime": "^6.26.0", - "emotion": "^9.1.3", "global": "^4.3.2", "prop-types": "^15.6.1", - "react-emotion": "^9.1.3" + "react-emotion": "^9.1.3", + "util-deprecate": "^1.0.2" }, "peerDependencies": { "react": "*" diff --git a/addons/jest/src/components/Panel.js b/addons/jest/src/components/Panel.js index 23b3a20ee98..2512aefb450 100644 --- a/addons/jest/src/components/Panel.js +++ b/addons/jest/src/components/Panel.js @@ -2,8 +2,6 @@ import React from 'react'; import PropTypes from 'prop-types'; import styled from 'react-emotion'; -import { baseFonts } from '@storybook/components'; - import Indicator from './Indicator'; import Result, { FailedResult } from './Result'; import provideJestResult from '../hoc/provideJestResult'; @@ -25,7 +23,6 @@ const Item = styled('li')({ const NoTests = styled('div')({ padding: '10px 20px', flex: 1, - ...baseFonts, }); const FileTitle = styled('h2')({ @@ -139,7 +136,6 @@ const Content = styled(({ tests, className }) => ( ))({ padding: '10px 20px', flex: '1 1 0%', - ...baseFonts, }); const Panel = ({ tests }) => diff --git a/addons/jest/src/hoc/provideJestResult.js b/addons/jest/src/hoc/provideJestResult.js index aec752bc7e0..a72169a21ee 100644 --- a/addons/jest/src/hoc/provideJestResult.js +++ b/addons/jest/src/hoc/provideJestResult.js @@ -1,7 +1,7 @@ import React from 'react'; import PropTypes from 'prop-types'; -const provideTests = Component => { +const provideTests = Component => class TestProvider extends React.Component { static propTypes = { channel: PropTypes.shape({ @@ -11,21 +11,22 @@ const provideTests = Component => { api: PropTypes.shape({ onStory: PropTypes.func, }).isRequired, + active: PropTypes.bool, + }; + static defaultProps = { + active: true, }; - constructor(props) { - super(props); - - this.state = {}; - this.onAddTests = this.onAddTests.bind(this); - } + state = {}; componentDidMount() { - this.stopListeningOnStory = this.props.api.onStory(() => { + const { channel, api } = this.props; + + this.stopListeningOnStory = api.onStory(() => { this.onAddTests({}); }); - this.props.channel.on('storybook/tests/add_tests', this.onAddTests); + channel.on('storybook/tests/add_tests', this.onAddTests); } componentWillUnmount() { @@ -35,16 +36,14 @@ const provideTests = Component => { this.props.channel.removeListener('storybook/tests/add_tests', this.onAddTests); } - onAddTests({ kind, storyName, tests }) { + onAddTests = ({ kind, storyName, tests }) => { this.setState({ kind, storyName, tests }); - } + }; render() { - return ; + const { active } = this.props; + return active ? : null; } - } - - return TestProvider; -}; + }; export default provideTests; diff --git a/addons/jest/src/index.js b/addons/jest/src/index.js index e302a7a0f07..49c7cca457b 100644 --- a/addons/jest/src/index.js +++ b/addons/jest/src/index.js @@ -1,7 +1,8 @@ import addons from '@storybook/addons'; +import deprecate from 'util-deprecate'; const findTestResults = (testFiles, jestTestResults, jestTestFilesExt) => - testFiles.map(name => { + Array.from(testFiles).map(name => { if (jestTestResults && jestTestResults.testResults) { return { name, @@ -27,9 +28,27 @@ export const withTests = userOptions => { }; const options = Object.assign({}, defaultOptions, userOptions); - return (...testFiles) => (storyFn, { kind, story }) => { - emitAddTests({ kind, story, testFiles, options }); + return (...args) => { + if (typeof args[0] === 'string') { + return deprecate((story, { kind }) => { + emitAddTests({ kind, story, testFiles: args, options }); - return storyFn(); + return story(); + }, 'Passing component filenames to the `@storybook/addon-jest` via `withTests` is deprecated. Instead, use the `jest` story parameter'); + } + + const [ + story, + { + kind, + parameters: { jest: testFiles }, + }, + ] = args; + + if (testFiles && !testFiles.disable) { + emitAddTests({ kind, story, testFiles, options }); + } + + return story(); }; }; diff --git a/addons/jest/src/register.js b/addons/jest/src/register.js index 66639bb3b40..7300e64299b 100644 --- a/addons/jest/src/register.js +++ b/addons/jest/src/register.js @@ -4,11 +4,11 @@ import addons from '@storybook/addons'; import PanelTitle from './components/PanelTitle'; import Panel from './components/Panel'; -// Register the addon with a unique name. addons.register('storybook/tests', api => { - // Also need to set a unique name to the panel. + const channel = addons.getChannel(); addons.addPanel('storybook/tests/panel', { title: , - render: () => , + // eslint-disable-next-line react/prop-types + render: ({ active }) => , }); }); diff --git a/addons/knobs/package.json b/addons/knobs/package.json index 5a7102487fe..f3465f75727 100644 --- a/addons/knobs/package.json +++ b/addons/knobs/package.json @@ -1,6 +1,6 @@ { "name": "@storybook/addon-knobs", - "version": "4.0.0-alpha.10", + "version": "4.0.0-alpha.14", "description": "Storybook Addon Prop Editor Component", "repository": { "type": "git", @@ -13,22 +13,20 @@ "prepare": "node ../../scripts/prepare.js" }, "dependencies": { - "@storybook/addons": "4.0.0-alpha.10", - "@storybook/components": "4.0.0-alpha.10", - "@storybook/core-events": "4.0.0-alpha.10", + "@storybook/addons": "4.0.0-alpha.14", + "@storybook/components": "4.0.0-alpha.14", + "@storybook/core-events": "4.0.0-alpha.14", "babel-runtime": "^6.26.0", - "deep-equal": "^1.0.1", + "copy-to-clipboard": "^3.0.8", "escape-html": "^1.0.3", + "fast-deep-equal": "^2.0.1", "global": "^4.3.2", - "insert-css": "^2.0.0", - "lodash.debounce": "^4.0.8", - "moment": "^2.22.2", "prop-types": "^15.6.1", + "qs": "^6.5.2", "react-color": "^2.14.1", "react-datetime": "^2.14.0", "react-emotion": "^9.1.3", "react-lifecycles-compat": "^3.0.4", - "react-textarea-autosize": "^6.1.0", "util-deprecate": "^1.0.2" }, "devDependencies": { diff --git a/addons/knobs/src/KnobManager.js b/addons/knobs/src/KnobManager.js index 9ae2ca937a7..bd183225a50 100644 --- a/addons/knobs/src/KnobManager.js +++ b/addons/knobs/src/KnobManager.js @@ -1,5 +1,5 @@ /* eslint no-underscore-dangle: 0 */ -import deepEqual from 'deep-equal'; +import deepEqual from 'fast-deep-equal'; import escape from 'escape-html'; import KnobStore from './KnobStore'; diff --git a/addons/knobs/src/KnobStore.js b/addons/knobs/src/KnobStore.js index 6474abfb79e..18b5125245a 100644 --- a/addons/knobs/src/KnobStore.js +++ b/addons/knobs/src/KnobStore.js @@ -1,3 +1,6 @@ +const callArg = fn => fn(); +const callAll = fns => fns.forEach(callArg); + export default class KnobStore { constructor() { this.store = {}; @@ -12,7 +15,12 @@ export default class KnobStore { this.store[key] = value; this.store[key].used = true; this.store[key].groupId = value.groupId; - this.callbacks.forEach(cb => cb()); + + // debounce the execution of the callbacks for 50 milliseconds + if (this.timer) { + clearTimeout(this.timer); + } + this.timer = setTimeout(callAll, 50, this.callbacks); } get(key) { diff --git a/addons/knobs/src/components/GroupTabs.js b/addons/knobs/src/components/GroupTabs.js deleted file mode 100644 index bed567cf6c9..00000000000 --- a/addons/knobs/src/components/GroupTabs.js +++ /dev/null @@ -1,110 +0,0 @@ -import PropTypes from 'prop-types'; -import React, { Component } from 'react'; -import styled from 'react-emotion'; - -import { Placeholder } from '@storybook/components'; - -const Wrapper = styled('div')({ - flex: '1 1 auto', - display: 'flex', - flexDirection: 'column', - background: 'white', - borderRadius: 4, - border: 'solid 1px rgb(236, 236, 236)', - width: '100%', -}); - -const Bar = styled('div')({ - display: 'flex', - flexWrap: 'wrap', - flexDirection: 'row', - justifyContent: 'flex-start', - alignItems: 'center', - borderBottom: 'solid 1px #eaeaea', -}); - -const Content = styled('div')({ - flex: '1 1 0', - display: 'flex', - overflow: 'auto', -}); - -const Tab = styled('button')(({ active }) => ({ - fontSize: 11, - letterSpacing: '1px', - padding: '10px 15px', - textTransform: 'uppercase', - transition: 'opacity 0.3s', - opacity: active ? 1 : 0.5, - maxHeight: 60, - overflow: 'hidden', - cursor: 'pointer', - background: 'transparent', - border: 'none', -})); - -class GroupTabs extends Component { - renderTab(name, group) { - const onClick = e => { - e.preventDefault(); - this.props.onGroupSelect(name); - }; - - let { title } = group; - if (typeof title === 'function') { - title = title(); - } - - return ( - - {title} - - ); - } - - render() { - const entries = this.props.groups ? Object.entries(this.props.groups) : null; - - return entries && entries.length ? ( - - {entries.map(([key, value]) => this.renderTab(key, value))} - - {entries.map(([key, value]) => { - const groupStyle = { display: 'none' }; - if (key === this.props.selectedGroup) { - Object.assign(groupStyle, { flex: 1, display: 'flex' }); - } - return ( -
- {value.render()} -
- ); - })} -
-
- ) : ( - no groups available - ); - } -} - -GroupTabs.defaultProps = { - groups: {}, - onGroupSelect: () => {}, - selectedGroup: null, -}; - -GroupTabs.propTypes = { - // eslint-disable-next-line react/forbid-prop-types - groups: PropTypes.object, - onGroupSelect: PropTypes.func, - selectedGroup: PropTypes.string, -}; - -export default GroupTabs; diff --git a/addons/knobs/src/components/Panel.js b/addons/knobs/src/components/Panel.js index 509a250849f..c7446455323 100644 --- a/addons/knobs/src/components/Panel.js +++ b/addons/knobs/src/components/Panel.js @@ -1,12 +1,14 @@ -import React from 'react'; +import React, { PureComponent } from 'react'; import PropTypes from 'prop-types'; - +import qs from 'qs'; +import { document } from 'global'; import styled from 'react-emotion'; +import copy from 'copy-to-clipboard'; + +import { Placeholder, TabWrapper, TabsState, ActionBar, ActionButton } from '@storybook/components'; -import { Placeholder } from '@storybook/components'; -import GroupTabs from './GroupTabs'; -import PropForm from './PropForm'; import Types from './types'; +import PropForm from './PropForm'; const getTimestamp = () => +new Date(); @@ -16,49 +18,21 @@ const PanelWrapper = styled('div')({ width: '100%', }); -const PanelInner = styled('div')({ - padding: '5px', - width: 'auto', - position: 'relative', -}); - -const ResetButton = styled('button')({ - position: 'absolute', - bottom: 11, - right: 10, - border: 'none', - borderTop: 'solid 1px rgba(0, 0, 0, 0.2)', - borderLeft: 'solid 1px rgba(0, 0, 0, 0.2)', - background: 'rgba(255, 255, 255, 0.5)', - padding: '5px 10px', - borderRadius: '4px 0 0 0', - color: 'rgba(0, 0, 0, 0.5)', - outline: 'none', -}); - -export default class Panel extends React.Component { +export default class Panel extends PureComponent { constructor(props) { super(props); - this.handleChange = this.handleChange.bind(this); - this.handleClick = this.handleClick.bind(this); - this.setKnobs = this.setKnobs.bind(this); - this.reset = this.reset.bind(this); - this.setOptions = this.setOptions.bind(this); - this.onGroupSelect = this.onGroupSelect.bind(this); - - this.state = { knobs: {}, groupId: DEFAULT_GROUP_ID }; + this.state = { knobs: {} }; this.options = {}; this.lastEdit = getTimestamp(); this.loadedFromUrl = false; } - componentDidMount() { this.props.channel.on('addon:knobs:setKnobs', this.setKnobs); this.props.channel.on('addon:knobs:setOptions', this.setOptions); this.stopListeningOnStory = this.props.api.onStory(() => { - this.setState({ knobs: [], groupId: DEFAULT_GROUP_ID }); + this.setState({ knobs: {} }); this.props.channel.emit('addon:knobs:reset'); }); } @@ -68,15 +42,11 @@ export default class Panel extends React.Component { this.stopListeningOnStory(); } - onGroupSelect(name) { - this.setState({ groupId: name }); - } - - setOptions(options = { timestamps: false }) { + setOptions = (options = { timestamps: false }) => { this.options = options; - } + }; - setKnobs({ knobs, timestamp }) { + setKnobs = ({ knobs, timestamp }) => { const queryParams = {}; const { api, channel } = this.props; @@ -86,7 +56,6 @@ export default class Panel extends React.Component { // For the first time, get values from the URL and set them. if (!this.loadedFromUrl) { const urlValue = api.getQueryParam(`knob-${name}`); - if (urlValue !== undefined) { // If the knob value present in url knob.value = Types[knob.type].deserialize(urlValue); @@ -94,48 +63,63 @@ export default class Panel extends React.Component { } } - queryParams[`knob-${name}`] = Types[knob.type].serialize(knob.value); + // set all knobsquery params to be deleted from URL + queryParams[`knob-${name}`] = null; }); - this.loadedFromUrl = true; + api.setQueryParams(queryParams); this.setState({ knobs }); + + this.loadedFromUrl = true; } - } + }; - reset() { + reset = () => { this.props.channel.emit('addon:knobs:reset'); - } + }; - emitChange(changedKnob) { - this.props.channel.emit('addon:knobs:knobChange', changedKnob); - } - - handleChange(changedKnob) { - this.lastEdit = getTimestamp(); - const { api } = this.props; + copy = () => { + const { location } = document; + const query = qs.parse(location.search.replace('?', '')); const { knobs } = this.state; - const { name, type, value } = changedKnob; + + Object.entries(knobs).forEach(([name, knob]) => { + query[`knob-${name}`] = Types[knob.type].serialize(knob.value); + }); + + copy(`${location.origin + location.pathname}?${qs.stringify(query)}`); + + // TODO: show some notification of this + }; + + emitChange = changedKnob => { + this.props.channel.emit('addon:knobs:knobChange', changedKnob); + }; + + handleChange = changedKnob => { + this.lastEdit = getTimestamp(); + const { knobs } = this.state; + const { name } = changedKnob; const newKnobs = { ...knobs }; newKnobs[name] = { ...newKnobs[name], ...changedKnob, }; - this.setState({ knobs: newKnobs }); - - const queryParams = {}; - queryParams[`knob-${name}`] = Types[type].serialize(value); - - api.setQueryParams(queryParams); this.setState({ knobs: newKnobs }, this.emitChange(changedKnob)); - } + }; - handleClick(knob) { + handleClick = knob => { this.props.channel.emit('addon:knobs:knobClick', knob); - } + }; render() { - const { knobs, groupId } = this.state; + const { knobs } = this.state; + const { active } = this.props; + + if (!active) { + return null; + } const groups = {}; const groupIds = []; @@ -146,20 +130,23 @@ export default class Panel extends React.Component { const knobKeyGroupId = knobs[key].groupId; groupIds.push(knobKeyGroupId); groups[knobKeyGroupId] = { - render: () =>
{knobKeyGroupId}
, + render: ({ active: groupActive, selected }) => ( + + knob.groupId === knobKeyGroupId)} + onFieldChange={this.handleChange} + onFieldClick={this.handleClick} + /> + + ), title: knobKeyGroupId, }; }); - if (groupIds.length > 0) { - groups[DEFAULT_GROUP_ID] = { - render: () =>
{DEFAULT_GROUP_ID}
, - title: DEFAULT_GROUP_ID, - }; - if (groupId !== DEFAULT_GROUP_ID) { - knobsArray = knobsArray.filter(key => knobs[key].groupId === groupId); - } - } + groups[DEFAULT_GROUP_ID] = { + render: () => null, + title: DEFAULT_GROUP_ID, + }; knobsArray = knobsArray.map(key => knobs[key]); @@ -169,33 +156,38 @@ export default class Panel extends React.Component { return ( - {groupIds.length > 0 && ( - - )} - + {groupIds.length > 0 ? ( + + {Object.entries(groups).map(([k, v]) => ( +
+ {v.render} +
+ ))} +
+ ) : ( -
- RESET + )} + + COPY + RESET +
); } } Panel.propTypes = { + active: PropTypes.bool.isRequired, + onReset: PropTypes.object, // eslint-disable-line channel: PropTypes.shape({ emit: PropTypes.func, on: PropTypes.func, removeListener: PropTypes.func, }).isRequired, - onReset: PropTypes.object, // eslint-disable-line api: PropTypes.shape({ onStory: PropTypes.func, getQueryParam: PropTypes.func, diff --git a/addons/knobs/src/components/PropField.js b/addons/knobs/src/components/PropField.js deleted file mode 100644 index 888d5a9e75b..00000000000 --- a/addons/knobs/src/components/PropField.js +++ /dev/null @@ -1,43 +0,0 @@ -import PropTypes from 'prop-types'; -import React from 'react'; -import styled from 'react-emotion'; - -import TypeMap from './types'; - -const InvalidType = () => Invalid Type; - -const Field = styled('div')({ - display: 'table-row', - padding: '5px', -}); -const Label = styled('label')({ - display: 'table-cell', - boxSizing: 'border-box', - verticalAlign: 'top', - paddingRight: 5, - paddingTop: 5, - textAlign: 'right', - width: 80, - fontSize: 12, - color: 'rgb(68, 68, 68)', - fontWeight: 600, -}); - -export default function PropField({ onChange, onClick, knob }) { - const InputType = TypeMap[knob.type] || InvalidType; - return ( - - - - - ); -} - -PropField.propTypes = { - knob: PropTypes.shape({ - name: PropTypes.string, - value: PropTypes.any, - }).isRequired, - onChange: PropTypes.func.isRequired, - onClick: PropTypes.func.isRequired, -}; diff --git a/addons/knobs/src/components/PropForm.js b/addons/knobs/src/components/PropForm.js index 821b2410e33..0b07504a829 100644 --- a/addons/knobs/src/components/PropForm.js +++ b/addons/knobs/src/components/PropForm.js @@ -1,19 +1,19 @@ -import React from 'react'; +import React, { Component } from 'react'; import PropTypes from 'prop-types'; import styled from 'react-emotion'; -import PropField from './PropField'; +import { Field } from '@storybook/components'; +import TypeMap from './types'; const Form = styled('form')({ - display: 'table', boxSizing: 'border-box', width: '100%', - borderCollapse: 'separate', - borderSpacing: '5px', }); -export default class propForm extends React.Component { +const InvalidType = () => Invalid Type; + +export default class PropForm extends Component { makeChangeHandler(name, type) { return value => { const change = { name, type, value }; @@ -28,16 +28,12 @@ export default class propForm extends React.Component {
{knobs.map(knob => { const changeHandler = this.makeChangeHandler(knob.name, knob.type); + const InputType = TypeMap[knob.type] || InvalidType; + return ( - + + + ); })} @@ -45,19 +41,15 @@ export default class propForm extends React.Component { } } -propForm.displayName = 'propForm'; +PropForm.displayName = 'PropForm'; -propForm.defaultProps = { - knobs: [], -}; - -propForm.propTypes = { +PropForm.propTypes = { knobs: PropTypes.arrayOf( PropTypes.shape({ name: PropTypes.string, value: PropTypes.any, }) - ), + ).isRequired, onFieldChange: PropTypes.func.isRequired, onFieldClick: PropTypes.func.isRequired, }; diff --git a/addons/knobs/src/components/__tests__/GroupTabs.js b/addons/knobs/src/components/__tests__/GroupTabs.js deleted file mode 100644 index aa978bedb96..00000000000 --- a/addons/knobs/src/components/__tests__/GroupTabs.js +++ /dev/null @@ -1,62 +0,0 @@ -import React from 'react'; -import { shallow } from 'enzyme'; -import GroupTabs from '../GroupTabs'; - -describe('GroupTabs', () => { - test('should render only the selected group with display set other than "none"', () => { - const groups = { - test1: { - render() { - return
TEST 1
; - }, - }, - test2: { - render() { - return
TEST 2
; - }, - }, - }; - - const onGroupSelect = () => 'onGroupSelect'; - - const wrapper = shallow( - - ); - - expect(wrapper.find('#test1').parent()).toHaveStyle('display', 'none'); - expect(wrapper.find('#test2').parent()).not.toHaveStyle('display', 'none'); - }); - - test('should set onGroupSelected as onClick handlers of tabs', () => { - const groups = { - test1: { - title: 'test 1', - render() { - return
TEST 1
; - }, - }, - }; - const onGroupSelect = jest.fn(); - const preventDefault = jest.fn(); - const wrapper = shallow( - - ); - wrapper - .find('Styled(button)') - .dive() - .simulate('click', { preventDefault }); - - expect(onGroupSelect).toHaveBeenCalled(); - expect(preventDefault).toHaveBeenCalled(); - }); - - describe('when no groups are given', () => { - test('should render "no groups available"', () => { - const groups = {}; - const onGroupSelect = () => 'onGroupSelect'; - const wrapper = shallow(); - - expect(wrapper.contains('no groups available')).toBe(true); - }); - }); -}); diff --git a/addons/knobs/src/components/__tests__/Panel.js b/addons/knobs/src/components/__tests__/Panel.js index ef0389705b6..dc9b17d163f 100644 --- a/addons/knobs/src/components/__tests__/Panel.js +++ b/addons/knobs/src/components/__tests__/Panel.js @@ -6,14 +6,14 @@ describe('Panel', () => { it('should subscribe to setKnobs event of channel', () => { const testChannel = { on: jest.fn() }; const testApi = { onStory: jest.fn() }; - shallow(); + shallow(); expect(testChannel.on).toHaveBeenCalledWith('addon:knobs:setKnobs', jasmine.any(Function)); }); it('should subscribe to onStory event', () => { const testChannel = { on: jest.fn() }; const testApi = { onStory: jest.fn() }; - shallow(); + shallow(); expect(testApi.onStory).toHaveBeenCalled(); expect(testChannel.on).toHaveBeenCalledWith('addon:knobs:setKnobs', jasmine.any(Function)); @@ -41,7 +41,7 @@ describe('Panel', () => { onStory: jest.fn(), }; - shallow(); + shallow(); const setKnobsHandler = handlers['addon:knobs:setKnobs']; const knobs = { @@ -67,7 +67,7 @@ describe('Panel', () => { expect(testChannel.emit).toHaveBeenCalledWith(e, knobFromUrl); }); - it('should set query params when url params are already read', () => { + it('should remove query params when url params are already read', () => { const handlers = {}; const testChannel = { @@ -88,7 +88,7 @@ describe('Panel', () => { onStory: jest.fn(), }; - const wrapper = shallow(); + const wrapper = shallow(); const setKnobsHandler = handlers['addon:knobs:setKnobs']; const knobs = { @@ -109,8 +109,8 @@ describe('Panel', () => { setKnobsHandler({ knobs, timestamp: +new Date() }); const knobFromStory = { - 'knob-foo': knobs.foo.value, - 'knob-baz': knobs.baz.value, + 'knob-foo': null, + 'knob-baz': null, }; expect(testApi.setQueryParams).toHaveBeenCalledWith(knobFromStory); @@ -130,7 +130,7 @@ describe('Panel', () => { onStory: jest.fn(), }; - const wrapper = shallow(); + const wrapper = shallow(); const testChangedKnob = { name: 'foo', @@ -140,8 +140,8 @@ describe('Panel', () => { wrapper.instance().handleChange(testChangedKnob); expect(testChannel.emit).toHaveBeenCalledWith('addon:knobs:knobChange', testChangedKnob); - const paramsChange = { 'knob-foo': 'changed text' }; - expect(testApi.setQueryParams).toHaveBeenCalledWith(paramsChange); + // const paramsChange = { 'knob-foo': 'changed text' }; + // expect(testApi.setQueryParams).toHaveBeenCalledWith(paramsChange); }); }); }); diff --git a/addons/knobs/src/components/types/Array.js b/addons/knobs/src/components/types/Array.js index 6ccfedb1ae2..28e2851dd37 100644 --- a/addons/knobs/src/components/types/Array.js +++ b/addons/knobs/src/components/types/Array.js @@ -1,24 +1,7 @@ import PropTypes from 'prop-types'; import React from 'react'; -import styled from 'react-emotion'; -import Textarea from 'react-textarea-autosize'; -import debounce from 'lodash.debounce'; - -const StyledTextarea = styled(Textarea)({ - display: 'table-cell', - boxSizing: 'border-box', - verticalAlign: 'middle', - height: '26px', - width: '100%', - maxWidth: '100%', - outline: 'none', - border: '1px solid #f7f4f4', - borderRadius: 2, - fontSize: 11, - padding: '5px', - color: '#555', -}); +import { Textarea } from '@storybook/components'; function formatArray(value, separator) { if (value === '') { @@ -28,34 +11,21 @@ function formatArray(value, separator) { } class ArrayType extends React.Component { - constructor(props, context) { - super(props, context); - - this.state = { - value: props.knob.value.join(props.knob.separator), - }; - - this.onChange = debounce(this.props.onChange, 200); - } - - componentWillUnmount() { - this.onChange.cancel(); + shouldComponentUpdate(nextProps) { + return nextProps.knob.value !== this.props.knob.value; } handleChange = e => { const { knob } = this.props; const { value } = e.target; const newVal = formatArray(value, knob.separator); - - this.setState({ value }); - this.onChange(newVal); + this.props.onChange(newVal); }; render() { const { knob } = this.props; - const { value } = this.state; - return ; + return ); -storiesOf('Addons|Info.GitHub issues', module).add( - '#1814', - withInfo('Allow Duplicate DisplayNames for HOC #1814')(() => ( -
- - , + , + , +] +`; + +exports[`Storyshots Components|Form/Textarea sizes 1`] = ` +Array [ + , + , + , +] +`; + +exports[`Storyshots Components|Form/Textarea validations 1`] = ` +Array [ + , + , + , + , +] +`; + +exports[`Storyshots Components|Form/Textarea with knobs (controlled) 1`] = ` + +`; diff --git a/lib/components/src/form/button.js b/lib/components/src/form/button.js deleted file mode 100644 index 41812381b2c..00000000000 --- a/lib/components/src/form/button.js +++ /dev/null @@ -1,26 +0,0 @@ -import styled from 'react-emotion'; -import { baseFonts } from '../theme'; - -export default styled('button')({ - ...baseFonts, - border: 'none', - boxShadow: '0 0 0 1px rgba(0, 0, 0, 0.2)', - backgroundColor: 'rgb(255, 255, 255)', - padding: '4px 10px 7px', - borderRadius: 4, - cursor: 'pointer', - transition: 'box-shadow 0.15s ease-out', - ':hover': { - transition: 'background-color 0.15s ease-out', - boxShadow: '0 0 0 1px rgba(0, 0, 0, 0.3)', - }, - ':focus': { - transition: 'background-color 0.15s ease-out', - outline: 'none', - boxShadow: '0 0 0 2px rgba(0, 0, 0, 0.3)', - }, - ':active': { - transition: 'none', - backgroundColor: 'rgb(247, 247, 247)', - }, -}); diff --git a/lib/components/src/form/button.stories.js b/lib/components/src/form/button.stories.js deleted file mode 100644 index caba82eab12..00000000000 --- a/lib/components/src/form/button.stories.js +++ /dev/null @@ -1,10 +0,0 @@ -import React from 'react'; -import { storiesOf } from '@storybook/react'; -import { action } from '@storybook/addon-actions'; -import { withKnobs, text } from '@storybook/addon-knobs'; - -import Button from './button'; - -storiesOf('Components|Form/Button', module) - .addDecorator(withKnobs) - .add('with onclick', () => ); diff --git a/lib/components/src/form/field.js b/lib/components/src/form/field.js new file mode 100644 index 00000000000..c7730ff9a67 --- /dev/null +++ b/lib/components/src/form/field.js @@ -0,0 +1,47 @@ +import React from 'react'; +import PropTypes from 'prop-types'; +import styled from 'react-emotion'; + +const Wrapper = styled('label')(({ theme }) => ({ + width: '100%', + boxSizing: 'border-box', + display: 'flex', + margin: '8px 0', + borderBottom: theme.mainBorder, + padding: '0 8px 8px 8px', + + '&:last-child': { + borderBottom: '0 none', + paddingBottom: 0, + }, +})); + +const Label = styled('span')({ + minWidth: 100, + minHeight: 32, + marginRight: 16, + display: 'flex', + justifyContent: 'flex-start', + alignItems: 'center', + lineHeight: '16px', +}); + +const Field = ({ label, children }) => ( + + {label ? ( + + ) : null} + {children} + +); +Field.propTypes = { + label: PropTypes.node, + children: PropTypes.node.isRequired, +}; +Field.defaultProps = { + label: undefined, +}; + +export default Field; diff --git a/lib/components/src/form/form.stories.js b/lib/components/src/form/form.stories.js new file mode 100644 index 00000000000..5c2e1bc9efb --- /dev/null +++ b/lib/components/src/form/form.stories.js @@ -0,0 +1,93 @@ +import React, { Fragment } from 'react'; +import { storiesOf } from '@storybook/react'; +import { action } from '@storybook/addon-actions'; +import { withKnobs, text } from '@storybook/addon-knobs'; + +import { Button, Select, Textarea } from './input'; + +storiesOf('Components|Form/Select', module) + .addDecorator(withKnobs) + .add('sizes', () => ( + + {['auto', 'flex', '100%'].map(size => ( + + ))} + + )) + .add('validations', () => ( + + {['error', 'warn', 'valid', null].map(valid => ( + + ))} + + )); +storiesOf('Components|Form/Button', module) + .addDecorator(withKnobs) + .add('with knobs', () => ( + + )) + .add('sizes', () => ( + + {['auto', 'flex', '100%'].map(size => ( + + ))} + + )) + .add('validations', () => ( + + {['error', 'warn', 'valid', null].map(valid => ( + + ))} + + )) + .add('alignment', () => ( + + {['end', 'center', 'start'].map(align => ( + + ))} + + )); + +storiesOf('Components|Form/Textarea', module) + .addDecorator(withKnobs) + .add('with knobs (controlled)', () => ( +