UNPKG

devextreme

Version:

HTML5 JavaScript Component Suite for Responsive Web Development

301 lines (290 loc) • 12.2 kB
/** * DevExtreme (esm/viz/series/helpers/range_data_calculator.js) * Version: 21.1.4 * Build date: Mon Jun 21 2021 * * Copyright (c) 2012 - 2021 Developer Express Inc. ALL RIGHTS RESERVED * Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/ */ import { unique, getAddFunction, getLog } from "../../core/utils"; import { isDefined, isObject } from "../../../core/utils/type"; import { noop } from "../../../core/utils/common"; var DISCRETE = "discrete"; var { abs: abs, floor: floor, ceil: ceil, min: min } = Math; function continuousRangeCalculator(range, minValue, maxValue) { range.min = range.min < minValue ? range.min : minValue; range.max = range.max > maxValue ? range.max : maxValue } function createGetLogFunction(axisType, axis) { if ("logarithmic" !== axisType) { return null } var base = axis.getOptions().logarithmBase; return value => { var log = getLog(abs(value), base); var round = log < 0 ? floor : ceil; return round(log) } } function getRangeCalculator(axisType, axis, getLog) { var rangeCalculator = continuousRangeCalculator; if (axisType === DISCRETE) { rangeCalculator = function(range, minValue, maxValue) { if (minValue !== maxValue) { range.categories.push(maxValue) } range.categories.push(minValue) } } else if (axis) { rangeCalculator = function(range, value) { var interval = axis.calculateInterval(value, range.prevValue); var minInterval = range.interval; range.interval = (minInterval < interval ? minInterval : interval) || minInterval; range.prevValue = value; continuousRangeCalculator(range, value, value) } } if (getLog) { return (range, minValue, maxValue) => { var minArgs = []; rangeCalculator(range, minValue, maxValue); 0 !== minValue && minArgs.push(getLog(minValue)); 0 !== maxValue && minArgs.push(getLog(maxValue)); var linearThreshold = min.apply(null, minArgs); range.linearThreshold = range.linearThreshold < linearThreshold ? range.linearThreshold : linearThreshold } } return rangeCalculator } function getInitialRange(axisType, dataType, firstValue) { var range = { axisType: axisType, dataType: dataType }; if (axisType === DISCRETE) { range.categories = [] } else { range.min = isObject(firstValue) ? firstValue.min : firstValue; range.max = isObject(firstValue) ? firstValue.max : firstValue } return range } function processCategories(range) { if (range.categories) { range.categories = unique(range.categories) } return range } function getValueForArgument(point, extraPoint, x, range) { if (extraPoint && isDefined(extraPoint.value)) { var y1 = point.value; var y2 = extraPoint.value; var x1 = point.argument; var x2 = extraPoint.argument; var r = (x - x1) * (y2 - y1) / (x2 - x1) + y1.valueOf(); return "datetime" === range.dataType ? new Date(r) : r } else { return point.value } } function calculateRangeBetweenPoints(rangeCalculator, range, point, prevPoint, bound) { var value = getValueForArgument(point, prevPoint, bound, range); rangeCalculator(range, value, value) } function isLineSeries(series) { return series.type.toLowerCase().indexOf("line") >= 0 || series.type.toLowerCase().indexOf("area") >= 0 } function getViewportReducer(series) { var rangeCalculator = getRangeCalculator(series.valueAxisType); var argumentAxis = series.getArgumentAxis(); var viewport = argumentAxis && series.getArgumentAxis().visualRange() || {}; var calculatePointBetweenPoints = isLineSeries(series) ? calculateRangeBetweenPoints : noop; if (argumentAxis && argumentAxis.getMarginOptions().checkInterval) { var range = series.getArgumentAxis().getTranslator().getBusinessRange(); var add = getAddFunction(range, false); var interval = range.interval; if (isFinite(interval) && isDefined(viewport.startValue) && isDefined(viewport.endValue)) { viewport.startValue = add(viewport.startValue, interval, -1); viewport.endValue = add(viewport.endValue, interval) } } var viewportFilter = getViewPortFilter(viewport); return function(range, point, index, points) { var argument = point.argument; if (!point.hasValue()) { return range } if (viewportFilter(argument)) { if (!range.startCalc) { range.startCalc = true; calculatePointBetweenPoints(rangeCalculator, range, point, points[index - 1], viewport.startValue) } rangeCalculator(range, point.getMinValue(), point.getMaxValue()) } else if (!viewport.categories && isDefined(viewport.startValue) && argument > viewport.startValue) { if (!range.startCalc) { calculatePointBetweenPoints(rangeCalculator, range, point, points[index - 1], viewport.startValue) } range.endCalc = true; calculatePointBetweenPoints(rangeCalculator, range, point, points[index - 1], viewport.endValue) } return range } } function getViewPortFilter(viewport) { if (viewport.categories) { var dictionary = viewport.categories.reduce((result, category) => { result[category.valueOf()] = true; return result }, {}); return argument => isDefined(argument) && dictionary[argument.valueOf()] } if (!isDefined(viewport.startValue) && !isDefined(viewport.endValue)) { return () => true } if (!isDefined(viewport.endValue)) { return argument => argument >= viewport.startValue } if (!isDefined(viewport.startValue)) { return argument => argument <= viewport.endValue } return argument => argument >= viewport.startValue && argument <= viewport.endValue } export default { getViewPortFilter: getViewPortFilter, getArgumentRange: function(series) { var data = series._data || []; var range = {}; if (data.length) { if (series.argumentAxisType === DISCRETE) { range = { categories: data.map(item => item.argument) } } else { var interval; if (data.length > 1) { var i1 = series.getArgumentAxis().calculateInterval(data[0].argument, data[1].argument); var i2 = series.getArgumentAxis().calculateInterval(data[data.length - 1].argument, data[data.length - 2].argument); interval = min(i1, i2) } range = { min: data[0].argument, max: data[data.length - 1].argument, interval: interval } } } return processCategories(range) }, getRangeData: function(series) { var points = series.getPoints(); var useAggregation = series.useAggregation(); var argumentCalculator = getRangeCalculator(series.argumentAxisType, points.length > 1 && series.getArgumentAxis(), createGetLogFunction(series.argumentAxisType, series.getArgumentAxis())); var valueRangeCalculator = getRangeCalculator(series.valueAxisType, null, createGetLogFunction(series.valueAxisType, series.getValueAxis())); var viewportReducer = getViewportReducer(series); var range = points.reduce((function(range, point, index, points) { var argument = point.argument; if (!point.isArgumentCorrect()) { return range } argumentCalculator(range.arg, argument, argument); if (point.hasValue()) { valueRangeCalculator(range.val, point.getMinValue(), point.getMaxValue()); viewportReducer(range.viewport, point, index, points) } return range }), { arg: getInitialRange(series.argumentAxisType, series.argumentType, series.getArgumentRangeInitialValue()), val: getInitialRange(series.valueAxisType, series.valueType, points.length ? series.getValueRangeInitialValue() : void 0), viewport: getInitialRange(series.valueAxisType, series.valueType, points.length ? series.getValueRangeInitialValue() : void 0) }); if (useAggregation) { var argumentRange = this.getArgumentRange(series); if (series.argumentAxisType === DISCRETE) { range.arg = argumentRange } else { var viewport = series.getArgumentAxis().getViewport(); if (isDefined(viewport.startValue) || isDefined(viewport.length)) { argumentCalculator(range.arg, argumentRange.min, argumentRange.min) } if (isDefined(viewport.endValue) || isDefined(viewport.length) && isDefined(viewport.startValue)) { argumentCalculator(range.arg, argumentRange.max, argumentRange.max) } } } processCategories(range.arg); processCategories(range.val); return range }, getViewport: function(series) { var points = series.getPoints(); var range; var reducer = getViewportReducer(series); range = getInitialRange(series.valueAxisType, series.valueType, points.length ? series.getValueRangeInitialValue() : void 0); points.some((function(point, index) { reducer(range, point, index, points); return range.endCalc })); return range }, getPointsInViewPort: function(series) { var argumentViewPortFilter = getViewPortFilter(series.getArgumentAxis().visualRange() || {}); var valueViewPort = series.getValueAxis().visualRange() || {}; var valueViewPortFilter = getViewPortFilter(valueViewPort); var points = series.getPoints(); var addValue = function(values, point, isEdge) { var minValue = point.getMinValue(); var maxValue = point.getMaxValue(); var isMinValueInViewPort = valueViewPortFilter(minValue); var isMaxValueInViewPort = valueViewPortFilter(maxValue); if (isMinValueInViewPort) { values.push(minValue) } if (maxValue !== minValue && isMaxValueInViewPort) { values.push(maxValue) } if (isEdge && !isMinValueInViewPort && !isMaxValueInViewPort) { if (!values.length) { values.push(valueViewPort.startValue) } else { values.push(valueViewPort.endValue) } } }; var addEdgePoints = isLineSeries(series) ? function(result, points, index) { var point = points[index]; var prevPoint = points[index - 1]; var nextPoint = points[index + 1]; if (nextPoint && argumentViewPortFilter(nextPoint.argument)) { addValue(result[1], point, true) } if (prevPoint && argumentViewPortFilter(prevPoint.argument)) { addValue(result[1], point, true) } } : noop; return points.reduce((function(result, point, index) { if (argumentViewPortFilter(point.argument)) { addValue(result[0], point) } else { addEdgePoints(result, points, index) } return result }), [ [], [] ]) } };