mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 06:01:22 +08:00
MIGRATE more of lib/ui
This commit is contained in:
parent
4e61e0ef5f
commit
1f940ff62c
@ -2,6 +2,8 @@ import { Module } from '../index';
|
|||||||
|
|
||||||
export interface Notification {
|
export interface Notification {
|
||||||
id: string;
|
id: string;
|
||||||
|
link: string;
|
||||||
|
content: string;
|
||||||
onClear?: () => void;
|
onClear?: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,7 +139,7 @@ export interface TabsProps {
|
|||||||
selected?: string;
|
selected?: string;
|
||||||
actions?: {
|
actions?: {
|
||||||
onSelect: (id: string) => void;
|
onSelect: (id: string) => void;
|
||||||
};
|
} & Record<string, any>;
|
||||||
backgroundColor?: string;
|
backgroundColor?: string;
|
||||||
absolute?: boolean;
|
absolute?: boolean;
|
||||||
bordered?: boolean;
|
bordered?: boolean;
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import React from 'react';
|
import React, { FunctionComponent } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import { State } from '@storybook/api';
|
||||||
import { styled, lighten, darken } from '@storybook/theming';
|
import { styled, lighten, darken } from '@storybook/theming';
|
||||||
import { Link } from '@storybook/router';
|
import { Link } from '@storybook/router';
|
||||||
|
|
||||||
const baseStyle = ({ theme }) => ({
|
const Notification = styled.div(({ theme }) => ({
|
||||||
display: 'block',
|
display: 'block',
|
||||||
padding: '16px 20px',
|
padding: '16px 20px',
|
||||||
borderRadius: 10,
|
borderRadius: 10,
|
||||||
@ -15,26 +15,21 @@ const baseStyle = ({ theme }) => ({
|
|||||||
backgroundColor:
|
backgroundColor:
|
||||||
theme.base === 'light' ? darken(theme.background.app) : lighten(theme.background.app),
|
theme.base === 'light' ? darken(theme.background.app) : lighten(theme.background.app),
|
||||||
textDecoration: 'none',
|
textDecoration: 'none',
|
||||||
});
|
}));
|
||||||
|
const NotificationLink = Notification.withComponent(Link);
|
||||||
const NotificationLink = styled(Link)(baseStyle);
|
|
||||||
const Notification = styled.div(baseStyle);
|
|
||||||
|
|
||||||
export const NotificationItemSpacer = styled.div({
|
export const NotificationItemSpacer = styled.div({
|
||||||
height: 48,
|
height: 48,
|
||||||
});
|
});
|
||||||
|
|
||||||
export default function NotificationItem({ notification: { content, link } }) {
|
const NotificationItem: FunctionComponent<{
|
||||||
|
notification: State['notifications'][0];
|
||||||
|
}> = ({ notification: { content, link } }) => {
|
||||||
return link ? (
|
return link ? (
|
||||||
<NotificationLink to={link}>{content}</NotificationLink>
|
<NotificationLink to={link}>{content}</NotificationLink>
|
||||||
) : (
|
) : (
|
||||||
<Notification>{content}</Notification>
|
<Notification>{content}</Notification>
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
NotificationItem.propTypes = {
|
|
||||||
notification: PropTypes.shape({
|
|
||||||
content: PropTypes.string.isRequired,
|
|
||||||
link: PropTypes.string,
|
|
||||||
}).isRequired,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export default NotificationItem;
|
@ -1,55 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { styled } from '@storybook/theming';
|
|
||||||
|
|
||||||
import NotificationItem from './item';
|
|
||||||
|
|
||||||
const List = styled.div(
|
|
||||||
{
|
|
||||||
zIndex: 10,
|
|
||||||
|
|
||||||
'> * + *': {
|
|
||||||
marginTop: 10,
|
|
||||||
},
|
|
||||||
'&:empty': {
|
|
||||||
display: 'none',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
({ placement }) =>
|
|
||||||
placement || {
|
|
||||||
bottom: 0,
|
|
||||||
left: 0,
|
|
||||||
right: 0,
|
|
||||||
position: 'fixed',
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
export default function NotificationList({ notifications, placement }) {
|
|
||||||
return (
|
|
||||||
<List placement={placement}>
|
|
||||||
{notifications.map(notification => (
|
|
||||||
<NotificationItem key={notification.id} notification={notification} />
|
|
||||||
))}
|
|
||||||
</List>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
NotificationList.propTypes = {
|
|
||||||
placement: PropTypes.shape({
|
|
||||||
position: PropTypes.string,
|
|
||||||
left: PropTypes.number,
|
|
||||||
right: PropTypes.number,
|
|
||||||
top: PropTypes.number,
|
|
||||||
bottom: PropTypes.number,
|
|
||||||
}),
|
|
||||||
notifications: PropTypes.arrayOf(
|
|
||||||
PropTypes.shape({
|
|
||||||
id: PropTypes.string,
|
|
||||||
content: PropTypes.string.isRequired,
|
|
||||||
link: PropTypes.string,
|
|
||||||
}).isRequired
|
|
||||||
).isRequired,
|
|
||||||
};
|
|
||||||
NotificationList.defaultProps = {
|
|
||||||
placement: undefined,
|
|
||||||
};
|
|
40
lib/ui/src/components/notifications/notifications.tsx
Normal file
40
lib/ui/src/components/notifications/notifications.tsx
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
import React, { FunctionComponent } from 'react';
|
||||||
|
import { State } from '@storybook/api';
|
||||||
|
import { styled, CSSObject } from '@storybook/theming';
|
||||||
|
|
||||||
|
import NotificationItem from './item';
|
||||||
|
|
||||||
|
const List = styled.div<{ placement?: CSSObject }>(
|
||||||
|
{
|
||||||
|
zIndex: 10,
|
||||||
|
|
||||||
|
'> * + *': {
|
||||||
|
marginTop: 10,
|
||||||
|
},
|
||||||
|
'&:empty': {
|
||||||
|
display: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
({ placement }) =>
|
||||||
|
placement || {
|
||||||
|
bottom: 0,
|
||||||
|
left: 0,
|
||||||
|
right: 0,
|
||||||
|
position: 'fixed',
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const NotificationList: FunctionComponent<{
|
||||||
|
placement: CSSObject;
|
||||||
|
notifications: State['notifications'];
|
||||||
|
}> = ({ notifications, placement = undefined }) => {
|
||||||
|
return (
|
||||||
|
<List placement={placement}>
|
||||||
|
{notifications.map(notification => (
|
||||||
|
<NotificationItem key={notification.id} notification={notification} />
|
||||||
|
))}
|
||||||
|
</List>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default NotificationList;
|
@ -1,99 +0,0 @@
|
|||||||
import React, { Component, Fragment } from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { styled } from '@storybook/theming';
|
|
||||||
import { Tabs, Icons, IconButton } from '@storybook/components';
|
|
||||||
|
|
||||||
const DesktopOnlyIconButton = styled(IconButton)({
|
|
||||||
// Hides full screen icon at mobile breakpoint defined in app.js
|
|
||||||
'@media (max-width: 599px)': {
|
|
||||||
display: 'none',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
class SafeTab extends Component {
|
|
||||||
constructor(props) {
|
|
||||||
super(props);
|
|
||||||
this.state = { hasError: false };
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidCatch(error, info) {
|
|
||||||
this.setState({ hasError: true });
|
|
||||||
// eslint-disable-next-line no-console
|
|
||||||
console.error(error, info);
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
|
||||||
const { hasError } = this.state;
|
|
||||||
const { children, title, id } = this.props;
|
|
||||||
if (hasError) {
|
|
||||||
return <h1>Something went wrong.</h1>;
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<div id={id} title={title}>
|
|
||||||
{children}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SafeTab.propTypes = {
|
|
||||||
children: PropTypes.oneOfType([PropTypes.node, PropTypes.func]),
|
|
||||||
title: PropTypes.string.isRequired,
|
|
||||||
id: PropTypes.string.isRequired,
|
|
||||||
};
|
|
||||||
SafeTab.defaultProps = {
|
|
||||||
children: null,
|
|
||||||
};
|
|
||||||
|
|
||||||
const AddonPanel = React.memo(
|
|
||||||
({ panels, actions, selectedPanel, panelPosition, absolute = true }) => (
|
|
||||||
<Tabs
|
|
||||||
absolute={absolute}
|
|
||||||
selected={selectedPanel}
|
|
||||||
actions={actions}
|
|
||||||
flex
|
|
||||||
tools={
|
|
||||||
<Fragment>
|
|
||||||
<DesktopOnlyIconButton
|
|
||||||
key="position"
|
|
||||||
onClick={actions.togglePosition}
|
|
||||||
title="Change orientation"
|
|
||||||
>
|
|
||||||
<Icons icon={panelPosition === 'bottom' ? 'bottombar' : 'sidebaralt'} />
|
|
||||||
</DesktopOnlyIconButton>
|
|
||||||
<DesktopOnlyIconButton
|
|
||||||
key="visibility"
|
|
||||||
onClick={actions.toggleVisibility}
|
|
||||||
title="Hide addons"
|
|
||||||
>
|
|
||||||
<Icons icon="close" />
|
|
||||||
</DesktopOnlyIconButton>
|
|
||||||
</Fragment>
|
|
||||||
}
|
|
||||||
id="storybook-panel-root"
|
|
||||||
>
|
|
||||||
{Object.entries(panels).map(([k, v]) => (
|
|
||||||
<SafeTab key={k} id={k} title={v.title}>
|
|
||||||
{v.render}
|
|
||||||
</SafeTab>
|
|
||||||
))}
|
|
||||||
</Tabs>
|
|
||||||
)
|
|
||||||
);
|
|
||||||
AddonPanel.displayName = 'AddonPanel';
|
|
||||||
AddonPanel.propTypes = {
|
|
||||||
selectedPanel: PropTypes.string,
|
|
||||||
actions: PropTypes.shape({
|
|
||||||
togglePosition: PropTypes.func,
|
|
||||||
toggleVisibility: PropTypes.func,
|
|
||||||
}).isRequired,
|
|
||||||
panels: PropTypes.shape({}).isRequired,
|
|
||||||
panelPosition: PropTypes.oneOf(['bottom', 'right']),
|
|
||||||
absolute: PropTypes.bool,
|
|
||||||
};
|
|
||||||
AddonPanel.defaultProps = {
|
|
||||||
selectedPanel: null,
|
|
||||||
panelPosition: 'right',
|
|
||||||
absolute: true,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default AddonPanel;
|
|
83
lib/ui/src/components/panel/panel.tsx
Normal file
83
lib/ui/src/components/panel/panel.tsx
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
import React, { Component, Fragment } from 'react';
|
||||||
|
import { styled } from '@storybook/theming';
|
||||||
|
import { Tabs, Icons, IconButton } from '@storybook/components';
|
||||||
|
|
||||||
|
const DesktopOnlyIconButton = styled(IconButton)({
|
||||||
|
// Hides full screen icon at mobile breakpoint defined in app.js
|
||||||
|
'@media (max-width: 599px)': {
|
||||||
|
display: 'none',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
interface SafeTabProps {
|
||||||
|
title: string;
|
||||||
|
id: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
class SafeTab extends Component<SafeTabProps, { hasError: boolean }> {
|
||||||
|
constructor(props: SafeTabProps) {
|
||||||
|
super(props);
|
||||||
|
this.state = { hasError: false };
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidCatch(error: Error, info: any) {
|
||||||
|
this.setState({ hasError: true });
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.error(error, info);
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { hasError } = this.state;
|
||||||
|
const { children, title, id } = this.props;
|
||||||
|
if (hasError) {
|
||||||
|
return <h1>Something went wrong.</h1>;
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<div id={id} title={title}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const AddonPanel = React.memo<{
|
||||||
|
selectedPanel: string;
|
||||||
|
actions: { onSelect: (id: string) => void } & Record<string, any>;
|
||||||
|
panels: Record<string, any>;
|
||||||
|
panelPosition: 'bottom' | 'right';
|
||||||
|
absolute: boolean;
|
||||||
|
}>(({ panels, actions, selectedPanel = null, panelPosition = 'right', absolute = true }) => (
|
||||||
|
<Tabs
|
||||||
|
absolute={absolute}
|
||||||
|
selected={selectedPanel}
|
||||||
|
actions={actions}
|
||||||
|
tools={
|
||||||
|
<Fragment>
|
||||||
|
<DesktopOnlyIconButton
|
||||||
|
key="position"
|
||||||
|
onClick={actions.togglePosition}
|
||||||
|
title="Change orientation"
|
||||||
|
>
|
||||||
|
<Icons icon={panelPosition === 'bottom' ? 'bottombar' : 'sidebaralt'} />
|
||||||
|
</DesktopOnlyIconButton>
|
||||||
|
<DesktopOnlyIconButton
|
||||||
|
key="visibility"
|
||||||
|
onClick={actions.toggleVisibility}
|
||||||
|
title="Hide addons"
|
||||||
|
>
|
||||||
|
<Icons icon="close" />
|
||||||
|
</DesktopOnlyIconButton>
|
||||||
|
</Fragment>
|
||||||
|
}
|
||||||
|
id="storybook-panel-root"
|
||||||
|
>
|
||||||
|
{Object.entries(panels).map(([k, v]) => (
|
||||||
|
<SafeTab key={k} id={k} title={v.title}>
|
||||||
|
{v.render}
|
||||||
|
</SafeTab>
|
||||||
|
))}
|
||||||
|
</Tabs>
|
||||||
|
));
|
||||||
|
AddonPanel.displayName = 'AddonPanel';
|
||||||
|
|
||||||
|
export default AddonPanel;
|
@ -1,10 +1,11 @@
|
|||||||
import { DOCS_MODE } from 'global';
|
import { DOCS_MODE } from 'global';
|
||||||
import React from 'react';
|
import React, { FunctionComponent } from 'react';
|
||||||
import memoize from 'memoizerific';
|
import memoize from 'memoizerific';
|
||||||
|
|
||||||
import { Badge } from '@storybook/components';
|
import { Badge } from '@storybook/components';
|
||||||
import { Consumer } from '@storybook/api';
|
import { Consumer, Combo } from '@storybook/api';
|
||||||
|
|
||||||
|
import { StoriesHash } from '@storybook/api/dist/modules/stories';
|
||||||
import { shortcutToHumanString } from '../libs/shortcut';
|
import { shortcutToHumanString } from '../libs/shortcut';
|
||||||
|
|
||||||
import ListItemIcon from '../components/sidebar/ListItemIcon';
|
import ListItemIcon from '../components/sidebar/ListItemIcon';
|
||||||
@ -16,7 +17,7 @@ const focusableUIElements = {
|
|||||||
storyPanelRoot: 'storybook-panel-root',
|
storyPanelRoot: 'storybook-panel-root',
|
||||||
};
|
};
|
||||||
|
|
||||||
const shortcutToHumanStringIfEnabled = (shortcuts, enableShortcuts) =>
|
const shortcutToHumanStringIfEnabled = (shortcuts: string[], enableShortcuts: boolean) =>
|
||||||
enableShortcuts ? shortcutToHumanString(shortcuts) : null;
|
enableShortcuts ? shortcutToHumanString(shortcuts) : null;
|
||||||
|
|
||||||
const createMenu = memoize(1)(
|
const createMenu = memoize(1)(
|
||||||
@ -108,9 +109,9 @@ const createMenu = memoize(1)(
|
|||||||
]
|
]
|
||||||
);
|
);
|
||||||
|
|
||||||
export const collapseAllStories = stories => {
|
export const collapseAllStories = (stories: StoriesHash) => {
|
||||||
// keep track of component IDs that have been rewritten to the ID of their first leaf child
|
// keep track of component IDs that have been rewritten to the ID of their first leaf child
|
||||||
const componentIdToLeafId = {};
|
const componentIdToLeafId: Record<string, string> = {};
|
||||||
|
|
||||||
// 1) remove all leaves
|
// 1) remove all leaves
|
||||||
const leavesRemoved = Object.values(stories).filter(
|
const leavesRemoved = Object.values(stories).filter(
|
||||||
@ -126,8 +127,8 @@ export const collapseAllStories = stories => {
|
|||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
|
|
||||||
const nonLeafChildren = [];
|
const nonLeafChildren: string[] = [];
|
||||||
const leafChildren = [];
|
const leafChildren: string[] = [];
|
||||||
children.forEach(child => (stories[child].isLeaf ? leafChildren : nonLeafChildren).push(child));
|
children.forEach(child => (stories[child].isLeaf ? leafChildren : nonLeafChildren).push(child));
|
||||||
|
|
||||||
if (leafChildren.length === 0) {
|
if (leafChildren.length === 0) {
|
||||||
@ -135,7 +136,13 @@ export const collapseAllStories = stories => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const leafId = leafChildren[0];
|
const leafId = leafChildren[0];
|
||||||
const component = { ...rest, id: leafId, isLeaf: true, isComponent: true };
|
const component = {
|
||||||
|
...rest,
|
||||||
|
id: leafId,
|
||||||
|
isLeaf: true,
|
||||||
|
isComponent: true,
|
||||||
|
children: [] as string[],
|
||||||
|
};
|
||||||
componentIdToLeafId[id] = leafId;
|
componentIdToLeafId[id] = leafId;
|
||||||
|
|
||||||
// this is a component, so it should not have any non-leaf children
|
// this is a component, so it should not have any non-leaf children
|
||||||
@ -160,16 +167,16 @@ export const collapseAllStories = stories => {
|
|||||||
return { children: rewritten, ...rest };
|
return { children: rewritten, ...rest };
|
||||||
});
|
});
|
||||||
|
|
||||||
const result = {};
|
const result = {} as StoriesHash;
|
||||||
childrenRewritten.forEach(item => {
|
childrenRewritten.forEach(item => {
|
||||||
result[item.id] = item;
|
result[item.id] = item;
|
||||||
});
|
});
|
||||||
return result;
|
return result;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const collapseDocsOnlyStories = storiesHash => {
|
export const collapseDocsOnlyStories = (storiesHash: StoriesHash) => {
|
||||||
// keep track of component IDs that have been rewritten to the ID of their first leaf child
|
// keep track of component IDs that have been rewritten to the ID of their first leaf child
|
||||||
const componentIdToLeafId = {};
|
const componentIdToLeafId: Record<string, string> = {};
|
||||||
const docsOnlyStoriesRemoved = Object.values(storiesHash).filter(item => {
|
const docsOnlyStoriesRemoved = Object.values(storiesHash).filter(item => {
|
||||||
if (item.isLeaf && item.parameters && item.parameters.docsOnly) {
|
if (item.isLeaf && item.parameters && item.parameters.docsOnly) {
|
||||||
componentIdToLeafId[item.parent] = item.id;
|
componentIdToLeafId[item.parent] = item.id;
|
||||||
@ -187,7 +194,7 @@ export const collapseDocsOnlyStories = storiesHash => {
|
|||||||
...item,
|
...item,
|
||||||
id: leafId,
|
id: leafId,
|
||||||
isLeaf: true,
|
isLeaf: true,
|
||||||
children: undefined,
|
children: [] as string[],
|
||||||
};
|
};
|
||||||
return collapsed;
|
return collapsed;
|
||||||
}
|
}
|
||||||
@ -203,14 +210,14 @@ export const collapseDocsOnlyStories = storiesHash => {
|
|||||||
return item;
|
return item;
|
||||||
});
|
});
|
||||||
|
|
||||||
const result = {};
|
const result = {} as StoriesHash;
|
||||||
docsOnlyComponentsCollapsed.forEach(item => {
|
docsOnlyComponentsCollapsed.forEach(item => {
|
||||||
result[item.id] = item;
|
result[item.id] = item;
|
||||||
});
|
});
|
||||||
return result;
|
return result;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const mapper = ({ state, api }) => {
|
export const mapper = ({ state, api }: Combo) => {
|
||||||
const {
|
const {
|
||||||
ui: { name, url, enableShortcuts },
|
ui: { name, url, enableShortcuts },
|
||||||
viewMode,
|
viewMode,
|
||||||
@ -236,6 +243,10 @@ export const mapper = ({ state, api }) => {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export default props => (
|
const Nav: FunctionComponent<any> = props => (
|
||||||
<Consumer filter={mapper}>{fromState => <Sidebar {...props} {...fromState} />}</Consumer>
|
<Consumer filter={mapper}>
|
||||||
|
{(fromState: ReturnType<typeof mapper>) => <Sidebar {...props} {...fromState} />}
|
||||||
|
</Consumer>
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export default Nav;
|
@ -1,19 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
|
|
||||||
import { Consumer } from '@storybook/api';
|
|
||||||
|
|
||||||
import Notifications from '../components/notifications/notifications';
|
|
||||||
|
|
||||||
export const mapper = ({ state }) => {
|
|
||||||
const { notifications } = state;
|
|
||||||
|
|
||||||
return {
|
|
||||||
notifications,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
const NotificationConnect = props => (
|
|
||||||
<Consumer filter={mapper}>{fromState => <Notifications {...props} {...fromState} />}</Consumer>
|
|
||||||
);
|
|
||||||
|
|
||||||
export default NotificationConnect;
|
|
21
lib/ui/src/containers/notifications.tsx
Normal file
21
lib/ui/src/containers/notifications.tsx
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import React, { FunctionComponent } from 'react';
|
||||||
|
|
||||||
|
import { Consumer, Combo } from '@storybook/api';
|
||||||
|
|
||||||
|
import Notifications from '../components/notifications/notifications';
|
||||||
|
|
||||||
|
export const mapper = ({ state }: Combo) => {
|
||||||
|
const { notifications } = state;
|
||||||
|
|
||||||
|
return {
|
||||||
|
notifications,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const NotificationConnect: FunctionComponent<any> = props => (
|
||||||
|
<Consumer filter={mapper}>
|
||||||
|
{(fromState: ReturnType<typeof mapper>) => <Notifications {...props} {...fromState} />}
|
||||||
|
</Consumer>
|
||||||
|
);
|
||||||
|
|
||||||
|
export default NotificationConnect;
|
@ -1,22 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import memoize from 'memoizerific';
|
|
||||||
import { Consumer } from '@storybook/api';
|
|
||||||
|
|
||||||
import AddonPanel from '../components/panel/panel';
|
|
||||||
|
|
||||||
const createPanelActions = memoize(1)(api => ({
|
|
||||||
onSelect: panel => api.setSelectedPanel(panel),
|
|
||||||
toggleVisibility: () => api.togglePanel(),
|
|
||||||
togglePosition: () => api.togglePanelPosition(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
const mapper = ({ state, api }) => ({
|
|
||||||
panels: api.getStoryPanels(),
|
|
||||||
selectedPanel: api.getSelectedPanel(),
|
|
||||||
panelPosition: state.layout.panelPosition,
|
|
||||||
actions: createPanelActions(api),
|
|
||||||
});
|
|
||||||
|
|
||||||
export default props => (
|
|
||||||
<Consumer filter={mapper}>{customProps => <AddonPanel {...props} {...customProps} />}</Consumer>
|
|
||||||
);
|
|
26
lib/ui/src/containers/panel.tsx
Normal file
26
lib/ui/src/containers/panel.tsx
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import React, { FunctionComponent } from 'react';
|
||||||
|
import memoize from 'memoizerific';
|
||||||
|
import { Consumer, Combo } from '@storybook/api';
|
||||||
|
|
||||||
|
import AddonPanel from '../components/panel/panel';
|
||||||
|
|
||||||
|
const createPanelActions = memoize(1)(api => ({
|
||||||
|
onSelect: (panel: string) => api.setSelectedPanel(panel),
|
||||||
|
toggleVisibility: () => api.togglePanel(),
|
||||||
|
togglePosition: () => api.togglePanelPosition(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
const mapper = ({ state, api }: Combo) => ({
|
||||||
|
panels: api.getStoryPanels(),
|
||||||
|
selectedPanel: api.getSelectedPanel(),
|
||||||
|
panelPosition: state.layout.panelPosition,
|
||||||
|
actions: createPanelActions(api),
|
||||||
|
});
|
||||||
|
|
||||||
|
const Panel: FunctionComponent<any> = props => (
|
||||||
|
<Consumer filter={mapper}>
|
||||||
|
{(customProps: ReturnType<typeof mapper>) => <AddonPanel {...props} {...customProps} />}
|
||||||
|
</Consumer>
|
||||||
|
);
|
||||||
|
|
||||||
|
export default Panel;
|
@ -1,21 +1,31 @@
|
|||||||
import { PREVIEW_URL } from 'global';
|
import { PREVIEW_URL } from 'global';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
import { Consumer } from '@storybook/api';
|
import { Consumer, Combo } from '@storybook/api';
|
||||||
|
|
||||||
|
import { StoriesHash } from '@storybook/api/dist/modules/stories';
|
||||||
import { Preview } from '../components/preview/preview';
|
import { Preview } from '../components/preview/preview';
|
||||||
|
|
||||||
const nonAlphanumSpace = /[^a-z0-9 ]/gi;
|
const nonAlphanumSpace = /[^a-z0-9 ]/gi;
|
||||||
const doubleSpace = /\s\s/gi;
|
const doubleSpace = /\s\s/gi;
|
||||||
const replacer = match => ` ${match} `;
|
const replacer = (match: string) => ` ${match} `;
|
||||||
const addExtraWhiteSpace = input =>
|
|
||||||
|
const addExtraWhiteSpace = (input: string) =>
|
||||||
input.replace(nonAlphanumSpace, replacer).replace(doubleSpace, ' ');
|
input.replace(nonAlphanumSpace, replacer).replace(doubleSpace, ' ');
|
||||||
const getDescription = (storiesHash, storyId) => {
|
|
||||||
|
const getDescription = (storiesHash: StoriesHash, storyId: string) => {
|
||||||
const storyInfo = storiesHash[storyId];
|
const storyInfo = storiesHash[storyId];
|
||||||
return storyInfo ? addExtraWhiteSpace(`${storyInfo.kind} - ${storyInfo.name}`) : '';
|
|
||||||
|
if (storyInfo) {
|
||||||
|
// @ts-ignore
|
||||||
|
const { kind, name } = storyInfo;
|
||||||
|
return kind && name ? addExtraWhiteSpace(`${kind} - ${name}`) : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
};
|
};
|
||||||
|
|
||||||
const mapper = ({ api, state }) => {
|
const mapper = ({ api, state }: Combo) => {
|
||||||
const { layout, location, customQueryParams, storiesHash, storyId } = state;
|
const { layout, location, customQueryParams, storiesHash, storyId } = state;
|
||||||
const { parameters } = storiesHash[storyId] || {};
|
const { parameters } = storiesHash[storyId] || {};
|
||||||
return {
|
return {
|
||||||
@ -25,13 +35,13 @@ const mapper = ({ api, state }) => {
|
|||||||
description: getDescription(storiesHash, storyId),
|
description: getDescription(storiesHash, storyId),
|
||||||
...api.getUrlState(),
|
...api.getUrlState(),
|
||||||
queryParams: customQueryParams,
|
queryParams: customQueryParams,
|
||||||
docsOnly: parameters && parameters.docsOnly,
|
docsOnly: (parameters && parameters.docsOnly) as boolean,
|
||||||
location,
|
location,
|
||||||
parameters,
|
parameters,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
function getBaseUrl() {
|
function getBaseUrl(): string {
|
||||||
try {
|
try {
|
||||||
return PREVIEW_URL || 'iframe.html';
|
return PREVIEW_URL || 'iframe.html';
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@ -41,7 +51,7 @@ function getBaseUrl() {
|
|||||||
|
|
||||||
const PreviewConnected = React.memo(props => (
|
const PreviewConnected = React.memo(props => (
|
||||||
<Consumer filter={mapper}>
|
<Consumer filter={mapper}>
|
||||||
{fromState => (
|
{(fromState: ReturnType<typeof mapper>) => (
|
||||||
<Preview
|
<Preview
|
||||||
{...props}
|
{...props}
|
||||||
baseUrl={getBaseUrl()}
|
baseUrl={getBaseUrl()}
|
Loading…
x
Reference in New Issue
Block a user