| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515 | 
/** 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 zrUtil = require("zrender/lib/core/util");var matrix = require("zrender/lib/core/matrix");var layoutUtil = require("../../util/layout");var axisHelper = require("../../coord/axisHelper");var ParallelAxis = require("./ParallelAxis");var graphic = require("../../util/graphic");var numberUtil = require("../../util/number");var sliderMove = require("../../component/helper/sliderMove");/** 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.*//** * Parallel Coordinates * <https://en.wikipedia.org/wiki/Parallel_coordinates> */var each = zrUtil.each;var mathMin = Math.min;var mathMax = Math.max;var mathFloor = Math.floor;var mathCeil = Math.ceil;var round = numberUtil.round;var PI = Math.PI;function Parallel(parallelModel, ecModel, api) {  /**   * key: dimension   * @type {Object.<string, module:echarts/coord/parallel/Axis>}   * @private   */  this._axesMap = zrUtil.createHashMap();  /**   * key: dimension   * value: {position: [], rotation, }   * @type {Object.<string, Object>}   * @private   */  this._axesLayout = {};  /**   * Always follow axis order.   * @type {Array.<string>}   * @readOnly   */  this.dimensions = parallelModel.dimensions;  /**   * @type {module:zrender/core/BoundingRect}   */  this._rect;  /**   * @type {module:echarts/coord/parallel/ParallelModel}   */  this._model = parallelModel;  this._init(parallelModel, ecModel, api);}Parallel.prototype = {  type: 'parallel',  constructor: Parallel,  /**   * Initialize cartesian coordinate systems   * @private   */  _init: function (parallelModel, ecModel, api) {    var dimensions = parallelModel.dimensions;    var parallelAxisIndex = parallelModel.parallelAxisIndex;    each(dimensions, function (dim, idx) {      var axisIndex = parallelAxisIndex[idx];      var axisModel = ecModel.getComponent('parallelAxis', axisIndex);      var axis = this._axesMap.set(dim, new ParallelAxis(dim, axisHelper.createScaleByModel(axisModel), [0, 0], axisModel.get('type'), axisIndex));      var isCategory = axis.type === 'category';      axis.onBand = isCategory && axisModel.get('boundaryGap');      axis.inverse = axisModel.get('inverse'); // Injection      axisModel.axis = axis;      axis.model = axisModel;      axis.coordinateSystem = axisModel.coordinateSystem = this;    }, this);  },  /**   * Update axis scale after data processed   * @param  {module:echarts/model/Global} ecModel   * @param  {module:echarts/ExtensionAPI} api   */  update: function (ecModel, api) {    this._updateAxesFromSeries(this._model, ecModel);  },  /**   * @override   */  containPoint: function (point) {    var layoutInfo = this._makeLayoutInfo();    var axisBase = layoutInfo.axisBase;    var layoutBase = layoutInfo.layoutBase;    var pixelDimIndex = layoutInfo.pixelDimIndex;    var pAxis = point[1 - pixelDimIndex];    var pLayout = point[pixelDimIndex];    return pAxis >= axisBase && pAxis <= axisBase + layoutInfo.axisLength && pLayout >= layoutBase && pLayout <= layoutBase + layoutInfo.layoutLength;  },  getModel: function () {    return this._model;  },  /**   * Update properties from series   * @private   */  _updateAxesFromSeries: function (parallelModel, ecModel) {    ecModel.eachSeries(function (seriesModel) {      if (!parallelModel.contains(seriesModel, ecModel)) {        return;      }      var data = seriesModel.getData();      each(this.dimensions, function (dim) {        var axis = this._axesMap.get(dim);        axis.scale.unionExtentFromData(data, data.mapDimension(dim));        axisHelper.niceScaleExtent(axis.scale, axis.model);      }, this);    }, this);  },  /**   * Resize the parallel coordinate system.   * @param {module:echarts/coord/parallel/ParallelModel} parallelModel   * @param {module:echarts/ExtensionAPI} api   */  resize: function (parallelModel, api) {    this._rect = layoutUtil.getLayoutRect(parallelModel.getBoxLayoutParams(), {      width: api.getWidth(),      height: api.getHeight()    });    this._layoutAxes();  },  /**   * @return {module:zrender/core/BoundingRect}   */  getRect: function () {    return this._rect;  },  /**   * @private   */  _makeLayoutInfo: function () {    var parallelModel = this._model;    var rect = this._rect;    var xy = ['x', 'y'];    var wh = ['width', 'height'];    var layout = parallelModel.get('layout');    var pixelDimIndex = layout === 'horizontal' ? 0 : 1;    var layoutLength = rect[wh[pixelDimIndex]];    var layoutExtent = [0, layoutLength];    var axisCount = this.dimensions.length;    var axisExpandWidth = restrict(parallelModel.get('axisExpandWidth'), layoutExtent);    var axisExpandCount = restrict(parallelModel.get('axisExpandCount') || 0, [0, axisCount]);    var axisExpandable = parallelModel.get('axisExpandable') && axisCount > 3 && axisCount > axisExpandCount && axisExpandCount > 1 && axisExpandWidth > 0 && layoutLength > 0; // `axisExpandWindow` is According to the coordinates of [0, axisExpandLength],    // for sake of consider the case that axisCollapseWidth is 0 (when screen is narrow),    // where collapsed axes should be overlapped.    var axisExpandWindow = parallelModel.get('axisExpandWindow');    var winSize;    if (!axisExpandWindow) {      winSize = restrict(axisExpandWidth * (axisExpandCount - 1), layoutExtent);      var axisExpandCenter = parallelModel.get('axisExpandCenter') || mathFloor(axisCount / 2);      axisExpandWindow = [axisExpandWidth * axisExpandCenter - winSize / 2];      axisExpandWindow[1] = axisExpandWindow[0] + winSize;    } else {      winSize = restrict(axisExpandWindow[1] - axisExpandWindow[0], layoutExtent);      axisExpandWindow[1] = axisExpandWindow[0] + winSize;    }    var axisCollapseWidth = (layoutLength - winSize) / (axisCount - axisExpandCount); // Avoid axisCollapseWidth is too small.    axisCollapseWidth < 3 && (axisCollapseWidth = 0); // Find the first and last indices > ewin[0] and < ewin[1].    var winInnerIndices = [mathFloor(round(axisExpandWindow[0] / axisExpandWidth, 1)) + 1, mathCeil(round(axisExpandWindow[1] / axisExpandWidth, 1)) - 1]; // Pos in ec coordinates.    var axisExpandWindow0Pos = axisCollapseWidth / axisExpandWidth * axisExpandWindow[0];    return {      layout: layout,      pixelDimIndex: pixelDimIndex,      layoutBase: rect[xy[pixelDimIndex]],      layoutLength: layoutLength,      axisBase: rect[xy[1 - pixelDimIndex]],      axisLength: rect[wh[1 - pixelDimIndex]],      axisExpandable: axisExpandable,      axisExpandWidth: axisExpandWidth,      axisCollapseWidth: axisCollapseWidth,      axisExpandWindow: axisExpandWindow,      axisCount: axisCount,      winInnerIndices: winInnerIndices,      axisExpandWindow0Pos: axisExpandWindow0Pos    };  },  /**   * @private   */  _layoutAxes: function () {    var rect = this._rect;    var axes = this._axesMap;    var dimensions = this.dimensions;    var layoutInfo = this._makeLayoutInfo();    var layout = layoutInfo.layout;    axes.each(function (axis) {      var axisExtent = [0, layoutInfo.axisLength];      var idx = axis.inverse ? 1 : 0;      axis.setExtent(axisExtent[idx], axisExtent[1 - idx]);    });    each(dimensions, function (dim, idx) {      var posInfo = (layoutInfo.axisExpandable ? layoutAxisWithExpand : layoutAxisWithoutExpand)(idx, layoutInfo);      var positionTable = {        horizontal: {          x: posInfo.position,          y: layoutInfo.axisLength        },        vertical: {          x: 0,          y: posInfo.position        }      };      var rotationTable = {        horizontal: PI / 2,        vertical: 0      };      var position = [positionTable[layout].x + rect.x, positionTable[layout].y + rect.y];      var rotation = rotationTable[layout];      var transform = matrix.create();      matrix.rotate(transform, transform, rotation);      matrix.translate(transform, transform, position); // TODO      // tick等排布信息。      // TODO      // 根据axis order 更新 dimensions顺序。      this._axesLayout[dim] = {        position: position,        rotation: rotation,        transform: transform,        axisNameAvailableWidth: posInfo.axisNameAvailableWidth,        axisLabelShow: posInfo.axisLabelShow,        nameTruncateMaxWidth: posInfo.nameTruncateMaxWidth,        tickDirection: 1,        labelDirection: 1      };    }, this);  },  /**   * Get axis by dim.   * @param {string} dim   * @return {module:echarts/coord/parallel/ParallelAxis} [description]   */  getAxis: function (dim) {    return this._axesMap.get(dim);  },  /**   * Convert a dim value of a single item of series data to Point.   * @param {*} value   * @param {string} dim   * @return {Array}   */  dataToPoint: function (value, dim) {    return this.axisCoordToPoint(this._axesMap.get(dim).dataToCoord(value), dim);  },  /**   * Travel data for one time, get activeState of each data item.   * @param {module:echarts/data/List} data   * @param {Functio} cb param: {string} activeState 'active' or 'inactive' or 'normal'   *                            {number} dataIndex   * @param {number} [start=0] the start dataIndex that travel from.   * @param {number} [end=data.count()] the next dataIndex of the last dataIndex will be travel.   */  eachActiveState: function (data, callback, start, end) {    start == null && (start = 0);    end == null && (end = data.count());    var axesMap = this._axesMap;    var dimensions = this.dimensions;    var dataDimensions = [];    var axisModels = [];    zrUtil.each(dimensions, function (axisDim) {      dataDimensions.push(data.mapDimension(axisDim));      axisModels.push(axesMap.get(axisDim).model);    });    var hasActiveSet = this.hasAxisBrushed();    for (var dataIndex = start; dataIndex < end; dataIndex++) {      var activeState;      if (!hasActiveSet) {        activeState = 'normal';      } else {        activeState = 'active';        var values = data.getValues(dataDimensions, dataIndex);        for (var j = 0, lenj = dimensions.length; j < lenj; j++) {          var state = axisModels[j].getActiveState(values[j]);          if (state === 'inactive') {            activeState = 'inactive';            break;          }        }      }      callback(activeState, dataIndex);    }  },  /**   * Whether has any activeSet.   * @return {boolean}   */  hasAxisBrushed: function () {    var dimensions = this.dimensions;    var axesMap = this._axesMap;    var hasActiveSet = false;    for (var j = 0, lenj = dimensions.length; j < lenj; j++) {      if (axesMap.get(dimensions[j]).model.getActiveState() !== 'normal') {        hasActiveSet = true;      }    }    return hasActiveSet;  },  /**   * Convert coords of each axis to Point.   *  Return point. For example: [10, 20]   * @param {Array.<number>} coords   * @param {string} dim   * @return {Array.<number>}   */  axisCoordToPoint: function (coord, dim) {    var axisLayout = this._axesLayout[dim];    return graphic.applyTransform([coord, 0], axisLayout.transform);  },  /**   * Get axis layout.   */  getAxisLayout: function (dim) {    return zrUtil.clone(this._axesLayout[dim]);  },  /**   * @param {Array.<number>} point   * @return {Object} {axisExpandWindow, delta, behavior: 'jump' | 'slide' | 'none'}.   */  getSlidedAxisExpandWindow: function (point) {    var layoutInfo = this._makeLayoutInfo();    var pixelDimIndex = layoutInfo.pixelDimIndex;    var axisExpandWindow = layoutInfo.axisExpandWindow.slice();    var winSize = axisExpandWindow[1] - axisExpandWindow[0];    var extent = [0, layoutInfo.axisExpandWidth * (layoutInfo.axisCount - 1)]; // Out of the area of coordinate system.    if (!this.containPoint(point)) {      return {        behavior: 'none',        axisExpandWindow: axisExpandWindow      };    } // Conver the point from global to expand coordinates.    var pointCoord = point[pixelDimIndex] - layoutInfo.layoutBase - layoutInfo.axisExpandWindow0Pos; // For dragging operation convenience, the window should not be    // slided when mouse is the center area of the window.    var delta;    var behavior = 'slide';    var axisCollapseWidth = layoutInfo.axisCollapseWidth;    var triggerArea = this._model.get('axisExpandSlideTriggerArea'); // But consider touch device, jump is necessary.    var useJump = triggerArea[0] != null;    if (axisCollapseWidth) {      if (useJump && axisCollapseWidth && pointCoord < winSize * triggerArea[0]) {        behavior = 'jump';        delta = pointCoord - winSize * triggerArea[2];      } else if (useJump && axisCollapseWidth && pointCoord > winSize * (1 - triggerArea[0])) {        behavior = 'jump';        delta = pointCoord - winSize * (1 - triggerArea[2]);      } else {        (delta = pointCoord - winSize * triggerArea[1]) >= 0 && (delta = pointCoord - winSize * (1 - triggerArea[1])) <= 0 && (delta = 0);      }      delta *= layoutInfo.axisExpandWidth / axisCollapseWidth;      delta ? sliderMove(delta, axisExpandWindow, extent, 'all') // Avoid nonsense triger on mousemove.      : behavior = 'none';    } // When screen is too narrow, make it visible and slidable, although it is hard to interact.    else {        var winSize = axisExpandWindow[1] - axisExpandWindow[0];        var pos = extent[1] * pointCoord / winSize;        axisExpandWindow = [mathMax(0, pos - winSize / 2)];        axisExpandWindow[1] = mathMin(extent[1], axisExpandWindow[0] + winSize);        axisExpandWindow[0] = axisExpandWindow[1] - winSize;      }    return {      axisExpandWindow: axisExpandWindow,      behavior: behavior    };  }};function restrict(len, extent) {  return mathMin(mathMax(len, extent[0]), extent[1]);}function layoutAxisWithoutExpand(axisIndex, layoutInfo) {  var step = layoutInfo.layoutLength / (layoutInfo.axisCount - 1);  return {    position: step * axisIndex,    axisNameAvailableWidth: step,    axisLabelShow: true  };}function layoutAxisWithExpand(axisIndex, layoutInfo) {  var layoutLength = layoutInfo.layoutLength;  var axisExpandWidth = layoutInfo.axisExpandWidth;  var axisCount = layoutInfo.axisCount;  var axisCollapseWidth = layoutInfo.axisCollapseWidth;  var winInnerIndices = layoutInfo.winInnerIndices;  var position;  var axisNameAvailableWidth = axisCollapseWidth;  var axisLabelShow = false;  var nameTruncateMaxWidth;  if (axisIndex < winInnerIndices[0]) {    position = axisIndex * axisCollapseWidth;    nameTruncateMaxWidth = axisCollapseWidth;  } else if (axisIndex <= winInnerIndices[1]) {    position = layoutInfo.axisExpandWindow0Pos + axisIndex * axisExpandWidth - layoutInfo.axisExpandWindow[0];    axisNameAvailableWidth = axisExpandWidth;    axisLabelShow = true;  } else {    position = layoutLength - (axisCount - 1 - axisIndex) * axisCollapseWidth;    nameTruncateMaxWidth = axisCollapseWidth;  }  return {    position: position,    axisNameAvailableWidth: axisNameAvailableWidth,    axisLabelShow: axisLabelShow,    nameTruncateMaxWidth: nameTruncateMaxWidth  };}var _default = Parallel;module.exports = _default;
 |