mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 16:11:33 +08:00
REFACTOR preview in prep for Addon type wrapper
I think we even need need an addon type Context... hmm
This commit is contained in:
parent
7e5ac469e5
commit
292d9152f7
@ -1,5 +1,5 @@
|
|||||||
import { document } from 'global';
|
import { document } from 'global';
|
||||||
import React, { Component, Fragment } from 'react';
|
import React, { Component } from 'react';
|
||||||
import memoize from 'memoizerific';
|
import memoize from 'memoizerific';
|
||||||
import { styled } from '@storybook/theming';
|
import { styled } from '@storybook/theming';
|
||||||
|
|
||||||
@ -17,10 +17,6 @@ const ColorIcon = styled.span(
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
const Hidden = styled.div(() => ({
|
|
||||||
display: 'none',
|
|
||||||
}));
|
|
||||||
|
|
||||||
class ColorBlindness extends Component {
|
class ColorBlindness extends Component {
|
||||||
state = {
|
state = {
|
||||||
filter: false,
|
filter: false,
|
||||||
@ -44,118 +40,54 @@ class ColorBlindness extends Component {
|
|||||||
const { filter } = this.state;
|
const { filter } = this.state;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Fragment>
|
<Popout key="filters">
|
||||||
<Hidden>
|
<IconButton key="filter" active={!!filter} title="Color Blindness Emulation">
|
||||||
<svg key="svg">
|
<Icons icon="mirror" />
|
||||||
<defs>
|
</IconButton>
|
||||||
<filter id="protanopia">
|
{({ hide }) => (
|
||||||
<feColorMatrix
|
<List>
|
||||||
in="SourceGraphic"
|
{[
|
||||||
type="matrix"
|
'protanopia',
|
||||||
values="0.567, 0.433, 0, 0, 0 0.558, 0.442, 0, 0, 0 0, 0.242, 0.758, 0, 0 0, 0, 0, 1, 0"
|
'protanomaly',
|
||||||
/>
|
'deuteranopia',
|
||||||
</filter>
|
'deuteranomaly',
|
||||||
<filter id="protanomaly">
|
'tritanopia',
|
||||||
<feColorMatrix
|
'tritanomaly',
|
||||||
in="SourceGraphic"
|
'achromatopsia',
|
||||||
type="matrix"
|
'achromatomaly',
|
||||||
values="0.817, 0.183, 0, 0, 0 0.333, 0.667, 0, 0, 0 0, 0.125, 0.875, 0, 0 0, 0, 0, 1, 0"
|
].map(i => (
|
||||||
/>
|
|
||||||
</filter>
|
|
||||||
<filter id="deuteranopia">
|
|
||||||
<feColorMatrix
|
|
||||||
in="SourceGraphic"
|
|
||||||
type="matrix"
|
|
||||||
values="0.625, 0.375, 0, 0, 0 0.7, 0.3, 0, 0, 0 0, 0.3, 0.7, 0, 0 0, 0, 0, 1, 0"
|
|
||||||
/>
|
|
||||||
</filter>
|
|
||||||
<filter id="deuteranomaly">
|
|
||||||
<feColorMatrix
|
|
||||||
in="SourceGraphic"
|
|
||||||
type="matrix"
|
|
||||||
values="0.8, 0.2, 0, 0, 0 0.258, 0.742, 0, 0, 0 0, 0.142, 0.858, 0, 0 0, 0, 0, 1, 0"
|
|
||||||
/>
|
|
||||||
</filter>
|
|
||||||
<filter id="tritanopia">
|
|
||||||
<feColorMatrix
|
|
||||||
in="SourceGraphic"
|
|
||||||
type="matrix"
|
|
||||||
values="0.95, 0.05, 0, 0, 0 0, 0.433, 0.567, 0, 0 0, 0.475, 0.525, 0, 0 0, 0, 0, 1, 0"
|
|
||||||
/>
|
|
||||||
</filter>
|
|
||||||
<filter id="tritanomaly">
|
|
||||||
<feColorMatrix
|
|
||||||
in="SourceGraphic"
|
|
||||||
type="matrix"
|
|
||||||
values="0.967, 0.033, 0, 0, 0 0, 0.733, 0.267, 0, 0 0, 0.183, 0.817, 0, 0 0, 0, 0, 1, 0"
|
|
||||||
/>
|
|
||||||
</filter>
|
|
||||||
<filter id="achromatopsia">
|
|
||||||
<feColorMatrix
|
|
||||||
in="SourceGraphic"
|
|
||||||
type="matrix"
|
|
||||||
values="0.299, 0.587, 0.114, 0, 0 0.299, 0.587, 0.114, 0, 0 0.299, 0.587, 0.114, 0, 0 0, 0, 0, 1, 0"
|
|
||||||
/>
|
|
||||||
</filter>
|
|
||||||
<filter id="achromatomaly">
|
|
||||||
<feColorMatrix
|
|
||||||
in="SourceGraphic"
|
|
||||||
type="matrix"
|
|
||||||
values="0.618, 0.320, 0.062, 0, 0 0.163, 0.775, 0.062, 0, 0 0.163, 0.320, 0.516, 0, 0 0, 0, 0, 1, 0"
|
|
||||||
/>
|
|
||||||
</filter>
|
|
||||||
</defs>
|
|
||||||
</svg>
|
|
||||||
</Hidden>
|
|
||||||
<Popout key="filters">
|
|
||||||
<IconButton key="filter" active={!!filter} title="Color Blindness Emulation">
|
|
||||||
<Icons icon="mirror" />
|
|
||||||
</IconButton>
|
|
||||||
{({ hide }) => (
|
|
||||||
<List>
|
|
||||||
{[
|
|
||||||
'protanopia',
|
|
||||||
'protanomaly',
|
|
||||||
'deuteranopia',
|
|
||||||
'deuteranomaly',
|
|
||||||
'tritanopia',
|
|
||||||
'tritanomaly',
|
|
||||||
'achromatopsia',
|
|
||||||
'achromatomaly',
|
|
||||||
].map(i => (
|
|
||||||
<Item
|
|
||||||
key={i}
|
|
||||||
onClick={() => {
|
|
||||||
this.setFilter(filter === i ? null : i);
|
|
||||||
hide();
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Icon type={<ColorIcon filter={i} />} />
|
|
||||||
<Title>{i}</Title>
|
|
||||||
</Item>
|
|
||||||
))}
|
|
||||||
<Item
|
<Item
|
||||||
|
key={i}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
this.setFilter(filter === 'mono' ? null : 'mono');
|
this.setFilter(filter === i ? null : i);
|
||||||
hide();
|
hide();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Icon type={<ColorIcon filter="mono" />} />
|
<Icon type={<ColorIcon filter={i} />} />
|
||||||
<Title>mono</Title>
|
<Title>{i}</Title>
|
||||||
</Item>
|
</Item>
|
||||||
<Item
|
))}
|
||||||
onClick={() => {
|
<Item
|
||||||
this.setFilter(null);
|
onClick={() => {
|
||||||
hide();
|
this.setFilter(filter === 'mono' ? null : 'mono');
|
||||||
}}
|
hide();
|
||||||
>
|
}}
|
||||||
<Icon type={<ColorIcon />} />
|
>
|
||||||
<Title>Off</Title>
|
<Icon type={<ColorIcon filter="mono" />} />
|
||||||
</Item>
|
<Title>mono</Title>
|
||||||
</List>
|
</Item>
|
||||||
)}
|
<Item
|
||||||
</Popout>
|
onClick={() => {
|
||||||
</Fragment>
|
this.setFilter(null);
|
||||||
|
hide();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Icon type={<ColorIcon />} />
|
||||||
|
<Title>Off</Title>
|
||||||
|
</Item>
|
||||||
|
</List>
|
||||||
|
)}
|
||||||
|
</Popout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,20 @@
|
|||||||
import React from 'react';
|
import React, { Fragment } from 'react';
|
||||||
import addons, { types } from '@storybook/addons';
|
import addons, { types } from '@storybook/addons';
|
||||||
|
import { styled } from '@storybook/theming';
|
||||||
|
|
||||||
import Panel from './components/Panel';
|
import Panel from './components/Panel';
|
||||||
import ColorBlindness from './components/ColorBlindness';
|
import ColorBlindness from './components/ColorBlindness';
|
||||||
|
|
||||||
import { ADDON_ID, PANEL_ID } from './constants';
|
import { ADDON_ID, PANEL_ID } from './constants';
|
||||||
|
|
||||||
|
const Hidden = styled.div(() => ({
|
||||||
|
display: 'none',
|
||||||
|
}));
|
||||||
|
|
||||||
addons.register(ADDON_ID, api => {
|
addons.register(ADDON_ID, api => {
|
||||||
addons.add(PANEL_ID, {
|
addons.add(PANEL_ID, {
|
||||||
type: types.TOOL,
|
type: types.TOOL,
|
||||||
|
match: ({ viewMode }) => viewMode === 'story',
|
||||||
render: () => <ColorBlindness />,
|
render: () => <ColorBlindness />,
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -18,4 +24,75 @@ addons.register(ADDON_ID, api => {
|
|||||||
// eslint-disable-next-line react/prop-types
|
// eslint-disable-next-line react/prop-types
|
||||||
render: ({ active, key }) => <Panel key={key} api={api} active={active} />,
|
render: ({ active, key }) => <Panel key={key} api={api} active={active} />,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
addons.add(PANEL_ID, {
|
||||||
|
type: types.PREVIEW,
|
||||||
|
render: ({ children }) => (
|
||||||
|
<Fragment>
|
||||||
|
{children}
|
||||||
|
<Hidden>
|
||||||
|
<svg key="svg">
|
||||||
|
<defs>
|
||||||
|
<filter id="protanopia">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.567, 0.433, 0, 0, 0 0.558, 0.442, 0, 0, 0 0, 0.242, 0.758, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
<filter id="protanomaly">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.817, 0.183, 0, 0, 0 0.333, 0.667, 0, 0, 0 0, 0.125, 0.875, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
<filter id="deuteranopia">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.625, 0.375, 0, 0, 0 0.7, 0.3, 0, 0, 0 0, 0.3, 0.7, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
<filter id="deuteranomaly">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.8, 0.2, 0, 0, 0 0.258, 0.742, 0, 0, 0 0, 0.142, 0.858, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
<filter id="tritanopia">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.95, 0.05, 0, 0, 0 0, 0.433, 0.567, 0, 0 0, 0.475, 0.525, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
<filter id="tritanomaly">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.967, 0.033, 0, 0, 0 0, 0.733, 0.267, 0, 0 0, 0.183, 0.817, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
<filter id="achromatopsia">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.299, 0.587, 0.114, 0, 0 0.299, 0.587, 0.114, 0, 0 0.299, 0.587, 0.114, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
<filter id="achromatomaly">
|
||||||
|
<feColorMatrix
|
||||||
|
in="SourceGraphic"
|
||||||
|
type="matrix"
|
||||||
|
values="0.618, 0.320, 0.062, 0, 0 0.163, 0.775, 0.062, 0, 0 0.163, 0.320, 0.516, 0, 0 0, 0, 0, 1, 0"
|
||||||
|
/>
|
||||||
|
</filter>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
</Hidden>
|
||||||
|
</Fragment>
|
||||||
|
),
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -92,46 +92,44 @@ export default class BackgroundTool extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Fragment>
|
<Popout key="backgrounds">
|
||||||
<Popout key="backgrounds">
|
<IconButton key="background" title="Backgrounds">
|
||||||
<IconButton key="background" title="Backgrounds">
|
<Icons icon="photo" />
|
||||||
<Icons icon="photo" />
|
</IconButton>
|
||||||
</IconButton>
|
{({ hide }) => (
|
||||||
{({ hide }) => (
|
<List>
|
||||||
<List>
|
{selected !== undefined ? (
|
||||||
{selected !== undefined ? (
|
<Fragment>
|
||||||
<Fragment>
|
|
||||||
<Item
|
|
||||||
key="reset"
|
|
||||||
onClick={() => {
|
|
||||||
hide();
|
|
||||||
this.change(undefined);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Icon type="undo" />
|
|
||||||
<Title>Reset</Title>
|
|
||||||
<Detail>transparent</Detail>
|
|
||||||
</Item>
|
|
||||||
</Fragment>
|
|
||||||
) : null}
|
|
||||||
|
|
||||||
{list.map(([key, value]) => (
|
|
||||||
<Item
|
<Item
|
||||||
key={key}
|
key="reset"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
hide();
|
hide();
|
||||||
this.change(key);
|
this.change(undefined);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Icon type={<S.ColorIcon background={value} />} />
|
<Icon type="undo" />
|
||||||
<Title>{key}</Title>
|
<Title>Reset</Title>
|
||||||
<Detail>{value}</Detail>
|
<Detail>transparent</Detail>
|
||||||
</Item>
|
</Item>
|
||||||
))}
|
</Fragment>
|
||||||
</List>
|
) : null}
|
||||||
)}
|
|
||||||
</Popout>
|
{list.map(([key, value]) => (
|
||||||
</Fragment>
|
<Item
|
||||||
|
key={key}
|
||||||
|
onClick={() => {
|
||||||
|
hide();
|
||||||
|
this.change(key);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Icon type={<S.ColorIcon background={value} />} />
|
||||||
|
<Title>{key}</Title>
|
||||||
|
<Detail>{value}</Detail>
|
||||||
|
</Item>
|
||||||
|
))}
|
||||||
|
</List>
|
||||||
|
)}
|
||||||
|
</Popout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,6 +7,7 @@ import Tool from './Tool';
|
|||||||
addons.register(ADDON_ID, api => {
|
addons.register(ADDON_ID, api => {
|
||||||
addons.add(ADDON_ID, {
|
addons.add(ADDON_ID, {
|
||||||
type: types.TOOL,
|
type: types.TOOL,
|
||||||
|
match: ({ viewMode }) => viewMode === 'story',
|
||||||
render: () => <Tool api={api} />,
|
render: () => <Tool api={api} />,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -10,6 +10,7 @@ addons.register(ADDON_ID, api => {
|
|||||||
addons.add(ADDON_ID, {
|
addons.add(ADDON_ID, {
|
||||||
type: types.TOOL,
|
type: types.TOOL,
|
||||||
title: 'viewport / media-queries',
|
title: 'viewport / media-queries',
|
||||||
|
match: ({ viewMode }) => viewMode === 'story',
|
||||||
render: () => <Tool channel={channel} api={api} />,
|
render: () => <Tool channel={channel} api={api} />,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -2,6 +2,7 @@ export enum types {
|
|||||||
TAB = 'tab',
|
TAB = 'tab',
|
||||||
PANEL = 'panel',
|
PANEL = 'panel',
|
||||||
TOOL = 'tool',
|
TOOL = 'tool',
|
||||||
|
PREVIEW = 'preview',
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Types = types | string;
|
export type Types = types | string;
|
||||||
|
28
lib/components/src/preview/background.js
Normal file
28
lib/components/src/preview/background.js
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
const Context = React.createContext();
|
||||||
|
|
||||||
|
class Provider extends Component {
|
||||||
|
state = {
|
||||||
|
value: 'transparent',
|
||||||
|
grid: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
setValue = value => this.setState({ value });
|
||||||
|
|
||||||
|
setGrid = grid => this.setState({ grid });
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { children } = this.props;
|
||||||
|
const { setValue, setGrid } = this;
|
||||||
|
const { value, grid } = this.state;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Context.Provider value={{ value, setValue, grid, setGrid }}>{children}</Context.Provider>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const { Consumer } = Context;
|
||||||
|
|
||||||
|
export { Consumer as BackgroundConsumer, Provider as BackgroundProvider };
|
33
lib/components/src/preview/iframe.js
Normal file
33
lib/components/src/preview/iframe.js
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
import React, { Component, Fragment } from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
|
||||||
|
import { Global, css } from '@storybook/theming';
|
||||||
|
|
||||||
|
export class IFrame extends Component {
|
||||||
|
// this component renders an iframe, which gets updates via post-messages
|
||||||
|
shouldComponentUpdate() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { id, title, src, allowFullScreen, ...rest } = this.props;
|
||||||
|
return (
|
||||||
|
<Fragment>
|
||||||
|
<Global
|
||||||
|
styles={css({
|
||||||
|
iframe: {
|
||||||
|
border: '0 none',
|
||||||
|
},
|
||||||
|
})}
|
||||||
|
/>
|
||||||
|
<iframe id={id} title={title} src={src} allowFullScreen={allowFullScreen} {...rest} />
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
IFrame.propTypes = {
|
||||||
|
id: PropTypes.string.isRequired,
|
||||||
|
title: PropTypes.string.isRequired,
|
||||||
|
src: PropTypes.string.isRequired,
|
||||||
|
allowFullScreen: PropTypes.bool.isRequired,
|
||||||
|
};
|
@ -1,11 +1,10 @@
|
|||||||
import { window } from 'global';
|
import { window } from 'global';
|
||||||
import React, { Component, Fragment } from 'react';
|
import React, { Component, Fragment } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
import memoize from 'memoizerific';
|
||||||
|
|
||||||
import Events from '@storybook/core-events';
|
import { SET_CURRENT_STORY } from '@storybook/core-events';
|
||||||
import { types } from '@storybook/addons';
|
import { types } from '@storybook/addons';
|
||||||
import { Global, css } from '@storybook/theming';
|
|
||||||
import { Route } from '@storybook/router';
|
|
||||||
|
|
||||||
import Helmet from 'react-helmet-async';
|
import Helmet from 'react-helmet-async';
|
||||||
|
|
||||||
@ -17,60 +16,82 @@ import Zoom from './tools/zoom';
|
|||||||
import { Grid, Background } from './tools/background';
|
import { Grid, Background } from './tools/background';
|
||||||
import * as S from './components';
|
import * as S from './components';
|
||||||
|
|
||||||
class IFrame extends Component {
|
import { ZoomProvider, ZoomConsumer } from './zoom';
|
||||||
shouldComponentUpdate() {
|
import { BackgroundProvider, BackgroundConsumer } from './background';
|
||||||
// this component renders an iframe, which gets updates via post-messages
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
import { IFrame } from './iframe';
|
||||||
const { id, title, src, allowFullScreen, ...rest } = this.props;
|
|
||||||
return (
|
const renderIframe = ({ storyId, id }) => (
|
||||||
<Fragment>
|
<IFrame
|
||||||
<Global
|
key="iframe"
|
||||||
styles={css({
|
id="storybook-preview-iframe"
|
||||||
iframe: {
|
title={id || 'preview'}
|
||||||
border: '0 none',
|
src={`iframe.html?id=${storyId}`}
|
||||||
},
|
allowFullScreen
|
||||||
})}
|
/>
|
||||||
/>
|
);
|
||||||
<iframe id={id} title={title} src={src} allowFullScreen={allowFullScreen} {...rest} />
|
|
||||||
</Fragment>
|
const getElementList = memoize(10)((getFn, type, base) => base.concat(Object.values(getFn(type))));
|
||||||
);
|
|
||||||
}
|
const ActualPreview = ({ wrappers, id, storyId, active }) =>
|
||||||
}
|
wrappers.reduceRight(
|
||||||
IFrame.propTypes = {
|
(acc, wrapper, index) => wrapper.render({ index, children: acc, id, storyId, active }),
|
||||||
id: PropTypes.string.isRequired,
|
renderIframe({ id, storyId })
|
||||||
title: PropTypes.string.isRequired,
|
);
|
||||||
src: PropTypes.string.isRequired,
|
|
||||||
allowFullScreen: PropTypes.bool.isRequired,
|
const defaultWrappers = [
|
||||||
};
|
{ render: ({ children, active }) => <div hidden={!active}>{children}</div> },
|
||||||
|
{
|
||||||
|
render: ({ children }) => (
|
||||||
|
<BackgroundConsumer>
|
||||||
|
{({ value, grid }) => (
|
||||||
|
<Background id="storybook-preview-background" value={value}>
|
||||||
|
{grid ? <Grid /> : null}
|
||||||
|
{children}
|
||||||
|
</Background>
|
||||||
|
)}
|
||||||
|
</BackgroundConsumer>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
render: ({ children }) => (
|
||||||
|
<ZoomConsumer>
|
||||||
|
{({ value }) => (
|
||||||
|
<S.Frame
|
||||||
|
style={{
|
||||||
|
width: `${100 * value}%`,
|
||||||
|
height: `${100 * value}%`,
|
||||||
|
transform: `scale(${1 / value})`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</S.Frame>
|
||||||
|
)}
|
||||||
|
</ZoomConsumer>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const RightTools = () => <Fragment />;
|
||||||
|
|
||||||
// eslint-disable-next-line react/no-multi-comp
|
// eslint-disable-next-line react/no-multi-comp
|
||||||
class Preview extends Component {
|
class Preview extends Component {
|
||||||
state = {
|
shouldComponentUpdate({ storyId, viewMode, options }) {
|
||||||
zoom: 1,
|
const { props } = this;
|
||||||
grid: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
shouldComponentUpdate({ storyId, viewMode, options }, { zoom, grid }) {
|
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
options.isFullscreen !== props.options.isFullscreen ||
|
options.isFullscreen !== props.options.isFullscreen ||
|
||||||
options.isToolshown !== props.options.isToolshown ||
|
options.isToolshown !== props.options.isToolshown ||
|
||||||
viewMode !== props.viewMode ||
|
viewMode !== props.viewMode ||
|
||||||
storyId !== props.storyId ||
|
storyId !== props.storyId
|
||||||
zoom !== state.zoom ||
|
|
||||||
grid !== state.grid
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidUpdate(prevProps) {
|
componentDidUpdate(prevProps) {
|
||||||
const { api, storyId } = this.props;
|
const { api, storyId } = this.props;
|
||||||
const { path: prevStoryId } = prevProps;
|
const { storyId: prevStoryId } = prevProps;
|
||||||
if (storyId && storyId !== prevStoryId) {
|
if (storyId && storyId !== prevStoryId) {
|
||||||
api.emit(Events.SET_CURRENT_STORY, { storyId });
|
api.emit(SET_CURRENT_STORY, { storyId });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -87,102 +108,130 @@ class Preview extends Component {
|
|||||||
description,
|
description,
|
||||||
} = this.props;
|
} = this.props;
|
||||||
|
|
||||||
const { zoom, grid } = this.state;
|
const toolbarHeight = options.isToolshown ? 41 : 0;
|
||||||
const tools = getElements(types.TOOL);
|
|
||||||
const toolList = Object.values(tools);
|
|
||||||
const toolbarHeight = options.isToolshown ? 40 : 0;
|
|
||||||
|
|
||||||
const panels = getElements(types.TAB);
|
const wrappers = getElementList(getElements, types.PREVIEW, defaultWrappers);
|
||||||
const panelList = Object.values(panels);
|
const panels = getElementList(getElements, types.TAB, [
|
||||||
const tabsList = [
|
|
||||||
{
|
{
|
||||||
route: () => `/story/${storyId}`,
|
route: ({ storyId }) => `/story/${storyId}`,
|
||||||
match: () => viewMode === 'story',
|
match: ({ viewMode }) => viewMode === 'story',
|
||||||
|
render: ({ active, id, storyId }) => (
|
||||||
|
<ActualPreview active={active} wrappers={wrappers} id={id} storyId={storyId} />
|
||||||
|
),
|
||||||
title: 'Canvas',
|
title: 'Canvas',
|
||||||
key: 'canvas',
|
key: 'canvas',
|
||||||
},
|
},
|
||||||
].concat(panelList);
|
]);
|
||||||
|
const tools = getElementList(getElements, types.TOOL, [
|
||||||
|
{
|
||||||
|
render: () => (
|
||||||
|
<TabBar key="tabs" scroll={false}>
|
||||||
|
{panels.map((t, index) => {
|
||||||
|
const to = t.route({ storyId, viewMode, path, location });
|
||||||
|
const isActive = t.match({ storyId, viewMode, path, location });
|
||||||
|
return (
|
||||||
|
<S.UnstyledLink key={t.id || `l${index}`} to={to}>
|
||||||
|
<TabButton active={isActive}>{t.title}</TabButton>
|
||||||
|
</S.UnstyledLink>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</TabBar>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
match: ({ viewmode }) => viewMode === 'story',
|
||||||
|
render: () => (
|
||||||
|
<ZoomConsumer>
|
||||||
|
{({ set, value }) => (
|
||||||
|
<Zoom key="zoom" current={value} set={v => set(value * v)} reset={() => set(1)} />
|
||||||
|
)}
|
||||||
|
</ZoomConsumer>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
match: ({ viewmode }) => viewMode === 'story',
|
||||||
|
render: () => (
|
||||||
|
<BackgroundConsumer>
|
||||||
|
{({ setGrid, grid }) => (
|
||||||
|
<IconButton active={!!grid} key="grid" onClick={() => setGrid(!grid)}>
|
||||||
|
<Icons icon="grid" />
|
||||||
|
</IconButton>
|
||||||
|
)}
|
||||||
|
</BackgroundConsumer>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
const extraTools = [
|
||||||
|
{
|
||||||
|
match: ({ viewmode }) => viewMode === 'story',
|
||||||
|
render: () => (
|
||||||
|
<IconButton key="full" onClick={actions.toggleFullscreen}>
|
||||||
|
<Icons icon={options.isFullscreen ? 'cross' : 'expand'} />
|
||||||
|
</IconButton>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
match: ({ viewmode }) => viewMode === 'story',
|
||||||
|
render: () => (
|
||||||
|
<IconButton key="opener" onClick={() => window.open(`iframe.html?id=${storyId}`)}>
|
||||||
|
<Icons icon="share" />
|
||||||
|
</IconButton>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const left = tools
|
||||||
|
.filter(item => !item.match || item.match({ storyId, viewMode, location, path }))
|
||||||
|
.reduce((acc, item, index) => {
|
||||||
|
const content = item.render();
|
||||||
|
|
||||||
|
return content ? (
|
||||||
|
<Fragment key={item.id || item.key || `tool-${index}`}>
|
||||||
|
{acc}
|
||||||
|
{index > 0 ? <Separator title={index} key={`separator-${index}`} /> : null}
|
||||||
|
{content}
|
||||||
|
</Fragment>
|
||||||
|
) : (
|
||||||
|
acc
|
||||||
|
);
|
||||||
|
}, '');
|
||||||
|
|
||||||
|
const right = extraTools
|
||||||
|
.filter(item => !item.match || item.match({ storyId, viewMode, location, path }))
|
||||||
|
.reduce((acc, item, index) => {
|
||||||
|
const content = item.render();
|
||||||
|
|
||||||
|
return content ? (
|
||||||
|
<Fragment key={item.id || item.key || `tool-${index}`}>
|
||||||
|
{acc}
|
||||||
|
{index > 0 ? <Separator title={index} key={`separator-${index}`} /> : null}
|
||||||
|
{content}
|
||||||
|
</Fragment>
|
||||||
|
) : (
|
||||||
|
acc
|
||||||
|
);
|
||||||
|
}, '');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Fragment>
|
<BackgroundProvider>
|
||||||
{id === 'main' && (
|
<ZoomProvider>
|
||||||
<Helmet>
|
<Fragment>
|
||||||
<title>{description ? `${description} ⋅ ` : ''}Storybook</title>
|
{id === 'main' && (
|
||||||
</Helmet>
|
<Helmet key="description">
|
||||||
)}
|
<title>{description ? `${description} ⋅ ` : ''}Storybook</title>
|
||||||
<Toolbar
|
</Helmet>
|
||||||
key="toolbar"
|
)}
|
||||||
shown={options.isToolshown}
|
<Toolbar key="toolbar" shown={options.isToolshown} left={left} right={right} />
|
||||||
left={[]
|
<S.FrameWrap key="frame" offset={toolbarHeight}>
|
||||||
.concat(
|
{panels.map(({ id, key, render, match }) => (
|
||||||
tabsList.length > 1
|
<Fragment key={id || key}>
|
||||||
? [
|
{render({ active: match({ storyId, viewMode, location, path }) })}
|
||||||
<TabBar key="tabs" scroll={false}>
|
</Fragment>
|
||||||
{tabsList.map((t, index) => {
|
))}
|
||||||
const to = t.route({ storyId, viewMode, path, location });
|
</S.FrameWrap>
|
||||||
const isActive = t.match({ storyId, viewMode, path, location });
|
</Fragment>
|
||||||
return (
|
</ZoomProvider>
|
||||||
<S.UnstyledLink key={t.id || `l${index}`} to={to}>
|
</BackgroundProvider>
|
||||||
<TabButton active={isActive}>{t.title}</TabButton>
|
|
||||||
</S.UnstyledLink>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</TabBar>,
|
|
||||||
<Separator key="1" />,
|
|
||||||
]
|
|
||||||
: [<Spacer key="1" />]
|
|
||||||
)
|
|
||||||
.concat([
|
|
||||||
<Zoom
|
|
||||||
key="zoom"
|
|
||||||
current={zoom}
|
|
||||||
set={v => this.setState({ zoom: zoom * v })}
|
|
||||||
reset={() => this.setState({ zoom: 1 })}
|
|
||||||
/>,
|
|
||||||
<Separator key="2" />,
|
|
||||||
<IconButton active={!!grid} key="grid" onClick={() => this.setState({ grid: !grid })}>
|
|
||||||
<Icons icon="grid" />
|
|
||||||
</IconButton>,
|
|
||||||
...toolList.map((t, index) => (
|
|
||||||
<Fragment key={t.id || `t${index}`}>{t.render()}</Fragment>
|
|
||||||
)),
|
|
||||||
])}
|
|
||||||
right={[
|
|
||||||
<Separator key="1" />,
|
|
||||||
<IconButton key="full" onClick={actions.toggleFullscreen}>
|
|
||||||
<Icons icon={options.isFullscreen ? 'cross' : 'expand'} />
|
|
||||||
</IconButton>,
|
|
||||||
<Separator key="2" />,
|
|
||||||
<IconButton key="opener" onClick={() => window.open(`iframe.html?id=${storyId}`)}>
|
|
||||||
<Icons icon="share" />
|
|
||||||
</IconButton>,
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
<S.FrameWrap key="frame" offset={toolbarHeight}>
|
|
||||||
<div hidden={viewMode !== 'story'}>
|
|
||||||
<S.Frame
|
|
||||||
style={{
|
|
||||||
width: `${100 * zoom}%`,
|
|
||||||
height: `${100 * zoom}%`,
|
|
||||||
transform: `scale(${1 / zoom})`,
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Background id="storybook-preview-background">{grid ? <Grid /> : null}</Background>
|
|
||||||
<IFrame
|
|
||||||
id="storybook-preview-iframe"
|
|
||||||
title={id || 'preview'}
|
|
||||||
src={`iframe.html?id=${storyId}`}
|
|
||||||
allowFullScreen
|
|
||||||
/>
|
|
||||||
</S.Frame>
|
|
||||||
{panelList.map(panel => (
|
|
||||||
<Fragment key={panel.id}>
|
|
||||||
{panel.render({ active: panel.match({ storyId, viewMode, location, path }) })}
|
|
||||||
</Fragment>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
</S.FrameWrap>
|
|
||||||
</Fragment>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -63,7 +63,6 @@ const Wrapper = styled.div(({ theme, shown }) => ({
|
|||||||
left: 0,
|
left: 0,
|
||||||
height: 40,
|
height: 40,
|
||||||
width: '100%',
|
width: '100%',
|
||||||
boxSizing: 'border-box',
|
|
||||||
borderBottom: theme.mainBorder,
|
borderBottom: theme.mainBorder,
|
||||||
background: theme.barFill,
|
background: theme.barFill,
|
||||||
color: '#999999',
|
color: '#999999',
|
||||||
|
23
lib/components/src/preview/zoom.js
Normal file
23
lib/components/src/preview/zoom.js
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
const Context = React.createContext();
|
||||||
|
|
||||||
|
class Provider extends Component {
|
||||||
|
state = {
|
||||||
|
value: 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
set = value => this.setState({ value });
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { children } = this.props;
|
||||||
|
const { set } = this;
|
||||||
|
const { value } = this.state;
|
||||||
|
|
||||||
|
return <Context.Provider value={{ value, set }}>{children}</Context.Provider>;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const { Consumer } = Context;
|
||||||
|
|
||||||
|
export { Consumer as ZoomConsumer, Provider as ZoomProvider };
|
@ -55,7 +55,7 @@ export default function(options) {
|
|||||||
logger.line();
|
logger.line();
|
||||||
try {
|
try {
|
||||||
previewReject(error);
|
previewReject(error);
|
||||||
previewProcess.close();
|
// previewProcess.close();
|
||||||
logger.warn('force closed preview build');
|
logger.warn('force closed preview build');
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
logger.warn('Unable to close preview build!');
|
logger.warn('Unable to close preview build!');
|
||||||
|
@ -15,15 +15,18 @@ export const previewProps = {
|
|||||||
path: 'string',
|
path: 'string',
|
||||||
viewMode: 'story',
|
viewMode: 'story',
|
||||||
location: {},
|
location: {},
|
||||||
getElements: () => [
|
getElements: type =>
|
||||||
{
|
type === types.TAB
|
||||||
type: types.TAB,
|
? [
|
||||||
title: 'Notes',
|
{
|
||||||
route: ({ storyId }) => `/info/${storyId}`, // todo add type
|
type: types.TAB,
|
||||||
match: ({ viewMode }) => viewMode === 'info', // todo add type
|
title: 'Notes',
|
||||||
render: () => null,
|
route: ({ storyId }) => `/info/${storyId}`, // todo add type
|
||||||
},
|
match: ({ viewMode }) => viewMode === 'info', // todo add type
|
||||||
],
|
render: () => null,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: [],
|
||||||
options: {
|
options: {
|
||||||
isFullscreen: false,
|
isFullscreen: false,
|
||||||
isToolshown: true,
|
isToolshown: true,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user