278 lines
10 KiB
JavaScript
278 lines
10 KiB
JavaScript
"use strict";
|
|
|
|
Object.defineProperty(exports, "__esModule", {
|
|
value: true
|
|
});
|
|
exports.getWordsByLines = exports.Text = void 0;
|
|
exports.isValidTextAnchor = isValidTextAnchor;
|
|
var _react = _interopRequireWildcard(require("react"));
|
|
var React = _react;
|
|
var _clsx = require("clsx");
|
|
var _DataUtils = require("../util/DataUtils");
|
|
var _Global = require("../util/Global");
|
|
var _DOMUtils = require("../util/DOMUtils");
|
|
var _ReduceCSSCalc = require("../util/ReduceCSSCalc");
|
|
var _svgPropertiesAndEvents = require("../util/svgPropertiesAndEvents");
|
|
var _resolveDefaultProps2 = require("../util/resolveDefaultProps");
|
|
var _isWellBehavedNumber = require("../util/isWellBehavedNumber");
|
|
var _excluded = ["x", "y", "lineHeight", "capHeight", "fill", "scaleToFit", "textAnchor", "verticalAnchor"],
|
|
_excluded2 = ["dx", "dy", "angle", "className", "breakAll"];
|
|
function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function _interopRequireWildcard(e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != typeof e && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (var _t in e) "default" !== _t && {}.hasOwnProperty.call(e, _t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, _t)) && (i.get || i.set) ? o(f, _t, i) : f[_t] = e[_t]); return f; })(e, t); }
|
|
function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); }
|
|
function _objectWithoutProperties(e, t) { if (null == e) return {}; var o, r, i = _objectWithoutPropertiesLoose(e, t); if (Object.getOwnPropertySymbols) { var n = Object.getOwnPropertySymbols(e); for (r = 0; r < n.length; r++) o = n[r], -1 === t.indexOf(o) && {}.propertyIsEnumerable.call(e, o) && (i[o] = e[o]); } return i; }
|
|
function _objectWithoutPropertiesLoose(r, e) { if (null == r) return {}; var t = {}; for (var n in r) if ({}.hasOwnProperty.call(r, n)) { if (-1 !== e.indexOf(n)) continue; t[n] = r[n]; } return t; }
|
|
var BREAKING_SPACES = /[ \f\n\r\t\v\u2028\u2029]+/;
|
|
var calculateWordWidths = _ref => {
|
|
var {
|
|
children,
|
|
breakAll,
|
|
style
|
|
} = _ref;
|
|
try {
|
|
var words = [];
|
|
if (!(0, _DataUtils.isNullish)(children)) {
|
|
if (breakAll) {
|
|
words = children.toString().split('');
|
|
} else {
|
|
words = children.toString().split(BREAKING_SPACES);
|
|
}
|
|
}
|
|
var wordsWithComputedWidth = words.map(word => ({
|
|
word,
|
|
width: (0, _DOMUtils.getStringSize)(word, style).width
|
|
}));
|
|
var spaceWidth = breakAll ? 0 : (0, _DOMUtils.getStringSize)('\u00A0', style).width;
|
|
return {
|
|
wordsWithComputedWidth,
|
|
spaceWidth
|
|
};
|
|
} catch (_unused) {
|
|
return null;
|
|
}
|
|
};
|
|
function isValidTextAnchor(value) {
|
|
return value === 'start' || value === 'middle' || value === 'end' || value === 'inherit';
|
|
}
|
|
var calculate = (words, lineWidth, spaceWidth, scaleToFit) => words.reduce((result, _ref2) => {
|
|
var {
|
|
word,
|
|
width
|
|
} = _ref2;
|
|
var currentLine = result[result.length - 1];
|
|
if (currentLine && width != null && (lineWidth == null || scaleToFit || currentLine.width + width + spaceWidth < Number(lineWidth))) {
|
|
// Word can be added to an existing line
|
|
currentLine.words.push(word);
|
|
currentLine.width += width + spaceWidth;
|
|
} else {
|
|
// Add first word to line or word is too long to scaleToFit on existing line
|
|
var newLine = {
|
|
words: [word],
|
|
width
|
|
};
|
|
result.push(newLine);
|
|
}
|
|
return result;
|
|
}, []);
|
|
var findLongestLine = words => words.reduce((a, b) => a.width > b.width ? a : b);
|
|
var suffix = '…';
|
|
var checkOverflow = (text, index, breakAll, style, maxLines, lineWidth, spaceWidth, scaleToFit) => {
|
|
var tempText = text.slice(0, index);
|
|
var words = calculateWordWidths({
|
|
breakAll,
|
|
style,
|
|
children: tempText + suffix
|
|
});
|
|
if (!words) {
|
|
return [false, []];
|
|
}
|
|
var result = calculate(words.wordsWithComputedWidth, lineWidth, spaceWidth, scaleToFit);
|
|
var doesOverflow = result.length > maxLines || findLongestLine(result).width > Number(lineWidth);
|
|
return [doesOverflow, result];
|
|
};
|
|
var calculateWordsByLines = (_ref3, initialWordsWithComputedWith, spaceWidth, lineWidth, scaleToFit) => {
|
|
var {
|
|
maxLines,
|
|
children,
|
|
style,
|
|
breakAll
|
|
} = _ref3;
|
|
var shouldLimitLines = (0, _DataUtils.isNumber)(maxLines);
|
|
var text = String(children);
|
|
var originalResult = calculate(initialWordsWithComputedWith, lineWidth, spaceWidth, scaleToFit);
|
|
if (!shouldLimitLines || scaleToFit) {
|
|
return originalResult;
|
|
}
|
|
var overflows = originalResult.length > maxLines || findLongestLine(originalResult).width > Number(lineWidth);
|
|
if (!overflows) {
|
|
return originalResult;
|
|
}
|
|
var start = 0;
|
|
var end = text.length - 1;
|
|
var iterations = 0;
|
|
var trimmedResult;
|
|
while (start <= end && iterations <= text.length - 1) {
|
|
var middle = Math.floor((start + end) / 2);
|
|
var prev = middle - 1;
|
|
var [doesPrevOverflow, result] = checkOverflow(text, prev, breakAll, style, maxLines, lineWidth, spaceWidth, scaleToFit);
|
|
var [doesMiddleOverflow] = checkOverflow(text, middle, breakAll, style, maxLines, lineWidth, spaceWidth, scaleToFit);
|
|
if (!doesPrevOverflow && !doesMiddleOverflow) {
|
|
start = middle + 1;
|
|
}
|
|
if (doesPrevOverflow && doesMiddleOverflow) {
|
|
end = middle - 1;
|
|
}
|
|
if (!doesPrevOverflow && doesMiddleOverflow) {
|
|
trimmedResult = result;
|
|
break;
|
|
}
|
|
iterations++;
|
|
}
|
|
|
|
// Fallback to originalResult (result without trimming) if we cannot find the
|
|
// where to trim. This should not happen :tm:
|
|
return trimmedResult || originalResult;
|
|
};
|
|
var getWordsWithoutCalculate = children => {
|
|
var words = !(0, _DataUtils.isNullish)(children) ? children.toString().split(BREAKING_SPACES) : [];
|
|
return [{
|
|
words,
|
|
width: undefined
|
|
}];
|
|
};
|
|
var getWordsByLines = _ref4 => {
|
|
var {
|
|
width,
|
|
scaleToFit,
|
|
children,
|
|
style,
|
|
breakAll,
|
|
maxLines
|
|
} = _ref4;
|
|
// Only perform calculations if using features that require them (multiline, scaleToFit)
|
|
if ((width || scaleToFit) && !_Global.Global.isSsr) {
|
|
var wordsWithComputedWidth, spaceWidth;
|
|
var wordWidths = calculateWordWidths({
|
|
breakAll,
|
|
children,
|
|
style
|
|
});
|
|
if (wordWidths) {
|
|
var {
|
|
wordsWithComputedWidth: wcw,
|
|
spaceWidth: sw
|
|
} = wordWidths;
|
|
wordsWithComputedWidth = wcw;
|
|
spaceWidth = sw;
|
|
} else {
|
|
return getWordsWithoutCalculate(children);
|
|
}
|
|
return calculateWordsByLines({
|
|
breakAll,
|
|
children,
|
|
maxLines,
|
|
style
|
|
}, wordsWithComputedWidth, spaceWidth, width, Boolean(scaleToFit));
|
|
}
|
|
return getWordsWithoutCalculate(children);
|
|
};
|
|
exports.getWordsByLines = getWordsByLines;
|
|
var DEFAULT_FILL = '#808080';
|
|
var textDefaultProps = {
|
|
breakAll: false,
|
|
// Magic number from d3
|
|
capHeight: '0.71em',
|
|
fill: DEFAULT_FILL,
|
|
lineHeight: '1em',
|
|
scaleToFit: false,
|
|
textAnchor: 'start',
|
|
// Maintain compat with existing charts / default SVG behavior
|
|
verticalAnchor: 'end',
|
|
x: 0,
|
|
y: 0
|
|
};
|
|
var Text = exports.Text = /*#__PURE__*/(0, _react.forwardRef)((outsideProps, ref) => {
|
|
var _resolveDefaultProps = (0, _resolveDefaultProps2.resolveDefaultProps)(outsideProps, textDefaultProps),
|
|
{
|
|
x: propsX,
|
|
y: propsY,
|
|
lineHeight,
|
|
capHeight,
|
|
fill,
|
|
scaleToFit,
|
|
textAnchor,
|
|
verticalAnchor
|
|
} = _resolveDefaultProps,
|
|
props = _objectWithoutProperties(_resolveDefaultProps, _excluded);
|
|
var wordsByLines = (0, _react.useMemo)(() => {
|
|
return getWordsByLines({
|
|
breakAll: props.breakAll,
|
|
children: props.children,
|
|
maxLines: props.maxLines,
|
|
scaleToFit,
|
|
style: props.style,
|
|
width: props.width
|
|
});
|
|
}, [props.breakAll, props.children, props.maxLines, scaleToFit, props.style, props.width]);
|
|
var {
|
|
dx,
|
|
dy,
|
|
angle,
|
|
className,
|
|
breakAll
|
|
} = props,
|
|
textProps = _objectWithoutProperties(props, _excluded2);
|
|
if (!(0, _DataUtils.isNumOrStr)(propsX) || !(0, _DataUtils.isNumOrStr)(propsY) || wordsByLines.length === 0) {
|
|
return null;
|
|
}
|
|
var x = Number(propsX) + ((0, _DataUtils.isNumber)(dx) ? dx : 0);
|
|
var y = Number(propsY) + ((0, _DataUtils.isNumber)(dy) ? dy : 0);
|
|
if (!(0, _isWellBehavedNumber.isWellBehavedNumber)(x) || !(0, _isWellBehavedNumber.isWellBehavedNumber)(y)) {
|
|
return null;
|
|
}
|
|
var startDy;
|
|
switch (verticalAnchor) {
|
|
case 'start':
|
|
startDy = (0, _ReduceCSSCalc.reduceCSSCalc)("calc(".concat(capHeight, ")"));
|
|
break;
|
|
case 'middle':
|
|
startDy = (0, _ReduceCSSCalc.reduceCSSCalc)("calc(".concat((wordsByLines.length - 1) / 2, " * -").concat(lineHeight, " + (").concat(capHeight, " / 2))"));
|
|
break;
|
|
default:
|
|
startDy = (0, _ReduceCSSCalc.reduceCSSCalc)("calc(".concat(wordsByLines.length - 1, " * -").concat(lineHeight, ")"));
|
|
break;
|
|
}
|
|
var transforms = [];
|
|
if (scaleToFit) {
|
|
var lineWidth = wordsByLines[0].width;
|
|
var {
|
|
width
|
|
} = props;
|
|
transforms.push("scale(".concat((0, _DataUtils.isNumber)(width) && (0, _DataUtils.isNumber)(lineWidth) ? width / lineWidth : 1, ")"));
|
|
}
|
|
if (angle) {
|
|
transforms.push("rotate(".concat(angle, ", ").concat(x, ", ").concat(y, ")"));
|
|
}
|
|
if (transforms.length) {
|
|
textProps.transform = transforms.join(' ');
|
|
}
|
|
return /*#__PURE__*/React.createElement("text", _extends({}, (0, _svgPropertiesAndEvents.svgPropertiesAndEvents)(textProps), {
|
|
ref: ref,
|
|
x: x,
|
|
y: y,
|
|
className: (0, _clsx.clsx)('recharts-text', className),
|
|
textAnchor: textAnchor,
|
|
fill: fill.includes('url') ? DEFAULT_FILL : fill
|
|
}), wordsByLines.map((line, index) => {
|
|
var words = line.words.join(breakAll ? '' : ' ');
|
|
return (
|
|
/*#__PURE__*/
|
|
// duplicate words will cause duplicate keys which is why we add the array index here
|
|
React.createElement("tspan", {
|
|
x: x,
|
|
dy: index === 0 ? startDy : lineHeight,
|
|
key: "".concat(words, "-").concat(index)
|
|
}, words)
|
|
);
|
|
}));
|
|
});
|
|
Text.displayName = 'Text'; |