Merge branch 'next' into fix/theming-flash

This commit is contained in:
Norbert de Langen 2019-03-02 18:08:48 +01:00
commit 242e6111ec
11 changed files with 199 additions and 10 deletions

View File

@ -1,3 +1,33 @@
## 5.0.0-rc.8 (March 1, 2019)
### Features
* Core: Allow local decorators via params ([#5806](https://github.com/storybooks/storybook/pull/5806))
### Bug Fixes
* UI: Sort storiesHash so grouped keys appear together ([#5805](https://github.com/storybooks/storybook/pull/5805))
* UI: Close tooltips on iframe clicks on keypresses ([#5807](https://github.com/storybooks/storybook/pull/5807))
* Addon-Info: Add font family to info panel ([#5759](https://github.com/storybooks/storybook/pull/5759))
## 5.0.0-rc.7 (February 28, 2019)
### Features
* UI: Page load animation and `STORIES_CONFIGURED` event ([#5756](https://github.com/storybooks/storybook/pull/5756))
* Theming: Improve `brand` API ([#5733](https://github.com/storybooks/storybook/pull/5733))
* UI: Fuzzy search improvement ([#5748](https://github.com/storybooks/storybook/pull/5748))
* UI: Add toolbar animation ([#5742](https://github.com/storybooks/storybook/pull/5742))
### Bug Fixes
* UI: Fix update notifications placement ([#5716](https://github.com/storybooks/storybook/pull/5716))
* Angular: Fix global style imports ([#5776](https://github.com/storybooks/storybook/pull/5776))
* Addon-options: Add backwards compatibility ([#5758](https://github.com/storybooks/storybook/pull/5758))
* Addon-options: Fix deprecated url/name options ([#5773](https://github.com/storybooks/storybook/pull/5773))
* Addon-knobs: Remove call to `forceReRender()` on `STORY_CHANGED` ([#5753](https://github.com/storybooks/storybook/pull/5753))
* UI: Fix active state in addon-background, addon-viewport tools ([#5749](https://github.com/storybooks/storybook/pull/5749))
## 5.0.0-rc.6 (February 25, 2019) ## 5.0.0-rc.6 (February 25, 2019)
### Bug Fixes ### Bug Fixes

View File

@ -179,3 +179,10 @@ vanilla:
demo: https://vanilla-framework.github.io/vanilla-framework-react/ demo: https://vanilla-framework.github.io/vanilla-framework-react/
source: https://github.com/vanilla-framework/vanilla-framework-react source: https://github.com/vanilla-framework/vanilla-framework-react
site: https://vanillaframework.io/ site: https://vanillaframework.io/
govuk:
thumbnail: gov-uk.png
title: GOV.UK react
description: An implementation of the GOV.UK Design System in React using CSSinJS
demo: https://govuk-react.github.io/govuk-react/
source: https://github.com/govuk-react/govuk-react
site: https://design-system.service.gov.uk/

Binary file not shown.

After

Width:  |  Height:  |  Size: 53 KiB

View File

@ -1,8 +1,8 @@
{ {
"next": { "next": {
"version": "5.0.0-rc.6", "version": "5.0.0-rc.8",
"info": { "info": {
"plain": "### Bug Fixes\n\n* Addon-actions: FIX performance by upgrading to telejson 2.1 ([#5751](https://github.com/storybooks/storybook/pull/5751))\n* UI: FIX bad treeview mockdata ([#5741](https://github.com/storybooks/storybook/pull/5741))\n* UI: About page styling fixes ([#5732](https://github.com/storybooks/storybook/pull/5732))\n* UI: Restore the toolbar eject button ([#5737](https://github.com/storybooks/storybook/pull/5737))" "plain": "### Features\n\n* Core: Allow local decorators via params ([#5806](https://github.com/storybooks/storybook/pull/5806))\n\n### Bug Fixes\n\n* UI: Sort storiesHash so grouped keys appear together ([#5805](https://github.com/storybooks/storybook/pull/5805))\n* UI: Close tooltips on iframe clicks on keypresses ([#5807](https://github.com/storybooks/storybook/pull/5807))\n* Addon-Info: Add font family to info panel ([#5759](https://github.com/storybooks/storybook/pull/5759))"
} }
}, },
"latest": { "latest": {

View File

@ -0,0 +1,39 @@
import React from 'react';
// We would need to add this in config.js idomatically however that would make this file a bit confusing
import { addDecorator } from '@storybook/react';
addDecorator((s, { kind }) =>
kind === 'Core|Decorators' ? (
<>
<p>Global Decorator</p>
{s()}
</>
) : (
s()
)
);
export default {
title: 'Core|Decorators',
decorators: [
s => (
<>
<p>Kind Decorator</p>
{s()}
</>
),
],
};
export const all = () => <p>Story</p>;
all.parameters = {
decorators: [
s => (
<>
<p>Local Decorator</p>
{s()}
</>
),
],
};

View File

@ -5077,6 +5077,23 @@ exports[`Storyshots Basics|ScrollArea vertical 1`] = `
</div> </div>
`; `;
exports[`Storyshots Core|Decorators all 1`] = `
Array [
<p>
Global Decorator
</p>,
<p>
Kind Decorator
</p>,
<p>
Local Decorator
</p>,
<p>
Story
</p>,
]
`;
exports[`Storyshots Core|Events Force re-render 1`] = ` exports[`Storyshots Core|Events Force re-render 1`] = `
.emotion-0 { .emotion-0 {
border: 0; border: 0;

View File

@ -203,7 +203,12 @@ export default class ClientApi {
}, },
{ {
applyDecorators: this._decorateStory, applyDecorators: this._decorateStory,
getDecorators: () => [...localDecorators, ..._globalDecorators, withSubscriptionTracking], getDecorators: () => [
...(allParam.decorators || []),
...localDecorators,
..._globalDecorators,
withSubscriptionTracking,
],
} }
); );
return api; return api;

View File

@ -1,7 +1,8 @@
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { withState } from 'recompose'; import { withState, lifecycle } from 'recompose';
import { document } from 'global';
import TooltipTrigger from 'react-popper-tooltip'; import TooltipTrigger from 'react-popper-tooltip';
import Tooltip from './Tooltip'; import Tooltip from './Tooltip';
@ -17,7 +18,8 @@ const TargetSvgContainer = styled.g`
cursor: ${props => (props.mode === 'hover' ? 'default' : 'pointer')}; cursor: ${props => (props.mode === 'hover' ? 'default' : 'pointer')};
`; `;
const WithTooltip = ({ // Pure, does not bind to the body
const WithTooltipPure = ({
svg, svg,
trigger, trigger,
closeOnClick, closeOnClick,
@ -69,7 +71,7 @@ const WithTooltip = ({
); );
}; };
WithTooltip.propTypes = { WithTooltipPure.propTypes = {
svg: PropTypes.bool, svg: PropTypes.bool,
trigger: PropTypes.string, trigger: PropTypes.string,
closeOnClick: PropTypes.bool, closeOnClick: PropTypes.bool,
@ -82,7 +84,7 @@ WithTooltip.propTypes = {
onVisibilityChange: PropTypes.func.isRequired, onVisibilityChange: PropTypes.func.isRequired,
}; };
WithTooltip.defaultProps = { WithTooltipPure.defaultProps = {
svg: false, svg: false,
trigger: 'hover', trigger: 'hover',
closeOnClick: false, closeOnClick: false,
@ -92,6 +94,44 @@ WithTooltip.defaultProps = {
tooltipShown: false, tooltipShown: false,
}; };
const WithTooltip = lifecycle({
componentDidMount() {
const { onVisibilityChange } = this.props;
const hide = () => onVisibilityChange(false);
document.addEventListener('keydown', hide, false);
// Find all iframes on the screen and bind to clicks inside them (waiting until the iframe is ready)
const iframes = Array.from(document.getElementsByTagName('iframe'));
const unbinders = [];
iframes.forEach(iframe => {
const bind = () => {
iframe.contentDocument.addEventListener('click', hide);
unbinders.push(() => {
iframe.contentDocument.removeEventListener('click', hide);
});
};
bind(); // I don't know how to find out if it's already loaded so I potentially will bind twice
iframe.addEventListener('load', bind);
unbinders.push(() => {
iframe.removeEventListener('load', bind);
});
});
this.unbind = () => {
document.removeEventListener('keydown', hide);
unbinders.forEach(unbind => {
unbind();
});
};
},
componentWillUnmount() {
if (this.unbind) {
this.unbind();
}
},
})(WithTooltipPure);
export default WithTooltip; export default WithTooltip;
const WithToolTipState = withState( const WithToolTipState = withState(
@ -100,4 +140,4 @@ const WithToolTipState = withState(
({ startOpen }) => startOpen ({ startOpen }) => startOpen
)(WithTooltip); )(WithTooltip);
export { WithToolTipState }; export { WithTooltipPure, WithToolTipState };

View File

@ -81,7 +81,6 @@ export default ({
resolve: { resolve: {
extensions: ['.mjs', '.js', '.jsx', '.json'], extensions: ['.mjs', '.js', '.jsx', '.json'],
modules: ['node_modules'].concat(raw.NODE_PATH || []), modules: ['node_modules'].concat(raw.NODE_PATH || []),
mainFields: ['browser', 'main', 'module'],
alias: { alias: {
'core-js': path.dirname(require.resolve('core-js/package.json')), 'core-js': path.dirname(require.resolve('core-js/package.json')),
...reactPaths, ...reactPaths,

View File

@ -114,7 +114,8 @@ const initStoriesApi = ({
}); });
const setStories = input => { const setStories = input => {
const storiesHash = Object.values(input).reduce((acc, item) => { // This doesn't quite have the right order -- it does not group the top-level keys, see #5518
const storiesHashOutOfOrder = Object.values(input).reduce((acc, item) => {
const { kind, parameters } = item; const { kind, parameters } = item;
const { const {
hierarchyRootSeparator: rootSeparator, hierarchyRootSeparator: rootSeparator,
@ -162,6 +163,22 @@ const initStoriesApi = ({
return acc; return acc;
}, {}); }, {});
// When adding a group, also add all of its children, depth first
function addItem(acc, item) {
if (!acc[item]) {
// If we were already inserted as part of a group, that's great.
acc[item.id] = item;
const { children } = item;
if (children) {
children.forEach(id => addItem(acc, storiesHashOutOfOrder[id]));
}
}
return acc;
}
// Now create storiesHash by reordering the above by group
const storiesHash = Object.values(storiesHashOutOfOrder).reduce(addItem, {});
const { storyId, viewMode } = store.getState(); const { storyId, viewMode } = store.getState();
if (!storyId || !storiesHash[storyId]) { if (!storyId || !storiesHash[storyId]) {

View File

@ -144,6 +144,41 @@ describe('stories API', () => {
}); });
}); });
// Stories can get out of order for a few reasons -- see reproductions on
// https://github.com/storybooks/storybook/issues/5518
it('does the right thing for out of order stories', () => {
const navigate = jest.fn();
const store = createMockStore();
const {
api: { setStories },
} = initStories({ store, navigate });
setStories({
'a--1': { kind: 'a', name: '1', parameters, path: 'a--1', id: 'a--1' },
'b--1': { kind: 'b', name: '1', parameters, path: 'b--1', id: 'b--1' },
'a--2': { kind: 'a', name: '2', parameters, path: 'a--2', id: 'a--2' },
});
const { storiesHash: storedStoriesHash } = store.getState();
// We need exact key ordering, even if in theory JS doens't guarantee it
expect(Object.keys(storedStoriesHash)).toEqual(['a', 'a--1', 'a--2', 'b', 'b--1']);
expect(storedStoriesHash.a).toMatchObject({
id: 'a',
children: ['a--1', 'a--2'],
isRoot: false,
isComponent: true,
});
expect(storedStoriesHash.b).toMatchObject({
id: 'b',
children: ['b--1'],
isRoot: false,
isComponent: true,
});
});
it('navigates to the first story in the store if there is none selected', () => { it('navigates to the first story in the store if there is none selected', () => {
const navigate = jest.fn(); const navigate = jest.fn();
const store = { getState: () => ({ viewMode: 'story' }), setState: jest.fn() }; const store = { getState: () => ({ viewMode: 'story' }), setState: jest.fn() };