mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-26 12:49:49 +00:00
d7343da279
Fix an issue where boolean props with a default value of `true` are not rendered when their value is `false`
264 lines
7.2 KiB
JavaScript
264 lines
7.2 KiB
JavaScript
import React, {isValidElement as isElement} from 'react';
|
|
import collapse from 'collapse-white-space';
|
|
import isPlainObject from 'is-plain-object';
|
|
import stringify from 'stringify-object';
|
|
import sortobject from 'sortobject';
|
|
import traverse from 'traverse';
|
|
import {fill} from 'lodash';
|
|
|
|
export default function reactElementToJSXString(
|
|
ReactElement, {
|
|
displayName,
|
|
filterProps = [],
|
|
showDefaultProps = true,
|
|
showFunctions = false,
|
|
tabStop = 2,
|
|
useBooleanShorthandSyntax = true,
|
|
} = {}
|
|
) {
|
|
const getDisplayName = displayName || getDefaultDisplayName;
|
|
|
|
return toJSXString({ReactElement});
|
|
|
|
function toJSXString({ReactElement: Element = null, lvl = 0, inline = false}) {
|
|
if (typeof Element === 'string' || typeof Element === 'number') {
|
|
return Element;
|
|
} else if (!isElement(Element)) {
|
|
throw new Error(
|
|
`react-element-to-jsx-string: Expected a ReactElement,
|
|
got \`${typeof Element}\``
|
|
);
|
|
}
|
|
|
|
const tagName = getDisplayName(Element);
|
|
|
|
let out = `<${tagName}`;
|
|
const props = formatProps(Element.props, getDefaultProps(Element), inline, lvl);
|
|
let attributes = [];
|
|
const children = React.Children.toArray(Element.props.children)
|
|
.filter(onlyMeaningfulChildren);
|
|
|
|
if (Element.ref !== null) {
|
|
attributes.push(getJSXAttribute('ref', Element.ref, inline, lvl));
|
|
}
|
|
|
|
if (Element.key !== null &&
|
|
// React automatically add key=".X" when there are some children
|
|
!(/^\./).test(Element.key)) {
|
|
attributes.push(getJSXAttribute('key', Element.key, inline, lvl));
|
|
}
|
|
|
|
attributes = attributes
|
|
.concat(props)
|
|
.filter(({name}) => filterProps.indexOf(name) === -1);
|
|
|
|
let containsMultilineAttr = false;
|
|
attributes.forEach(attribute => {
|
|
let isMultilineAttr = false;
|
|
if (['plainObject', 'array', 'function'].indexOf(attribute.type) > -1) {
|
|
isMultilineAttr = attribute.value.indexOf('\n') > -1;
|
|
}
|
|
|
|
if (isMultilineAttr) {
|
|
containsMultilineAttr = true;
|
|
}
|
|
|
|
if ((attributes.length === 1 || inline) && !isMultilineAttr) {
|
|
out += ' ';
|
|
} else {
|
|
out += `\n${spacer(lvl + 1, tabStop)}`;
|
|
}
|
|
|
|
if (useBooleanShorthandSyntax && attribute.value === '{true}') {
|
|
out += `${attribute.name}`;
|
|
} else {
|
|
out += `${attribute.name}=${attribute.value}`;
|
|
}
|
|
});
|
|
|
|
if ((attributes.length > 1 || containsMultilineAttr) && !inline) {
|
|
out += `\n${spacer(lvl, tabStop)}`;
|
|
}
|
|
|
|
if (children.length > 0) {
|
|
out += '>';
|
|
lvl++;
|
|
if (!inline) {
|
|
out += '\n';
|
|
out += spacer(lvl, tabStop);
|
|
}
|
|
|
|
if (typeof children === 'string') {
|
|
out += children;
|
|
} else {
|
|
out += children
|
|
.reduce(mergePlainStringChildren, [])
|
|
.map(
|
|
recurse({lvl, inline})
|
|
).join(`\n${spacer(lvl, tabStop)}`);
|
|
}
|
|
if (!inline) {
|
|
out += '\n';
|
|
out += spacer(lvl - 1, tabStop);
|
|
}
|
|
out += `</${tagName}>`;
|
|
} else {
|
|
if (attributes.length <= 1) {
|
|
out += ' ';
|
|
}
|
|
|
|
out += '/>';
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
function formatProps(props, defaultProps, inline, lvl) {
|
|
let formatted = Object
|
|
.keys(props)
|
|
.filter(noChildren);
|
|
|
|
if (useBooleanShorthandSyntax) {
|
|
formatted = formatted.filter(key => noFalse(props[key], defaultProps[key]));
|
|
}
|
|
|
|
if (!showDefaultProps) {
|
|
formatted = formatted.filter(key => defaultProps[key] ? defaultProps[key] !== props[key] : true);
|
|
}
|
|
|
|
return formatted
|
|
.sort()
|
|
.map(propName => getJSXAttribute(propName, props[propName], inline, lvl));
|
|
}
|
|
|
|
function getJSXAttribute(name, value, inline, lvl) {
|
|
return {
|
|
name,
|
|
type: getValueType(value),
|
|
value: formatJSXAttribute(value, inline, lvl)
|
|
.replace(/'?<__reactElementToJSXString__Wrapper__>/g, '')
|
|
.replace(/<\/__reactElementToJSXString__Wrapper__>'?/g, ''),
|
|
};
|
|
}
|
|
|
|
function formatJSXAttribute(propValue, inline, lvl) {
|
|
if (typeof propValue === 'string') {
|
|
return `"${propValue}"`;
|
|
}
|
|
|
|
return `{${formatValue(propValue, inline, lvl)}}`;
|
|
}
|
|
|
|
function getValueType(value) {
|
|
if (isElement(value)) {
|
|
return 'element';
|
|
}
|
|
|
|
if (isPlainObject(value)) {
|
|
return 'plainObject';
|
|
}
|
|
|
|
if (Array.isArray(value)) {
|
|
return 'array';
|
|
}
|
|
|
|
return typeof value;
|
|
}
|
|
|
|
function formatValue(value, inline, lvl) {
|
|
const wrapper = '__reactElementToJSXString__Wrapper__';
|
|
|
|
if (typeof value === 'function' && !showFunctions) {
|
|
return function noRefCheck() {};
|
|
} else if (isElement(value)) {
|
|
// we use this delimiter hack in cases where the react element is a property
|
|
// of an object from a root prop
|
|
// i.e.
|
|
// reactElementToJSXString(<div a={{b: <div />}} />
|
|
// // <div a={{b: <div />}} />
|
|
// we then remove the whole wrapping
|
|
// otherwise, the element would be surrounded by quotes: <div a={{b: '<div />'}} />
|
|
return `<${wrapper}>${toJSXString({ReactElement: value, inline: true})}</${wrapper}>`;
|
|
} else if (isPlainObject(value) || Array.isArray(value)) {
|
|
return `<${wrapper}>${stringifyObject(value, inline, lvl)}</${wrapper}>`;
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
function recurse({lvl, inline}) {
|
|
return Element => toJSXString({ReactElement: Element, lvl, inline});
|
|
}
|
|
|
|
function stringifyObject(obj, inline, lvl) {
|
|
if (Object.keys(obj).length > 0 || obj.length > 0) {
|
|
// eslint-disable-next-line array-callback-return
|
|
obj = traverse(obj).map(function(value) {
|
|
if (isElement(value) || this.isLeaf) {
|
|
this.update(formatValue(value, inline, lvl));
|
|
}
|
|
});
|
|
|
|
obj = sortobject(obj);
|
|
}
|
|
|
|
const stringified = stringify(obj);
|
|
|
|
if (inline) {
|
|
return collapse(stringified)
|
|
.replace(/{ /g, '{')
|
|
.replace(/ }/g, '}')
|
|
.replace(/\[ /g, '[')
|
|
.replace(/ ]/g, ']');
|
|
}
|
|
|
|
// Replace tabs with spaces, and add necessary indentation in front of each new line
|
|
return stringified
|
|
.replace(/\t/g, spacer(1, tabStop))
|
|
.replace(/\n([^$])/g, `\n${spacer(lvl + 1, tabStop)}$1`);
|
|
}
|
|
}
|
|
|
|
function getDefaultDisplayName(ReactElement) {
|
|
return ReactElement.type.displayName ||
|
|
ReactElement.type.name || // function name
|
|
(typeof ReactElement.type === 'function' ? // function without a name, you should provide one
|
|
'No Display Name' :
|
|
ReactElement.type);
|
|
}
|
|
|
|
function getDefaultProps(ReactElement) {
|
|
return ReactElement.type.defaultProps || {};
|
|
}
|
|
|
|
function mergePlainStringChildren(prev, cur) {
|
|
const lastItem = prev[prev.length - 1];
|
|
|
|
if (typeof cur === 'number') {
|
|
cur = String(cur);
|
|
}
|
|
|
|
if (typeof lastItem === 'string' && typeof cur === 'string') {
|
|
prev[prev.length - 1] += cur;
|
|
} else {
|
|
prev.push(cur);
|
|
}
|
|
|
|
return prev;
|
|
}
|
|
|
|
function spacer(times, tabStop) {
|
|
return times === 0 ? '' : fill(new Array(times * tabStop), ' ').join('');
|
|
}
|
|
|
|
function noChildren(propName) {
|
|
return propName !== 'children';
|
|
}
|
|
|
|
function noFalse(propValue, defaultValue) {
|
|
return typeof propValue !== 'boolean' || propValue || Boolean(defaultValue);
|
|
}
|
|
|
|
function onlyMeaningfulChildren(children) {
|
|
return children !== true && children !== false && children !== null && children !== '';
|
|
}
|