mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-04 19:01:05 +08:00
168 lines
4.3 KiB
JavaScript
168 lines
4.3 KiB
JavaScript
/* eslint-disable no-underscore-dangle */
|
|
|
|
import PropTypes from 'prop-types';
|
|
import React from 'react';
|
|
|
|
import { Table, Td, Th } from '@storybook/components';
|
|
import PropVal from './PropVal';
|
|
import PrettyPropType from './types/PrettyPropType';
|
|
|
|
const PropTypesMap = new Map();
|
|
|
|
Object.keys(PropTypes).forEach(typeName => {
|
|
const type = PropTypes[typeName];
|
|
|
|
PropTypesMap.set(type, typeName);
|
|
PropTypesMap.set(type.isRequired, typeName);
|
|
});
|
|
|
|
const isNotEmpty = obj => obj && obj.props && Object.keys(obj.props).length > 0;
|
|
|
|
const hasDocgen = type => isNotEmpty(type.__docgenInfo);
|
|
|
|
const propsFromDocgen = type => {
|
|
const props = {};
|
|
const docgenInfoProps = type.__docgenInfo.props;
|
|
|
|
Object.keys(docgenInfoProps).forEach(property => {
|
|
const docgenInfoProp = docgenInfoProps[property];
|
|
const defaultValueDesc = docgenInfoProp.defaultValue || {};
|
|
const propType = docgenInfoProp.flowType || docgenInfoProp.type || 'other';
|
|
|
|
props[property] = {
|
|
property,
|
|
propType,
|
|
required: docgenInfoProp.required,
|
|
description: docgenInfoProp.description,
|
|
defaultValue: defaultValueDesc.value,
|
|
};
|
|
});
|
|
|
|
return props;
|
|
};
|
|
|
|
const propsFromPropTypes = type => {
|
|
const props = {};
|
|
|
|
if (type.propTypes) {
|
|
Object.keys(type.propTypes).forEach(property => {
|
|
const typeInfo = type.propTypes[property];
|
|
const required = typeInfo.isRequired === undefined;
|
|
const docgenInfo =
|
|
type.__docgenInfo && type.__docgenInfo.props && type.__docgenInfo.props[property];
|
|
const description = docgenInfo ? docgenInfo.description : null;
|
|
let propType = PropTypesMap.get(typeInfo) || 'other';
|
|
|
|
if (propType === 'other') {
|
|
if (docgenInfo && docgenInfo.type) {
|
|
propType = docgenInfo.type.name;
|
|
}
|
|
}
|
|
|
|
props[property] = { property, propType, required, description };
|
|
});
|
|
}
|
|
|
|
if (type.defaultProps) {
|
|
Object.keys(type.defaultProps).forEach(property => {
|
|
const value = type.defaultProps[property];
|
|
|
|
if (value === undefined) {
|
|
return;
|
|
}
|
|
|
|
if (!props[property]) {
|
|
props[property] = { property };
|
|
}
|
|
|
|
props[property].defaultValue = value;
|
|
});
|
|
}
|
|
|
|
return props;
|
|
};
|
|
|
|
export const multiLineText = input => {
|
|
if (!input) return input;
|
|
const text = String(input);
|
|
const arrayOfText = text.split(/\r?\n|\r/g);
|
|
const isSingleLine = arrayOfText.length < 2;
|
|
return isSingleLine
|
|
? text
|
|
: arrayOfText.map((
|
|
lineOfText,
|
|
i // note: lineOfText is the closest we will get to a unique key
|
|
) => (
|
|
<span key={lineOfText}>
|
|
{i > 0 && <br />} {lineOfText}
|
|
</span>
|
|
));
|
|
};
|
|
|
|
export default function PropTable(props) {
|
|
const { type, maxPropObjectKeys, maxPropArrayLength, maxPropStringLength } = props;
|
|
|
|
if (!type) {
|
|
return null;
|
|
}
|
|
|
|
const accumProps = hasDocgen(type) ? propsFromDocgen(type) : propsFromPropTypes(type);
|
|
const array = Object.values(accumProps);
|
|
|
|
if (!array.length) {
|
|
return <small>No propTypes defined!</small>;
|
|
}
|
|
|
|
const propValProps = {
|
|
maxPropObjectKeys,
|
|
maxPropArrayLength,
|
|
maxPropStringLength,
|
|
};
|
|
|
|
return (
|
|
<Table>
|
|
<thead>
|
|
<tr>
|
|
<Th bordered>property</Th>
|
|
<Th bordered>propType</Th>
|
|
<Th bordered>required</Th>
|
|
<Th bordered>default</Th>
|
|
<Th bordered>description</Th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{array.map(row => (
|
|
<tr key={row.property}>
|
|
<Td bordered code>
|
|
{row.property}
|
|
</Td>
|
|
<Td bordered code>
|
|
<PrettyPropType propType={row.propType} />
|
|
</Td>
|
|
<Td bordered>{row.required ? 'yes' : '-'}</Td>
|
|
<Td bordered>
|
|
{row.defaultValue === undefined ? (
|
|
'-'
|
|
) : (
|
|
<PropVal val={row.defaultValue} {...propValProps} />
|
|
)}
|
|
</Td>
|
|
<Td bordered>{multiLineText(row.description)}</Td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</Table>
|
|
);
|
|
}
|
|
|
|
PropTable.displayName = 'PropTable';
|
|
PropTable.defaultProps = {
|
|
type: null,
|
|
};
|
|
PropTable.propTypes = {
|
|
type: PropTypes.func,
|
|
maxPropObjectKeys: PropTypes.number.isRequired,
|
|
maxPropArrayLength: PropTypes.number.isRequired,
|
|
maxPropStringLength: PropTypes.number.isRequired,
|
|
};
|