List.js 19.4 KB
Newer Older
L
lang 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
/**
 * List for data storage
 * @module echarts/data/List
 */
define(function (require) {

    var UNDEFINED = 'undefined';
    var global = window;
    var Float32Array = typeof global.Float32Array === UNDEFINED
        ? Array : global.Float32Array;
    var Int32Array = typeof global.Int32Array === UNDEFINED
        ? Array : global.Int32Array;

    var dataCtors = {
        float: Float32Array,
        int: Int32Array,
        'number': Array
    }
L
lang 已提交
19

L
lang 已提交
20
    var Model = require('../model/Model');
L
lang 已提交
21
    var DataDiffer = require('./DataDiffer');
L
lang 已提交
22

L
lang 已提交
23 24
    var zrUtil = require('zrender/core/util');
    var isObject = zrUtil.isObject;
L
lang 已提交
25 26

    /**
L
lang 已提交
27 28
     * @constructor
     * @alias module:echarts/data/List
L
lang 已提交
29
     */
L
lang 已提交
30 31 32 33 34 35 36 37 38 39 40 41 42 43
    var List = function (dimensions, seriesModel) {

        dimensions = dimensions || ['x', 'y'];

        var dimensionInfos = [];
        var dimensionNames = [];
        for (var i = 0; i < dimensions.length; i++) {
            var dimensionName;
            var dimensionInfo = {};
            if (typeof dimensions[i] === 'string') {
                dimensionName = dimensions[i];
                dimensionInfo = {
                    name: dimensionName,
                    // Type can be 'float', 'int', 'number'
L
lang 已提交
44
                    // Default is number, Precision of float may not enough
L
lang 已提交
45 46
                    type: 'number'
                };
L
lang 已提交
47
            }
L
lang 已提交
48 49 50 51 52 53 54
            else {
                dimensionInfo = dimensions[i];
                dimensionName = dimensionInfo.name;
                dimensionInfo.type = dimensionInfo.type || 'float'
            }
            dimensionNames.push(dimensionName);
            dimensionInfos.push(dimensionInfo);
L
lang 已提交
55
        }
L
lang 已提交
56 57 58 59 60
        /**
         * @readOnly
         * @type {Array.<string>}
         */
        this.dimensions = dimensionNames;
P
pah100 已提交
61

L
lang 已提交
62 63 64 65
        /**
         * Infomation of each data dimension, like data type.
         */
        this._dimensionInfos = dimensionInfos;
L
lang 已提交
66

L
lang 已提交
67 68 69 70
        /**
         * @type {module:echarts/model/Model}
         */
        this.seriesModel = seriesModel;
L
lang 已提交
71

72
        /**
L
lang 已提交
73 74 75 76
         * Indices stores the indices of data subset after filtered.
         * This data subset will be used in chart.
         * @type {Array.<number>}
         * @readOnly
77
         */
L
lang 已提交
78
        this.indices = [];
79 80

        /**
L
lang 已提交
81 82
         * Dimensions hint for regenerating the raw value
         * @type {Array.<string>}
83
         */
L
lang 已提交
84
        this._rawValueDims = ['x'];
85 86

        /**
L
lang 已提交
87 88 89
         * Data storage
         * @type {Object.<key, TypedArray|Array>}
         * @private
90
         */
L
lang 已提交
91
        this._storage = {};
92

L
lang 已提交
93
        /**
L
lang 已提交
94 95 96
         * Models of data option is stored sparse for optimizing memory cost
         * @type {Array.<module:echarts/model/Model>}
         * @private
L
lang 已提交
97
         */
L
lang 已提交
98
        this._optionModels = [];
L
lang 已提交
99 100

        /**
L
lang 已提交
101
         * @param {module:echarts/data/List}
L
lang 已提交
102
         */
L
lang 已提交
103
        this.stackedOn = null;
L
lang 已提交
104

105
        /**
L
lang 已提交
106 107 108
         * Global visual properties after visual coding
         * @type {Object}
         * @private
109
         */
L
lang 已提交
110
        this._visual = {};
L
lang 已提交
111

L
lang 已提交
112
        /**
L
lang 已提交
113 114 115
         * Item visual properties after visual coding
         * @type {Array.<Object>}
         * @private
L
lang 已提交
116
         */
L
lang 已提交
117
        this._itemVisuals = [];
L
lang 已提交
118

L
lang 已提交
119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175
        /**
         * Item layout properties after layout
         * @type {Array.<Object>}
         * @private
         */
        this._itemLayouts = [];

        /**
         * Graphic elemnents
         * @type {Array.<module:zrender/Element>}
         * @private
         */
        this._graphicEls = [];
    }

    var listProto = List.prototype;

    listProto.type = 'list';

    /**
     * Initialize from data
     */
    listProto.initData = function (data) {
        // Clear
        var optionModels = this._optionModels = [];
        var storage = this._storage = {};
        var indices = this.indices = [];

        var dimensions = this.dimensions;
        var size = data.length;

        // Init storage
        for (var i = 0; i < dimensions.length; i++) {
            var dimInfo = this._dimensionInfos[i];
            var DataCtor = dataCtors[dimInfo.type];
            storage[dimensions[i]] = new DataCtor(size);
        }

        // Special storage of indices of option model
        // It is used for indexing the model in List#_optionModels
        var optionModelIndices = storage.$optionModelIndices = new Int32Array(size);

        var tempValue = [];
        var rawValue1D = false;
        for (var idx = 0; idx < data.length; idx++) {
            var value = data[idx];
            // Each data item contains value and option
            if (data[idx] != null && data[idx].hasOwnProperty('value')) {
                value = data[idx].value;
                var model = new Model(data[idx], this.seriesModel);
                var modelIdx = optionModels.length;
                optionModelIndices[idx] = modelIdx;
                optionModels.push(model);
            }
            else {
                // Reference to the undefined
                optionModelIndices[idx] = -1;
L
lang 已提交
176
            }
L
lang 已提交
177 178 179 180 181 182 183 184
            // Bar chart, line chart which uses category axis
            // only gives the 'y' value. 'x' value is the indices of cateogry
            if (typeof (value) === 'number') {
                // Use a tempValue to normalize the value to be a (x, y) value
                tempValue[0] = idx;
                tempValue[1] = value;
                value = tempValue;
                rawValue1D = true;
185 186
            }

L
lang 已提交
187 188 189 190 191 192 193 194
            // Store the data by dimensions
            for (var k = 0; k < dimensions.length; k++) {
                var dim = dimensions[k];
                var dimStorage = storage[dim];
                var dimValue = value[k];
                // PENDING NULL is empty or zero
                if (dimValue == null || dimValue === '-') {
                    dimValue = NaN;
195
                }
L
lang 已提交
196
                dimStorage[idx] = dimValue;
197 198
            }

L
lang 已提交
199 200
            indices.push(idx);
        }
L
lang 已提交
201

L
lang 已提交
202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
        this._rawValueDims = rawValue1D ? dimensions.slice(1, 2) : dimensions.slice();
    };

    /**
     * @return {number}
     */
    listProto.count = function () {
        return this.indices.length;
    };

    /**
     * Get value
     * @param {string} dim
     * @param {number} idx
     * @param {boolean} stack
     * @return {number}
     */
    listProto.get = function (dim, idx, stack) {
        var storage = this._storage;
        var dataIndex = this.indices[idx];

        var value = storage[dim] && storage[dim][dataIndex];
        if (stack && this.stackedOn) {
            var stackedValue = this.stackedOn.get(dim, idx, stack);
L
lang 已提交
226 227 228 229 230
            // Considering positive stack, negative stack and empty data
            if ((value >= 0 && stackedValue > 0)  // Positive stack
                || (value <= 0 && stackedValue < 0) // Negative stack
            ) {
                value += stackedValue;
L
lang 已提交
231 232 233 234
            }
        }
        return value;
    };
L
lang 已提交
235

L
lang 已提交
236 237 238 239 240 241 242 243 244 245 246
    /**
     * If value is NaN. Inlcuding '-'
     * @param {string} dim
     * @param {number} idx
     * @return {number}
     */
    listProto.hasValue = function (idx) {
        var dimensions = this.dimensions;
        for (var i = 0, len = dimensions.length; i < len; i++) {
            if (isNaN(this.get(dimensions[i], idx))) {
                return false;
L
lang 已提交
247
            }
L
lang 已提交
248 249 250
        }
        return true;
    }
L
lang 已提交
251

L
lang 已提交
252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270
    /**
     * Get extent of data in one dimension
     * @param {string} dim
     * @param {boolean} stack
     */
    listProto.getDataExtent = function (dim, stack) {
        var dimData = this._storage[dim];
        var min = Infinity;
        var max = -Infinity;
        var value;
        if (dimData) {
            for (var i = 0, len = this.count(); i < len; i++) {
                value = this.get(dim, i, stack);
                value < min && (min = value);
                value > max && (max = value);
            }
        }
        return [min, max];
    };
L
lang 已提交
271

L
lang 已提交
272 273 274 275 276 277 278 279 280 281 282
    /**
     * Get raw value
     * @param {number} idx
     * @return {number}
     */
    listProto.getRawValue = function (idx) {
        var rawValueDims = this._rawValueDims;
        var storage = this._storage;
        if (rawValueDims.length === 1) {
            var dimData = storage[rawValueDims[0]];
            return dimData && dimData[idx];
L
lang 已提交
283
        }
L
lang 已提交
284 285 286 287
        else {
            var value = [];
            for (var i = 0; i < rawValueDims.length; i++) {
                value[i] = this.get(rawValueDims[i], idx);
288
            }
L
lang 已提交
289 290 291
            return value;
        }
    };
L
lang 已提交
292

L
lang 已提交
293 294 295
    /**
     * Get raw data index
     */
L
lang 已提交
296
    listProto.getRawIndex = function (idx) {
L
lang 已提交
297 298
        return this.indices[idx];
    };
L
lang 已提交
299

L
lang 已提交
300 301 302 303 304 305
    function normalizeDimensions(dimensions) {
        if (typeof (dimensions) === 'string') {
            dimensions = [dimensions];
        }
        return dimensions;
    }
L
lang 已提交
306

L
lang 已提交
307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322
    function getStackDimMap(stackDim, dimensions) {
        if (! stackDim) {
            return {};
        }
        if (typeof stackDim === 'string') {
            stackDim = [stackDim];
        }
        var stackDimMap = {};
        // Avoid get the undefined value
        for (var i = 0; i < dimensions.length; i++) {
            stackDimMap[dimensions[i]] = false;
        }
        for (var i = 0; i < stackDim.length; i++) {
            stackDimMap[stackDim[i]] = true;
        }
        return stackDimMap;
L
lang 已提交
323
    }
L
lang 已提交
324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342
    /**
     * Data iteration
     * @param {string|Array.<string>}
     * @param {Function} cb
     * @param {boolean} [stack=false]
     * @param {*} [context=this]
     *
     * @example
     *  list.each('x', function (x, idx) {});
     *  list.each(['x', 'y'], function (x, y, idx) {});
     *  list.each(function (idx) {})
     */
    listProto.each = function (dimensions, cb, stack, context) {
        if (typeof dimensions === 'function') {
            context = stack;
            stack = cb;
            cb = dimensions;
            dimensions = [];
        }
L
lang 已提交
343

L
lang 已提交
344
        dimensions = normalizeDimensions(dimensions);
L
lang 已提交
345

L
lang 已提交
346 347 348
        var value = [];
        var dimSize = dimensions.length;
        var indices = this.indices;
L
lang 已提交
349

L
lang 已提交
350 351 352 353
        // Only stacked on the value axis
        var stackDimMap = getStackDimMap(this._rawValueDims, dimensions);
        // Optimizing for 1 dim case
        var firstDimStack = stackDimMap[dimensions[0]];
L
lang 已提交
354

L
lang 已提交
355
        context = context || this;
L
lang 已提交
356

L
lang 已提交
357 358 359
        for (var i = 0; i < indices.length; i++) {
            if (dimSize === 0) {
                // FIXME Pass value as parameter ?
L
lang 已提交
360
                cb.call(context, i);
L
lang 已提交
361 362 363
            }
            // Simple optimization
            else if (dimSize === 1) {
L
lang 已提交
364
                cb.call(context, this.get(dimensions[0], i, firstDimStack), i);
L
lang 已提交
365 366 367 368 369 370 371 372 373 374 375
            }
            else {
                for (var k = 0; k < dimSize; k++) {
                    value[k] = this.get(dimensions[k], i, stackDimMap[dimensions[k]]);
                }
                // Index
                value[k] = i;
                cb.apply(context, value);
            }
        }
    };
L
lang 已提交
376

L
lang 已提交
377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402
    /**
     * Data filter
     * @param {string|Array.<string>}
     * @param {Function} cb
     * @param {boolean} [stack=false]
     * @param {*} [context=this]
     */
    listProto.filterSelf = function (dimensions, cb, stack, context) {
        dimensions = normalizeDimensions(dimensions);

        var newIndices = [];
        var value = [];
        var dimSize = dimensions.length;
        var indices = this.indices;

        // Only stacked on the value axis
        var stackDimMap = getStackDimMap(this._rawValueDims, dimensions);
        // Optimizing for 1 dim case
        var firstDimStack = stackDimMap[dimensions[0]];

        context = context || this;

        for (var i = 0; i < indices.length; i++) {
            var keep;
            // Simple optimization
            if (dimSize === 1) {
L
lang 已提交
403
                keep = cb.call(
L
lang 已提交
404 405 406 407 408 409 410 411 412 413 414 415 416 417
                    context, this.get(dimensions[0], i, firstDimStack), i
                );
            }
            else {
                for (var k = 0; k < dimSize; k++) {
                    value[k] = this.get(dimensions[k], i, stackDimMap[dimensions[k]]);
                }
                value[k] = i;
                keep = cb.apply(context, value);
            }
            if (keep) {
                newIndices.push(indices[i]);
            }
        }
P
pah100 已提交
418

L
lang 已提交
419
        this.indices = newIndices;
L
lang 已提交
420

L
lang 已提交
421 422
        return this;
    };
L
lang 已提交
423

L
lang 已提交
424 425 426 427 428 429 430 431 432 433 434 435 436 437
    /**
     * Data mapping
     * @param {string|Array.<string>}
     * @param {Function} cb
     * @param {boolean} [stack=false]
     * @param {*} [context=this]
     */
    listProto.map = function (dimensions, cb, stack, context) {
        if (typeof dimensions === 'function') {
            context = stack;
            stack = cb;
            cb = dimensions;
            dimensions = [];
        }
L
lang 已提交
438

L
lang 已提交
439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500
        var result = [];
        this.each(dimensions, function () {
            result.push(cb && cb.apply(this, arguments));
        }, stack, context);
        return result;
    };

    var temporaryModel = new Model(null);
    /**
     * Get model of one data item.
     * It will create a temporary model if value on idx is not an option.
     */
    listProto.getItemModel = function (idx) {
        var storage = this._storage;
        var optionModelIndices = storage.$optionModelIndices;
        var modelIndex = optionModelIndices && optionModelIndices[idx];

        var model = this._optionModels[modelIndex];

        if (! model) {
            // Use a temporary model proxy if value on idx is not an option.
            // FIXME Create a new one may cause memory leak
            model = temporaryModel;
            model.parentModel = this.seriesModel;
        }
        return model;
    };

    /**
     * Create a data differ
     * @param {module:echarts/data/List} oldList
     * @return {module:echarts/data/DataDiffer}
     */
    listProto.diff = function (oldList) {
        return new DataDiffer(oldList ? oldList.indices : [], this.indices);
    };

    /**
     * Get visual property.
     * @param {string} key
     */
    listProto.getVisual = function (key) {
        var visual = this._visual;
        return visual && visual[key];
    };

    /**
     * Set visual property
     * @param {string|Object} key
     * @param {*} [value]
     *
     * @example
     *  setVisual('color', color);
     *  setVisual({
     *      'color': color
     *  });
     */
    listProto.setVisual = function (key, val) {
        if (isObject(key)) {
            for (var name in key) {
                if (key.hasOwnProperty(name)) {
                    this.setVisual(name, key[name]);
L
lang 已提交
501 502
                }
            }
L
lang 已提交
503 504 505 506 507
            return;
        }
        this._visual = this._visual || {};
        this._visual[key] = val;
    };
L
lang 已提交
508

L
lang 已提交
509 510 511 512 513 514 515
    /**
     * Get layout of single data item
     * @param {number} idx
     */
    listProto.getItemLayout = function (idx) {
        return this._itemLayouts[idx];
    },
L
lang 已提交
516

L
lang 已提交
517 518 519 520 521 522 523 524
    /**
     * Set layout of single data item
     * @param {number} idx
     * @param {Object} layout
     */
    listProto.setItemLayout = function (idx, layout) {
        this._itemLayouts[idx] = layout;
    },
L
lang 已提交
525

L
lang 已提交
526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562
    /**
     * Get visual property of single data item
     * @param {number} idx
     * @param {string} key
     */
    listProto.getItemVisual = function (idx, key) {
        var itemVisual = this._itemVisuals[idx];
        var val = itemVisual && itemVisual[key];
        if (val == null) {
            // Use global visual property
            return this.getVisual(key);
        }
        return val;
    },

    /**
     * Set visual property of single data item
     *
     * @param {number} idx
     * @param {string|Object} key
     * @param {*} [value]
     *
     * @example
     *  setItemVisual(0, 'color', color);
     *  setItemVisual(0, {
     *      'color': color
     *  });
     */
    listProto.setItemVisual = function (idx, key, value) {
        var itemVisual = this._itemVisuals[idx] || {};
        this._itemVisuals[idx] = itemVisual;

        if (isObject(key)) {
            for (var name in key) {
                if (key.hasOwnProperty(name)) {
                    itemVisual[key] = key[name];
                }
L
lang 已提交
563
            }
L
lang 已提交
564
            return;
L
lang 已提交
565
        }
L
lang 已提交
566
        itemVisual[key] = value;
L
lang 已提交
567 568
    };

L
lang 已提交
569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606
    /**
     * @param {number} idx
     * @param {module:zrender/Element} el
     */
    listProto.setItemGraphicEl = function (idx, el) {
        this._graphicEls[idx] = el;
    };

    /**
     * @param {number} idx
     * @return {module:zrender/Element}
     */
    listProto.getItemGraphicEl = function (idx) {
        return this._graphicEls[idx];
    };

    /**
     * @param {Function} cb
     * @param {*} context
     */
    listProto.eachItemGraphicEl = function (cb, context) {
        zrUtil.each(this._graphicEls, cb, context);
    };

    /**
     * Shallow clone a new list except visual and layout properties, and graph elements.
     * New list only change the indices.
     */
    listProto.cloneShallow = function () {
        var list = new List(this._dimensionInfos, this.seriesModel);
        list.stackedOn = this.stackedOn;

        // FIXME
        list._storage = this._storage;
        list._optionModels = this._optionModels;
        list._rawValueDims = this._rawValueDims;

        list.indices = this.indices.slice();
P
pah100 已提交
607

L
lang 已提交
608 609
        return list;
    };
L
lang 已提交
610

L
lang 已提交
611 612 613
    /**
     * Helper function to create a list from option data
     */
614
    List.fromArray = function (data, seriesModel, ecModel) {
L
lang 已提交
615
        var coordinateSystem = seriesModel.get('coordinateSystem');
L
lang 已提交
616
        var dimensions;
617

L
lang 已提交
618
        var categoryAxisModel;
L
lang 已提交
619 620 621 622 623 624
        // FIXME
        // 这里 List 跟几个坐标系和坐标系 Model 耦合了
        if (coordinateSystem === 'cartesian2d') {
            var xAxisModel = ecModel.getComponent('xAxis', seriesModel.get('xAxisIndex'));
            var yAxisModel = ecModel.getComponent('yAxis', seriesModel.get('yAxisIndex'));
            if (xAxisModel.get('type') === 'category') {
L
lang 已提交
625
                dimensions = ['x', 'y'];
L
lang 已提交
626 627

                categoryAxisModel = xAxisModel;
L
lang 已提交
628
            }
L
lang 已提交
629
            else if (yAxisModel.get('type') === 'category') {
L
lang 已提交
630
                dimensions = ['y', 'x'];
L
lang 已提交
631

P
pah100 已提交
632
                categoryAxisModel = yAxisModel;
L
lang 已提交
633 634 635
            }
            else {
                // PENDING
L
lang 已提交
636 637 638
                var dimSize = data[0] && data[0].length;
                if (dimSize === 2) {
                    dimensions = ['x', 'y'];
L
lang 已提交
639
                }
L
lang 已提交
640 641
                else if (dimSize === 3) {
                    dimensions = ['x', 'y', 'z'];
L
lang 已提交
642 643 644 645
                }
            }
        }
        else if (coordinateSystem === 'polar') {
L
lang 已提交
646
            var axisFinder = function (axisModel) {
L
lang 已提交
647
                return axisModel.get('polarIndex') === polarIndex;
P
pah100 已提交
648
            }
L
lang 已提交
649 650 651 652 653
            var polarIndex = seriesModel.get('polarIndex') || 0;
            var angleAxisModel = ecModel.findComponent('angleAxis', axisFinder);
            var radiusAxisModel = ecModel.findComponent('radiusAxis', axisFinder);

            if (angleAxisModel.get('type') === 'category') {
L
lang 已提交
654
                dimensions = ['angle', 'radius'];
L
lang 已提交
655 656

                categoryAxisModel = angleAxisModel;
L
lang 已提交
657 658
            }
            else if (radiusAxisModel.get('type') === 'category') {
L
lang 已提交
659
                dimensions = ['radius', 'angle'];
L
lang 已提交
660 661

                categoryAxisModel = radiusAxisModel;
L
lang 已提交
662 663
            }
            else {
L
lang 已提交
664
                // PENDING
L
lang 已提交
665 666 667
                var dimSize = data[0] && data[0].length;
                if (dimSize === 2) {
                    dimensions = ['radius', 'angle'];
L
lang 已提交
668
                }
L
lang 已提交
669 670
                else if (dimSize === 3) {
                    dimensions = ['radius', 'angle', 'value'];
L
lang 已提交
671
                }
L
lang 已提交
672 673
            }
        }
674

L
lang 已提交
675 676 677
        var list = new List(dimensions, seriesModel);

        list.initData(data);
678

L
lang 已提交
679
        return list;
L
lang 已提交
680 681
    };

L
lang 已提交
682 683
    return List;
});