if (!Array.prototype.indexOf) { Array.prototype.indexOf = function(value, from) { var len = this.length >>> 0; from = Number(from) || 0; from = Math[from < 0 ? 'ceil' : 'floor'](from); if (from < 0) { from += len; } for (; from < len; from++) { if (from in this && this[from] === value) { return from; } } return -1; }; } if (!Array.prototype.forEach) { Array.prototype.forEach = function(fn, context) { for (var i = 0, len = this.length >>> 0; i < len; i++) { if (i in this) { fn.call(context, this[i], i, this); } } }; } if (!Array.prototype.map) { Array.prototype.map = function(fn, context) { var result = [ ]; for (var i = 0, len = this.length >>> 0; i < len; i++) { if (i in this) { result[i] = fn.call(context, this[i], i, this); } } return result; }; } if (!Array.prototype.every) { Array.prototype.every = function(fn, context) { for (var i = 0, len = this.length >>> 0; i < len; i++) { if (i in this && !fn.call(context, this[i], i, this)) { return false; } } return true; }; } if (!Array.prototype.some) { Array.prototype.some = function(fn, context) { for (var i = 0, len = this.length >>> 0; i < len; i++) { if (i in this && fn.call(context, this[i], i, this)) { return true; } } return false; }; } if (!Array.prototype.filter) { Array.prototype.filter = function(fn, context) { var result = [ ], val; for (var i = 0, len = this.length >>> 0; i < len; i++) { if (i in this) { val = this[i]; // in case fn mutates this if (fn.call(context, val, i, this)) { result.push(val); } } } return result; }; } if (!Array.prototype.reduce) { Array.prototype.reduce = function(fn /*, initial*/) { var len = this.length >>> 0, i = 0, rv; if (arguments.length > 1) { rv = arguments[1]; } else { do { if (i in this) { rv = this[i++]; break; } // if array contains no values, no initial value to return if (++i >= len) { throw new TypeError(); } } while (true); } for (; i < len; i++) { if (i in this) { rv = fn.call(null, rv, this[i], i, this); } } return rv; }; } /** * Invokes method on all items in a given array * @method invoke * @memberOf fabric.util.array * @param {Array} array Array to iterate over * @param {String} method Name of a method to invoke */ function invoke(array, method) { var args = slice.call(arguments, 2), result = [ ]; for (var i = 0, len = array.length; i < len; i++) { result[i] = args.length ? array[i][method].apply(array[i], args) : array[i][method].call(array[i]); } return result; } /** * Finds maximum value in array (not necessarily "first" one) * @method max * @memberOf fabric.util.array * @param {Array} array Array to iterate over * @param {String} byProperty */ function max(array, byProperty) { var i = array.length - 1, result = byProperty ? array[i][byProperty] : array[i]; if (byProperty) { while (i--) { if (array[i][byProperty] >= result) { result = array[i][byProperty]; } } } else { while (i--) { if (array[i] >= result) { result = array[i]; } } } return result; } /** * Finds minimum value in array (not necessarily "first" one) * @method min * @memberOf fabric.util.array * @param {Array} array Array to iterate over * @param {String} byProperty */ function min(array, byProperty) { var i = array.length - 1, result = byProperty ? array[i][byProperty] : array[i]; if (byProperty) { while (i--) { if (array[i][byProperty] < result) { result = array[i][byProperty]; } } } else { while (i--) { if (array[i] < result) { result = array[i]; } } } return result; } /** @namespace */ fabric.util.array = { invoke: invoke, min: min, max: max };