mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 13:21:14 +08:00
408 lines
9.7 KiB
JavaScript
408 lines
9.7 KiB
JavaScript
/* eslint no-underscore-dangle: 0 */
|
||
|
||
import React, { createElement } from 'react';
|
||
import PropTypes from 'prop-types';
|
||
import global from 'global';
|
||
import { baseFonts } from '@storybook/components';
|
||
import { ThemeProvider } from 'glamorous';
|
||
|
||
import marksy from 'marksy';
|
||
|
||
import Node from './Node';
|
||
import { Pre } from './markdown';
|
||
|
||
global.STORYBOOK_REACT_CLASSES = global.STORYBOOK_REACT_CLASSES || [];
|
||
const { STORYBOOK_REACT_CLASSES } = global;
|
||
|
||
const getName = type => type.displayName || type.name;
|
||
|
||
const stylesheet = {
|
||
button: {
|
||
base: {
|
||
fontFamily: 'sans-serif',
|
||
fontSize: '12px',
|
||
display: 'block',
|
||
position: 'fixed',
|
||
border: 'none',
|
||
background: '#28c',
|
||
color: '#fff',
|
||
padding: '5px 15px',
|
||
cursor: 'pointer',
|
||
},
|
||
topRight: {
|
||
top: 0,
|
||
right: 0,
|
||
borderRadius: '0 0 0 5px',
|
||
},
|
||
},
|
||
info: {
|
||
position: 'fixed',
|
||
background: 'white',
|
||
top: 0,
|
||
bottom: 0,
|
||
left: 0,
|
||
right: 0,
|
||
padding: '0 40px',
|
||
overflow: 'auto',
|
||
zIndex: 99999,
|
||
},
|
||
children: {
|
||
position: 'relative',
|
||
zIndex: 0,
|
||
},
|
||
infoBody: {
|
||
...baseFonts,
|
||
fontWeight: 300,
|
||
lineHeight: 1.45,
|
||
fontSize: '15px',
|
||
border: '1px solid #eee',
|
||
padding: '20px 40px 40px',
|
||
borderRadius: '2px',
|
||
boxShadow: '0px 2px 3px rgba(0, 0, 0, 0.05)',
|
||
backgroundColor: '#fff',
|
||
marginTop: '20px',
|
||
marginBottom: '20px',
|
||
},
|
||
infoContent: {
|
||
marginBottom: 0,
|
||
},
|
||
infoStory: {},
|
||
jsxInfoContent: {
|
||
borderTop: '1px solid #eee',
|
||
margin: '20px 0 0 0',
|
||
},
|
||
header: {
|
||
h1: {
|
||
margin: 0,
|
||
padding: 0,
|
||
fontSize: '35px',
|
||
},
|
||
h2: {
|
||
margin: '0 0 10px 0',
|
||
padding: 0,
|
||
fontWeight: 400,
|
||
fontSize: '22px',
|
||
},
|
||
body: {
|
||
borderBottom: '1px solid #eee',
|
||
paddingTop: 10,
|
||
marginBottom: 10,
|
||
},
|
||
},
|
||
source: {
|
||
h1: {
|
||
margin: '20px 0 0 0',
|
||
padding: '0 0 5px 0',
|
||
fontSize: '25px',
|
||
borderBottom: '1px solid #EEE',
|
||
},
|
||
},
|
||
propTableHead: {
|
||
margin: '20px 0 0 0',
|
||
},
|
||
};
|
||
|
||
export default class Story extends React.Component {
|
||
constructor(...args) {
|
||
super(...args);
|
||
this.state = {
|
||
open: false,
|
||
stylesheet: this.props.styles(stylesheet),
|
||
};
|
||
this.marksy = marksy({
|
||
createElement,
|
||
elements: this.props.components,
|
||
});
|
||
}
|
||
|
||
componentWillReceiveProps(nextProps) {
|
||
this.setState({
|
||
stylesheet: nextProps.styles(stylesheet),
|
||
});
|
||
}
|
||
|
||
_renderStory() {
|
||
return <div style={this.state.stylesheet.infoStory}>{this.props.children}</div>;
|
||
}
|
||
|
||
_renderInline() {
|
||
return (
|
||
<div>
|
||
{this._renderInlineHeader()}
|
||
{this._renderStory()}
|
||
<div style={this.state.stylesheet.infoPage}>
|
||
<div style={this.state.stylesheet.infoBody}>
|
||
{this._getInfoContent()}
|
||
{this._getComponentDescription()}
|
||
{this._getSourceCode()}
|
||
{this._getPropTables()}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
);
|
||
}
|
||
|
||
_renderInlineHeader() {
|
||
const infoHeader = this._getInfoHeader();
|
||
|
||
return (
|
||
infoHeader && (
|
||
<div style={this.state.stylesheet.infoPage}>
|
||
<div style={this.state.stylesheet.infoBody}>{infoHeader}</div>
|
||
</div>
|
||
)
|
||
);
|
||
}
|
||
|
||
_renderOverlay() {
|
||
const buttonStyle = {
|
||
...this.state.stylesheet.button.base,
|
||
...this.state.stylesheet.button.topRight,
|
||
};
|
||
|
||
const infoStyle = Object.assign({}, this.state.stylesheet.info);
|
||
if (!this.state.open) {
|
||
infoStyle.display = 'none';
|
||
}
|
||
|
||
const openOverlay = () => {
|
||
this.setState({ open: true });
|
||
return false;
|
||
};
|
||
|
||
const closeOverlay = () => {
|
||
this.setState({ open: false });
|
||
return false;
|
||
};
|
||
|
||
return (
|
||
<div>
|
||
<div style={this.state.stylesheet.children}>{this.props.children}</div>
|
||
<button type="button" style={buttonStyle} onClick={openOverlay}>
|
||
Show Info
|
||
</button>
|
||
<div style={infoStyle}>
|
||
<button type="button" style={buttonStyle} onClick={closeOverlay}>
|
||
×
|
||
</button>
|
||
<div style={this.state.stylesheet.infoPage}>
|
||
<div style={this.state.stylesheet.infoBody}>
|
||
{this._getInfoHeader()}
|
||
{this._getInfoContent()}
|
||
{this._getComponentDescription()}
|
||
{this._getSourceCode()}
|
||
{this._getPropTables()}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
);
|
||
}
|
||
|
||
_getInfoHeader() {
|
||
if (!this.props.context || !this.props.showHeader) {
|
||
return null;
|
||
}
|
||
|
||
return (
|
||
<div style={this.state.stylesheet.header.body}>
|
||
<h1 style={this.state.stylesheet.header.h1}>{this.props.context.kind}</h1>
|
||
<h2 style={this.state.stylesheet.header.h2}>{this.props.context.story}</h2>
|
||
</div>
|
||
);
|
||
}
|
||
|
||
_getInfoContent() {
|
||
if (!this.props.info) {
|
||
return '';
|
||
}
|
||
|
||
if (React.isValidElement(this.props.info)) {
|
||
return (
|
||
<div
|
||
style={
|
||
this.props.showInline
|
||
? this.state.stylesheet.jsxInfoContent
|
||
: this.state.stylesheet.infoContent
|
||
}
|
||
>
|
||
{this.props.info}
|
||
</div>
|
||
);
|
||
}
|
||
|
||
const lines = this.props.info.split('\n');
|
||
while (lines[0].trim() === '') {
|
||
lines.shift();
|
||
}
|
||
let padding = 0;
|
||
const matches = lines[0].match(/^ */);
|
||
if (matches) {
|
||
padding = matches[0].length;
|
||
}
|
||
const source = lines.map(s => s.slice(padding)).join('\n');
|
||
return <div style={this.state.stylesheet.infoContent}>{this.marksy(source).tree}</div>;
|
||
}
|
||
|
||
_getComponentDescription() {
|
||
let retDiv = null;
|
||
|
||
if (Object.keys(STORYBOOK_REACT_CLASSES).length) {
|
||
Object.keys(STORYBOOK_REACT_CLASSES).forEach(key => {
|
||
if (STORYBOOK_REACT_CLASSES[key].name === this.props.context.story) {
|
||
retDiv = <div>{STORYBOOK_REACT_CLASSES[key].docgenInfo.description}</div>;
|
||
}
|
||
});
|
||
}
|
||
|
||
return retDiv;
|
||
}
|
||
|
||
_getSourceCode() {
|
||
if (!this.props.showSource) {
|
||
return null;
|
||
}
|
||
|
||
const {
|
||
maxPropsIntoLine,
|
||
maxPropObjectKeys,
|
||
maxPropArrayLength,
|
||
maxPropStringLength,
|
||
} = this.props;
|
||
|
||
return (
|
||
<div>
|
||
<h1 style={this.state.stylesheet.source.h1}>Story Source</h1>
|
||
<Pre>
|
||
{React.Children.map(this.props.children, (root, idx) => (
|
||
<Node
|
||
key={idx}
|
||
node={root}
|
||
depth={0}
|
||
maxPropsIntoLine={maxPropsIntoLine}
|
||
maxPropObjectKeys={maxPropObjectKeys}
|
||
maxPropArrayLength={maxPropArrayLength}
|
||
maxPropStringLength={maxPropStringLength}
|
||
/>
|
||
))}
|
||
</Pre>
|
||
</div>
|
||
);
|
||
}
|
||
|
||
_getPropTables() {
|
||
const types = new Map();
|
||
|
||
if (this.props.propTables === null) {
|
||
return null;
|
||
}
|
||
|
||
if (!this.props.children) {
|
||
return null;
|
||
}
|
||
|
||
if (this.props.propTables) {
|
||
this.props.propTables.forEach(type => {
|
||
types.set(type, true);
|
||
});
|
||
}
|
||
|
||
// depth-first traverse and collect types
|
||
const extract = children => {
|
||
if (!children) {
|
||
return;
|
||
}
|
||
if (Array.isArray(children)) {
|
||
children.forEach(extract);
|
||
return;
|
||
}
|
||
if (children.props && children.props.children) {
|
||
extract(children.props.children);
|
||
}
|
||
if (
|
||
typeof children === 'string' ||
|
||
typeof children.type === 'string' ||
|
||
(Array.isArray(this.props.propTablesExclude) && // also ignore excluded types
|
||
~this.props.propTablesExclude.indexOf(children.type)) // eslint-disable-line no-bitwise
|
||
) {
|
||
return;
|
||
}
|
||
if (children.type && !types.has(children.type)) {
|
||
types.set(children.type, true);
|
||
}
|
||
};
|
||
|
||
// extract components from children
|
||
extract(this.props.children);
|
||
|
||
const array = Array.from(types.keys());
|
||
array.sort((a, b) => getName(a) > getName(b));
|
||
|
||
const { maxPropObjectKeys, maxPropArrayLength, maxPropStringLength } = this.props;
|
||
const propTables = array.map((type, i) => (
|
||
// eslint-disable-next-line react/no-array-index-key
|
||
<div key={`${getName(type)}_${i}`}>
|
||
<h2 style={this.state.stylesheet.propTableHead}>"{getName(type)}" Component</h2>
|
||
<this.props.PropTable
|
||
type={type}
|
||
maxPropObjectKeys={maxPropObjectKeys}
|
||
maxPropArrayLength={maxPropArrayLength}
|
||
maxPropStringLength={maxPropStringLength}
|
||
/>
|
||
</div>
|
||
));
|
||
|
||
if (!propTables || propTables.length === 0) {
|
||
return null;
|
||
}
|
||
|
||
return (
|
||
<div>
|
||
<h1 style={this.state.stylesheet.source.h1}>Prop Types</h1>
|
||
{propTables}
|
||
</div>
|
||
);
|
||
}
|
||
|
||
render() {
|
||
return (
|
||
<ThemeProvider theme={this.state.stylesheet}>
|
||
{this.props.showInline ? this._renderInline() : this._renderOverlay()}
|
||
</ThemeProvider>
|
||
);
|
||
}
|
||
}
|
||
|
||
Story.displayName = 'Story';
|
||
|
||
Story.propTypes = {
|
||
context: PropTypes.shape({
|
||
kind: PropTypes.string,
|
||
story: PropTypes.string,
|
||
}),
|
||
info: PropTypes.oneOfType([PropTypes.string, PropTypes.node]),
|
||
propTables: PropTypes.arrayOf(PropTypes.func),
|
||
propTablesExclude: PropTypes.arrayOf(PropTypes.func),
|
||
showInline: PropTypes.bool,
|
||
showHeader: PropTypes.bool,
|
||
showSource: PropTypes.bool,
|
||
styles: PropTypes.func.isRequired,
|
||
children: PropTypes.oneOfType([PropTypes.object, PropTypes.array]),
|
||
components: PropTypes.shape({}),
|
||
maxPropsIntoLine: PropTypes.number.isRequired,
|
||
maxPropObjectKeys: PropTypes.number.isRequired,
|
||
maxPropArrayLength: PropTypes.number.isRequired,
|
||
maxPropStringLength: PropTypes.number.isRequired,
|
||
};
|
||
Story.defaultProps = {
|
||
context: null,
|
||
info: '',
|
||
children: null,
|
||
propTables: null,
|
||
propTablesExclude: [],
|
||
showInline: false,
|
||
showHeader: true,
|
||
showSource: true,
|
||
components: {},
|
||
};
|