Frontend Enhancements: - Complete React TypeScript frontend with modern UI components - Distributed workflows management interface with real-time updates - Socket.IO integration for live agent status monitoring - Agent management dashboard with cluster visualization - Project management interface with metrics and task tracking - Responsive design with proper error handling and loading states Backend Infrastructure: - Distributed coordinator for multi-agent workflow orchestration - Cluster management API with comprehensive agent operations - Enhanced database models for agents and projects - Project service for filesystem-based project discovery - Performance monitoring and metrics collection - Comprehensive API documentation and error handling Documentation: - Complete distributed development guide (README_DISTRIBUTED.md) - Comprehensive development report with architecture insights - System configuration templates and deployment guides The platform now provides a complete web interface for managing the distributed AI cluster with real-time monitoring, workflow orchestration, and agent coordination capabilities. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
2099 lines
96 KiB
JavaScript
2099 lines
96 KiB
JavaScript
var _excluded = ["item"],
|
|
_excluded2 = ["children", "className", "width", "height", "style", "compact", "title", "desc"];
|
|
function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
|
|
function _extends() { _extends = Object.assign ? Object.assign.bind() : function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
|
|
function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
|
|
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
|
|
function _iterableToArrayLimit(r, l) { var t = null == r ? null : "undefined" != typeof Symbol && r[Symbol.iterator] || r["@@iterator"]; if (null != t) { var e, n, i, u, a = [], f = !0, o = !1; try { if (i = (t = t.call(r)).next, 0 === l) { if (Object(t) !== t) return; f = !1; } else for (; !(f = (e = i.call(t)).done) && (a.push(e.value), a.length !== l); f = !0); } catch (r) { o = !0, n = r; } finally { try { if (!f && null != t["return"] && (u = t["return"](), Object(u) !== u)) return; } finally { if (o) throw n; } } return a; } }
|
|
function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
|
|
function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
|
|
function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } } return target; }
|
|
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
|
|
function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, _toPropertyKey(descriptor.key), descriptor); } }
|
|
function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, "prototype", { writable: false }); return Constructor; }
|
|
function _callSuper(t, o, e) { return o = _getPrototypeOf(o), _possibleConstructorReturn(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], _getPrototypeOf(t).constructor) : o.apply(t, e)); }
|
|
function _possibleConstructorReturn(self, call) { if (call && (_typeof(call) === "object" || typeof call === "function")) { return call; } else if (call !== void 0) { throw new TypeError("Derived constructors may only return object or undefined"); } return _assertThisInitialized(self); }
|
|
function _assertThisInitialized(self) { if (self === void 0) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return self; }
|
|
function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); }
|
|
function _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf.bind() : function _getPrototypeOf(o) { return o.__proto__ || Object.getPrototypeOf(o); }; return _getPrototypeOf(o); }
|
|
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function"); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, writable: true, configurable: true } }); Object.defineProperty(subClass, "prototype", { writable: false }); if (superClass) _setPrototypeOf(subClass, superClass); }
|
|
function _setPrototypeOf(o, p) { _setPrototypeOf = Object.setPrototypeOf ? Object.setPrototypeOf.bind() : function _setPrototypeOf(o, p) { o.__proto__ = p; return o; }; return _setPrototypeOf(o, p); }
|
|
function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); }
|
|
function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
|
|
function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
|
|
function _iterableToArray(iter) { if (typeof Symbol !== "undefined" && iter[Symbol.iterator] != null || iter["@@iterator"] != null) return Array.from(iter); }
|
|
function _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); }
|
|
function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i]; return arr2; }
|
|
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
|
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
|
|
function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : i + ""; }
|
|
function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
|
|
import React, { Component, cloneElement, isValidElement, forwardRef } from 'react';
|
|
import isNil from 'lodash/isNil';
|
|
import isFunction from 'lodash/isFunction';
|
|
import range from 'lodash/range';
|
|
import get from 'lodash/get';
|
|
import sortBy from 'lodash/sortBy';
|
|
import throttle from 'lodash/throttle';
|
|
import clsx from 'clsx';
|
|
// eslint-disable-next-line no-restricted-imports
|
|
|
|
import invariant from 'tiny-invariant';
|
|
import { Surface } from '../container/Surface';
|
|
import { Layer } from '../container/Layer';
|
|
import { Tooltip } from '../component/Tooltip';
|
|
import { Legend } from '../component/Legend';
|
|
import { Dot } from '../shape/Dot';
|
|
import { isInRectangle } from '../shape/Rectangle';
|
|
import { filterProps, findAllByType, findChildByType, getDisplayName, getReactEventByType, isChildrenEqual, parseChildIndex, renderByOrder, validateWidthHeight } from '../util/ReactUtils';
|
|
import { Brush } from '../cartesian/Brush';
|
|
import { getOffset } from '../util/DOMUtils';
|
|
import { findEntryInArray, getAnyElementOfObject, hasDuplicate, isNumber, uniqueId } from '../util/DataUtils';
|
|
import { appendOffsetOfLegend, calculateActiveTickIndex, combineEventHandlers, getBandSizeOfAxis, getBarPosition, getBarSizeList, getDomainOfDataByKey, getDomainOfItemsWithSameAxis, getDomainOfStackGroups, getLegendProps, getMainColorOfGraphicItem, getStackedDataOfItem, getStackGroupsByAxisId, getTicksOfAxis, getTooltipItem, isCategoricalAxis, parseDomainOfCategoryAxis, parseErrorBarsOfAxis, parseSpecifiedDomain } from '../util/ChartUtils';
|
|
import { detectReferenceElementsDomain } from '../util/DetectReferenceElementsDomain';
|
|
import { inRangeOfSector, polarToCartesian } from '../util/PolarUtils';
|
|
import { shallowEqual } from '../util/ShallowEqual';
|
|
import { eventCenter, SYNC_EVENT } from '../util/Events';
|
|
import { adaptEventHandlers } from '../util/types';
|
|
import { AccessibilityManager } from './AccessibilityManager';
|
|
import { isDomainSpecifiedByUser } from '../util/isDomainSpecifiedByUser';
|
|
import { getActiveShapeIndexForTooltip, isFunnel, isPie, isScatter } from '../util/ActiveShapeUtils';
|
|
import { Cursor } from '../component/Cursor';
|
|
import { ChartLayoutContextProvider } from '../context/chartLayoutContext';
|
|
var ORIENT_MAP = {
|
|
xAxis: ['bottom', 'top'],
|
|
yAxis: ['left', 'right']
|
|
};
|
|
var FULL_WIDTH_AND_HEIGHT = {
|
|
width: '100%',
|
|
height: '100%'
|
|
};
|
|
var originCoordinate = {
|
|
x: 0,
|
|
y: 0
|
|
};
|
|
|
|
/**
|
|
* This function exists as a temporary workaround.
|
|
*
|
|
* Why? generateCategoricalChart does not render `{children}` directly;
|
|
* instead it passes them through `renderByOrder` function which reads their handlers.
|
|
*
|
|
* So, this is a handler that does nothing.
|
|
* Once we get rid of `renderByOrder` and switch to JSX only, we can get rid of this handler too.
|
|
*
|
|
* @param {JSX} element as is in JSX
|
|
* @returns {JSX} the same element
|
|
*/
|
|
function renderAsIs(element) {
|
|
return element;
|
|
}
|
|
var calculateTooltipPos = function calculateTooltipPos(rangeObj, layout) {
|
|
if (layout === 'horizontal') {
|
|
return rangeObj.x;
|
|
}
|
|
if (layout === 'vertical') {
|
|
return rangeObj.y;
|
|
}
|
|
if (layout === 'centric') {
|
|
return rangeObj.angle;
|
|
}
|
|
return rangeObj.radius;
|
|
};
|
|
var getActiveCoordinate = function getActiveCoordinate(layout, tooltipTicks, activeIndex, rangeObj) {
|
|
var entry = tooltipTicks.find(function (tick) {
|
|
return tick && tick.index === activeIndex;
|
|
});
|
|
if (entry) {
|
|
if (layout === 'horizontal') {
|
|
return {
|
|
x: entry.coordinate,
|
|
y: rangeObj.y
|
|
};
|
|
}
|
|
if (layout === 'vertical') {
|
|
return {
|
|
x: rangeObj.x,
|
|
y: entry.coordinate
|
|
};
|
|
}
|
|
if (layout === 'centric') {
|
|
var _angle = entry.coordinate;
|
|
var _radius = rangeObj.radius;
|
|
return _objectSpread(_objectSpread(_objectSpread({}, rangeObj), polarToCartesian(rangeObj.cx, rangeObj.cy, _radius, _angle)), {}, {
|
|
angle: _angle,
|
|
radius: _radius
|
|
});
|
|
}
|
|
var radius = entry.coordinate;
|
|
var angle = rangeObj.angle;
|
|
return _objectSpread(_objectSpread(_objectSpread({}, rangeObj), polarToCartesian(rangeObj.cx, rangeObj.cy, radius, angle)), {}, {
|
|
angle: angle,
|
|
radius: radius
|
|
});
|
|
}
|
|
return originCoordinate;
|
|
};
|
|
var getDisplayedData = function getDisplayedData(data, _ref) {
|
|
var graphicalItems = _ref.graphicalItems,
|
|
dataStartIndex = _ref.dataStartIndex,
|
|
dataEndIndex = _ref.dataEndIndex;
|
|
var itemsData = (graphicalItems !== null && graphicalItems !== void 0 ? graphicalItems : []).reduce(function (result, child) {
|
|
var itemData = child.props.data;
|
|
if (itemData && itemData.length) {
|
|
return [].concat(_toConsumableArray(result), _toConsumableArray(itemData));
|
|
}
|
|
return result;
|
|
}, []);
|
|
if (itemsData.length > 0) {
|
|
return itemsData;
|
|
}
|
|
if (data && data.length && isNumber(dataStartIndex) && isNumber(dataEndIndex)) {
|
|
return data.slice(dataStartIndex, dataEndIndex + 1);
|
|
}
|
|
return [];
|
|
};
|
|
function getDefaultDomainByAxisType(axisType) {
|
|
return axisType === 'number' ? [0, 'auto'] : undefined;
|
|
}
|
|
|
|
/**
|
|
* Get the content to be displayed in the tooltip
|
|
* @param {Object} state Current state
|
|
* @param {Array} chartData The data defined in chart
|
|
* @param {Number} activeIndex Active index of data
|
|
* @param {String} activeLabel Active label of data
|
|
* @return {Array} The content of tooltip
|
|
*/
|
|
var getTooltipContent = function getTooltipContent(state, chartData, activeIndex, activeLabel) {
|
|
var graphicalItems = state.graphicalItems,
|
|
tooltipAxis = state.tooltipAxis;
|
|
var displayedData = getDisplayedData(chartData, state);
|
|
if (activeIndex < 0 || !graphicalItems || !graphicalItems.length || activeIndex >= displayedData.length) {
|
|
return null;
|
|
}
|
|
// get data by activeIndex when the axis don't allow duplicated category
|
|
return graphicalItems.reduce(function (result, child) {
|
|
var _child$props$data;
|
|
/**
|
|
* Fixes: https://github.com/recharts/recharts/issues/3669
|
|
* Defaulting to chartData below to fix an edge case where the tooltip does not include data from all charts
|
|
* when a separate dataset is passed to chart prop data and specified on Line/Area/etc prop data
|
|
*/
|
|
var data = (_child$props$data = child.props.data) !== null && _child$props$data !== void 0 ? _child$props$data : chartData;
|
|
if (data && state.dataStartIndex + state.dataEndIndex !== 0 &&
|
|
// https://github.com/recharts/recharts/issues/4717
|
|
// The data is sliced only when the active index is within the start/end index range.
|
|
state.dataEndIndex - state.dataStartIndex >= activeIndex) {
|
|
data = data.slice(state.dataStartIndex, state.dataEndIndex + 1);
|
|
}
|
|
var payload;
|
|
if (tooltipAxis.dataKey && !tooltipAxis.allowDuplicatedCategory) {
|
|
// graphic child has data props
|
|
var entries = data === undefined ? displayedData : data;
|
|
payload = findEntryInArray(entries, tooltipAxis.dataKey, activeLabel);
|
|
} else {
|
|
payload = data && data[activeIndex] || displayedData[activeIndex];
|
|
}
|
|
if (!payload) {
|
|
return result;
|
|
}
|
|
return [].concat(_toConsumableArray(result), [getTooltipItem(child, payload)]);
|
|
}, []);
|
|
};
|
|
|
|
/**
|
|
* Returns tooltip data based on a mouse position (as a parameter or in state)
|
|
* @param {Object} state current state
|
|
* @param {Array} chartData the data defined in chart
|
|
* @param {String} layout The layout type of chart
|
|
* @param {Object} rangeObj { x, y } coordinates
|
|
* @return {Object} Tooltip data data
|
|
*/
|
|
var getTooltipData = function getTooltipData(state, chartData, layout, rangeObj) {
|
|
var rangeData = rangeObj || {
|
|
x: state.chartX,
|
|
y: state.chartY
|
|
};
|
|
var pos = calculateTooltipPos(rangeData, layout);
|
|
var ticks = state.orderedTooltipTicks,
|
|
axis = state.tooltipAxis,
|
|
tooltipTicks = state.tooltipTicks;
|
|
var activeIndex = calculateActiveTickIndex(pos, ticks, tooltipTicks, axis);
|
|
if (activeIndex >= 0 && tooltipTicks) {
|
|
var activeLabel = tooltipTicks[activeIndex] && tooltipTicks[activeIndex].value;
|
|
var activePayload = getTooltipContent(state, chartData, activeIndex, activeLabel);
|
|
var activeCoordinate = getActiveCoordinate(layout, ticks, activeIndex, rangeData);
|
|
return {
|
|
activeTooltipIndex: activeIndex,
|
|
activeLabel: activeLabel,
|
|
activePayload: activePayload,
|
|
activeCoordinate: activeCoordinate
|
|
};
|
|
}
|
|
return null;
|
|
};
|
|
|
|
/**
|
|
* Get the configuration of axis by the options of axis instance
|
|
* @param {Object} props Latest props
|
|
* @param {Array} axes The instance of axes
|
|
* @param {Array} graphicalItems The instances of item
|
|
* @param {String} axisType The type of axis, xAxis - x-axis, yAxis - y-axis
|
|
* @param {String} axisIdKey The unique id of an axis
|
|
* @param {Object} stackGroups The items grouped by axisId and stackId
|
|
* @param {Number} dataStartIndex The start index of the data series when a brush is applied
|
|
* @param {Number} dataEndIndex The end index of the data series when a brush is applied
|
|
* @return {Object} Configuration
|
|
*/
|
|
export var getAxisMapByAxes = function getAxisMapByAxes(props, _ref2) {
|
|
var axes = _ref2.axes,
|
|
graphicalItems = _ref2.graphicalItems,
|
|
axisType = _ref2.axisType,
|
|
axisIdKey = _ref2.axisIdKey,
|
|
stackGroups = _ref2.stackGroups,
|
|
dataStartIndex = _ref2.dataStartIndex,
|
|
dataEndIndex = _ref2.dataEndIndex;
|
|
var layout = props.layout,
|
|
children = props.children,
|
|
stackOffset = props.stackOffset;
|
|
var isCategorical = isCategoricalAxis(layout, axisType);
|
|
|
|
// Eliminate duplicated axes
|
|
return axes.reduce(function (result, child) {
|
|
var _childProps$domain2;
|
|
var childProps = child.type.defaultProps !== undefined ? _objectSpread(_objectSpread({}, child.type.defaultProps), child.props) : child.props;
|
|
var type = childProps.type,
|
|
dataKey = childProps.dataKey,
|
|
allowDataOverflow = childProps.allowDataOverflow,
|
|
allowDuplicatedCategory = childProps.allowDuplicatedCategory,
|
|
scale = childProps.scale,
|
|
ticks = childProps.ticks,
|
|
includeHidden = childProps.includeHidden;
|
|
var axisId = childProps[axisIdKey];
|
|
if (result[axisId]) {
|
|
return result;
|
|
}
|
|
var displayedData = getDisplayedData(props.data, {
|
|
graphicalItems: graphicalItems.filter(function (item) {
|
|
var _defaultProps;
|
|
var itemAxisId = axisIdKey in item.props ? item.props[axisIdKey] : (_defaultProps = item.type.defaultProps) === null || _defaultProps === void 0 ? void 0 : _defaultProps[axisIdKey];
|
|
return itemAxisId === axisId;
|
|
}),
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
});
|
|
var len = displayedData.length;
|
|
var domain, duplicateDomain, categoricalDomain;
|
|
|
|
/*
|
|
* This is a hack to short-circuit the domain creation here to enhance performance.
|
|
* Usually, the data is used to determine the domain, but when the user specifies
|
|
* a domain upfront (via props), there is no need to calculate the domain start and end,
|
|
* which is very expensive for a larger amount of data.
|
|
* The only thing that would prohibit short-circuiting is when the user doesn't allow data overflow,
|
|
* because the axis is supposed to ignore the specified domain that way.
|
|
*/
|
|
if (isDomainSpecifiedByUser(childProps.domain, allowDataOverflow, type)) {
|
|
domain = parseSpecifiedDomain(childProps.domain, null, allowDataOverflow);
|
|
/* The chart can be categorical and have the domain specified in numbers
|
|
* we still need to calculate the categorical domain
|
|
* TODO: refactor this more
|
|
*/
|
|
if (isCategorical && (type === 'number' || scale !== 'auto')) {
|
|
categoricalDomain = getDomainOfDataByKey(displayedData, dataKey, 'category');
|
|
}
|
|
}
|
|
|
|
// if the domain is defaulted we need this for `originalDomain` as well
|
|
var defaultDomain = getDefaultDomainByAxisType(type);
|
|
|
|
// we didn't create the domain from user's props above, so we need to calculate it
|
|
if (!domain || domain.length === 0) {
|
|
var _childProps$domain;
|
|
var childDomain = (_childProps$domain = childProps.domain) !== null && _childProps$domain !== void 0 ? _childProps$domain : defaultDomain;
|
|
if (dataKey) {
|
|
// has dataKey in <Axis />
|
|
domain = getDomainOfDataByKey(displayedData, dataKey, type);
|
|
if (type === 'category' && isCategorical) {
|
|
// the field type is category data and this axis is categorical axis
|
|
var duplicate = hasDuplicate(domain);
|
|
if (allowDuplicatedCategory && duplicate) {
|
|
duplicateDomain = domain;
|
|
// When category axis has duplicated text, serial numbers are used to generate scale
|
|
domain = range(0, len);
|
|
} else if (!allowDuplicatedCategory) {
|
|
// remove duplicated category
|
|
domain = parseDomainOfCategoryAxis(childDomain, domain, child).reduce(function (finalDomain, entry) {
|
|
return finalDomain.indexOf(entry) >= 0 ? finalDomain : [].concat(_toConsumableArray(finalDomain), [entry]);
|
|
}, []);
|
|
}
|
|
} else if (type === 'category') {
|
|
// the field type is category data and this axis is numerical axis
|
|
if (!allowDuplicatedCategory) {
|
|
domain = parseDomainOfCategoryAxis(childDomain, domain, child).reduce(function (finalDomain, entry) {
|
|
return finalDomain.indexOf(entry) >= 0 || entry === '' || isNil(entry) ? finalDomain : [].concat(_toConsumableArray(finalDomain), [entry]);
|
|
}, []);
|
|
} else {
|
|
// eliminate undefined or null or empty string
|
|
domain = domain.filter(function (entry) {
|
|
return entry !== '' && !isNil(entry);
|
|
});
|
|
}
|
|
} else if (type === 'number') {
|
|
// the field type is numerical
|
|
var errorBarsDomain = parseErrorBarsOfAxis(displayedData, graphicalItems.filter(function (item) {
|
|
var _defaultProps2, _defaultProps3;
|
|
var itemAxisId = axisIdKey in item.props ? item.props[axisIdKey] : (_defaultProps2 = item.type.defaultProps) === null || _defaultProps2 === void 0 ? void 0 : _defaultProps2[axisIdKey];
|
|
var itemHide = 'hide' in item.props ? item.props.hide : (_defaultProps3 = item.type.defaultProps) === null || _defaultProps3 === void 0 ? void 0 : _defaultProps3.hide;
|
|
return itemAxisId === axisId && (includeHidden || !itemHide);
|
|
}), dataKey, axisType, layout);
|
|
if (errorBarsDomain) {
|
|
domain = errorBarsDomain;
|
|
}
|
|
}
|
|
if (isCategorical && (type === 'number' || scale !== 'auto')) {
|
|
categoricalDomain = getDomainOfDataByKey(displayedData, dataKey, 'category');
|
|
}
|
|
} else if (isCategorical) {
|
|
// the axis is a categorical axis
|
|
domain = range(0, len);
|
|
} else if (stackGroups && stackGroups[axisId] && stackGroups[axisId].hasStack && type === 'number') {
|
|
// when stackOffset is 'expand', the domain may be calculated as [0, 1.000000000002]
|
|
domain = stackOffset === 'expand' ? [0, 1] : getDomainOfStackGroups(stackGroups[axisId].stackGroups, dataStartIndex, dataEndIndex);
|
|
} else {
|
|
domain = getDomainOfItemsWithSameAxis(displayedData, graphicalItems.filter(function (item) {
|
|
var itemAxisId = axisIdKey in item.props ? item.props[axisIdKey] : item.type.defaultProps[axisIdKey];
|
|
var itemHide = 'hide' in item.props ? item.props.hide : item.type.defaultProps.hide;
|
|
return itemAxisId === axisId && (includeHidden || !itemHide);
|
|
}), type, layout, true);
|
|
}
|
|
if (type === 'number') {
|
|
// To detect wether there is any reference lines whose props alwaysShow is true
|
|
domain = detectReferenceElementsDomain(children, domain, axisId, axisType, ticks);
|
|
if (childDomain) {
|
|
domain = parseSpecifiedDomain(childDomain, domain, allowDataOverflow);
|
|
}
|
|
} else if (type === 'category' && childDomain) {
|
|
var axisDomain = childDomain;
|
|
var isDomainValid = domain.every(function (entry) {
|
|
return axisDomain.indexOf(entry) >= 0;
|
|
});
|
|
if (isDomainValid) {
|
|
domain = axisDomain;
|
|
}
|
|
}
|
|
}
|
|
return _objectSpread(_objectSpread({}, result), {}, _defineProperty({}, axisId, _objectSpread(_objectSpread({}, childProps), {}, {
|
|
axisType: axisType,
|
|
domain: domain,
|
|
categoricalDomain: categoricalDomain,
|
|
duplicateDomain: duplicateDomain,
|
|
originalDomain: (_childProps$domain2 = childProps.domain) !== null && _childProps$domain2 !== void 0 ? _childProps$domain2 : defaultDomain,
|
|
isCategorical: isCategorical,
|
|
layout: layout
|
|
})));
|
|
}, {});
|
|
};
|
|
|
|
/**
|
|
* Get the configuration of axis by the options of item,
|
|
* this kind of axis does not display in chart
|
|
* @param {Object} props Latest props
|
|
* @param {Array} graphicalItems The instances of item
|
|
* @param {ReactElement} Axis Axis Component
|
|
* @param {String} axisType The type of axis, xAxis - x-axis, yAxis - y-axis
|
|
* @param {String} axisIdKey The unique id of an axis
|
|
* @param {Object} stackGroups The items grouped by axisId and stackId
|
|
* @param {Number} dataStartIndex The start index of the data series when a brush is applied
|
|
* @param {Number} dataEndIndex The end index of the data series when a brush is applied
|
|
* @return {Object} Configuration
|
|
*/
|
|
var getAxisMapByItems = function getAxisMapByItems(props, _ref3) {
|
|
var graphicalItems = _ref3.graphicalItems,
|
|
Axis = _ref3.Axis,
|
|
axisType = _ref3.axisType,
|
|
axisIdKey = _ref3.axisIdKey,
|
|
stackGroups = _ref3.stackGroups,
|
|
dataStartIndex = _ref3.dataStartIndex,
|
|
dataEndIndex = _ref3.dataEndIndex;
|
|
var layout = props.layout,
|
|
children = props.children;
|
|
var displayedData = getDisplayedData(props.data, {
|
|
graphicalItems: graphicalItems,
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
});
|
|
var len = displayedData.length;
|
|
var isCategorical = isCategoricalAxis(layout, axisType);
|
|
var index = -1;
|
|
|
|
// The default type of x-axis is category axis,
|
|
// The default contents of x-axis is the serial numbers of data
|
|
// The default type of y-axis is number axis
|
|
// The default contents of y-axis is the domain of data
|
|
return graphicalItems.reduce(function (result, child) {
|
|
var childProps = child.type.defaultProps !== undefined ? _objectSpread(_objectSpread({}, child.type.defaultProps), child.props) : child.props;
|
|
var axisId = childProps[axisIdKey];
|
|
var originalDomain = getDefaultDomainByAxisType('number');
|
|
if (!result[axisId]) {
|
|
index++;
|
|
var domain;
|
|
if (isCategorical) {
|
|
domain = range(0, len);
|
|
} else if (stackGroups && stackGroups[axisId] && stackGroups[axisId].hasStack) {
|
|
domain = getDomainOfStackGroups(stackGroups[axisId].stackGroups, dataStartIndex, dataEndIndex);
|
|
domain = detectReferenceElementsDomain(children, domain, axisId, axisType);
|
|
} else {
|
|
domain = parseSpecifiedDomain(originalDomain, getDomainOfItemsWithSameAxis(displayedData, graphicalItems.filter(function (item) {
|
|
var _defaultProps4, _defaultProps5;
|
|
var itemAxisId = axisIdKey in item.props ? item.props[axisIdKey] : (_defaultProps4 = item.type.defaultProps) === null || _defaultProps4 === void 0 ? void 0 : _defaultProps4[axisIdKey];
|
|
var itemHide = 'hide' in item.props ? item.props.hide : (_defaultProps5 = item.type.defaultProps) === null || _defaultProps5 === void 0 ? void 0 : _defaultProps5.hide;
|
|
return itemAxisId === axisId && !itemHide;
|
|
}), 'number', layout), Axis.defaultProps.allowDataOverflow);
|
|
domain = detectReferenceElementsDomain(children, domain, axisId, axisType);
|
|
}
|
|
return _objectSpread(_objectSpread({}, result), {}, _defineProperty({}, axisId, _objectSpread(_objectSpread({
|
|
axisType: axisType
|
|
}, Axis.defaultProps), {}, {
|
|
hide: true,
|
|
orientation: get(ORIENT_MAP, "".concat(axisType, ".").concat(index % 2), null),
|
|
domain: domain,
|
|
originalDomain: originalDomain,
|
|
isCategorical: isCategorical,
|
|
layout: layout
|
|
// specify scale when no Axis
|
|
// scale: isCategorical ? 'band' : 'linear',
|
|
})));
|
|
}
|
|
return result;
|
|
}, {});
|
|
};
|
|
|
|
/**
|
|
* Get the configuration of all x-axis or y-axis
|
|
* @param {Object} props Latest props
|
|
* @param {String} axisType The type of axis
|
|
* @param {React.ComponentType} [AxisComp] Axis Component
|
|
* @param {Array} graphicalItems The instances of item
|
|
* @param {Object} stackGroups The items grouped by axisId and stackId
|
|
* @param {Number} dataStartIndex The start index of the data series when a brush is applied
|
|
* @param {Number} dataEndIndex The end index of the data series when a brush is applied
|
|
* @return {Object} Configuration
|
|
*/
|
|
var getAxisMap = function getAxisMap(props, _ref4) {
|
|
var _ref4$axisType = _ref4.axisType,
|
|
axisType = _ref4$axisType === void 0 ? 'xAxis' : _ref4$axisType,
|
|
AxisComp = _ref4.AxisComp,
|
|
graphicalItems = _ref4.graphicalItems,
|
|
stackGroups = _ref4.stackGroups,
|
|
dataStartIndex = _ref4.dataStartIndex,
|
|
dataEndIndex = _ref4.dataEndIndex;
|
|
var children = props.children;
|
|
var axisIdKey = "".concat(axisType, "Id");
|
|
// Get all the instance of Axis
|
|
var axes = findAllByType(children, AxisComp);
|
|
var axisMap = {};
|
|
if (axes && axes.length) {
|
|
axisMap = getAxisMapByAxes(props, {
|
|
axes: axes,
|
|
graphicalItems: graphicalItems,
|
|
axisType: axisType,
|
|
axisIdKey: axisIdKey,
|
|
stackGroups: stackGroups,
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
});
|
|
} else if (graphicalItems && graphicalItems.length) {
|
|
axisMap = getAxisMapByItems(props, {
|
|
Axis: AxisComp,
|
|
graphicalItems: graphicalItems,
|
|
axisType: axisType,
|
|
axisIdKey: axisIdKey,
|
|
stackGroups: stackGroups,
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
});
|
|
}
|
|
return axisMap;
|
|
};
|
|
var tooltipTicksGenerator = function tooltipTicksGenerator(axisMap) {
|
|
var axis = getAnyElementOfObject(axisMap);
|
|
var tooltipTicks = getTicksOfAxis(axis, false, true);
|
|
return {
|
|
tooltipTicks: tooltipTicks,
|
|
orderedTooltipTicks: sortBy(tooltipTicks, function (o) {
|
|
return o.coordinate;
|
|
}),
|
|
tooltipAxis: axis,
|
|
tooltipAxisBandSize: getBandSizeOfAxis(axis, tooltipTicks)
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Returns default, reset state for the categorical chart.
|
|
* @param {Object} props Props object to use when creating the default state
|
|
* @return {Object} Whole new state
|
|
*/
|
|
export var createDefaultState = function createDefaultState(props) {
|
|
var children = props.children,
|
|
defaultShowTooltip = props.defaultShowTooltip;
|
|
var brushItem = findChildByType(children, Brush);
|
|
var startIndex = 0;
|
|
var endIndex = 0;
|
|
if (props.data && props.data.length !== 0) {
|
|
endIndex = props.data.length - 1;
|
|
}
|
|
if (brushItem && brushItem.props) {
|
|
if (brushItem.props.startIndex >= 0) {
|
|
startIndex = brushItem.props.startIndex;
|
|
}
|
|
if (brushItem.props.endIndex >= 0) {
|
|
endIndex = brushItem.props.endIndex;
|
|
}
|
|
}
|
|
return {
|
|
chartX: 0,
|
|
chartY: 0,
|
|
dataStartIndex: startIndex,
|
|
dataEndIndex: endIndex,
|
|
activeTooltipIndex: -1,
|
|
isTooltipActive: Boolean(defaultShowTooltip)
|
|
};
|
|
};
|
|
var hasGraphicalBarItem = function hasGraphicalBarItem(graphicalItems) {
|
|
if (!graphicalItems || !graphicalItems.length) {
|
|
return false;
|
|
}
|
|
return graphicalItems.some(function (item) {
|
|
var name = getDisplayName(item && item.type);
|
|
return name && name.indexOf('Bar') >= 0;
|
|
});
|
|
};
|
|
var getAxisNameByLayout = function getAxisNameByLayout(layout) {
|
|
if (layout === 'horizontal') {
|
|
return {
|
|
numericAxisName: 'yAxis',
|
|
cateAxisName: 'xAxis'
|
|
};
|
|
}
|
|
if (layout === 'vertical') {
|
|
return {
|
|
numericAxisName: 'xAxis',
|
|
cateAxisName: 'yAxis'
|
|
};
|
|
}
|
|
if (layout === 'centric') {
|
|
return {
|
|
numericAxisName: 'radiusAxis',
|
|
cateAxisName: 'angleAxis'
|
|
};
|
|
}
|
|
return {
|
|
numericAxisName: 'angleAxis',
|
|
cateAxisName: 'radiusAxis'
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Calculate the offset of main part in the svg element
|
|
* @param {Object} params.props Latest props
|
|
* @param {Array} params.graphicalItems The instances of item
|
|
* @param {Object} params.xAxisMap The configuration of x-axis
|
|
* @param {Object} params.yAxisMap The configuration of y-axis
|
|
* @param {Object} prevLegendBBox The boundary box of legend
|
|
* @return {Object} The offset of main part in the svg element
|
|
*/
|
|
var calculateOffset = function calculateOffset(_ref5, prevLegendBBox) {
|
|
var props = _ref5.props,
|
|
graphicalItems = _ref5.graphicalItems,
|
|
_ref5$xAxisMap = _ref5.xAxisMap,
|
|
xAxisMap = _ref5$xAxisMap === void 0 ? {} : _ref5$xAxisMap,
|
|
_ref5$yAxisMap = _ref5.yAxisMap,
|
|
yAxisMap = _ref5$yAxisMap === void 0 ? {} : _ref5$yAxisMap;
|
|
var width = props.width,
|
|
height = props.height,
|
|
children = props.children;
|
|
var margin = props.margin || {};
|
|
var brushItem = findChildByType(children, Brush);
|
|
var legendItem = findChildByType(children, Legend);
|
|
var offsetH = Object.keys(yAxisMap).reduce(function (result, id) {
|
|
var entry = yAxisMap[id];
|
|
var orientation = entry.orientation;
|
|
if (!entry.mirror && !entry.hide) {
|
|
return _objectSpread(_objectSpread({}, result), {}, _defineProperty({}, orientation, result[orientation] + entry.width));
|
|
}
|
|
return result;
|
|
}, {
|
|
left: margin.left || 0,
|
|
right: margin.right || 0
|
|
});
|
|
var offsetV = Object.keys(xAxisMap).reduce(function (result, id) {
|
|
var entry = xAxisMap[id];
|
|
var orientation = entry.orientation;
|
|
if (!entry.mirror && !entry.hide) {
|
|
return _objectSpread(_objectSpread({}, result), {}, _defineProperty({}, orientation, get(result, "".concat(orientation)) + entry.height));
|
|
}
|
|
return result;
|
|
}, {
|
|
top: margin.top || 0,
|
|
bottom: margin.bottom || 0
|
|
});
|
|
var offset = _objectSpread(_objectSpread({}, offsetV), offsetH);
|
|
var brushBottom = offset.bottom;
|
|
if (brushItem) {
|
|
offset.bottom += brushItem.props.height || Brush.defaultProps.height;
|
|
}
|
|
if (legendItem && prevLegendBBox) {
|
|
// @ts-expect-error margin is optional in props but required in appendOffsetOfLegend
|
|
offset = appendOffsetOfLegend(offset, graphicalItems, props, prevLegendBBox);
|
|
}
|
|
var offsetWidth = width - offset.left - offset.right;
|
|
var offsetHeight = height - offset.top - offset.bottom;
|
|
return _objectSpread(_objectSpread({
|
|
brushBottom: brushBottom
|
|
}, offset), {}, {
|
|
// never return negative values for height and width
|
|
width: Math.max(offsetWidth, 0),
|
|
height: Math.max(offsetHeight, 0)
|
|
});
|
|
};
|
|
// Determine the size of the axis, used for calculation of relative bar sizes
|
|
var getCartesianAxisSize = function getCartesianAxisSize(axisObj, axisName) {
|
|
if (axisName === 'xAxis') {
|
|
return axisObj[axisName].width;
|
|
}
|
|
if (axisName === 'yAxis') {
|
|
return axisObj[axisName].height;
|
|
}
|
|
// This is only supported for Bar charts (i.e. charts with cartesian axes), so we should never get here
|
|
return undefined;
|
|
};
|
|
export var generateCategoricalChart = function generateCategoricalChart(_ref6) {
|
|
var chartName = _ref6.chartName,
|
|
GraphicalChild = _ref6.GraphicalChild,
|
|
_ref6$defaultTooltipE = _ref6.defaultTooltipEventType,
|
|
defaultTooltipEventType = _ref6$defaultTooltipE === void 0 ? 'axis' : _ref6$defaultTooltipE,
|
|
_ref6$validateTooltip = _ref6.validateTooltipEventTypes,
|
|
validateTooltipEventTypes = _ref6$validateTooltip === void 0 ? ['axis'] : _ref6$validateTooltip,
|
|
axisComponents = _ref6.axisComponents,
|
|
legendContent = _ref6.legendContent,
|
|
formatAxisMap = _ref6.formatAxisMap,
|
|
defaultProps = _ref6.defaultProps;
|
|
var getFormatItems = function getFormatItems(props, currentState) {
|
|
var graphicalItems = currentState.graphicalItems,
|
|
stackGroups = currentState.stackGroups,
|
|
offset = currentState.offset,
|
|
updateId = currentState.updateId,
|
|
dataStartIndex = currentState.dataStartIndex,
|
|
dataEndIndex = currentState.dataEndIndex;
|
|
var barSize = props.barSize,
|
|
layout = props.layout,
|
|
barGap = props.barGap,
|
|
barCategoryGap = props.barCategoryGap,
|
|
globalMaxBarSize = props.maxBarSize;
|
|
var _getAxisNameByLayout = getAxisNameByLayout(layout),
|
|
numericAxisName = _getAxisNameByLayout.numericAxisName,
|
|
cateAxisName = _getAxisNameByLayout.cateAxisName;
|
|
var hasBar = hasGraphicalBarItem(graphicalItems);
|
|
var formattedItems = [];
|
|
graphicalItems.forEach(function (item, index) {
|
|
var displayedData = getDisplayedData(props.data, {
|
|
graphicalItems: [item],
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
});
|
|
var itemProps = item.type.defaultProps !== undefined ? _objectSpread(_objectSpread({}, item.type.defaultProps), item.props) : item.props;
|
|
var dataKey = itemProps.dataKey,
|
|
childMaxBarSize = itemProps.maxBarSize;
|
|
// axisId of the numerical axis
|
|
var numericAxisId = itemProps["".concat(numericAxisName, "Id")];
|
|
// axisId of the categorical axis
|
|
var cateAxisId = itemProps["".concat(cateAxisName, "Id")];
|
|
var axisObjInitialValue = {};
|
|
var axisObj = axisComponents.reduce(function (result, entry) {
|
|
var _item$type$displayNam, _item$type;
|
|
// map of axisId to axis for a specific axis type
|
|
var axisMap = currentState["".concat(entry.axisType, "Map")];
|
|
// axisId of axis we are currently computing
|
|
var id = itemProps["".concat(entry.axisType, "Id")];
|
|
|
|
/**
|
|
* tell the user in dev mode that their configuration is incorrect if we cannot find a match between
|
|
* axisId on the chart and axisId on the axis. zAxis does not get passed in the map for ComposedChart,
|
|
* leave it out of the check for now.
|
|
*/
|
|
!(axisMap && axisMap[id] || entry.axisType === 'zAxis') ? process.env.NODE_ENV !== "production" ? invariant(false, "Specifying a(n) ".concat(entry.axisType, "Id requires a corresponding ").concat(entry.axisType
|
|
// @ts-expect-error we should stop reading data from ReactElements
|
|
, "Id on the targeted graphical component ").concat((_item$type$displayNam = item === null || item === void 0 || (_item$type = item.type) === null || _item$type === void 0 ? void 0 : _item$type.displayName) !== null && _item$type$displayNam !== void 0 ? _item$type$displayNam : '')) : invariant(false) : void 0;
|
|
|
|
// the axis we are currently formatting
|
|
var axis = axisMap[id];
|
|
return _objectSpread(_objectSpread({}, result), {}, _defineProperty(_defineProperty({}, entry.axisType, axis), "".concat(entry.axisType, "Ticks"), getTicksOfAxis(axis)));
|
|
}, axisObjInitialValue);
|
|
var cateAxis = axisObj[cateAxisName];
|
|
var cateTicks = axisObj["".concat(cateAxisName, "Ticks")];
|
|
var stackedData = stackGroups && stackGroups[numericAxisId] && stackGroups[numericAxisId].hasStack && getStackedDataOfItem(item, stackGroups[numericAxisId].stackGroups);
|
|
var itemIsBar = getDisplayName(item.type).indexOf('Bar') >= 0;
|
|
var bandSize = getBandSizeOfAxis(cateAxis, cateTicks);
|
|
var barPosition = [];
|
|
var sizeList = hasBar && getBarSizeList({
|
|
barSize: barSize,
|
|
stackGroups: stackGroups,
|
|
totalSize: getCartesianAxisSize(axisObj, cateAxisName)
|
|
});
|
|
if (itemIsBar) {
|
|
var _ref7, _getBandSizeOfAxis;
|
|
// If it is bar, calculate the position of bar
|
|
var maxBarSize = isNil(childMaxBarSize) ? globalMaxBarSize : childMaxBarSize;
|
|
var barBandSize = (_ref7 = (_getBandSizeOfAxis = getBandSizeOfAxis(cateAxis, cateTicks, true)) !== null && _getBandSizeOfAxis !== void 0 ? _getBandSizeOfAxis : maxBarSize) !== null && _ref7 !== void 0 ? _ref7 : 0;
|
|
barPosition = getBarPosition({
|
|
barGap: barGap,
|
|
barCategoryGap: barCategoryGap,
|
|
bandSize: barBandSize !== bandSize ? barBandSize : bandSize,
|
|
sizeList: sizeList[cateAxisId],
|
|
maxBarSize: maxBarSize
|
|
});
|
|
if (barBandSize !== bandSize) {
|
|
barPosition = barPosition.map(function (pos) {
|
|
return _objectSpread(_objectSpread({}, pos), {}, {
|
|
position: _objectSpread(_objectSpread({}, pos.position), {}, {
|
|
offset: pos.position.offset - barBandSize / 2
|
|
})
|
|
});
|
|
});
|
|
}
|
|
}
|
|
// @ts-expect-error we should stop reading data from ReactElements
|
|
var composedFn = item && item.type && item.type.getComposedData;
|
|
if (composedFn) {
|
|
formattedItems.push({
|
|
props: _objectSpread(_objectSpread({}, composedFn(_objectSpread(_objectSpread({}, axisObj), {}, {
|
|
displayedData: displayedData,
|
|
props: props,
|
|
dataKey: dataKey,
|
|
item: item,
|
|
bandSize: bandSize,
|
|
barPosition: barPosition,
|
|
offset: offset,
|
|
stackedData: stackedData,
|
|
layout: layout,
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
}))), {}, _defineProperty(_defineProperty(_defineProperty({
|
|
key: item.key || "item-".concat(index)
|
|
}, numericAxisName, axisObj[numericAxisName]), cateAxisName, axisObj[cateAxisName]), "animationId", updateId)),
|
|
childIndex: parseChildIndex(item, props.children),
|
|
item: item
|
|
});
|
|
}
|
|
});
|
|
return formattedItems;
|
|
};
|
|
|
|
/**
|
|
* The AxisMaps are expensive to render on large data sets
|
|
* so provide the ability to store them in state and only update them when necessary
|
|
* they are dependent upon the start and end index of
|
|
* the brush so it's important that this method is called _after_
|
|
* the state is updated with any new start/end indices
|
|
*
|
|
* @param {Object} props The props object to be used for updating the axismaps
|
|
* dataStartIndex: The start index of the data series when a brush is applied
|
|
* dataEndIndex: The end index of the data series when a brush is applied
|
|
* updateId: The update id
|
|
* @param {Object} prevState Prev state
|
|
* @return {Object} state New state to set
|
|
*/
|
|
var updateStateOfAxisMapsOffsetAndStackGroups = function updateStateOfAxisMapsOffsetAndStackGroups(_ref8, prevState) {
|
|
var props = _ref8.props,
|
|
dataStartIndex = _ref8.dataStartIndex,
|
|
dataEndIndex = _ref8.dataEndIndex,
|
|
updateId = _ref8.updateId;
|
|
if (!validateWidthHeight({
|
|
props: props
|
|
})) {
|
|
return null;
|
|
}
|
|
var children = props.children,
|
|
layout = props.layout,
|
|
stackOffset = props.stackOffset,
|
|
data = props.data,
|
|
reverseStackOrder = props.reverseStackOrder;
|
|
var _getAxisNameByLayout2 = getAxisNameByLayout(layout),
|
|
numericAxisName = _getAxisNameByLayout2.numericAxisName,
|
|
cateAxisName = _getAxisNameByLayout2.cateAxisName;
|
|
var graphicalItems = findAllByType(children, GraphicalChild);
|
|
var stackGroups = getStackGroupsByAxisId(data, graphicalItems, "".concat(numericAxisName, "Id"), "".concat(cateAxisName, "Id"), stackOffset, reverseStackOrder);
|
|
var axisObj = axisComponents.reduce(function (result, entry) {
|
|
var name = "".concat(entry.axisType, "Map");
|
|
return _objectSpread(_objectSpread({}, result), {}, _defineProperty({}, name, getAxisMap(props, _objectSpread(_objectSpread({}, entry), {}, {
|
|
graphicalItems: graphicalItems,
|
|
stackGroups: entry.axisType === numericAxisName && stackGroups,
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
}))));
|
|
}, {});
|
|
var offset = calculateOffset(_objectSpread(_objectSpread({}, axisObj), {}, {
|
|
props: props,
|
|
graphicalItems: graphicalItems
|
|
}), prevState === null || prevState === void 0 ? void 0 : prevState.legendBBox);
|
|
Object.keys(axisObj).forEach(function (key) {
|
|
axisObj[key] = formatAxisMap(props, axisObj[key], offset, key.replace('Map', ''), chartName);
|
|
});
|
|
var cateAxisMap = axisObj["".concat(cateAxisName, "Map")];
|
|
var ticksObj = tooltipTicksGenerator(cateAxisMap);
|
|
var formattedGraphicalItems = getFormatItems(props, _objectSpread(_objectSpread({}, axisObj), {}, {
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex,
|
|
updateId: updateId,
|
|
graphicalItems: graphicalItems,
|
|
stackGroups: stackGroups,
|
|
offset: offset
|
|
}));
|
|
return _objectSpread(_objectSpread({
|
|
formattedGraphicalItems: formattedGraphicalItems,
|
|
graphicalItems: graphicalItems,
|
|
offset: offset,
|
|
stackGroups: stackGroups
|
|
}, ticksObj), axisObj);
|
|
};
|
|
var CategoricalChartWrapper = /*#__PURE__*/function (_Component) {
|
|
function CategoricalChartWrapper(_props) {
|
|
var _props$id, _props$throttleDelay;
|
|
var _this;
|
|
_classCallCheck(this, CategoricalChartWrapper);
|
|
_this = _callSuper(this, CategoricalChartWrapper, [_props]);
|
|
_defineProperty(_this, "eventEmitterSymbol", Symbol('rechartsEventEmitter'));
|
|
_defineProperty(_this, "accessibilityManager", new AccessibilityManager());
|
|
_defineProperty(_this, "handleLegendBBoxUpdate", function (box) {
|
|
if (box) {
|
|
var _this$state = _this.state,
|
|
dataStartIndex = _this$state.dataStartIndex,
|
|
dataEndIndex = _this$state.dataEndIndex,
|
|
updateId = _this$state.updateId;
|
|
_this.setState(_objectSpread({
|
|
legendBBox: box
|
|
}, updateStateOfAxisMapsOffsetAndStackGroups({
|
|
props: _this.props,
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex,
|
|
updateId: updateId
|
|
}, _objectSpread(_objectSpread({}, _this.state), {}, {
|
|
legendBBox: box
|
|
}))));
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleReceiveSyncEvent", function (cId, data, emitter) {
|
|
if (_this.props.syncId === cId) {
|
|
if (emitter === _this.eventEmitterSymbol && typeof _this.props.syncMethod !== 'function') {
|
|
return;
|
|
}
|
|
_this.applySyncEvent(data);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleBrushChange", function (_ref9) {
|
|
var startIndex = _ref9.startIndex,
|
|
endIndex = _ref9.endIndex;
|
|
// Only trigger changes if the extents of the brush have actually changed
|
|
if (startIndex !== _this.state.dataStartIndex || endIndex !== _this.state.dataEndIndex) {
|
|
var updateId = _this.state.updateId;
|
|
_this.setState(function () {
|
|
return _objectSpread({
|
|
dataStartIndex: startIndex,
|
|
dataEndIndex: endIndex
|
|
}, updateStateOfAxisMapsOffsetAndStackGroups({
|
|
props: _this.props,
|
|
dataStartIndex: startIndex,
|
|
dataEndIndex: endIndex,
|
|
updateId: updateId
|
|
}, _this.state));
|
|
});
|
|
_this.triggerSyncEvent({
|
|
dataStartIndex: startIndex,
|
|
dataEndIndex: endIndex
|
|
});
|
|
}
|
|
});
|
|
/**
|
|
* The handler of mouse entering chart
|
|
* @param {Object} e Event object
|
|
* @return {Null} null
|
|
*/
|
|
_defineProperty(_this, "handleMouseEnter", function (e) {
|
|
var mouse = _this.getMouseInfo(e);
|
|
if (mouse) {
|
|
var _nextState = _objectSpread(_objectSpread({}, mouse), {}, {
|
|
isTooltipActive: true
|
|
});
|
|
_this.setState(_nextState);
|
|
_this.triggerSyncEvent(_nextState);
|
|
var onMouseEnter = _this.props.onMouseEnter;
|
|
if (isFunction(onMouseEnter)) {
|
|
onMouseEnter(_nextState, e);
|
|
}
|
|
}
|
|
});
|
|
_defineProperty(_this, "triggeredAfterMouseMove", function (e) {
|
|
var mouse = _this.getMouseInfo(e);
|
|
var nextState = mouse ? _objectSpread(_objectSpread({}, mouse), {}, {
|
|
isTooltipActive: true
|
|
}) : {
|
|
isTooltipActive: false
|
|
};
|
|
_this.setState(nextState);
|
|
_this.triggerSyncEvent(nextState);
|
|
var onMouseMove = _this.props.onMouseMove;
|
|
if (isFunction(onMouseMove)) {
|
|
onMouseMove(nextState, e);
|
|
}
|
|
});
|
|
/**
|
|
* The handler of mouse entering a scatter
|
|
* @param {Object} el The active scatter
|
|
* @return {Object} no return
|
|
*/
|
|
_defineProperty(_this, "handleItemMouseEnter", function (el) {
|
|
_this.setState(function () {
|
|
return {
|
|
isTooltipActive: true,
|
|
activeItem: el,
|
|
activePayload: el.tooltipPayload,
|
|
activeCoordinate: el.tooltipPosition || {
|
|
x: el.cx,
|
|
y: el.cy
|
|
}
|
|
};
|
|
});
|
|
});
|
|
/**
|
|
* The handler of mouse leaving a scatter
|
|
* @return {Object} no return
|
|
*/
|
|
_defineProperty(_this, "handleItemMouseLeave", function () {
|
|
_this.setState(function () {
|
|
return {
|
|
isTooltipActive: false
|
|
};
|
|
});
|
|
});
|
|
/**
|
|
* The handler of mouse moving in chart
|
|
* @param {React.MouseEvent} e Event object
|
|
* @return {void} no return
|
|
*/
|
|
_defineProperty(_this, "handleMouseMove", function (e) {
|
|
e.persist();
|
|
_this.throttleTriggeredAfterMouseMove(e);
|
|
});
|
|
/**
|
|
* The handler if mouse leaving chart
|
|
* @param {Object} e Event object
|
|
* @return {Null} no return
|
|
*/
|
|
_defineProperty(_this, "handleMouseLeave", function (e) {
|
|
_this.throttleTriggeredAfterMouseMove.cancel();
|
|
var nextState = {
|
|
isTooltipActive: false
|
|
};
|
|
_this.setState(nextState);
|
|
_this.triggerSyncEvent(nextState);
|
|
var onMouseLeave = _this.props.onMouseLeave;
|
|
if (isFunction(onMouseLeave)) {
|
|
onMouseLeave(nextState, e);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleOuterEvent", function (e) {
|
|
var eventName = getReactEventByType(e);
|
|
var event = get(_this.props, "".concat(eventName));
|
|
if (eventName && isFunction(event)) {
|
|
var _mouse;
|
|
var mouse;
|
|
if (/.*touch.*/i.test(eventName)) {
|
|
mouse = _this.getMouseInfo(e.changedTouches[0]);
|
|
} else {
|
|
mouse = _this.getMouseInfo(e);
|
|
}
|
|
event((_mouse = mouse) !== null && _mouse !== void 0 ? _mouse : {}, e);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleClick", function (e) {
|
|
var mouse = _this.getMouseInfo(e);
|
|
if (mouse) {
|
|
var _nextState2 = _objectSpread(_objectSpread({}, mouse), {}, {
|
|
isTooltipActive: true
|
|
});
|
|
_this.setState(_nextState2);
|
|
_this.triggerSyncEvent(_nextState2);
|
|
var onClick = _this.props.onClick;
|
|
if (isFunction(onClick)) {
|
|
onClick(_nextState2, e);
|
|
}
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleMouseDown", function (e) {
|
|
var onMouseDown = _this.props.onMouseDown;
|
|
if (isFunction(onMouseDown)) {
|
|
var _nextState3 = _this.getMouseInfo(e);
|
|
onMouseDown(_nextState3, e);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleMouseUp", function (e) {
|
|
var onMouseUp = _this.props.onMouseUp;
|
|
if (isFunction(onMouseUp)) {
|
|
var _nextState4 = _this.getMouseInfo(e);
|
|
onMouseUp(_nextState4, e);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleTouchMove", function (e) {
|
|
if (e.changedTouches != null && e.changedTouches.length > 0) {
|
|
_this.throttleTriggeredAfterMouseMove(e.changedTouches[0]);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleTouchStart", function (e) {
|
|
if (e.changedTouches != null && e.changedTouches.length > 0) {
|
|
_this.handleMouseDown(e.changedTouches[0]);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleTouchEnd", function (e) {
|
|
if (e.changedTouches != null && e.changedTouches.length > 0) {
|
|
_this.handleMouseUp(e.changedTouches[0]);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleDoubleClick", function (e) {
|
|
var onDoubleClick = _this.props.onDoubleClick;
|
|
if (isFunction(onDoubleClick)) {
|
|
var _nextState5 = _this.getMouseInfo(e);
|
|
onDoubleClick(_nextState5, e);
|
|
}
|
|
});
|
|
_defineProperty(_this, "handleContextMenu", function (e) {
|
|
var onContextMenu = _this.props.onContextMenu;
|
|
if (isFunction(onContextMenu)) {
|
|
var _nextState6 = _this.getMouseInfo(e);
|
|
onContextMenu(_nextState6, e);
|
|
}
|
|
});
|
|
_defineProperty(_this, "triggerSyncEvent", function (data) {
|
|
if (_this.props.syncId !== undefined) {
|
|
eventCenter.emit(SYNC_EVENT, _this.props.syncId, data, _this.eventEmitterSymbol);
|
|
}
|
|
});
|
|
_defineProperty(_this, "applySyncEvent", function (data) {
|
|
var _this$props = _this.props,
|
|
layout = _this$props.layout,
|
|
syncMethod = _this$props.syncMethod;
|
|
var updateId = _this.state.updateId;
|
|
var dataStartIndex = data.dataStartIndex,
|
|
dataEndIndex = data.dataEndIndex;
|
|
if (data.dataStartIndex !== undefined || data.dataEndIndex !== undefined) {
|
|
_this.setState(_objectSpread({
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex
|
|
}, updateStateOfAxisMapsOffsetAndStackGroups({
|
|
props: _this.props,
|
|
dataStartIndex: dataStartIndex,
|
|
dataEndIndex: dataEndIndex,
|
|
updateId: updateId
|
|
}, _this.state)));
|
|
} else if (data.activeTooltipIndex !== undefined) {
|
|
var chartX = data.chartX,
|
|
chartY = data.chartY;
|
|
var activeTooltipIndex = data.activeTooltipIndex;
|
|
var _this$state2 = _this.state,
|
|
offset = _this$state2.offset,
|
|
tooltipTicks = _this$state2.tooltipTicks;
|
|
if (!offset) {
|
|
return;
|
|
}
|
|
if (typeof syncMethod === 'function') {
|
|
// Call a callback function. If there is an application specific algorithm
|
|
activeTooltipIndex = syncMethod(tooltipTicks, data);
|
|
} else if (syncMethod === 'value') {
|
|
// Set activeTooltipIndex to the index with the same value as data.activeLabel
|
|
// For loop instead of findIndex because the latter is very slow in some browsers
|
|
activeTooltipIndex = -1; // in case we cannot find the element
|
|
for (var i = 0; i < tooltipTicks.length; i++) {
|
|
if (tooltipTicks[i].value === data.activeLabel) {
|
|
activeTooltipIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
var viewBox = _objectSpread(_objectSpread({}, offset), {}, {
|
|
x: offset.left,
|
|
y: offset.top
|
|
});
|
|
// When a categorical chart is combined with another chart, the value of chartX
|
|
// and chartY may beyond the boundaries.
|
|
var validateChartX = Math.min(chartX, viewBox.x + viewBox.width);
|
|
var validateChartY = Math.min(chartY, viewBox.y + viewBox.height);
|
|
var activeLabel = tooltipTicks[activeTooltipIndex] && tooltipTicks[activeTooltipIndex].value;
|
|
var activePayload = getTooltipContent(_this.state, _this.props.data, activeTooltipIndex);
|
|
var activeCoordinate = tooltipTicks[activeTooltipIndex] ? {
|
|
x: layout === 'horizontal' ? tooltipTicks[activeTooltipIndex].coordinate : validateChartX,
|
|
y: layout === 'horizontal' ? validateChartY : tooltipTicks[activeTooltipIndex].coordinate
|
|
} : originCoordinate;
|
|
_this.setState(_objectSpread(_objectSpread({}, data), {}, {
|
|
activeLabel: activeLabel,
|
|
activeCoordinate: activeCoordinate,
|
|
activePayload: activePayload,
|
|
activeTooltipIndex: activeTooltipIndex
|
|
}));
|
|
} else {
|
|
_this.setState(data);
|
|
}
|
|
});
|
|
_defineProperty(_this, "renderCursor", function (element) {
|
|
var _element$props$active;
|
|
var _this$state3 = _this.state,
|
|
isTooltipActive = _this$state3.isTooltipActive,
|
|
activeCoordinate = _this$state3.activeCoordinate,
|
|
activePayload = _this$state3.activePayload,
|
|
offset = _this$state3.offset,
|
|
activeTooltipIndex = _this$state3.activeTooltipIndex,
|
|
tooltipAxisBandSize = _this$state3.tooltipAxisBandSize;
|
|
var tooltipEventType = _this.getTooltipEventType();
|
|
// The cursor is a part of the Tooltip, and it should be shown (by default) when the Tooltip is active.
|
|
var isActive = (_element$props$active = element.props.active) !== null && _element$props$active !== void 0 ? _element$props$active : isTooltipActive;
|
|
var layout = _this.props.layout;
|
|
var key = element.key || '_recharts-cursor';
|
|
return /*#__PURE__*/React.createElement(Cursor, {
|
|
key: key,
|
|
activeCoordinate: activeCoordinate,
|
|
activePayload: activePayload,
|
|
activeTooltipIndex: activeTooltipIndex,
|
|
chartName: chartName,
|
|
element: element,
|
|
isActive: isActive,
|
|
layout: layout,
|
|
offset: offset,
|
|
tooltipAxisBandSize: tooltipAxisBandSize,
|
|
tooltipEventType: tooltipEventType
|
|
});
|
|
});
|
|
_defineProperty(_this, "renderPolarAxis", function (element, displayName, index) {
|
|
var axisType = get(element, 'type.axisType');
|
|
var axisMap = get(_this.state, "".concat(axisType, "Map"));
|
|
var elementDefaultProps = element.type.defaultProps;
|
|
var elementProps = elementDefaultProps !== undefined ? _objectSpread(_objectSpread({}, elementDefaultProps), element.props) : element.props;
|
|
var axisOption = axisMap && axisMap[elementProps["".concat(axisType, "Id")]];
|
|
return /*#__PURE__*/cloneElement(element, _objectSpread(_objectSpread({}, axisOption), {}, {
|
|
className: clsx(axisType, axisOption.className),
|
|
key: element.key || "".concat(displayName, "-").concat(index),
|
|
ticks: getTicksOfAxis(axisOption, true)
|
|
}));
|
|
});
|
|
_defineProperty(_this, "renderPolarGrid", function (element) {
|
|
var _element$props = element.props,
|
|
radialLines = _element$props.radialLines,
|
|
polarAngles = _element$props.polarAngles,
|
|
polarRadius = _element$props.polarRadius;
|
|
var _this$state4 = _this.state,
|
|
radiusAxisMap = _this$state4.radiusAxisMap,
|
|
angleAxisMap = _this$state4.angleAxisMap;
|
|
var radiusAxis = getAnyElementOfObject(radiusAxisMap);
|
|
var angleAxis = getAnyElementOfObject(angleAxisMap);
|
|
var cx = angleAxis.cx,
|
|
cy = angleAxis.cy,
|
|
innerRadius = angleAxis.innerRadius,
|
|
outerRadius = angleAxis.outerRadius;
|
|
return /*#__PURE__*/cloneElement(element, {
|
|
polarAngles: Array.isArray(polarAngles) ? polarAngles : getTicksOfAxis(angleAxis, true).map(function (entry) {
|
|
return entry.coordinate;
|
|
}),
|
|
polarRadius: Array.isArray(polarRadius) ? polarRadius : getTicksOfAxis(radiusAxis, true).map(function (entry) {
|
|
return entry.coordinate;
|
|
}),
|
|
cx: cx,
|
|
cy: cy,
|
|
innerRadius: innerRadius,
|
|
outerRadius: outerRadius,
|
|
key: element.key || 'polar-grid',
|
|
radialLines: radialLines
|
|
});
|
|
});
|
|
/**
|
|
* Draw legend
|
|
* @return {ReactElement} The instance of Legend
|
|
*/
|
|
_defineProperty(_this, "renderLegend", function () {
|
|
var formattedGraphicalItems = _this.state.formattedGraphicalItems;
|
|
var _this$props2 = _this.props,
|
|
children = _this$props2.children,
|
|
width = _this$props2.width,
|
|
height = _this$props2.height;
|
|
var margin = _this.props.margin || {};
|
|
var legendWidth = width - (margin.left || 0) - (margin.right || 0);
|
|
var props = getLegendProps({
|
|
children: children,
|
|
formattedGraphicalItems: formattedGraphicalItems,
|
|
legendWidth: legendWidth,
|
|
legendContent: legendContent
|
|
});
|
|
if (!props) {
|
|
return null;
|
|
}
|
|
var item = props.item,
|
|
otherProps = _objectWithoutProperties(props, _excluded);
|
|
return /*#__PURE__*/cloneElement(item, _objectSpread(_objectSpread({}, otherProps), {}, {
|
|
chartWidth: width,
|
|
chartHeight: height,
|
|
margin: margin,
|
|
onBBoxUpdate: _this.handleLegendBBoxUpdate
|
|
}));
|
|
});
|
|
/**
|
|
* Draw Tooltip
|
|
* @return {ReactElement} The instance of Tooltip
|
|
*/
|
|
_defineProperty(_this, "renderTooltip", function () {
|
|
var _tooltipItem$props$ac;
|
|
var _this$props3 = _this.props,
|
|
children = _this$props3.children,
|
|
accessibilityLayer = _this$props3.accessibilityLayer;
|
|
var tooltipItem = findChildByType(children, Tooltip);
|
|
if (!tooltipItem) {
|
|
return null;
|
|
}
|
|
var _this$state5 = _this.state,
|
|
isTooltipActive = _this$state5.isTooltipActive,
|
|
activeCoordinate = _this$state5.activeCoordinate,
|
|
activePayload = _this$state5.activePayload,
|
|
activeLabel = _this$state5.activeLabel,
|
|
offset = _this$state5.offset;
|
|
|
|
// The user can set isActive on the Tooltip,
|
|
// and we respect the user to enable customisation.
|
|
// The Tooltip is active if the user has set isActive, or if the tooltip is active due to a mouse event.
|
|
var isActive = (_tooltipItem$props$ac = tooltipItem.props.active) !== null && _tooltipItem$props$ac !== void 0 ? _tooltipItem$props$ac : isTooltipActive;
|
|
return /*#__PURE__*/cloneElement(tooltipItem, {
|
|
viewBox: _objectSpread(_objectSpread({}, offset), {}, {
|
|
x: offset.left,
|
|
y: offset.top
|
|
}),
|
|
active: isActive,
|
|
label: activeLabel,
|
|
payload: isActive ? activePayload : [],
|
|
coordinate: activeCoordinate,
|
|
accessibilityLayer: accessibilityLayer
|
|
});
|
|
});
|
|
_defineProperty(_this, "renderBrush", function (element) {
|
|
var _this$props4 = _this.props,
|
|
margin = _this$props4.margin,
|
|
data = _this$props4.data;
|
|
var _this$state6 = _this.state,
|
|
offset = _this$state6.offset,
|
|
dataStartIndex = _this$state6.dataStartIndex,
|
|
dataEndIndex = _this$state6.dataEndIndex,
|
|
updateId = _this$state6.updateId;
|
|
|
|
// TODO: update brush when children update
|
|
return /*#__PURE__*/cloneElement(element, {
|
|
key: element.key || '_recharts-brush',
|
|
onChange: combineEventHandlers(_this.handleBrushChange, element.props.onChange),
|
|
data: data,
|
|
x: isNumber(element.props.x) ? element.props.x : offset.left,
|
|
y: isNumber(element.props.y) ? element.props.y : offset.top + offset.height + offset.brushBottom - (margin.bottom || 0),
|
|
width: isNumber(element.props.width) ? element.props.width : offset.width,
|
|
startIndex: dataStartIndex,
|
|
endIndex: dataEndIndex,
|
|
updateId: "brush-".concat(updateId)
|
|
});
|
|
});
|
|
_defineProperty(_this, "renderReferenceElement", function (element, displayName, index) {
|
|
if (!element) {
|
|
return null;
|
|
}
|
|
var _this2 = _this,
|
|
clipPathId = _this2.clipPathId;
|
|
var _this$state7 = _this.state,
|
|
xAxisMap = _this$state7.xAxisMap,
|
|
yAxisMap = _this$state7.yAxisMap,
|
|
offset = _this$state7.offset;
|
|
var elementDefaultProps = element.type.defaultProps || {};
|
|
var _element$props2 = element.props,
|
|
_element$props2$xAxis = _element$props2.xAxisId,
|
|
xAxisId = _element$props2$xAxis === void 0 ? elementDefaultProps.xAxisId : _element$props2$xAxis,
|
|
_element$props2$yAxis = _element$props2.yAxisId,
|
|
yAxisId = _element$props2$yAxis === void 0 ? elementDefaultProps.yAxisId : _element$props2$yAxis;
|
|
return /*#__PURE__*/cloneElement(element, {
|
|
key: element.key || "".concat(displayName, "-").concat(index),
|
|
xAxis: xAxisMap[xAxisId],
|
|
yAxis: yAxisMap[yAxisId],
|
|
viewBox: {
|
|
x: offset.left,
|
|
y: offset.top,
|
|
width: offset.width,
|
|
height: offset.height
|
|
},
|
|
clipPathId: clipPathId
|
|
});
|
|
});
|
|
_defineProperty(_this, "renderActivePoints", function (_ref10) {
|
|
var item = _ref10.item,
|
|
activePoint = _ref10.activePoint,
|
|
basePoint = _ref10.basePoint,
|
|
childIndex = _ref10.childIndex,
|
|
isRange = _ref10.isRange;
|
|
var result = [];
|
|
// item is not a React Element so we don't need to resolve defaultProps.
|
|
var key = item.props.key;
|
|
var itemItemProps = item.item.type.defaultProps !== undefined ? _objectSpread(_objectSpread({}, item.item.type.defaultProps), item.item.props) : item.item.props;
|
|
var activeDot = itemItemProps.activeDot,
|
|
dataKey = itemItemProps.dataKey;
|
|
var dotProps = _objectSpread(_objectSpread({
|
|
index: childIndex,
|
|
dataKey: dataKey,
|
|
cx: activePoint.x,
|
|
cy: activePoint.y,
|
|
r: 4,
|
|
fill: getMainColorOfGraphicItem(item.item),
|
|
strokeWidth: 2,
|
|
stroke: '#fff',
|
|
payload: activePoint.payload,
|
|
value: activePoint.value
|
|
}, filterProps(activeDot, false)), adaptEventHandlers(activeDot));
|
|
result.push(CategoricalChartWrapper.renderActiveDot(activeDot, dotProps, "".concat(key, "-activePoint-").concat(childIndex)));
|
|
if (basePoint) {
|
|
result.push(CategoricalChartWrapper.renderActiveDot(activeDot, _objectSpread(_objectSpread({}, dotProps), {}, {
|
|
cx: basePoint.x,
|
|
cy: basePoint.y
|
|
}), "".concat(key, "-basePoint-").concat(childIndex)));
|
|
} else if (isRange) {
|
|
result.push(null);
|
|
}
|
|
return result;
|
|
});
|
|
_defineProperty(_this, "renderGraphicChild", function (element, displayName, index) {
|
|
var item = _this.filterFormatItem(element, displayName, index);
|
|
if (!item) {
|
|
return null;
|
|
}
|
|
var tooltipEventType = _this.getTooltipEventType();
|
|
var _this$state8 = _this.state,
|
|
isTooltipActive = _this$state8.isTooltipActive,
|
|
tooltipAxis = _this$state8.tooltipAxis,
|
|
activeTooltipIndex = _this$state8.activeTooltipIndex,
|
|
activeLabel = _this$state8.activeLabel;
|
|
var children = _this.props.children;
|
|
var tooltipItem = findChildByType(children, Tooltip);
|
|
// item is not a React Element so we don't need to resolve defaultProps
|
|
var _item$props = item.props,
|
|
points = _item$props.points,
|
|
isRange = _item$props.isRange,
|
|
baseLine = _item$props.baseLine;
|
|
var itemItemProps = item.item.type.defaultProps !== undefined ? _objectSpread(_objectSpread({}, item.item.type.defaultProps), item.item.props) : item.item.props;
|
|
var activeDot = itemItemProps.activeDot,
|
|
hide = itemItemProps.hide,
|
|
activeBar = itemItemProps.activeBar,
|
|
activeShape = itemItemProps.activeShape;
|
|
var hasActive = Boolean(!hide && isTooltipActive && tooltipItem && (activeDot || activeBar || activeShape));
|
|
var itemEvents = {};
|
|
if (tooltipEventType !== 'axis' && tooltipItem && tooltipItem.props.trigger === 'click') {
|
|
itemEvents = {
|
|
onClick: combineEventHandlers(_this.handleItemMouseEnter, element.props.onClick)
|
|
};
|
|
} else if (tooltipEventType !== 'axis') {
|
|
itemEvents = {
|
|
onMouseLeave: combineEventHandlers(_this.handleItemMouseLeave, element.props.onMouseLeave),
|
|
onMouseEnter: combineEventHandlers(_this.handleItemMouseEnter, element.props.onMouseEnter)
|
|
};
|
|
}
|
|
var graphicalItem = /*#__PURE__*/cloneElement(element, _objectSpread(_objectSpread({}, item.props), itemEvents));
|
|
function findWithPayload(entry) {
|
|
// TODO needs to verify dataKey is Function
|
|
return typeof tooltipAxis.dataKey === 'function' ? tooltipAxis.dataKey(entry.payload) : null;
|
|
}
|
|
if (hasActive) {
|
|
if (activeTooltipIndex >= 0) {
|
|
var activePoint, basePoint;
|
|
if (tooltipAxis.dataKey && !tooltipAxis.allowDuplicatedCategory) {
|
|
// number transform to string
|
|
var specifiedKey = typeof tooltipAxis.dataKey === 'function' ? findWithPayload : 'payload.'.concat(tooltipAxis.dataKey.toString());
|
|
activePoint = findEntryInArray(points, specifiedKey, activeLabel);
|
|
basePoint = isRange && baseLine && findEntryInArray(baseLine, specifiedKey, activeLabel);
|
|
} else {
|
|
activePoint = points === null || points === void 0 ? void 0 : points[activeTooltipIndex];
|
|
basePoint = isRange && baseLine && baseLine[activeTooltipIndex];
|
|
}
|
|
if (activeShape || activeBar) {
|
|
var activeIndex = element.props.activeIndex !== undefined ? element.props.activeIndex : activeTooltipIndex;
|
|
return [/*#__PURE__*/cloneElement(element, _objectSpread(_objectSpread(_objectSpread({}, item.props), itemEvents), {}, {
|
|
activeIndex: activeIndex
|
|
})), null, null];
|
|
}
|
|
if (!isNil(activePoint)) {
|
|
return [graphicalItem].concat(_toConsumableArray(_this.renderActivePoints({
|
|
item: item,
|
|
activePoint: activePoint,
|
|
basePoint: basePoint,
|
|
childIndex: activeTooltipIndex,
|
|
isRange: isRange
|
|
})));
|
|
}
|
|
} else {
|
|
var _this$getItemByXY;
|
|
/**
|
|
* We hit this block if consumer uses a Tooltip without XAxis and/or YAxis.
|
|
* In which case, this.state.activeTooltipIndex never gets set
|
|
* because the mouse events that trigger that value getting set never get trigged without the axis components.
|
|
*
|
|
* An example usage case is a FunnelChart
|
|
*/
|
|
var _ref11 = (_this$getItemByXY = _this.getItemByXY(_this.state.activeCoordinate)) !== null && _this$getItemByXY !== void 0 ? _this$getItemByXY : {
|
|
graphicalItem: graphicalItem
|
|
},
|
|
_ref11$graphicalItem = _ref11.graphicalItem,
|
|
_ref11$graphicalItem$ = _ref11$graphicalItem.item,
|
|
xyItem = _ref11$graphicalItem$ === void 0 ? element : _ref11$graphicalItem$,
|
|
childIndex = _ref11$graphicalItem.childIndex;
|
|
var elementProps = _objectSpread(_objectSpread(_objectSpread({}, item.props), itemEvents), {}, {
|
|
activeIndex: childIndex
|
|
});
|
|
return [/*#__PURE__*/cloneElement(xyItem, elementProps), null, null];
|
|
}
|
|
}
|
|
if (isRange) {
|
|
return [graphicalItem, null, null];
|
|
}
|
|
return [graphicalItem, null];
|
|
});
|
|
_defineProperty(_this, "renderCustomized", function (element, displayName, index) {
|
|
return /*#__PURE__*/cloneElement(element, _objectSpread(_objectSpread({
|
|
key: "recharts-customized-".concat(index)
|
|
}, _this.props), _this.state));
|
|
});
|
|
_defineProperty(_this, "renderMap", {
|
|
CartesianGrid: {
|
|
handler: renderAsIs,
|
|
once: true
|
|
},
|
|
ReferenceArea: {
|
|
handler: _this.renderReferenceElement
|
|
},
|
|
ReferenceLine: {
|
|
handler: renderAsIs
|
|
},
|
|
ReferenceDot: {
|
|
handler: _this.renderReferenceElement
|
|
},
|
|
XAxis: {
|
|
handler: renderAsIs
|
|
},
|
|
YAxis: {
|
|
handler: renderAsIs
|
|
},
|
|
Brush: {
|
|
handler: _this.renderBrush,
|
|
once: true
|
|
},
|
|
Bar: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
Line: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
Area: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
Radar: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
RadialBar: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
Scatter: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
Pie: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
Funnel: {
|
|
handler: _this.renderGraphicChild
|
|
},
|
|
Tooltip: {
|
|
handler: _this.renderCursor,
|
|
once: true
|
|
},
|
|
PolarGrid: {
|
|
handler: _this.renderPolarGrid,
|
|
once: true
|
|
},
|
|
PolarAngleAxis: {
|
|
handler: _this.renderPolarAxis
|
|
},
|
|
PolarRadiusAxis: {
|
|
handler: _this.renderPolarAxis
|
|
},
|
|
Customized: {
|
|
handler: _this.renderCustomized
|
|
}
|
|
});
|
|
_this.clipPathId = "".concat((_props$id = _props.id) !== null && _props$id !== void 0 ? _props$id : uniqueId('recharts'), "-clip");
|
|
|
|
// trigger 60fps
|
|
_this.throttleTriggeredAfterMouseMove = throttle(_this.triggeredAfterMouseMove, (_props$throttleDelay = _props.throttleDelay) !== null && _props$throttleDelay !== void 0 ? _props$throttleDelay : 1000 / 60);
|
|
_this.state = {};
|
|
return _this;
|
|
}
|
|
_inherits(CategoricalChartWrapper, _Component);
|
|
return _createClass(CategoricalChartWrapper, [{
|
|
key: "componentDidMount",
|
|
value: function componentDidMount() {
|
|
var _this$props$margin$le, _this$props$margin$to;
|
|
this.addListener();
|
|
this.accessibilityManager.setDetails({
|
|
container: this.container,
|
|
offset: {
|
|
left: (_this$props$margin$le = this.props.margin.left) !== null && _this$props$margin$le !== void 0 ? _this$props$margin$le : 0,
|
|
top: (_this$props$margin$to = this.props.margin.top) !== null && _this$props$margin$to !== void 0 ? _this$props$margin$to : 0
|
|
},
|
|
coordinateList: this.state.tooltipTicks,
|
|
mouseHandlerCallback: this.triggeredAfterMouseMove,
|
|
layout: this.props.layout
|
|
});
|
|
this.displayDefaultTooltip();
|
|
}
|
|
}, {
|
|
key: "displayDefaultTooltip",
|
|
value: function displayDefaultTooltip() {
|
|
var _this$props5 = this.props,
|
|
children = _this$props5.children,
|
|
data = _this$props5.data,
|
|
height = _this$props5.height,
|
|
layout = _this$props5.layout;
|
|
var tooltipElem = findChildByType(children, Tooltip);
|
|
// If the chart doesn't include a <Tooltip /> element, there's no tooltip to display
|
|
if (!tooltipElem) {
|
|
return;
|
|
}
|
|
var defaultIndex = tooltipElem.props.defaultIndex;
|
|
|
|
// Protect against runtime errors
|
|
if (typeof defaultIndex !== 'number' || defaultIndex < 0 || defaultIndex > this.state.tooltipTicks.length - 1) {
|
|
return;
|
|
}
|
|
var activeLabel = this.state.tooltipTicks[defaultIndex] && this.state.tooltipTicks[defaultIndex].value;
|
|
var activePayload = getTooltipContent(this.state, data, defaultIndex, activeLabel);
|
|
var independentAxisCoord = this.state.tooltipTicks[defaultIndex].coordinate;
|
|
var dependentAxisCoord = (this.state.offset.top + height) / 2;
|
|
var isHorizontal = layout === 'horizontal';
|
|
var activeCoordinate = isHorizontal ? {
|
|
x: independentAxisCoord,
|
|
y: dependentAxisCoord
|
|
} : {
|
|
y: independentAxisCoord,
|
|
x: dependentAxisCoord
|
|
};
|
|
|
|
// Unlike other chart types, scatter plot's tooltip positions rely on both X and Y coordinates. Only the scatter plot
|
|
// element knows its own Y coordinates.
|
|
// If there's a scatter plot, we'll want to grab that element for an interrogation.
|
|
var scatterPlotElement = this.state.formattedGraphicalItems.find(function (_ref12) {
|
|
var item = _ref12.item;
|
|
return item.type.name === 'Scatter';
|
|
});
|
|
if (scatterPlotElement) {
|
|
activeCoordinate = _objectSpread(_objectSpread({}, activeCoordinate), scatterPlotElement.props.points[defaultIndex].tooltipPosition);
|
|
activePayload = scatterPlotElement.props.points[defaultIndex].tooltipPayload;
|
|
}
|
|
var nextState = {
|
|
activeTooltipIndex: defaultIndex,
|
|
isTooltipActive: true,
|
|
activeLabel: activeLabel,
|
|
activePayload: activePayload,
|
|
activeCoordinate: activeCoordinate
|
|
};
|
|
this.setState(nextState);
|
|
this.renderCursor(tooltipElem);
|
|
|
|
// Make sure that anyone who keyboard-only users who tab to the chart will start their
|
|
// cursors at defaultIndex
|
|
this.accessibilityManager.setIndex(defaultIndex);
|
|
}
|
|
}, {
|
|
key: "getSnapshotBeforeUpdate",
|
|
value: function getSnapshotBeforeUpdate(prevProps, prevState) {
|
|
if (!this.props.accessibilityLayer) {
|
|
return null;
|
|
}
|
|
if (this.state.tooltipTicks !== prevState.tooltipTicks) {
|
|
this.accessibilityManager.setDetails({
|
|
coordinateList: this.state.tooltipTicks
|
|
});
|
|
}
|
|
if (this.props.layout !== prevProps.layout) {
|
|
this.accessibilityManager.setDetails({
|
|
layout: this.props.layout
|
|
});
|
|
}
|
|
if (this.props.margin !== prevProps.margin) {
|
|
var _this$props$margin$le2, _this$props$margin$to2;
|
|
this.accessibilityManager.setDetails({
|
|
offset: {
|
|
left: (_this$props$margin$le2 = this.props.margin.left) !== null && _this$props$margin$le2 !== void 0 ? _this$props$margin$le2 : 0,
|
|
top: (_this$props$margin$to2 = this.props.margin.top) !== null && _this$props$margin$to2 !== void 0 ? _this$props$margin$to2 : 0
|
|
}
|
|
});
|
|
}
|
|
|
|
// Something has to be returned for getSnapshotBeforeUpdate
|
|
return null;
|
|
}
|
|
}, {
|
|
key: "componentDidUpdate",
|
|
value: function componentDidUpdate(prevProps) {
|
|
// Check to see if the Tooltip updated. If so, re-check default tooltip position
|
|
if (!isChildrenEqual([findChildByType(prevProps.children, Tooltip)], [findChildByType(this.props.children, Tooltip)])) {
|
|
this.displayDefaultTooltip();
|
|
}
|
|
}
|
|
}, {
|
|
key: "componentWillUnmount",
|
|
value: function componentWillUnmount() {
|
|
this.removeListener();
|
|
this.throttleTriggeredAfterMouseMove.cancel();
|
|
}
|
|
}, {
|
|
key: "getTooltipEventType",
|
|
value: function getTooltipEventType() {
|
|
var tooltipItem = findChildByType(this.props.children, Tooltip);
|
|
if (tooltipItem && typeof tooltipItem.props.shared === 'boolean') {
|
|
var eventType = tooltipItem.props.shared ? 'axis' : 'item';
|
|
return validateTooltipEventTypes.indexOf(eventType) >= 0 ? eventType : defaultTooltipEventType;
|
|
}
|
|
return defaultTooltipEventType;
|
|
}
|
|
|
|
/**
|
|
* Get the information of mouse in chart, return null when the mouse is not in the chart
|
|
* @param {MousePointer} event The event object
|
|
* @return {Object} Mouse data
|
|
*/
|
|
}, {
|
|
key: "getMouseInfo",
|
|
value: function getMouseInfo(event) {
|
|
if (!this.container) {
|
|
return null;
|
|
}
|
|
var element = this.container;
|
|
var boundingRect = element.getBoundingClientRect();
|
|
var containerOffset = getOffset(boundingRect);
|
|
var e = {
|
|
chartX: Math.round(event.pageX - containerOffset.left),
|
|
chartY: Math.round(event.pageY - containerOffset.top)
|
|
};
|
|
var scale = boundingRect.width / element.offsetWidth || 1;
|
|
var rangeObj = this.inRange(e.chartX, e.chartY, scale);
|
|
if (!rangeObj) {
|
|
return null;
|
|
}
|
|
var _this$state9 = this.state,
|
|
xAxisMap = _this$state9.xAxisMap,
|
|
yAxisMap = _this$state9.yAxisMap;
|
|
var tooltipEventType = this.getTooltipEventType();
|
|
var toolTipData = getTooltipData(this.state, this.props.data, this.props.layout, rangeObj);
|
|
if (tooltipEventType !== 'axis' && xAxisMap && yAxisMap) {
|
|
var xScale = getAnyElementOfObject(xAxisMap).scale;
|
|
var yScale = getAnyElementOfObject(yAxisMap).scale;
|
|
var xValue = xScale && xScale.invert ? xScale.invert(e.chartX) : null;
|
|
var yValue = yScale && yScale.invert ? yScale.invert(e.chartY) : null;
|
|
return _objectSpread(_objectSpread({}, e), {}, {
|
|
xValue: xValue,
|
|
yValue: yValue
|
|
}, toolTipData);
|
|
}
|
|
if (toolTipData) {
|
|
return _objectSpread(_objectSpread({}, e), toolTipData);
|
|
}
|
|
return null;
|
|
}
|
|
}, {
|
|
key: "inRange",
|
|
value: function inRange(x, y) {
|
|
var scale = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 1;
|
|
var layout = this.props.layout;
|
|
var scaledX = x / scale,
|
|
scaledY = y / scale;
|
|
if (layout === 'horizontal' || layout === 'vertical') {
|
|
var offset = this.state.offset;
|
|
var isInRange = scaledX >= offset.left && scaledX <= offset.left + offset.width && scaledY >= offset.top && scaledY <= offset.top + offset.height;
|
|
return isInRange ? {
|
|
x: scaledX,
|
|
y: scaledY
|
|
} : null;
|
|
}
|
|
var _this$state10 = this.state,
|
|
angleAxisMap = _this$state10.angleAxisMap,
|
|
radiusAxisMap = _this$state10.radiusAxisMap;
|
|
if (angleAxisMap && radiusAxisMap) {
|
|
var angleAxis = getAnyElementOfObject(angleAxisMap);
|
|
return inRangeOfSector({
|
|
x: scaledX,
|
|
y: scaledY
|
|
}, angleAxis);
|
|
}
|
|
return null;
|
|
}
|
|
}, {
|
|
key: "parseEventsOfWrapper",
|
|
value: function parseEventsOfWrapper() {
|
|
var children = this.props.children;
|
|
var tooltipEventType = this.getTooltipEventType();
|
|
var tooltipItem = findChildByType(children, Tooltip);
|
|
var tooltipEvents = {};
|
|
if (tooltipItem && tooltipEventType === 'axis') {
|
|
if (tooltipItem.props.trigger === 'click') {
|
|
tooltipEvents = {
|
|
onClick: this.handleClick
|
|
};
|
|
} else {
|
|
tooltipEvents = {
|
|
onMouseEnter: this.handleMouseEnter,
|
|
onDoubleClick: this.handleDoubleClick,
|
|
onMouseMove: this.handleMouseMove,
|
|
onMouseLeave: this.handleMouseLeave,
|
|
onTouchMove: this.handleTouchMove,
|
|
onTouchStart: this.handleTouchStart,
|
|
onTouchEnd: this.handleTouchEnd,
|
|
onContextMenu: this.handleContextMenu
|
|
};
|
|
}
|
|
}
|
|
|
|
// @ts-expect-error adaptEventHandlers expects DOM Event but generateCategoricalChart works with React UIEvents
|
|
var outerEvents = adaptEventHandlers(this.props, this.handleOuterEvent);
|
|
return _objectSpread(_objectSpread({}, outerEvents), tooltipEvents);
|
|
}
|
|
}, {
|
|
key: "addListener",
|
|
value: function addListener() {
|
|
eventCenter.on(SYNC_EVENT, this.handleReceiveSyncEvent);
|
|
}
|
|
}, {
|
|
key: "removeListener",
|
|
value: function removeListener() {
|
|
eventCenter.removeListener(SYNC_EVENT, this.handleReceiveSyncEvent);
|
|
}
|
|
}, {
|
|
key: "filterFormatItem",
|
|
value: function filterFormatItem(item, displayName, childIndex) {
|
|
var formattedGraphicalItems = this.state.formattedGraphicalItems;
|
|
for (var i = 0, len = formattedGraphicalItems.length; i < len; i++) {
|
|
var entry = formattedGraphicalItems[i];
|
|
if (entry.item === item || entry.props.key === item.key || displayName === getDisplayName(entry.item.type) && childIndex === entry.childIndex) {
|
|
return entry;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
}, {
|
|
key: "renderClipPath",
|
|
value: function renderClipPath() {
|
|
var clipPathId = this.clipPathId;
|
|
var _this$state$offset = this.state.offset,
|
|
left = _this$state$offset.left,
|
|
top = _this$state$offset.top,
|
|
height = _this$state$offset.height,
|
|
width = _this$state$offset.width;
|
|
return /*#__PURE__*/React.createElement("defs", null, /*#__PURE__*/React.createElement("clipPath", {
|
|
id: clipPathId
|
|
}, /*#__PURE__*/React.createElement("rect", {
|
|
x: left,
|
|
y: top,
|
|
height: height,
|
|
width: width
|
|
})));
|
|
}
|
|
}, {
|
|
key: "getXScales",
|
|
value: function getXScales() {
|
|
var xAxisMap = this.state.xAxisMap;
|
|
return xAxisMap ? Object.entries(xAxisMap).reduce(function (res, _ref13) {
|
|
var _ref14 = _slicedToArray(_ref13, 2),
|
|
axisId = _ref14[0],
|
|
axisProps = _ref14[1];
|
|
return _objectSpread(_objectSpread({}, res), {}, _defineProperty({}, axisId, axisProps.scale));
|
|
}, {}) : null;
|
|
}
|
|
}, {
|
|
key: "getYScales",
|
|
value: function getYScales() {
|
|
var yAxisMap = this.state.yAxisMap;
|
|
return yAxisMap ? Object.entries(yAxisMap).reduce(function (res, _ref15) {
|
|
var _ref16 = _slicedToArray(_ref15, 2),
|
|
axisId = _ref16[0],
|
|
axisProps = _ref16[1];
|
|
return _objectSpread(_objectSpread({}, res), {}, _defineProperty({}, axisId, axisProps.scale));
|
|
}, {}) : null;
|
|
}
|
|
}, {
|
|
key: "getXScaleByAxisId",
|
|
value: function getXScaleByAxisId(axisId) {
|
|
var _this$state$xAxisMap;
|
|
return (_this$state$xAxisMap = this.state.xAxisMap) === null || _this$state$xAxisMap === void 0 || (_this$state$xAxisMap = _this$state$xAxisMap[axisId]) === null || _this$state$xAxisMap === void 0 ? void 0 : _this$state$xAxisMap.scale;
|
|
}
|
|
}, {
|
|
key: "getYScaleByAxisId",
|
|
value: function getYScaleByAxisId(axisId) {
|
|
var _this$state$yAxisMap;
|
|
return (_this$state$yAxisMap = this.state.yAxisMap) === null || _this$state$yAxisMap === void 0 || (_this$state$yAxisMap = _this$state$yAxisMap[axisId]) === null || _this$state$yAxisMap === void 0 ? void 0 : _this$state$yAxisMap.scale;
|
|
}
|
|
}, {
|
|
key: "getItemByXY",
|
|
value: function getItemByXY(chartXY) {
|
|
var _this$state11 = this.state,
|
|
formattedGraphicalItems = _this$state11.formattedGraphicalItems,
|
|
activeItem = _this$state11.activeItem;
|
|
if (formattedGraphicalItems && formattedGraphicalItems.length) {
|
|
for (var i = 0, len = formattedGraphicalItems.length; i < len; i++) {
|
|
var graphicalItem = formattedGraphicalItems[i];
|
|
// graphicalItem is not a React Element so we don't need to resolve defaultProps
|
|
var props = graphicalItem.props,
|
|
item = graphicalItem.item;
|
|
var itemProps = item.type.defaultProps !== undefined ? _objectSpread(_objectSpread({}, item.type.defaultProps), item.props) : item.props;
|
|
var itemDisplayName = getDisplayName(item.type);
|
|
if (itemDisplayName === 'Bar') {
|
|
var activeBarItem = (props.data || []).find(function (entry) {
|
|
return isInRectangle(chartXY, entry);
|
|
});
|
|
if (activeBarItem) {
|
|
return {
|
|
graphicalItem: graphicalItem,
|
|
payload: activeBarItem
|
|
};
|
|
}
|
|
} else if (itemDisplayName === 'RadialBar') {
|
|
var _activeBarItem = (props.data || []).find(function (entry) {
|
|
return inRangeOfSector(chartXY, entry);
|
|
});
|
|
if (_activeBarItem) {
|
|
return {
|
|
graphicalItem: graphicalItem,
|
|
payload: _activeBarItem
|
|
};
|
|
}
|
|
} else if (isFunnel(graphicalItem, activeItem) || isPie(graphicalItem, activeItem) || isScatter(graphicalItem, activeItem)) {
|
|
var activeIndex = getActiveShapeIndexForTooltip({
|
|
graphicalItem: graphicalItem,
|
|
activeTooltipItem: activeItem,
|
|
itemData: itemProps.data
|
|
});
|
|
var childIndex = itemProps.activeIndex === undefined ? activeIndex : itemProps.activeIndex;
|
|
return {
|
|
graphicalItem: _objectSpread(_objectSpread({}, graphicalItem), {}, {
|
|
childIndex: childIndex
|
|
}),
|
|
payload: isScatter(graphicalItem, activeItem) ? itemProps.data[activeIndex] : graphicalItem.props.data[activeIndex]
|
|
};
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
}, {
|
|
key: "render",
|
|
value: function render() {
|
|
var _this3 = this;
|
|
if (!validateWidthHeight(this)) {
|
|
return null;
|
|
}
|
|
var _this$props6 = this.props,
|
|
children = _this$props6.children,
|
|
className = _this$props6.className,
|
|
width = _this$props6.width,
|
|
height = _this$props6.height,
|
|
style = _this$props6.style,
|
|
compact = _this$props6.compact,
|
|
title = _this$props6.title,
|
|
desc = _this$props6.desc,
|
|
others = _objectWithoutProperties(_this$props6, _excluded2);
|
|
var attrs = filterProps(others, false);
|
|
|
|
// The "compact" mode is mainly used as the panorama within Brush
|
|
if (compact) {
|
|
return /*#__PURE__*/React.createElement(ChartLayoutContextProvider, {
|
|
state: this.state,
|
|
width: this.props.width,
|
|
height: this.props.height,
|
|
clipPathId: this.clipPathId
|
|
}, /*#__PURE__*/React.createElement(Surface, _extends({}, attrs, {
|
|
width: width,
|
|
height: height,
|
|
title: title,
|
|
desc: desc
|
|
}), this.renderClipPath(), renderByOrder(children, this.renderMap)));
|
|
}
|
|
if (this.props.accessibilityLayer) {
|
|
var _this$props$tabIndex, _this$props$role;
|
|
// Set tabIndex to 0 by default (can be overwritten)
|
|
attrs.tabIndex = (_this$props$tabIndex = this.props.tabIndex) !== null && _this$props$tabIndex !== void 0 ? _this$props$tabIndex : 0;
|
|
// Set role to img by default (can be overwritten)
|
|
attrs.role = (_this$props$role = this.props.role) !== null && _this$props$role !== void 0 ? _this$props$role : 'application';
|
|
attrs.onKeyDown = function (e) {
|
|
_this3.accessibilityManager.keyboardEvent(e);
|
|
// 'onKeyDown' is not currently a supported prop that can be passed through
|
|
// if it's added, this should be added: this.props.onKeyDown(e);
|
|
};
|
|
attrs.onFocus = function () {
|
|
_this3.accessibilityManager.focus();
|
|
// 'onFocus' is not currently a supported prop that can be passed through
|
|
// if it's added, the focus event should be forwarded to the prop
|
|
};
|
|
}
|
|
var events = this.parseEventsOfWrapper();
|
|
return /*#__PURE__*/React.createElement(ChartLayoutContextProvider, {
|
|
state: this.state,
|
|
width: this.props.width,
|
|
height: this.props.height,
|
|
clipPathId: this.clipPathId
|
|
}, /*#__PURE__*/React.createElement("div", _extends({
|
|
className: clsx('recharts-wrapper', className),
|
|
style: _objectSpread({
|
|
position: 'relative',
|
|
cursor: 'default',
|
|
width: width,
|
|
height: height
|
|
}, style)
|
|
}, events, {
|
|
ref: function ref(node) {
|
|
_this3.container = node;
|
|
}
|
|
}), /*#__PURE__*/React.createElement(Surface, _extends({}, attrs, {
|
|
width: width,
|
|
height: height,
|
|
title: title,
|
|
desc: desc,
|
|
style: FULL_WIDTH_AND_HEIGHT
|
|
}), this.renderClipPath(), renderByOrder(children, this.renderMap)), this.renderLegend(), this.renderTooltip()));
|
|
}
|
|
}]);
|
|
}(Component);
|
|
_defineProperty(CategoricalChartWrapper, "displayName", chartName);
|
|
// todo join specific chart propTypes
|
|
_defineProperty(CategoricalChartWrapper, "defaultProps", _objectSpread({
|
|
layout: 'horizontal',
|
|
stackOffset: 'none',
|
|
barCategoryGap: '10%',
|
|
barGap: 4,
|
|
margin: {
|
|
top: 5,
|
|
right: 5,
|
|
bottom: 5,
|
|
left: 5
|
|
},
|
|
reverseStackOrder: false,
|
|
syncMethod: 'index'
|
|
}, defaultProps));
|
|
_defineProperty(CategoricalChartWrapper, "getDerivedStateFromProps", function (nextProps, prevState) {
|
|
var dataKey = nextProps.dataKey,
|
|
data = nextProps.data,
|
|
children = nextProps.children,
|
|
width = nextProps.width,
|
|
height = nextProps.height,
|
|
layout = nextProps.layout,
|
|
stackOffset = nextProps.stackOffset,
|
|
margin = nextProps.margin;
|
|
var dataStartIndex = prevState.dataStartIndex,
|
|
dataEndIndex = prevState.dataEndIndex;
|
|
if (prevState.updateId === undefined) {
|
|
var defaultState = createDefaultState(nextProps);
|
|
return _objectSpread(_objectSpread(_objectSpread({}, defaultState), {}, {
|
|
updateId: 0
|
|
}, updateStateOfAxisMapsOffsetAndStackGroups(_objectSpread(_objectSpread({
|
|
props: nextProps
|
|
}, defaultState), {}, {
|
|
updateId: 0
|
|
}), prevState)), {}, {
|
|
prevDataKey: dataKey,
|
|
prevData: data,
|
|
prevWidth: width,
|
|
prevHeight: height,
|
|
prevLayout: layout,
|
|
prevStackOffset: stackOffset,
|
|
prevMargin: margin,
|
|
prevChildren: children
|
|
});
|
|
}
|
|
if (dataKey !== prevState.prevDataKey || data !== prevState.prevData || width !== prevState.prevWidth || height !== prevState.prevHeight || layout !== prevState.prevLayout || stackOffset !== prevState.prevStackOffset || !shallowEqual(margin, prevState.prevMargin)) {
|
|
var _defaultState = createDefaultState(nextProps);
|
|
|
|
// Fixes https://github.com/recharts/recharts/issues/2143
|
|
var keepFromPrevState = {
|
|
// (chartX, chartY) are (0,0) in default state, but we want to keep the last mouse position to avoid
|
|
// any flickering
|
|
chartX: prevState.chartX,
|
|
chartY: prevState.chartY,
|
|
// The tooltip should stay active when it was active in the previous render. If this is not
|
|
// the case, the tooltip disappears and immediately re-appears, causing a flickering effect
|
|
isTooltipActive: prevState.isTooltipActive
|
|
};
|
|
var updatesToState = _objectSpread(_objectSpread({}, getTooltipData(prevState, data, layout)), {}, {
|
|
updateId: prevState.updateId + 1
|
|
});
|
|
var newState = _objectSpread(_objectSpread(_objectSpread({}, _defaultState), keepFromPrevState), updatesToState);
|
|
return _objectSpread(_objectSpread(_objectSpread({}, newState), updateStateOfAxisMapsOffsetAndStackGroups(_objectSpread({
|
|
props: nextProps
|
|
}, newState), prevState)), {}, {
|
|
prevDataKey: dataKey,
|
|
prevData: data,
|
|
prevWidth: width,
|
|
prevHeight: height,
|
|
prevLayout: layout,
|
|
prevStackOffset: stackOffset,
|
|
prevMargin: margin,
|
|
prevChildren: children
|
|
});
|
|
}
|
|
if (!isChildrenEqual(children, prevState.prevChildren)) {
|
|
var _brush$props$startInd, _brush$props, _brush$props$endIndex, _brush$props2;
|
|
// specifically check for Brush - if it exists and the start and end indexes are different, re-render with the new ones
|
|
var brush = findChildByType(children, Brush);
|
|
var startIndex = brush ? (_brush$props$startInd = (_brush$props = brush.props) === null || _brush$props === void 0 ? void 0 : _brush$props.startIndex) !== null && _brush$props$startInd !== void 0 ? _brush$props$startInd : dataStartIndex : dataStartIndex;
|
|
var endIndex = brush ? (_brush$props$endIndex = (_brush$props2 = brush.props) === null || _brush$props2 === void 0 ? void 0 : _brush$props2.endIndex) !== null && _brush$props$endIndex !== void 0 ? _brush$props$endIndex : dataEndIndex : dataEndIndex;
|
|
var hasDifferentStartOrEndIndex = startIndex !== dataStartIndex || endIndex !== dataEndIndex;
|
|
|
|
// update configuration in children
|
|
var hasGlobalData = !isNil(data);
|
|
var newUpdateId = hasGlobalData && !hasDifferentStartOrEndIndex ? prevState.updateId : prevState.updateId + 1;
|
|
return _objectSpread(_objectSpread({
|
|
updateId: newUpdateId
|
|
}, updateStateOfAxisMapsOffsetAndStackGroups(_objectSpread(_objectSpread({
|
|
props: nextProps
|
|
}, prevState), {}, {
|
|
updateId: newUpdateId,
|
|
dataStartIndex: startIndex,
|
|
dataEndIndex: endIndex
|
|
}), prevState)), {}, {
|
|
prevChildren: children,
|
|
dataStartIndex: startIndex,
|
|
dataEndIndex: endIndex
|
|
});
|
|
}
|
|
return null;
|
|
});
|
|
_defineProperty(CategoricalChartWrapper, "renderActiveDot", function (option, props, key) {
|
|
var dot;
|
|
if ( /*#__PURE__*/isValidElement(option)) {
|
|
dot = /*#__PURE__*/cloneElement(option, props);
|
|
} else if (isFunction(option)) {
|
|
dot = option(props);
|
|
} else {
|
|
dot = /*#__PURE__*/React.createElement(Dot, props);
|
|
}
|
|
return /*#__PURE__*/React.createElement(Layer, {
|
|
className: "recharts-active-dot",
|
|
key: key
|
|
}, dot);
|
|
});
|
|
var CategoricalChart = /*#__PURE__*/forwardRef(function CategoricalChart(props, ref) {
|
|
return /*#__PURE__*/React.createElement(CategoricalChartWrapper, _extends({}, props, {
|
|
ref: ref
|
|
}));
|
|
});
|
|
CategoricalChart.displayName = CategoricalChartWrapper.displayName;
|
|
return CategoricalChart;
|
|
}; |