List.js 27.8 KB
Newer Older
L
lang 已提交
1 2 3 4 5 6 7
/**
 * List for data storage
 * @module echarts/data/List
 */
define(function (require) {

    var UNDEFINED = 'undefined';
L
lang 已提交
8 9 10 11 12
    var globalObj = typeof window === 'undefined' ? global : window;
    var Float64Array = typeof globalObj.Float64Array === UNDEFINED
        ? Array : globalObj.Float64Array;
    var Int32Array = typeof globalObj.Int32Array === UNDEFINED
        ? Array : globalObj.Int32Array;
L
lang 已提交
13 14

    var dataCtors = {
15
        'float': Float64Array,
P
pah100 已提交
16
        'int': Int32Array,
L
lang 已提交
17
        // Ordinal data type can be string or int
P
pah100 已提交
18
        'ordinal': Array,
19 20
        'number': Array,
        'time': Array
L
lang 已提交
21
    };
L
lang 已提交
22

L
lang 已提交
23
    var Model = require('../model/Model');
L
lang 已提交
24
    var DataDiffer = require('./DataDiffer');
L
lang 已提交
25

L
lang 已提交
26
    var zrUtil = require('zrender/core/util');
P
pah100 已提交
27
    var modelUtil = require('../util/model');
L
lang 已提交
28
    var isObject = zrUtil.isObject;
L
lang 已提交
29

L
lang 已提交
30
    var IMMUTABLE_PROPERTIES = [
L
lang 已提交
31
        'stackedOn', '_nameList', '_idList', '_rawData'
L
lang 已提交
32 33
    ];

34 35
    var transferImmuProperties = function (a, b, wrappedMethod) {
        zrUtil.each(IMMUTABLE_PROPERTIES.concat(wrappedMethod || []), function (propName) {
36 37 38
            if (b.hasOwnProperty(propName)) {
                a[propName] = b[propName];
            }
L
lang 已提交
39 40
        });
    };
L
lang 已提交
41

L
lang 已提交
42
    /**
L
lang 已提交
43 44
     * @constructor
     * @alias module:echarts/data/List
L
lang 已提交
45 46
     *
     * @param {Array.<string>} dimensions
L
lang 已提交
47
     *        Dimensions should be concrete names like x, y, z, lng, lat, angle, radius
L
lang 已提交
48
     * @param {module:echarts/model/Model} hostModel
L
lang 已提交
49
     */
L
lang 已提交
50
    var List = function (dimensions, hostModel) {
L
lang 已提交
51 52 53

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

L
lang 已提交
54
        var dimensionInfos = {};
L
lang 已提交
55 56 57 58 59 60 61 62
        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,
L
lang 已提交
63
                    stackable: false,
L
lang 已提交
64
                    // Type can be 'float', 'int', 'number'
L
lang 已提交
65
                    // Default is number, Precision of float may not enough
L
lang 已提交
66 67
                    type: 'number'
                };
L
lang 已提交
68
            }
L
lang 已提交
69 70 71
            else {
                dimensionInfo = dimensions[i];
                dimensionName = dimensionInfo.name;
L
lang 已提交
72
                dimensionInfo.type = dimensionInfo.type || 'number';
L
lang 已提交
73 74
            }
            dimensionNames.push(dimensionName);
L
lang 已提交
75
            dimensionInfos[dimensionName] = dimensionInfo;
L
lang 已提交
76
        }
L
lang 已提交
77 78 79 80 81
        /**
         * @readOnly
         * @type {Array.<string>}
         */
        this.dimensions = dimensionNames;
P
pah100 已提交
82

L
lang 已提交
83 84
        /**
         * Infomation of each data dimension, like data type.
L
lang 已提交
85
         * @type {Object}
L
lang 已提交
86 87
         */
        this._dimensionInfos = dimensionInfos;
L
lang 已提交
88

L
lang 已提交
89 90 91
        /**
         * @type {module:echarts/model/Model}
         */
L
lang 已提交
92
        this.hostModel = hostModel;
L
lang 已提交
93

94
        /**
L
lang 已提交
95 96 97 98
         * Indices stores the indices of data subset after filtered.
         * This data subset will be used in chart.
         * @type {Array.<number>}
         * @readOnly
99
         */
L
lang 已提交
100
        this.indices = [];
101 102

        /**
L
lang 已提交
103 104 105
         * Data storage
         * @type {Object.<key, TypedArray|Array>}
         * @private
106
         */
L
lang 已提交
107
        this._storage = {};
108

L
lang 已提交
109 110 111 112
        /**
         * @type {Array.<string>}
         */
        this._nameList = [];
L
lang 已提交
113 114 115 116
        /**
         * @type {Array.<string>}
         */
        this._idList = [];
L
lang 已提交
117
        /**
L
lang 已提交
118 119 120
         * Models of data option is stored sparse for optimizing memory cost
         * @type {Array.<module:echarts/model/Model>}
         * @private
L
lang 已提交
121
         */
L
lang 已提交
122
        this._optionModels = [];
L
lang 已提交
123 124

        /**
L
lang 已提交
125
         * @param {module:echarts/data/List}
L
lang 已提交
126
         */
L
lang 已提交
127
        this.stackedOn = null;
L
lang 已提交
128

129
        /**
L
lang 已提交
130 131 132
         * Global visual properties after visual coding
         * @type {Object}
         * @private
133
         */
L
lang 已提交
134
        this._visual = {};
L
lang 已提交
135

D
deqingli 已提交
136 137 138 139 140 141 142
        /**
         * Globel layout properties.
         * @type {Object}
         * @private
         */
        this._layout = {};

L
lang 已提交
143
        /**
L
lang 已提交
144 145 146
         * Item visual properties after visual coding
         * @type {Array.<Object>}
         * @private
L
lang 已提交
147
         */
L
lang 已提交
148
        this._itemVisuals = [];
L
lang 已提交
149

L
lang 已提交
150 151 152 153 154 155 156 157 158 159 160 161 162
        /**
         * Item layout properties after layout
         * @type {Array.<Object>}
         * @private
         */
        this._itemLayouts = [];

        /**
         * Graphic elemnents
         * @type {Array.<module:zrender/Element>}
         * @private
         */
        this._graphicEls = [];
L
lang 已提交
163 164

        /**
L
lang 已提交
165
         * @type {Array.<Array|Object>}
L
lang 已提交
166
         * @private
L
lang 已提交
167
         */
L
lang 已提交
168
        this._rawData;
L
lang 已提交
169
    };
L
lang 已提交
170 171 172 173 174

    var listProto = List.prototype;

    listProto.type = 'list';

L
lang 已提交
175 176 177 178 179 180 181 182 183 184 185 186
    /**
     * Get dimension name
     * @param {string|number} dim
     *        Dimension can be concrete names like x, y, z, lng, lat, angle, radius
     *        Or a ordinal number. For example getDimensionInfo(0) will return 'x' or 'lng' or 'radius'
     */
    listProto.getDimension = function (dim) {
        if (!isNaN(dim)) {
            dim = this.dimensions[dim] || dim;
        }
        return dim;
    };
L
lang 已提交
187 188
    /**
     * Get type and stackable info of particular dimension
L
lang 已提交
189 190 191
     * @param {string|number} dim
     *        Dimension can be concrete names like x, y, z, lng, lat, angle, radius
     *        Or a ordinal number. For example getDimensionInfo(0) will return 'x' or 'lng' or 'radius'
L
lang 已提交
192 193
     */
    listProto.getDimensionInfo = function (dim) {
L
lang 已提交
194
        return this._dimensionInfos[this.getDimension(dim)];
L
lang 已提交
195 196
    };

L
lang 已提交
197 198
    /**
     * Initialize from data
L
lang 已提交
199 200
     * @param {Array.<Object|number|Array>} data
     * @param {Array.<string>} [nameList]
L
lang 已提交
201
     * @param {Function} [dimValueGetter] (dataItem, dimName, dataIndex, dimIndex) => number
L
lang 已提交
202
     */
L
lang 已提交
203
    listProto.initData = function (data, nameList, dimValueGetter) {
L
Comma  
lang 已提交
204
        data = data || [];
L
lang 已提交
205

L
lang 已提交
206 207
        this._rawData = data;

L
lang 已提交
208 209 210 211 212 213
        // Clear
        var storage = this._storage = {};
        var indices = this.indices = [];

        var dimensions = this.dimensions;
        var size = data.length;
L
Add pie  
lang 已提交
214
        var dimensionInfoMap = this._dimensionInfos;
L
lang 已提交
215

L
lang 已提交
216 217 218
        var idList = [];
        var nameRepeatCount = {};

L
lang 已提交
219 220
        nameList = nameList || [];

L
lang 已提交
221 222
        // Init storage
        for (var i = 0; i < dimensions.length; i++) {
L
lang 已提交
223
            var dimInfo = dimensionInfoMap[dimensions[i]];
L
lang 已提交
224 225 226 227
            var DataCtor = dataCtors[dimInfo.type];
            storage[dimensions[i]] = new DataCtor(size);
        }

L
lang 已提交
228 229
        // Default dim value getter
        dimValueGetter = dimValueGetter || function (dataItem, dimName, dataIndex, dimIndex) {
P
pah100 已提交
230 231 232 233 234 235 236 237
            var value = modelUtil.getDataItemValue(dataItem);
            return modelUtil.converDataValue(
                zrUtil.isArray(value)
                    ? value[dimIndex]
                    // If value is a single number or something else not array.
                    : value,
                dimensionInfoMap[dimName]
            );
L
lang 已提交
238
        };
L
lang 已提交
239

L
lang 已提交
240
        for (var idx = 0; idx < data.length; idx++) {
L
lang 已提交
241
            var dataItem = data[idx];
L
lang 已提交
242 243 244
            // Each data item is value
            // [1, 2]
            // 2
L
lang 已提交
245 246
            // Bar chart, line chart which uses category axis
            // only gives the 'y' value. 'x' value is the indices of cateogry
L
Add pie  
lang 已提交
247
            // Use a tempValue to normalize the value to be a (x, y) value
248

L
lang 已提交
249 250 251 252 253
            // Store the data by dimensions
            for (var k = 0; k < dimensions.length; k++) {
                var dim = dimensions[k];
                var dimStorage = storage[dim];
                // PENDING NULL is empty or zero
L
lang 已提交
254
                dimStorage[idx] = dimValueGetter(dataItem, dim, idx, k);
255 256
            }

L
lang 已提交
257 258
            indices.push(idx);
        }
L
lang 已提交
259

L
lang 已提交
260
        // Use the name in option and create id
L
lang 已提交
261
        for (var i = 0; i < data.length; i++) {
L
lang 已提交
262
            var id = '';
L
lang 已提交
263
            if (!nameList[i]) {
L
lang 已提交
264 265 266
                nameList[i] = data[i].name;
                // Try using the id in option
                id = data[i].id;
L
lang 已提交
267 268 269 270 271 272 273 274
            }
            var name = nameList[i] || '';
            if (!id && name) {
                // Use name as id and add counter to avoid same name
                nameRepeatCount[name] = nameRepeatCount[name] || 0;
                id = name;
                if (nameRepeatCount[name] > 0) {
                    id += '__ec__' + nameRepeatCount[name];
275
                }
L
lang 已提交
276
                nameRepeatCount[name]++;
277
            }
L
lang 已提交
278
            id && (idList[i] = id);
279 280
        }

L
lang 已提交
281
        this._nameList = nameList;
L
lang 已提交
282
        this._idList = idList;
L
lang 已提交
283 284 285 286 287 288 289 290 291 292 293
    };

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

    /**
     * Get value
L
lang 已提交
294
     * @param {string} dim Dim must be concrete name.
L
lang 已提交
295 296 297 298 299 300 301 302 303
     * @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];
L
lang 已提交
304
        var dimensionInfo = this._dimensionInfos[dim];
L
lang 已提交
305
        // FIXME ordinal data type is not stackable
306 307 308 309 310 311 312 313 314 315 316 317
        if (stack && dimensionInfo && dimensionInfo.stackable) {
            var stackedOn = this.stackedOn;
            while (stackedOn) {
                // Get no stacked data of stacked on
                var stackedValue = stackedOn.get(dim, idx);
                // Considering positive stack, negative stack and empty data
                if ((value >= 0 && stackedValue > 0)  // Positive stack
                    || (value <= 0 && stackedValue < 0) // Negative stack
                ) {
                    value += stackedValue;
                }
                stackedOn = stackedOn.stackedOn;
L
lang 已提交
318 319 320 321
            }
        }
        return value;
    };
L
lang 已提交
322

P
pah100 已提交
323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345
    /**
     * Get value for multi dimensions.
     * @param {Array.<string>} [dimensions] If ignored, using all dimensions.
     * @param {number} idx
     * @param {boolean} stack
     * @return {number}
     */
    listProto.getValues = function (dimensions, idx, stack) {
        var values = [];

        if (!zrUtil.isArray(dimensions)) {
            stack = idx;
            idx = dimensions;
            dimensions = this.dimensions;
        }

        for (var i = 0, len = dimensions.length; i < len; i++) {
            values.push(this.get(dimensions[i], idx, stack));
        }

        return values;
    };

L
lang 已提交
346 347 348 349 350 351 352 353
    /**
     * If value is NaN. Inlcuding '-'
     * @param {string} dim
     * @param {number} idx
     * @return {number}
     */
    listProto.hasValue = function (idx) {
        var dimensions = this.dimensions;
L
lang 已提交
354
        var dimensionInfos = this._dimensionInfos;
L
lang 已提交
355
        for (var i = 0, len = dimensions.length; i < len; i++) {
L
lang 已提交
356 357 358 359 360
            if (
                // Ordinal type can be string or number
                dimensionInfos[dimensions[i]].type !== 'ordinal'
                && isNaN(this.get(dimensions[i], idx))
            ) {
L
lang 已提交
361
                return false;
L
lang 已提交
362
            }
L
lang 已提交
363 364
        }
        return true;
L
lang 已提交
365
    };
L
lang 已提交
366

L
lang 已提交
367 368 369 370 371 372 373
    /**
     * Get extent of data in one dimension
     * @param {string} dim
     * @param {boolean} stack
     */
    listProto.getDataExtent = function (dim, stack) {
        var dimData = this._storage[dim];
L
tweak  
lang 已提交
374 375 376
        var dimInfo = this.getDimensionInfo(dim);
        stack = (dimInfo && dimInfo.stackable) && stack;
        var dimExtent = (this._extent || (this._extent = {}))[dim + (!!stack)];
L
lang 已提交
377
        var value;
L
lang 已提交
378 379 380
        if (dimExtent) {
            return dimExtent;
        }
L
lang 已提交
381
        // var dimInfo = this._dimensionInfos[dim];
L
lang 已提交
382
        if (dimData) {
L
lang 已提交
383 384
            var min = Infinity;
            var max = -Infinity;
L
lang 已提交
385 386
            // var isOrdinal = dimInfo.type === 'ordinal';
            for (var i = 0, len = this.count(); i < len; i++) {
L
lang 已提交
387
                value = this.get(dim, i, stack);
L
lang 已提交
388 389 390 391 392
                // FIXME
                // if (isOrdinal && typeof value === 'string') {
                //     value = zrUtil.indexOf(dimData, value);
                //     console.log(value);
                // }
L
lang 已提交
393 394 395
                value < min && (min = value);
                value > max && (max = value);
            }
L
tweak  
lang 已提交
396
            return (this._extent[dim + stack] = [min, max]);
L
lang 已提交
397 398 399
        }
        else {
            return [Infinity, -Infinity];
L
lang 已提交
400 401
        }
    };
L
lang 已提交
402

L
Add pie  
lang 已提交
403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420
    /**
     * Get sum of data in one dimension
     * @param {string} dim
     * @param {boolean} stack
     */
    listProto.getSum = function (dim, stack) {
        var dimData = this._storage[dim];
        var sum = 0;
        if (dimData) {
            for (var i = 0, len = this.count(); i < len; i++) {
                var value = this.get(dim, i, stack);
                if (!isNaN(value)) {
                    sum += value;
                }
            }
        }
        return sum;
    };
L
lang 已提交
421

L
lang 已提交
422 423 424 425 426 427 428 429 430 431
    /**
     * Retreive the index with given value
     * @param {number} idx
     * @param {number} value
     * @return {number}
     */
    // FIXME Precision of float value
    listProto.indexOf = function (dim, value) {
        var storage = this._storage;
        var dimData = storage[dim];
L
lang 已提交
432
        var indices = this.indices;
L
lang 已提交
433 434

        if (dimData) {
L
lang 已提交
435 436 437
            for (var i = 0, len = indices.length; i < len; i++) {
                var rawIndex = indices[i];
                if (dimData[rawIndex] === value) {
L
lang 已提交
438 439 440 441 442 443 444
                    return i;
                }
            }
        }
        return -1;
    };

L
lang 已提交
445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464
    /**
     * Retreive the index with given name
     * @param {number} idx
     * @param {number} name
     * @return {number}
     */
    listProto.indexOfName = function (name) {
        var indices = this.indices;
        var nameList = this._nameList;

        for (var i = 0, len = indices.length; i < len; i++) {
            var rawIndex = indices[i];
            if (nameList[rawIndex] === name) {
                return i;
            }
        }

        return -1;
    };

L
lang 已提交
465 466
    /**
     * Retreive the index of nearest value
467
     * @param {string|Array.<string>} dim
L
lang 已提交
468
     * @param {number} value
L
lang 已提交
469
     * @param {boolean} stack If given value is after stacked
L
lang 已提交
470 471
     * @return {number}
     */
L
lang 已提交
472
    listProto.indexOfNearest = function (dim, value, stack) {
473 474 475
        if (!zrUtil.isArray(dim)) {
            dim = dim ? [dim] : [];
        }
L
lang 已提交
476 477 478 479 480 481
        var storage = this._storage;
        var dimData = storage[dim];

        if (dimData) {
            var minDist = Number.MAX_VALUE;
            var nearestIdx = -1;
482 483 484 485 486 487 488
            for (var j = 0, lenj = dim.length; j < lenj; j++) {
                for (var i = 0, len = this.count(); i < len; i++) {
                    var dist = Math.abs(this.get(dim[j], i, stack) - value);
                    if (dist <= minDist) {
                        minDist = dist;
                        nearestIdx = i;
                    }
L
lang 已提交
489 490 491 492 493
                }
            }
            return nearestIdx;
        }
        return -1;
L
lang 已提交
494
    };
L
lang 已提交
495

L
lang 已提交
496 497
    /**
     * Get raw data index
L
lang 已提交
498 499
     * @param {number} idx
     * @return {number}
L
lang 已提交
500
     */
L
lang 已提交
501
    listProto.getRawIndex = function (idx) {
502 503
        var rawIdx = this.indices[idx];
        return rawIdx == null ? -1 : rawIdx;
L
lang 已提交
504
    };
L
lang 已提交
505

L
lang 已提交
506 507
    /**
     * @param {number} idx
508
     * @param {boolean} [notDefaultIdx=false]
L
lang 已提交
509 510
     * @return {string}
     */
L
lang 已提交
511 512 513 514 515 516 517 518 519 520 521
    listProto.getName = function (idx) {
        return this._nameList[this.indices[idx]] || '';
    };

    /**
     * @param {number} idx
     * @param {boolean} [notDefaultIdx=false]
     * @return {string}
     */
    listProto.getId = function (idx) {
        return this._idList[this.indices[idx]] || (this.getRawIndex(idx) + '');
L
lang 已提交
522 523 524
    };


L
lang 已提交
525
    function normalizeDimensions(dimensions) {
526
        if (!zrUtil.isArray(dimensions)) {
L
lang 已提交
527 528 529 530
            dimensions = [dimensions];
        }
        return dimensions;
    }
L
lang 已提交
531

L
lang 已提交
532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550
    /**
     * 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 已提交
551

L
lang 已提交
552 553 554
        dimensions = zrUtil.map(
            normalizeDimensions(dimensions), this.getDimension, this
        );
L
lang 已提交
555

L
lang 已提交
556 557 558
        var value = [];
        var dimSize = dimensions.length;
        var indices = this.indices;
L
lang 已提交
559

L
lang 已提交
560
        context = context || this;
L
lang 已提交
561

L
lang 已提交
562 563
        for (var i = 0; i < indices.length; i++) {
            if (dimSize === 0) {
L
lang 已提交
564
                cb.call(context, i);
L
lang 已提交
565 566 567
            }
            // Simple optimization
            else if (dimSize === 1) {
L
lang 已提交
568
                cb.call(context, this.get(dimensions[0], i, stack), i);
L
lang 已提交
569 570 571
            }
            else {
                for (var k = 0; k < dimSize; k++) {
L
lang 已提交
572
                    value[k] = this.get(dimensions[k], i, stack);
L
lang 已提交
573 574 575 576 577 578 579
                }
                // Index
                value[k] = i;
                cb.apply(context, value);
            }
        }
    };
L
lang 已提交
580

L
lang 已提交
581 582 583 584 585 586 587 588
    /**
     * Data filter
     * @param {string|Array.<string>}
     * @param {Function} cb
     * @param {boolean} [stack=false]
     * @param {*} [context=this]
     */
    listProto.filterSelf = function (dimensions, cb, stack, context) {
L
lang 已提交
589 590 591 592 593 594 595
        if (typeof dimensions === 'function') {
            context = stack;
            stack = cb;
            cb = dimensions;
            dimensions = [];
        }

L
lang 已提交
596 597 598
        dimensions = zrUtil.map(
            normalizeDimensions(dimensions), this.getDimension, this
        );
L
lang 已提交
599 600 601 602 603 604 605 606 607 608 609 610

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

        context = context || this;

        for (var i = 0; i < indices.length; i++) {
            var keep;
            // Simple optimization
            if (dimSize === 1) {
L
lang 已提交
611
                keep = cb.call(
L
lang 已提交
612
                    context, this.get(dimensions[0], i, stack), i
L
lang 已提交
613 614 615 616
                );
            }
            else {
                for (var k = 0; k < dimSize; k++) {
L
lang 已提交
617
                    value[k] = this.get(dimensions[k], i, stack);
L
lang 已提交
618 619 620 621 622 623 624 625
                }
                value[k] = i;
                keep = cb.apply(context, value);
            }
            if (keep) {
                newIndices.push(indices[i]);
            }
        }
P
pah100 已提交
626

L
lang 已提交
627
        this.indices = newIndices;
L
lang 已提交
628

L
tweak  
lang 已提交
629 630 631
        // Reset data extent
        this._extent = {};

L
lang 已提交
632 633
        return this;
    };
L
lang 已提交
634

L
lang 已提交
635
    /**
L
lang 已提交
636 637
     * Data mapping to a plain array
     * @param {string|Array.<string>} [dimensions]
L
lang 已提交
638 639 640
     * @param {Function} cb
     * @param {boolean} [stack=false]
     * @param {*} [context=this]
L
lang 已提交
641
     * @return {Array}
L
lang 已提交
642
     */
L
lang 已提交
643
    listProto.mapArray = function (dimensions, cb, stack, context) {
L
lang 已提交
644 645 646 647 648 649
        if (typeof dimensions === 'function') {
            context = stack;
            stack = cb;
            cb = dimensions;
            dimensions = [];
        }
L
lang 已提交
650

L
lang 已提交
651 652 653 654 655 656 657
        var result = [];
        this.each(dimensions, function () {
            result.push(cb && cb.apply(this, arguments));
        }, stack, context);
        return result;
    };

L
lang 已提交
658 659 660 661 662 663 664 665 666
    /**
     * Data mapping to a new List with given dimensions
     * @param {string|Array.<string>} dimensions
     * @param {Function} cb
     * @param {boolean} [stack=false]
     * @param {*} [context=this]
     * @return {Array}
     */
    listProto.map = function (dimensions, cb, stack, context) {
L
lang 已提交
667 668 669
        dimensions = zrUtil.map(
            normalizeDimensions(dimensions), this.getDimension, this
        );
L
lang 已提交
670

L
lang 已提交
671
        var allDimensions = this.dimensions;
L
lang 已提交
672
        var list = new List(
L
lang 已提交
673
            zrUtil.map(allDimensions, this.getDimensionInfo, this),
L
lang 已提交
674 675 676 677 678 679 680
            this.hostModel
        );

        // Following properties are all immutable.
        // So we can reference to the same value
        var indices = list.indices = this.indices;

L
lang 已提交
681
        // FIXME If needs stackedOn, value may already been stacked
682
        transferImmuProperties(list, this, this._wrappedMethods);
L
lang 已提交
683 684 685 686 687

        var storage = list._storage = {};
        var thisStorage = this._storage;

        // Init storage
L
lang 已提交
688 689
        for (var i = 0; i < allDimensions.length; i++) {
            var dim = allDimensions[i];
L
lang 已提交
690
            var dimStore = thisStorage[dim];
L
lang 已提交
691
            if (zrUtil.indexOf(dimensions, dim) >= 0) {
L
lang 已提交
692 693 694 695
                storage[dim] = new dimStore.constructor(
                    thisStorage[dim].length
                );
            }
L
lang 已提交
696 697 698 699
            else {
                // Direct copy for other dimensions
                storage[dim] = thisStorage[dim];
            }
L
lang 已提交
700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725
        }

        var tmpRetValue = [];
        this.each(dimensions, function () {
            var idx = arguments[arguments.length - 1];
            var retValue = cb && cb.apply(this, arguments);
            if (retValue != null) {
                // a number
                if (typeof retValue === 'number') {
                    tmpRetValue[0] = retValue;
                    retValue = tmpRetValue;
                }
                for (var i = 0; i < retValue.length; i++) {
                    var dim = dimensions[i];
                    var dimStore = storage[dim];
                    var rawIdx = indices[idx];
                    if (dimStore) {
                        dimStore[rawIdx] = retValue[i];
                    }
                }
            }
        });

        return list;
    };

L
lang 已提交
726
    var temporaryModel = new Model(null);
L
lang 已提交
727 728 729 730 731
    // Since temporate model is shared by all data items. So we must make sure it can't be write.
    // PENDING may cause any performance problem?
    // if (Object.freeze) {
    //     Object.freeze(temporaryModel);
    // }
L
lang 已提交
732 733 734
    /**
     * Get model of one data item.
     * It will create a temporary model if value on idx is not an option.
L
lang 已提交
735 736
     *
     * @param {number} idx
L
lang 已提交
737
     * @param {boolean} [createNew=false]
L
lang 已提交
738
     */
L
lang 已提交
739
    // FIXME Model proxy ?
L
lang 已提交
740
    listProto.getItemModel = function (idx, createNew) {
L
lang 已提交
741
        var model;
L
lang 已提交
742
        var hostModel = this.hostModel;
L
lang 已提交
743 744 745 746 747 748 749 750
        idx = this.indices[idx];
        // Use a temporary model proxy
        // FIXME Create a new one may cause memory leak
        if (createNew) {
            model = new Model(null, hostModel);
        }
        else {
            model = temporaryModel;
L
lang 已提交
751
        }
752 753 754 755
        // FIXME If return null when idx not exists
        model.option = this._rawData[idx];
        model.parentModel = hostModel;
        model.ecModel = hostModel.ecModel;
L
lang 已提交
756 757 758 759 760
        return model;
    };

    /**
     * Create a data differ
L
tweak  
lang 已提交
761
     * @param {module:echarts/data/List} otherList
L
lang 已提交
762 763
     * @return {module:echarts/data/DataDiffer}
     */
L
tweak  
lang 已提交
764
    listProto.diff = function (otherList) {
L
lang 已提交
765
        var idList = this._idList;
766
        var otherIdList = otherList && otherList._idList;
767
        return new DataDiffer(
L
tweak  
lang 已提交
768
            otherList ? otherList.indices : [], this.indices, function (idx) {
769 770
                return otherIdList[idx] || (idx + '');
            }, function (idx) {
L
lang 已提交
771
                return idList[idx] || (idx + '');
772 773
            }
        );
L
lang 已提交
774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799
    };
    /**
     * 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 已提交
800 801
                }
            }
L
lang 已提交
802 803 804 805 806
            return;
        }
        this._visual = this._visual || {};
        this._visual[key] = val;
    };
L
lang 已提交
807

D
deqingli 已提交
808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825
    /**
     * Set layout property.
     * @param {string} key 
     * @param {*} val
     */
    listProto.setLayout = function (key, val) {
        this._layout[key] = val;
    };

    /**
     * Get layout property.
     * @param  {string} key.
     * @return {*}
     */
    listProto.getLayout = function (key) { 
        return this._layout[key];
    };

L
lang 已提交
826 827 828 829 830 831 832
    /**
     * Get layout of single data item
     * @param {number} idx
     */
    listProto.getItemLayout = function (idx) {
        return this._itemLayouts[idx];
    },
L
lang 已提交
833

L
lang 已提交
834 835 836 837
    /**
     * Set layout of single data item
     * @param {number} idx
     * @param {Object} layout
L
lang 已提交
838
     * @param {boolean=} [merge=false]
L
lang 已提交
839
     */
P
tree  
pah100 已提交
840 841 842 843
    listProto.setItemLayout = function (idx, layout, merge) {
        this._itemLayouts[idx] = merge
            ? zrUtil.extend(this._itemLayouts[idx] || {}, layout)
            : layout;
L
lang 已提交
844
    },
L
lang 已提交
845

L
lang 已提交
846 847 848 849
    /**
     * Get visual property of single data item
     * @param {number} idx
     * @param {string} key
L
lang 已提交
850
     * @param {boolean} ignoreParent
L
lang 已提交
851
     */
L
lang 已提交
852
    listProto.getItemVisual = function (idx, key, ignoreParent) {
L
lang 已提交
853 854
        var itemVisual = this._itemVisuals[idx];
        var val = itemVisual && itemVisual[key];
L
lang 已提交
855
        if (val == null && !ignoreParent) {
L
lang 已提交
856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881
            // 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)) {
P
pah100 已提交
882
                    itemVisual[name] = key[name];
L
lang 已提交
883
                }
L
lang 已提交
884
            }
L
lang 已提交
885
            return;
L
lang 已提交
886
        }
L
lang 已提交
887
        itemVisual[key] = value;
L
lang 已提交
888 889
    };

L
lang 已提交
890 891 892
    var setItemDataAndSeriesIndex = function (child) {
        child.seriesIndex = this.seriesIndex;
        child.dataIndex = this.dataIndex;
L
lang 已提交
893
    };
L
lang 已提交
894
    /**
895
     * Set graphic element relative to data. It can be set as null
L
lang 已提交
896
     * @param {number} idx
897
     * @param {module:zrender/Element} [el]
L
lang 已提交
898 899
     */
    listProto.setItemGraphicEl = function (idx, el) {
L
lang 已提交
900
        var hostModel = this.hostModel;
P
pah100 已提交
901

P
pah100 已提交
902 903 904 905 906 907 908 909
        if (el) {
            // Add data index and series index for indexing the data by element
            // Useful in tooltip
            el.dataIndex = idx;
            el.seriesIndex = hostModel && hostModel.seriesIndex;
            if (el.type === 'group') {
                el.traverse(setItemDataAndSeriesIndex, el);
            }
P
pah100 已提交
910
        }
911 912

        this._graphicEls[idx] = el;
P
pah100 已提交
913 914
    };

L
lang 已提交
915 916 917 918 919 920 921 922 923 924 925 926 927
    /**
     * @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) {
L
lang 已提交
928 929 930 931 932
        zrUtil.each(this._graphicEls, function (el, idx) {
            if (el) {
                cb && cb.call(context, el, idx);
            }
        });
L
lang 已提交
933 934 935 936 937 938 939
    };

    /**
     * Shallow clone a new list except visual and layout properties, and graph elements.
     * New list only change the indices.
     */
    listProto.cloneShallow = function () {
L
lang 已提交
940
        var dimensionInfoList = zrUtil.map(this.dimensions, this.getDimensionInfo, this);
L
lang 已提交
941
        var list = new List(dimensionInfoList, this.hostModel);
L
lang 已提交
942 943 944

        // FIXME
        list._storage = this._storage;
L
lang 已提交
945

946
        transferImmuProperties(list, this, this._wrappedMethods);
L
lang 已提交
947 948

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

L
lang 已提交
950 951
        return list;
    };
L
lang 已提交
952

953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970
    /**
     * Wrap some method to add more feature
     * @param {string} methodName
     * @param {Function} injectFunction
     */
    listProto.wrapMethod = function (methodName, injectFunction) {
        var originalMethod = this[methodName];
        if (typeof originalMethod !== 'function') {
            return;
        }
        this._wrappedMethods = this._wrappedMethods || [];
        this._wrappedMethods.push(methodName);
        this[methodName] = function () {
            var res = originalMethod.apply(this, arguments);
            return injectFunction.call(this, res);
        };
    };

L
lang 已提交
971 972
    return List;
});