UNPKG

devextreme

Version:

HTML5 JavaScript Component Suite for Responsive Web Development

424 lines (402 loc) • 17.2 kB
/** * DevExtreme (viz/core/series_family.js) * Version: 18.1.3 * Build date: Tue May 15 2018 * * Copyright (c) 2012 - 2018 Developer Express Inc. ALL RIGHTS RESERVED * Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/ */ "use strict"; var isNumeric = require("../../core/utils/type").isNumeric, extend = require("../../core/utils/extend").extend, each = require("../../core/utils/iterator").each, isDefined = require("../../core/utils/type").isDefined, _math = Math, _round = _math.round, _abs = _math.abs, _pow = _math.pow, _each = each, _noop = require("../../core/utils/common").noop, vizUtils = require("./utils"), DEFAULT_BAR_GROUP_PADDING = .3, _normalizeEnum = vizUtils.normalizeEnum; function validateBarPadding(barPadding) { return barPadding < 0 || barPadding > 1 ? void 0 : barPadding } function validateBarGroupPadding(barGroupPadding) { return barGroupPadding < 0 || barGroupPadding > 1 ? DEFAULT_BAR_GROUP_PADDING : barGroupPadding } function isStackExist(series, arg, equalBarWidth) { return series.some(function(s) { return equalBarWidth && !s.getOptions().ignoreEmptyPoints || s.getPointsByArg(arg, true).some(function(point) { return point.hasValue() }) }) } function correctStackCoordinates(series, currentStacks, arg, stack, parameters, barsArea, seriesStackIndexCallback) { series.forEach(function(series) { var extraParameters, stackIndex = seriesStackIndexCallback(currentStacks.indexOf(stack), currentStacks.length), points = series.getPointsByArg(arg, true), barPadding = validateBarPadding(series.getOptions().barPadding), barWidth = series.getOptions().barWidth, offset = getOffset(stackIndex, parameters), width = parameters.width; if (stackIndex === -1) { return } if (isDefined(barPadding) || isDefined(barWidth)) { extraParameters = calculateParams(barsArea, currentStacks.length, 1 - barPadding, barWidth); width = extraParameters.width; offset = getOffset(stackIndex, extraParameters) } correctPointCoordinates(points, width, offset) }) } function adjustBarSeriesDimensionsCore(series, options, seriesStackIndexCallback) { var stack, commonStacks = [], allArguments = [], seriesInStacks = {}, barWidth = options.barWidth, barGroupWidth = options.barGroupWidth, interval = series[0] && series[0].getArgumentAxis().getTranslator().getInterval(), barsArea = barGroupWidth ? interval > barGroupWidth ? barGroupWidth : interval : interval * (1 - validateBarGroupPadding(options.barGroupPadding)); series.forEach(function(s, i) { var argument, stackName = s.getStackName && s.getStackName() || i.toString(); for (argument in s.pointsByArgument) { if (allArguments.indexOf(argument.valueOf()) === -1) { allArguments.push(argument.valueOf()) } } if (commonStacks.indexOf(stackName) === -1) { commonStacks.push(stackName); seriesInStacks[stackName] = [] } seriesInStacks[stackName].push(s) }); allArguments.forEach(function(arg) { var parameters, currentStacks = []; for (stack in seriesInStacks) { if (isStackExist(seriesInStacks[stack], arg, options.equalBarWidth)) { currentStacks.push(stack) } } parameters = calculateParams(barsArea, currentStacks.length, barWidth); for (stack in seriesInStacks) { correctStackCoordinates(seriesInStacks[stack], currentStacks, arg, stack, parameters, barsArea, seriesStackIndexCallback) } }) } function calculateParams(barsArea, count, percentWidth, fixedBarWidth) { var spacing, width; if (fixedBarWidth) { width = Math.min(fixedBarWidth, _round(barsArea / count)); spacing = count > 1 ? _round((barsArea - width * count) / (count - 1)) : 0 } else { if (isDefined(percentWidth)) { width = _round(barsArea * percentWidth / count); spacing = _round(count > 1 ? (barsArea - barsArea * percentWidth) / (count - 1) : 0) } else { spacing = _round(barsArea / count * .2); width = _round((barsArea - spacing * (count - 1)) / count) } } return { width: width > 1 ? width : 1, spacing: spacing, middleIndex: count / 2 } } function getOffset(stackIndex, parameters) { return (stackIndex - parameters.middleIndex + .5) * parameters.width - (parameters.middleIndex - stackIndex - .5) * parameters.spacing } function correctPointCoordinates(points, width, offset) { _each(points, function(_, point) { point.correctCoordinates({ width: width, offset: offset }) }) } function getValueType(value) { return value >= 0 ? "positive" : "negative" } function getVisibleSeries(that) { return that.series.filter(function(s) { return s.isVisible() }) } function getAbsStackSumByArg(stackKeepers, stackName, argument) { var positiveStackValue = (stackKeepers.positive[stackName] || {})[argument] || 0, negativeStackValue = -(stackKeepers.negative[stackName] || {})[argument] || 0; return positiveStackValue + negativeStackValue } function getStackSumByArg(stackKeepers, stackName, argument) { var positiveStackValue = (stackKeepers.positive[stackName] || {})[argument] || 0, negativeStackValue = (stackKeepers.negative[stackName] || {})[argument] || 0; return positiveStackValue + negativeStackValue } function getSeriesStackIndexCallback(rotated) { if (!rotated) { return function(index) { return index } } else { return function(index, stackCount) { return stackCount - index - 1 } } } function adjustBarSeriesDimensions() { adjustBarSeriesDimensionsCore(getVisibleSeries(this), this._options, getSeriesStackIndexCallback(this.rotated)) } function adjustStackedSeriesValues() { var that = this, negativesAsZeroes = that._options.negativesAsZeroes, series = getVisibleSeries(that), stackKeepers = { positive: {}, negative: {} }, holesStack = { left: {}, right: {} }, lastSeriesInStack = {}; series.forEach(function(singleSeries) { var stackName = singleSeries.getStackName(), hole = false; singleSeries._prevSeries = lastSeriesInStack[stackName]; lastSeriesInStack[stackName] = singleSeries; singleSeries.holes = extend(true, {}, holesStack); singleSeries.getPoints().forEach(function(point, index, points) { var currentStack, value = point.initialValue, argument = point.argument.valueOf(), stacks = value >= 0 ? stackKeepers.positive : stackKeepers.negative; if (negativesAsZeroes && value < 0) { stacks = stackKeepers.positive; value = 0; point.resetValue() } stacks[stackName] = stacks[stackName] || {}; currentStack = stacks[stackName]; if (currentStack[argument]) { point.correctValue(currentStack[argument]); currentStack[argument] += value } else { currentStack[argument] = value; point.resetCorrection() } if (!point.hasValue()) { var prevPoint = points[index - 1]; if (!hole && prevPoint && prevPoint.hasValue()) { argument = prevPoint.argument.valueOf(); prevPoint._skipSetRightHole = true; holesStack.right[argument] = (holesStack.right[argument] || 0) + (prevPoint.value - (isFinite(prevPoint.minValue) ? prevPoint.minValue : 0)) } hole = true } else { if (hole) { hole = false; holesStack.left[argument] = (holesStack.left[argument] || 0) + (point.value - (isFinite(point.minValue) ? point.minValue : 0)); point._skipSetLeftHole = true } } }) }); series.forEach(function(singleSeries) { var holes = singleSeries.holes; singleSeries.getPoints().forEach(function(point) { var argument = point.argument.valueOf(); point.resetHoles(); !point._skipSetLeftHole && point.setHole(holes.left[argument] || holesStack.left[argument] && 0, "left"); !point._skipSetRightHole && point.setHole(holes.right[argument] || holesStack.right[argument] && 0, "right"); point._skipSetLeftHole = null; point._skipSetRightHole = null }) }); that._stackKeepers = stackKeepers; series.forEach(function(singleSeries) { singleSeries.getPoints().forEach(function(point) { var argument = point.argument.valueOf(), stackName = singleSeries.getStackName(), absTotal = getAbsStackSumByArg(stackKeepers, stackName, argument), total = getStackSumByArg(stackKeepers, stackName, argument); point.setPercentValue(absTotal, total, holesStack.left[argument], holesStack.right[argument]) }) }) } function updateStackedSeriesValues() { var that = this, series = getVisibleSeries(that), stack = that._stackKeepers, stackKeepers = { positive: {}, negative: {} }; _each(series, function(_, singleSeries) { var minBarSize = singleSeries.getOptions().minBarSize, valueAxisTranslator = singleSeries.getValueAxis().getTranslator(), minShownBusinessValue = minBarSize && valueAxisTranslator.getMinBarSize(minBarSize), stackName = singleSeries.getStackName(); _each(singleSeries.getPoints(), function(index, point) { if (!point.hasValue()) { return } var updateValue, valueType, currentStack, value = point.initialValue, argument = point.argument.valueOf(); if (that.fullStacked) { value = value / getAbsStackSumByArg(stack, stackName, argument) || 0 } updateValue = valueAxisTranslator.checkMinBarSize(value, minShownBusinessValue, point.value); valueType = getValueType(updateValue); currentStack = stackKeepers[valueType][stackName] = stackKeepers[valueType][stackName] || {}; if (currentStack[argument]) { point.minValue = currentStack[argument]; currentStack[argument] += updateValue } else { currentStack[argument] = updateValue } point.value = currentStack[argument] }) }); if (that.fullStacked) { updateFullStackedSeriesValues(series, stackKeepers) } } function updateFullStackedSeriesValues(series, stackKeepers) { _each(series, function(_, singleSeries) { var stackName = singleSeries.getStackName ? singleSeries.getStackName() : "default"; _each(singleSeries.getPoints(), function(index, point) { var stackSum = getAbsStackSumByArg(stackKeepers, stackName, point.argument.valueOf()); point.value = point.value / stackSum; if (isNumeric(point.minValue)) { point.minValue = point.minValue / stackSum } }) }) } function updateBarSeriesValues() { _each(this.series, function(_, singleSeries) { var minBarSize = singleSeries.getOptions().minBarSize, valueAxisTranslator = singleSeries.getValueAxis().getTranslator(), minShownBusinessValue = minBarSize && valueAxisTranslator.getMinBarSize(minBarSize); if (minShownBusinessValue) { _each(singleSeries.getPoints(), function(index, point) { if (point.hasValue()) { point.value = valueAxisTranslator.checkMinBarSize(point.initialValue, minShownBusinessValue) } }) } }) } function adjustCandlestickSeriesDimensions() { adjustBarSeriesDimensionsCore(getVisibleSeries(this), { barWidth: null, equalBarWidth: true, barGroupPadding: .3 }, getSeriesStackIndexCallback(this.rotated)) } function adjustBubbleSeriesDimensions() { var series = getVisibleSeries(this); if (!series.length) { return } var pointSize, bubbleArea, sizeProportion, sizeDispersion, areaDispersion, options = this._options, argTranslator = series[0].getArgumentAxis().getTranslator(), valTranslator = series[0].getValueAxis().getTranslator(), visibleAreaX = argTranslator.getCanvasVisibleArea(), visibleAreaY = valTranslator.getCanvasVisibleArea(), min = _math.min(visibleAreaX.max - visibleAreaX.min, visibleAreaY.max - visibleAreaY.min), minBubbleArea = _pow(options.minBubbleSize, 2), maxBubbleArea = _pow(min * options.maxBubbleSize, 2), equalBubbleSize = (min * options.maxBubbleSize + options.minBubbleSize) / 2, minPointSize = 1 / 0, maxPointSize = 0; _each(series, function(_, seriesItem) { _each(seriesItem.getPoints(), function(_, point) { maxPointSize = maxPointSize > point.size ? maxPointSize : point.size; minPointSize = minPointSize < point.size ? minPointSize : point.size }) }); sizeDispersion = maxPointSize - minPointSize; areaDispersion = _abs(maxBubbleArea - minBubbleArea); minPointSize = minPointSize < 0 ? 0 : minPointSize; _each(series, function(_, seriesItem) { _each(seriesItem.getPoints(), function(_, point) { if (maxPointSize === minPointSize) { pointSize = _round(equalBubbleSize) } else { sizeProportion = _abs(point.size - minPointSize) / sizeDispersion; bubbleArea = areaDispersion * sizeProportion + minBubbleArea; pointSize = _round(_math.sqrt(bubbleArea)) } point.correctCoordinates(pointSize) }) }) } function SeriesFamily(options) { var that = this; that.type = _normalizeEnum(options.type); that.pane = options.pane; that.rotated = options.rotated; that.series = []; that.updateOptions(options); switch (that.type) { case "bar": that.adjustSeriesDimensions = adjustBarSeriesDimensions; that.updateSeriesValues = updateBarSeriesValues; break; case "rangebar": that.adjustSeriesDimensions = adjustBarSeriesDimensions; break; case "fullstackedbar": that.fullStacked = true; that.adjustSeriesDimensions = adjustBarSeriesDimensions; that.adjustSeriesValues = adjustStackedSeriesValues; that.updateSeriesValues = updateStackedSeriesValues; break; case "stackedbar": that.adjustSeriesDimensions = adjustBarSeriesDimensions; that.adjustSeriesValues = adjustStackedSeriesValues; that.updateSeriesValues = updateStackedSeriesValues; break; case "fullstackedarea": case "fullstackedline": case "fullstackedspline": case "fullstackedsplinearea": that.fullStacked = true; that.adjustSeriesValues = adjustStackedSeriesValues; break; case "stackedarea": case "stackedsplinearea": case "stackedline": case "stackedspline": that.adjustSeriesValues = adjustStackedSeriesValues; break; case "candlestick": case "stock": that.adjustSeriesDimensions = adjustCandlestickSeriesDimensions; break; case "bubble": that.adjustSeriesDimensions = adjustBubbleSeriesDimensions } } exports.SeriesFamily = SeriesFamily; SeriesFamily.prototype = { constructor: SeriesFamily, adjustSeriesDimensions: _noop, adjustSeriesValues: _noop, updateSeriesValues: _noop, updateOptions: function(options) { this._options = options }, dispose: function() { this.series = null }, add: function(series) { var type = this.type; this.series = vizUtils.map(series, function(singleSeries) { return singleSeries.type === type ? singleSeries : null }) } };