| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683 | 
/** 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.*/var _util = require("zrender/lib/core/util");var retrieve = _util.retrieve;var defaults = _util.defaults;var extend = _util.extend;var each = _util.each;var formatUtil = require("../../util/format");var graphic = require("../../util/graphic");var Model = require("../../model/Model");var _number = require("../../util/number");var isRadianAroundZero = _number.isRadianAroundZero;var remRadian = _number.remRadian;var _symbol = require("../../util/symbol");var createSymbol = _symbol.createSymbol;var matrixUtil = require("zrender/lib/core/matrix");var _vector = require("zrender/lib/core/vector");var v2ApplyTransform = _vector.applyTransform;var _axisHelper = require("../../coord/axisHelper");var shouldShowAllLabels = _axisHelper.shouldShowAllLabels;/** 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.*/var PI = Math.PI;/** * A final axis is translated and rotated from a "standard axis". * So opt.position and opt.rotation is required. * * A standard axis is and axis from [0, 0] to [0, axisExtent[1]], * for example: (0, 0) ------------> (0, 50) * * nameDirection or tickDirection or labelDirection is 1 means tick * or label is below the standard axis, whereas is -1 means above * the standard axis. labelOffset means offset between label and axis, * which is useful when 'onZero', where axisLabel is in the grid and * label in outside grid. * * Tips: like always, * positive rotation represents anticlockwise, and negative rotation * represents clockwise. * The direction of position coordinate is the same as the direction * of screen coordinate. * * Do not need to consider axis 'inverse', which is auto processed by * axis extent. * * @param {module:zrender/container/Group} group * @param {Object} axisModel * @param {Object} opt Standard axis parameters. * @param {Array.<number>} opt.position [x, y] * @param {number} opt.rotation by radian * @param {number} [opt.nameDirection=1] 1 or -1 Used when nameLocation is 'middle' or 'center'. * @param {number} [opt.tickDirection=1] 1 or -1 * @param {number} [opt.labelDirection=1] 1 or -1 * @param {number} [opt.labelOffset=0] Usefull when onZero. * @param {string} [opt.axisLabelShow] default get from axisModel. * @param {string} [opt.axisName] default get from axisModel. * @param {number} [opt.axisNameAvailableWidth] * @param {number} [opt.labelRotate] by degree, default get from axisModel. * @param {number} [opt.strokeContainThreshold] Default label interval when label * @param {number} [opt.nameTruncateMaxWidth] */var AxisBuilder = function (axisModel, opt) {  /**   * @readOnly   */  this.opt = opt;  /**   * @readOnly   */  this.axisModel = axisModel; // Default value  defaults(opt, {    labelOffset: 0,    nameDirection: 1,    tickDirection: 1,    labelDirection: 1,    silent: true  });  /**   * @readOnly   */  this.group = new graphic.Group(); // FIXME Not use a seperate text group?  var dumbGroup = new graphic.Group({    position: opt.position.slice(),    rotation: opt.rotation  }); // this.group.add(dumbGroup);  // this._dumbGroup = dumbGroup;  dumbGroup.updateTransform();  this._transform = dumbGroup.transform;  this._dumbGroup = dumbGroup;};AxisBuilder.prototype = {  constructor: AxisBuilder,  hasBuilder: function (name) {    return !!builders[name];  },  add: function (name) {    builders[name].call(this);  },  getGroup: function () {    return this.group;  }};var builders = {  /**   * @private   */  axisLine: function () {    var opt = this.opt;    var axisModel = this.axisModel;    if (!axisModel.get('axisLine.show')) {      return;    }    var extent = this.axisModel.axis.getExtent();    var matrix = this._transform;    var pt1 = [extent[0], 0];    var pt2 = [extent[1], 0];    if (matrix) {      v2ApplyTransform(pt1, pt1, matrix);      v2ApplyTransform(pt2, pt2, matrix);    }    var lineStyle = extend({      lineCap: 'round'    }, axisModel.getModel('axisLine.lineStyle').getLineStyle());    this.group.add(new graphic.Line({      // Id for animation      anid: 'line',      subPixelOptimize: true,      shape: {        x1: pt1[0],        y1: pt1[1],        x2: pt2[0],        y2: pt2[1]      },      style: lineStyle,      strokeContainThreshold: opt.strokeContainThreshold || 5,      silent: true,      z2: 1    }));    var arrows = axisModel.get('axisLine.symbol');    var arrowSize = axisModel.get('axisLine.symbolSize');    var arrowOffset = axisModel.get('axisLine.symbolOffset') || 0;    if (typeof arrowOffset === 'number') {      arrowOffset = [arrowOffset, arrowOffset];    }    if (arrows != null) {      if (typeof arrows === 'string') {        // Use the same arrow for start and end point        arrows = [arrows, arrows];      }      if (typeof arrowSize === 'string' || typeof arrowSize === 'number') {        // Use the same size for width and height        arrowSize = [arrowSize, arrowSize];      }      var symbolWidth = arrowSize[0];      var symbolHeight = arrowSize[1];      each([{        rotate: opt.rotation + Math.PI / 2,        offset: arrowOffset[0],        r: 0      }, {        rotate: opt.rotation - Math.PI / 2,        offset: arrowOffset[1],        r: Math.sqrt((pt1[0] - pt2[0]) * (pt1[0] - pt2[0]) + (pt1[1] - pt2[1]) * (pt1[1] - pt2[1]))      }], function (point, index) {        if (arrows[index] !== 'none' && arrows[index] != null) {          var symbol = createSymbol(arrows[index], -symbolWidth / 2, -symbolHeight / 2, symbolWidth, symbolHeight, lineStyle.stroke, true); // Calculate arrow position with offset          var r = point.r + point.offset;          var pos = [pt1[0] + r * Math.cos(opt.rotation), pt1[1] - r * Math.sin(opt.rotation)];          symbol.attr({            rotation: point.rotate,            position: pos,            silent: true,            z2: 11          });          this.group.add(symbol);        }      }, this);    }  },  /**   * @private   */  axisTickLabel: function () {    var axisModel = this.axisModel;    var opt = this.opt;    var ticksEls = buildAxisMajorTicks(this, axisModel, opt);    var labelEls = buildAxisLabel(this, axisModel, opt);    fixMinMaxLabelShow(axisModel, labelEls, ticksEls);    buildAxisMinorTicks(this, axisModel, opt);  },  /**   * @private   */  axisName: function () {    var opt = this.opt;    var axisModel = this.axisModel;    var name = retrieve(opt.axisName, axisModel.get('name'));    if (!name) {      return;    }    var nameLocation = axisModel.get('nameLocation');    var nameDirection = opt.nameDirection;    var textStyleModel = axisModel.getModel('nameTextStyle');    var gap = axisModel.get('nameGap') || 0;    var extent = this.axisModel.axis.getExtent();    var gapSignal = extent[0] > extent[1] ? -1 : 1;    var pos = [nameLocation === 'start' ? extent[0] - gapSignal * gap : nameLocation === 'end' ? extent[1] + gapSignal * gap : (extent[0] + extent[1]) / 2, // 'middle'    // Reuse labelOffset.    isNameLocationCenter(nameLocation) ? opt.labelOffset + nameDirection * gap : 0];    var labelLayout;    var nameRotation = axisModel.get('nameRotate');    if (nameRotation != null) {      nameRotation = nameRotation * PI / 180; // To radian.    }    var axisNameAvailableWidth;    if (isNameLocationCenter(nameLocation)) {      labelLayout = innerTextLayout(opt.rotation, nameRotation != null ? nameRotation : opt.rotation, // Adapt to axis.      nameDirection);    } else {      labelLayout = endTextLayout(opt, nameLocation, nameRotation || 0, extent);      axisNameAvailableWidth = opt.axisNameAvailableWidth;      if (axisNameAvailableWidth != null) {        axisNameAvailableWidth = Math.abs(axisNameAvailableWidth / Math.sin(labelLayout.rotation));        !isFinite(axisNameAvailableWidth) && (axisNameAvailableWidth = null);      }    }    var textFont = textStyleModel.getFont();    var truncateOpt = axisModel.get('nameTruncate', true) || {};    var ellipsis = truncateOpt.ellipsis;    var maxWidth = retrieve(opt.nameTruncateMaxWidth, truncateOpt.maxWidth, axisNameAvailableWidth); // FIXME    // truncate rich text? (consider performance)    var truncatedText = ellipsis != null && maxWidth != null ? formatUtil.truncateText(name, maxWidth, textFont, ellipsis, {      minChar: 2,      placeholder: truncateOpt.placeholder    }) : name;    var tooltipOpt = axisModel.get('tooltip', true);    var mainType = axisModel.mainType;    var formatterParams = {      componentType: mainType,      name: name,      $vars: ['name']    };    formatterParams[mainType + 'Index'] = axisModel.componentIndex;    var textEl = new graphic.Text({      // Id for animation      anid: 'name',      __fullText: name,      __truncatedText: truncatedText,      position: pos,      rotation: labelLayout.rotation,      silent: isLabelSilent(axisModel),      z2: 1,      tooltip: tooltipOpt && tooltipOpt.show ? extend({        content: name,        formatter: function () {          return name;        },        formatterParams: formatterParams      }, tooltipOpt) : null    });    graphic.setTextStyle(textEl.style, textStyleModel, {      text: truncatedText,      textFont: textFont,      textFill: textStyleModel.getTextColor() || axisModel.get('axisLine.lineStyle.color'),      textAlign: textStyleModel.get('align') || labelLayout.textAlign,      textVerticalAlign: textStyleModel.get('verticalAlign') || labelLayout.textVerticalAlign    });    if (axisModel.get('triggerEvent')) {      textEl.eventData = makeAxisEventDataBase(axisModel);      textEl.eventData.targetType = 'axisName';      textEl.eventData.name = name;    } // FIXME    this._dumbGroup.add(textEl);    textEl.updateTransform();    this.group.add(textEl);    textEl.decomposeTransform();  }};var makeAxisEventDataBase = AxisBuilder.makeAxisEventDataBase = function (axisModel) {  var eventData = {    componentType: axisModel.mainType,    componentIndex: axisModel.componentIndex  };  eventData[axisModel.mainType + 'Index'] = axisModel.componentIndex;  return eventData;};/** * @public * @static * @param {Object} opt * @param {number} axisRotation in radian * @param {number} textRotation in radian * @param {number} direction * @return {Object} { *  rotation, // according to axis *  textAlign, *  textVerticalAlign * } */var innerTextLayout = AxisBuilder.innerTextLayout = function (axisRotation, textRotation, direction) {  var rotationDiff = remRadian(textRotation - axisRotation);  var textAlign;  var textVerticalAlign;  if (isRadianAroundZero(rotationDiff)) {    // Label is parallel with axis line.    textVerticalAlign = direction > 0 ? 'top' : 'bottom';    textAlign = 'center';  } else if (isRadianAroundZero(rotationDiff - PI)) {    // Label is inverse parallel with axis line.    textVerticalAlign = direction > 0 ? 'bottom' : 'top';    textAlign = 'center';  } else {    textVerticalAlign = 'middle';    if (rotationDiff > 0 && rotationDiff < PI) {      textAlign = direction > 0 ? 'right' : 'left';    } else {      textAlign = direction > 0 ? 'left' : 'right';    }  }  return {    rotation: rotationDiff,    textAlign: textAlign,    textVerticalAlign: textVerticalAlign  };};function endTextLayout(opt, textPosition, textRotate, extent) {  var rotationDiff = remRadian(textRotate - opt.rotation);  var textAlign;  var textVerticalAlign;  var inverse = extent[0] > extent[1];  var onLeft = textPosition === 'start' && !inverse || textPosition !== 'start' && inverse;  if (isRadianAroundZero(rotationDiff - PI / 2)) {    textVerticalAlign = onLeft ? 'bottom' : 'top';    textAlign = 'center';  } else if (isRadianAroundZero(rotationDiff - PI * 1.5)) {    textVerticalAlign = onLeft ? 'top' : 'bottom';    textAlign = 'center';  } else {    textVerticalAlign = 'middle';    if (rotationDiff < PI * 1.5 && rotationDiff > PI / 2) {      textAlign = onLeft ? 'left' : 'right';    } else {      textAlign = onLeft ? 'right' : 'left';    }  }  return {    rotation: rotationDiff,    textAlign: textAlign,    textVerticalAlign: textVerticalAlign  };}var isLabelSilent = AxisBuilder.isLabelSilent = function (axisModel) {  var tooltipOpt = axisModel.get('tooltip');  return axisModel.get('silent') // Consider mouse cursor, add these restrictions.  || !(axisModel.get('triggerEvent') || tooltipOpt && tooltipOpt.show);};function fixMinMaxLabelShow(axisModel, labelEls, tickEls) {  if (shouldShowAllLabels(axisModel.axis)) {    return;  } // If min or max are user set, we need to check  // If the tick on min(max) are overlap on their neighbour tick  // If they are overlapped, we need to hide the min(max) tick label  var showMinLabel = axisModel.get('axisLabel.showMinLabel');  var showMaxLabel = axisModel.get('axisLabel.showMaxLabel'); // FIXME  // Have not consider onBand yet, where tick els is more than label els.  labelEls = labelEls || [];  tickEls = tickEls || [];  var firstLabel = labelEls[0];  var nextLabel = labelEls[1];  var lastLabel = labelEls[labelEls.length - 1];  var prevLabel = labelEls[labelEls.length - 2];  var firstTick = tickEls[0];  var nextTick = tickEls[1];  var lastTick = tickEls[tickEls.length - 1];  var prevTick = tickEls[tickEls.length - 2];  if (showMinLabel === false) {    ignoreEl(firstLabel);    ignoreEl(firstTick);  } else if (isTwoLabelOverlapped(firstLabel, nextLabel)) {    if (showMinLabel) {      ignoreEl(nextLabel);      ignoreEl(nextTick);    } else {      ignoreEl(firstLabel);      ignoreEl(firstTick);    }  }  if (showMaxLabel === false) {    ignoreEl(lastLabel);    ignoreEl(lastTick);  } else if (isTwoLabelOverlapped(prevLabel, lastLabel)) {    if (showMaxLabel) {      ignoreEl(prevLabel);      ignoreEl(prevTick);    } else {      ignoreEl(lastLabel);      ignoreEl(lastTick);    }  }}function ignoreEl(el) {  el && (el.ignore = true);}function isTwoLabelOverlapped(current, next, labelLayout) {  // current and next has the same rotation.  var firstRect = current && current.getBoundingRect().clone();  var nextRect = next && next.getBoundingRect().clone();  if (!firstRect || !nextRect) {    return;  } // When checking intersect of two rotated labels, we use mRotationBack  // to avoid that boundingRect is enlarge when using `boundingRect.applyTransform`.  var mRotationBack = matrixUtil.identity([]);  matrixUtil.rotate(mRotationBack, mRotationBack, -current.rotation);  firstRect.applyTransform(matrixUtil.mul([], mRotationBack, current.getLocalTransform()));  nextRect.applyTransform(matrixUtil.mul([], mRotationBack, next.getLocalTransform()));  return firstRect.intersect(nextRect);}function isNameLocationCenter(nameLocation) {  return nameLocation === 'middle' || nameLocation === 'center';}function createTicks(ticksCoords, tickTransform, tickEndCoord, tickLineStyle, aniid) {  var tickEls = [];  var pt1 = [];  var pt2 = [];  for (var i = 0; i < ticksCoords.length; i++) {    var tickCoord = ticksCoords[i].coord;    pt1[0] = tickCoord;    pt1[1] = 0;    pt2[0] = tickCoord;    pt2[1] = tickEndCoord;    if (tickTransform) {      v2ApplyTransform(pt1, pt1, tickTransform);      v2ApplyTransform(pt2, pt2, tickTransform);    } // Tick line, Not use group transform to have better line draw    var tickEl = new graphic.Line({      // Id for animation      anid: aniid + '_' + ticksCoords[i].tickValue,      subPixelOptimize: true,      shape: {        x1: pt1[0],        y1: pt1[1],        x2: pt2[0],        y2: pt2[1]      },      style: tickLineStyle,      z2: 2,      silent: true    });    tickEls.push(tickEl);  }  return tickEls;}function buildAxisMajorTicks(axisBuilder, axisModel, opt) {  var axis = axisModel.axis;  var tickModel = axisModel.getModel('axisTick');  if (!tickModel.get('show') || axis.scale.isBlank()) {    return;  }  var lineStyleModel = tickModel.getModel('lineStyle');  var tickEndCoord = opt.tickDirection * tickModel.get('length');  var ticksCoords = axis.getTicksCoords();  var ticksEls = createTicks(ticksCoords, axisBuilder._transform, tickEndCoord, defaults(lineStyleModel.getLineStyle(), {    stroke: axisModel.get('axisLine.lineStyle.color')  }), 'ticks');  for (var i = 0; i < ticksEls.length; i++) {    axisBuilder.group.add(ticksEls[i]);  }  return ticksEls;}function buildAxisMinorTicks(axisBuilder, axisModel, opt) {  var axis = axisModel.axis;  var minorTickModel = axisModel.getModel('minorTick');  if (!minorTickModel.get('show') || axis.scale.isBlank()) {    return;  }  var minorTicksCoords = axis.getMinorTicksCoords();  if (!minorTicksCoords.length) {    return;  }  var lineStyleModel = minorTickModel.getModel('lineStyle');  var tickEndCoord = opt.tickDirection * minorTickModel.get('length');  var minorTickLineStyle = defaults(lineStyleModel.getLineStyle(), defaults(axisModel.getModel('axisTick').getLineStyle(), {    stroke: axisModel.get('axisLine.lineStyle.color')  }));  for (var i = 0; i < minorTicksCoords.length; i++) {    var minorTicksEls = createTicks(minorTicksCoords[i], axisBuilder._transform, tickEndCoord, minorTickLineStyle, 'minorticks_' + i);    for (var k = 0; k < minorTicksEls.length; k++) {      axisBuilder.group.add(minorTicksEls[k]);    }  }}function buildAxisLabel(axisBuilder, axisModel, opt) {  var axis = axisModel.axis;  var show = retrieve(opt.axisLabelShow, axisModel.get('axisLabel.show'));  if (!show || axis.scale.isBlank()) {    return;  }  var labelModel = axisModel.getModel('axisLabel');  var labelMargin = labelModel.get('margin');  var labels = axis.getViewLabels(); // Special label rotate.  var labelRotation = (retrieve(opt.labelRotate, labelModel.get('rotate')) || 0) * PI / 180;  var labelLayout = innerTextLayout(opt.rotation, labelRotation, opt.labelDirection);  var rawCategoryData = axisModel.getCategories && axisModel.getCategories(true);  var labelEls = [];  var silent = isLabelSilent(axisModel);  var triggerEvent = axisModel.get('triggerEvent');  each(labels, function (labelItem, index) {    var tickValue = labelItem.tickValue;    var formattedLabel = labelItem.formattedLabel;    var rawLabel = labelItem.rawLabel;    var itemLabelModel = labelModel;    if (rawCategoryData && rawCategoryData[tickValue] && rawCategoryData[tickValue].textStyle) {      itemLabelModel = new Model(rawCategoryData[tickValue].textStyle, labelModel, axisModel.ecModel);    }    var textColor = itemLabelModel.getTextColor() || axisModel.get('axisLine.lineStyle.color');    var tickCoord = axis.dataToCoord(tickValue);    var pos = [tickCoord, opt.labelOffset + opt.labelDirection * labelMargin];    var textEl = new graphic.Text({      // Id for animation      anid: 'label_' + tickValue,      position: pos,      rotation: labelLayout.rotation,      silent: silent,      z2: 10    });    graphic.setTextStyle(textEl.style, itemLabelModel, {      text: formattedLabel,      textAlign: itemLabelModel.getShallow('align', true) || labelLayout.textAlign,      textVerticalAlign: itemLabelModel.getShallow('verticalAlign', true) || itemLabelModel.getShallow('baseline', true) || labelLayout.textVerticalAlign,      textFill: typeof textColor === 'function' ? textColor( // (1) 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.      // (2) Compatible with previous version, which always use formatted label as      // input. But in interval scale the formatted label is like '223,445', which      // maked user repalce ','. So we modify it to return original val but remain      // it as 'string' to avoid error in replacing.      axis.type === 'category' ? rawLabel : axis.type === 'value' ? tickValue + '' : tickValue, index) : textColor    }); // Pack data for mouse event    if (triggerEvent) {      textEl.eventData = makeAxisEventDataBase(axisModel);      textEl.eventData.targetType = 'axisLabel';      textEl.eventData.value = rawLabel;    } // FIXME    axisBuilder._dumbGroup.add(textEl);    textEl.updateTransform();    labelEls.push(textEl);    axisBuilder.group.add(textEl);    textEl.decomposeTransform();  });  return labelEls;}var _default = AxisBuilder;module.exports = _default;
 |