/* 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
{this.props.children}
; } _renderInline() { return (
{this._renderInlineHeader()} {this._renderStory()}
{this._getInfoContent()} {this._getComponentDescription()} {this._getSourceCode()} {this._getPropTables()}
); } _renderInlineHeader() { const infoHeader = this._getInfoHeader(); return ( infoHeader && (
{infoHeader}
) ); } _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 (
{this.props.children}
{this._getInfoHeader()} {this._getInfoContent()} {this._getComponentDescription()} {this._getSourceCode()} {this._getPropTables()}
); } _getInfoHeader() { if (!this.props.context || !this.props.showHeader) { return null; } return (

{this.props.context.kind}

{this.props.context.story}

); } _getInfoContent() { if (!this.props.info) { return ''; } if (React.isValidElement(this.props.info)) { return (
{this.props.info}
); } 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
{this.marksy(source).tree}
; } _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 =
{STORYBOOK_REACT_CLASSES[key].docgenInfo.description}
; } }); } return retDiv; } _getSourceCode() { if (!this.props.showSource) { return null; } const { maxPropsIntoLine, maxPropObjectKeys, maxPropArrayLength, maxPropStringLength, } = this.props; return (

Story Source

          {React.Children.map(this.props.children, (root, idx) => (
            
          ))}
        
); } _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

"{getName(type)}" Component

)); if (!propTables || propTables.length === 0) { return null; } return (

Prop Types

{propTables}
); } render() { return ( {this.props.showInline ? this._renderInline() : this._renderOverlay()} ); } } 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: {}, };