mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 15:31:16 +08:00
169 lines
4.1 KiB
JavaScript
169 lines
4.1 KiB
JavaScript
import React, { Component, Fragment } from 'react';
|
|
import PropTypes from 'prop-types';
|
|
import { document } from 'global';
|
|
import { Global, css } from '@emotion/core';
|
|
import memoize from 'memoizerific';
|
|
|
|
import { Popout, Item, Icons, Icon, IconButton, Title, List } from '@storybook/components';
|
|
import { STORY_CHANGED } from '@storybook/core-events';
|
|
|
|
import { PARAM_KEY } from './constants';
|
|
|
|
const toList = memoize(50)(viewports => Object.entries(viewports));
|
|
|
|
export default class ViewportTool extends Component {
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
viewports: {},
|
|
selected: undefined,
|
|
isRotated: false,
|
|
};
|
|
}
|
|
|
|
componentDidMount() {
|
|
const { api } = this.props;
|
|
this.iframe = document.getElementById('storybook-preview-iframe');
|
|
this.iframeClass = 'storybook-preview-iframe-viewport';
|
|
|
|
if (!this.iframe) {
|
|
throw new Error('Cannot find Storybook iframe');
|
|
}
|
|
|
|
api.on(STORY_CHANGED, this.onStoryChange);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
const { api } = this.props;
|
|
api.off(STORY_CHANGED, this.onStoryChange);
|
|
}
|
|
|
|
onStoryChange = id => {
|
|
const { api } = this.props;
|
|
const viewports = api.getParameters(id, PARAM_KEY);
|
|
|
|
if (viewports) {
|
|
this.setState({ viewports });
|
|
}
|
|
};
|
|
|
|
change = key => {
|
|
this.setState({ selected: key }, () => {
|
|
this.apply();
|
|
});
|
|
};
|
|
|
|
rotate = () => {
|
|
const { isRotated } = this.state;
|
|
this.setState({ isRotated: !isRotated }, () => {
|
|
this.apply();
|
|
});
|
|
};
|
|
|
|
apply = () => {
|
|
const { iframe, iframeClass } = this;
|
|
const { isRotated, selected, viewports } = this.state;
|
|
|
|
if (selected) {
|
|
const {
|
|
styles: { width: a, height: b },
|
|
} = viewports[selected];
|
|
iframe.style.width = isRotated ? b : a;
|
|
iframe.style.height = isRotated ? a : b;
|
|
|
|
if (!iframe.classList.item(iframeClass)) {
|
|
iframe.classList.add(iframeClass);
|
|
}
|
|
} else {
|
|
iframe.style.width = '100%';
|
|
iframe.style.height = '100%';
|
|
iframe.classList.remove(iframeClass);
|
|
}
|
|
};
|
|
|
|
render() {
|
|
const { iframeClass } = this;
|
|
const { viewports, selected } = this.state;
|
|
|
|
const list = toList(viewports);
|
|
|
|
if (!list.length) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<Fragment>
|
|
<Global
|
|
styles={css({
|
|
[`.${iframeClass}`]: {
|
|
border: '10px solid black',
|
|
borderRadius: 4,
|
|
margin: 10,
|
|
},
|
|
})}
|
|
/>
|
|
<Popout key="viewports">
|
|
<IconButton key="viewport" title="Change Viewport">
|
|
<Icons icon="grow" />
|
|
</IconButton>
|
|
{({ hide }) => (
|
|
<List>
|
|
{selected !== undefined ? (
|
|
<Fragment>
|
|
<Item
|
|
key="reset"
|
|
onClick={() => {
|
|
hide();
|
|
this.change(undefined);
|
|
}}
|
|
>
|
|
<Icon type="undo" />
|
|
<Title>Reset (responsive)</Title>
|
|
</Item>
|
|
<Item
|
|
key="rotate"
|
|
onClick={() => {
|
|
hide();
|
|
this.rotate();
|
|
}}
|
|
>
|
|
<Icon type="sync" />
|
|
<Title>Rotate</Title>
|
|
</Item>
|
|
</Fragment>
|
|
) : null}
|
|
|
|
{list.map(([key, { name, type }]) => (
|
|
<Item
|
|
key={key}
|
|
onClick={() => {
|
|
hide();
|
|
this.change(key);
|
|
}}
|
|
>
|
|
<Icon type={type} />
|
|
<Title>{name}</Title>
|
|
</Item>
|
|
))}
|
|
</List>
|
|
)}
|
|
</Popout>
|
|
</Fragment>
|
|
);
|
|
}
|
|
}
|
|
|
|
ViewportTool.propTypes = {
|
|
channel: PropTypes.shape({
|
|
on: PropTypes.func,
|
|
emit: PropTypes.func,
|
|
removeListener: PropTypes.func,
|
|
}).isRequired,
|
|
api: PropTypes.shape({
|
|
onStory: PropTypes.func,
|
|
getQueryParam: PropTypes.func,
|
|
setQueryParams: PropTypes.func,
|
|
}).isRequired,
|
|
};
|