REFACTOR preview in prep for Addon type wrapper

I think we even need need an addon type Context... hmm
This commit is contained in:
Norbert de Langen 2019-01-27 10:53:49 +01:00
parent 7e5ac469e5
commit 292d9152f7
13 changed files with 433 additions and 288 deletions

View File

@ -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>
); );
} }
} }

View File

@ -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>
),
});
}); });

View File

@ -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>
); );
} }
} }

View File

@ -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} />,
}); });
}); });

View File

@ -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} />,
}); });
}); });

View File

@ -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;

View 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 };

View 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,
};

View File

@ -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>
); );
} }
} }

View File

@ -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',

View 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 };

View File

@ -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!');

View File

@ -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,