123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348 |
- /*
- * Licensed to the Apache Software Foundation (ASF) under one
- * or more contributor license agreements. See the NOTICE file
- * distributed with this work for additional information
- * regarding copyright ownership. The ASF licenses this file
- * to you under the Apache License, Version 2.0 (the
- * "License"); you may not use this file except in compliance
- * with the License. You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing,
- * software distributed under the License is distributed on an
- * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
- * KIND, either express or implied. See the License for the
- * specific language governing permissions and limitations
- * under the License.
- */
- /**
- * AUTO-GENERATED FILE. DO NOT MODIFY.
- */
- /*
- * Licensed to the Apache Software Foundation (ASF) under one
- * or more contributor license agreements. See the NOTICE file
- * distributed with this work for additional information
- * regarding copyright ownership. The ASF licenses this file
- * to you under the Apache License, Version 2.0 (the
- * "License"); you may not use this file except in compliance
- * with the License. You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing,
- * software distributed under the License is distributed on an
- * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
- * KIND, either express or implied. See the License for the
- * specific language governing permissions and limitations
- * under the License.
- */
- import * as zrUtil from 'zrender/lib/core/util.js';
- import OrdinalScale from '../scale/Ordinal.js';
- import IntervalScale from '../scale/Interval.js';
- import Scale from '../scale/Scale.js';
- import { prepareLayoutBarSeries, makeColumnLayout, retrieveColumnLayout } from '../layout/barGrid.js';
- import BoundingRect from 'zrender/lib/core/BoundingRect.js';
- import TimeScale from '../scale/Time.js';
- import LogScale from '../scale/Log.js';
- import { getStackedDimension } from '../data/helper/dataStackHelper.js';
- import { ensureScaleRawExtentInfo } from './scaleRawExtentInfo.js';
- /**
- * Get axis scale extent before niced.
- * Item of returned array can only be number (including Infinity and NaN).
- *
- * Caution:
- * Precondition of calling this method:
- * The scale extent has been initialized using series data extent via
- * `scale.setExtent` or `scale.unionExtentFromData`;
- */
- export function getScaleExtent(scale, model) {
- var scaleType = scale.type;
- var rawExtentResult = ensureScaleRawExtentInfo(scale, model, scale.getExtent()).calculate();
- scale.setBlank(rawExtentResult.isBlank);
- var min = rawExtentResult.min;
- var max = rawExtentResult.max;
- // If bars are placed on a base axis of type time or interval account for axis boundary overflow and current axis
- // is base axis
- // FIXME
- // (1) Consider support value axis, where below zero and axis `onZero` should be handled properly.
- // (2) Refactor the logic with `barGrid`. Is it not need to `makeBarWidthAndOffsetInfo` twice with different extent?
- // Should not depend on series type `bar`?
- // (3) Fix that might overlap when using dataZoom.
- // (4) Consider other chart types using `barGrid`?
- // See #6728, #4862, `test/bar-overflow-time-plot.html`
- var ecModel = model.ecModel;
- if (ecModel && scaleType === 'time' /* || scaleType === 'interval' */) {
- var barSeriesModels = prepareLayoutBarSeries('bar', ecModel);
- var isBaseAxisAndHasBarSeries_1 = false;
- zrUtil.each(barSeriesModels, function (seriesModel) {
- isBaseAxisAndHasBarSeries_1 = isBaseAxisAndHasBarSeries_1 || seriesModel.getBaseAxis() === model.axis;
- });
- if (isBaseAxisAndHasBarSeries_1) {
- // Calculate placement of bars on axis. TODO should be decoupled
- // with barLayout
- var barWidthAndOffset = makeColumnLayout(barSeriesModels);
- // Adjust axis min and max to account for overflow
- var adjustedScale = adjustScaleForOverflow(min, max, model, barWidthAndOffset);
- min = adjustedScale.min;
- max = adjustedScale.max;
- }
- }
- return {
- extent: [min, max],
- // "fix" means "fixed", the value should not be
- // changed in the subsequent steps.
- fixMin: rawExtentResult.minFixed,
- fixMax: rawExtentResult.maxFixed
- };
- }
- function adjustScaleForOverflow(min, max, model,
- // Only support cartesian coord yet.
- barWidthAndOffset) {
- // Get Axis Length
- var axisExtent = model.axis.getExtent();
- var axisLength = axisExtent[1] - axisExtent[0];
- // Get bars on current base axis and calculate min and max overflow
- var barsOnCurrentAxis = retrieveColumnLayout(barWidthAndOffset, model.axis);
- if (barsOnCurrentAxis === undefined) {
- return {
- min: min,
- max: max
- };
- }
- var minOverflow = Infinity;
- zrUtil.each(barsOnCurrentAxis, function (item) {
- minOverflow = Math.min(item.offset, minOverflow);
- });
- var maxOverflow = -Infinity;
- zrUtil.each(barsOnCurrentAxis, function (item) {
- maxOverflow = Math.max(item.offset + item.width, maxOverflow);
- });
- minOverflow = Math.abs(minOverflow);
- maxOverflow = Math.abs(maxOverflow);
- var totalOverFlow = minOverflow + maxOverflow;
- // Calculate required buffer based on old range and overflow
- var oldRange = max - min;
- var oldRangePercentOfNew = 1 - (minOverflow + maxOverflow) / axisLength;
- var overflowBuffer = oldRange / oldRangePercentOfNew - oldRange;
- max += overflowBuffer * (maxOverflow / totalOverFlow);
- min -= overflowBuffer * (minOverflow / totalOverFlow);
- return {
- min: min,
- max: max
- };
- }
- // Precondition of calling this method:
- // The scale extent has been initialized using series data extent via
- // `scale.setExtent` or `scale.unionExtentFromData`;
- export function niceScaleExtent(scale, inModel) {
- var model = inModel;
- var extentInfo = getScaleExtent(scale, model);
- var extent = extentInfo.extent;
- var splitNumber = model.get('splitNumber');
- if (scale instanceof LogScale) {
- scale.base = model.get('logBase');
- }
- var scaleType = scale.type;
- var interval = model.get('interval');
- var isIntervalOrTime = scaleType === 'interval' || scaleType === 'time';
- scale.setExtent(extent[0], extent[1]);
- scale.calcNiceExtent({
- splitNumber: splitNumber,
- fixMin: extentInfo.fixMin,
- fixMax: extentInfo.fixMax,
- minInterval: isIntervalOrTime ? model.get('minInterval') : null,
- maxInterval: isIntervalOrTime ? model.get('maxInterval') : null
- });
- // If some one specified the min, max. And the default calculated interval
- // is not good enough. He can specify the interval. It is often appeared
- // in angle axis with angle 0 - 360. Interval calculated in interval scale is hard
- // to be 60.
- // FIXME
- if (interval != null) {
- scale.setInterval && scale.setInterval(interval);
- }
- }
- /**
- * @param axisType Default retrieve from model.type
- */
- export function createScaleByModel(model, axisType) {
- axisType = axisType || model.get('type');
- if (axisType) {
- switch (axisType) {
- // Buildin scale
- case 'category':
- return new OrdinalScale({
- ordinalMeta: model.getOrdinalMeta ? model.getOrdinalMeta() : model.getCategories(),
- extent: [Infinity, -Infinity]
- });
- case 'time':
- return new TimeScale({
- locale: model.ecModel.getLocaleModel(),
- useUTC: model.ecModel.get('useUTC')
- });
- default:
- // case 'value'/'interval', 'log', or others.
- return new (Scale.getClass(axisType) || IntervalScale)();
- }
- }
- }
- /**
- * Check if the axis cross 0
- */
- export function ifAxisCrossZero(axis) {
- var dataExtent = axis.scale.getExtent();
- var min = dataExtent[0];
- var max = dataExtent[1];
- return !(min > 0 && max > 0 || min < 0 && max < 0);
- }
- /**
- * @param axis
- * @return Label formatter function.
- * param: {number} tickValue,
- * param: {number} idx, the index in all ticks.
- * If category axis, this param is not required.
- * return: {string} label string.
- */
- export function makeLabelFormatter(axis) {
- var labelFormatter = axis.getLabelModel().get('formatter');
- var categoryTickStart = axis.type === 'category' ? axis.scale.getExtent()[0] : null;
- if (axis.scale.type === 'time') {
- return function (tpl) {
- return function (tick, idx) {
- return axis.scale.getFormattedLabel(tick, idx, tpl);
- };
- }(labelFormatter);
- } else if (zrUtil.isString(labelFormatter)) {
- return function (tpl) {
- return function (tick) {
- // For category axis, get raw value; for numeric axis,
- // get formatted label like '1,333,444'.
- var label = axis.scale.getLabel(tick);
- var text = tpl.replace('{value}', label != null ? label : '');
- return text;
- };
- }(labelFormatter);
- } else if (zrUtil.isFunction(labelFormatter)) {
- return function (cb) {
- return function (tick, idx) {
- // The original intention of `idx` is "the index of the tick in all ticks".
- // But the previous implementation of category axis do not consider the
- // `axisLabel.interval`, which cause that, for example, the `interval` is
- // `1`, then the ticks "name5", "name7", "name9" are displayed, where the
- // corresponding `idx` are `0`, `2`, `4`, but not `0`, `1`, `2`. So we keep
- // the definition here for back compatibility.
- if (categoryTickStart != null) {
- idx = tick.value - categoryTickStart;
- }
- return cb(getAxisRawValue(axis, tick), idx, tick.level != null ? {
- level: tick.level
- } : null);
- };
- }(labelFormatter);
- } else {
- return function (tick) {
- return axis.scale.getLabel(tick);
- };
- }
- }
- export function getAxisRawValue(axis, tick) {
- // In category axis with data zoom, tick is not the original
- // index of axis.data. So tick should not be exposed to user
- // in category axis.
- return axis.type === 'category' ? axis.scale.getLabel(tick) : tick.value;
- }
- /**
- * @param axis
- * @return Be null/undefined if no labels.
- */
- export function estimateLabelUnionRect(axis) {
- var axisModel = axis.model;
- var scale = axis.scale;
- if (!axisModel.get(['axisLabel', 'show']) || scale.isBlank()) {
- return;
- }
- var realNumberScaleTicks;
- var tickCount;
- var categoryScaleExtent = scale.getExtent();
- // Optimize for large category data, avoid call `getTicks()`.
- if (scale instanceof OrdinalScale) {
- tickCount = scale.count();
- } else {
- realNumberScaleTicks = scale.getTicks();
- tickCount = realNumberScaleTicks.length;
- }
- var axisLabelModel = axis.getLabelModel();
- var labelFormatter = makeLabelFormatter(axis);
- var rect;
- var step = 1;
- // Simple optimization for large amount of labels
- if (tickCount > 40) {
- step = Math.ceil(tickCount / 40);
- }
- for (var i = 0; i < tickCount; i += step) {
- var tick = realNumberScaleTicks ? realNumberScaleTicks[i] : {
- value: categoryScaleExtent[0] + i
- };
- var label = labelFormatter(tick, i);
- var unrotatedSingleRect = axisLabelModel.getTextRect(label);
- var singleRect = rotateTextRect(unrotatedSingleRect, axisLabelModel.get('rotate') || 0);
- rect ? rect.union(singleRect) : rect = singleRect;
- }
- return rect;
- }
- function rotateTextRect(textRect, rotate) {
- var rotateRadians = rotate * Math.PI / 180;
- var beforeWidth = textRect.width;
- var beforeHeight = textRect.height;
- var afterWidth = beforeWidth * Math.abs(Math.cos(rotateRadians)) + Math.abs(beforeHeight * Math.sin(rotateRadians));
- var afterHeight = beforeWidth * Math.abs(Math.sin(rotateRadians)) + Math.abs(beforeHeight * Math.cos(rotateRadians));
- var rotatedRect = new BoundingRect(textRect.x, textRect.y, afterWidth, afterHeight);
- return rotatedRect;
- }
- /**
- * @param model axisLabelModel or axisTickModel
- * @return {number|String} Can be null|'auto'|number|function
- */
- export function getOptionCategoryInterval(model) {
- var interval = model.get('interval');
- return interval == null ? 'auto' : interval;
- }
- /**
- * Set `categoryInterval` as 0 implicitly indicates that
- * show all labels regardless of overlap.
- * @param {Object} axis axisModel.axis
- */
- export function shouldShowAllLabels(axis) {
- return axis.type === 'category' && getOptionCategoryInterval(axis.getLabelModel()) === 0;
- }
- export function getDataDimensionsOnAxis(data, axisDim) {
- // Remove duplicated dat dimensions caused by `getStackedDimension`.
- var dataDimMap = {};
- // Currently `mapDimensionsAll` will contain stack result dimension ('__\0ecstackresult').
- // PENDING: is it reasonable? Do we need to remove the original dim from "coord dim" since
- // there has been stacked result dim?
- zrUtil.each(data.mapDimensionsAll(axisDim), function (dataDim) {
- // For example, the extent of the original dimension
- // is [0.1, 0.5], the extent of the `stackResultDimension`
- // is [7, 9], the final extent should NOT include [0.1, 0.5],
- // because there is no graphic corresponding to [0.1, 0.5].
- // See the case in `test/area-stack.html` `main1`, where area line
- // stack needs `yAxis` not start from 0.
- dataDimMap[getStackedDimension(data, dataDim)] = true;
- });
- return zrUtil.keys(dataDimMap);
- }
- export function unionAxisExtentFromData(dataExtent, data, axisDim) {
- if (data) {
- zrUtil.each(getDataDimensionsOnAxis(data, axisDim), function (dim) {
- var seriesExtent = data.getApproximateExtent(dim);
- seriesExtent[0] < dataExtent[0] && (dataExtent[0] = seriesExtent[0]);
- seriesExtent[1] > dataExtent[1] && (dataExtent[1] = seriesExtent[1]);
- });
- }
- }
|