/** * @module ol/Collection */ import {inherits} from './util.js'; import AssertionError from './AssertionError.js'; import CollectionEventType from './CollectionEventType.js'; import BaseObject from './Object.js'; import Event from './events/Event.js'; /** * @enum {string} * @private */ const Property = { LENGTH: 'length' }; /** * @classdesc * Events emitted by {@link module:ol/Collection~Collection} instances are instances of this * type. * * @constructor * @extends {module:ol/events/Event} * @param {module:ol/CollectionEventType} type Type. * @param {*=} opt_element Element. */ export const CollectionEvent = function(type, opt_element) { Event.call(this, type); /** * The element that is added to or removed from the collection. * @type {*} * @api */ this.element = opt_element; }; inherits(CollectionEvent, Event); /** * @typedef {Object} Options * @property {boolean} [unique=false] Disallow the same item from being added to * the collection twice. */ /** * @classdesc * An expanded version of standard JS Array, adding convenience methods for * manipulation. Add and remove changes to the Collection trigger a Collection * event. Note that this does not cover changes to the objects _within_ the * Collection; they trigger events on the appropriate object, not on the * Collection as a whole. * * @constructor * @extends {module:ol/Object} * @fires module:ol/Collection~CollectionEvent * @param {Array.=} opt_array Array. * @param {module:ol/Collection~Options=} opt_options Collection options. * @template T * @api */ const Collection = function(opt_array, opt_options) { BaseObject.call(this); const options = opt_options || {}; /** * @private * @type {boolean} */ this.unique_ = !!options.unique; /** * @private * @type {!Array.} */ this.array_ = opt_array ? opt_array : []; if (this.unique_) { for (let i = 0, ii = this.array_.length; i < ii; ++i) { this.assertUnique_(this.array_[i], i); } } this.updateLength_(); }; inherits(Collection, BaseObject); /** * Remove all elements from the collection. * @api */ Collection.prototype.clear = function() { while (this.getLength() > 0) { this.pop(); } }; /** * Add elements to the collection. This pushes each item in the provided array * to the end of the collection. * @param {!Array.} arr Array. * @return {module:ol/Collection.} This collection. * @api */ Collection.prototype.extend = function(arr) { for (let i = 0, ii = arr.length; i < ii; ++i) { this.push(arr[i]); } return this; }; /** * Iterate over each element, calling the provided callback. * @param {function(T, number, Array.): *} f The function to call * for every element. This function takes 3 arguments (the element, the * index and the array). The return value is ignored. * @api */ Collection.prototype.forEach = function(f) { const array = this.array_; for (let i = 0, ii = array.length; i < ii; ++i) { f(array[i], i, array); } }; /** * Get a reference to the underlying Array object. Warning: if the array * is mutated, no events will be dispatched by the collection, and the * collection's "length" property won't be in sync with the actual length * of the array. * @return {!Array.} Array. * @api */ Collection.prototype.getArray = function() { return this.array_; }; /** * Get the element at the provided index. * @param {number} index Index. * @return {T} Element. * @api */ Collection.prototype.item = function(index) { return this.array_[index]; }; /** * Get the length of this collection. * @return {number} The length of the array. * @observable * @api */ Collection.prototype.getLength = function() { return /** @type {number} */ (this.get(Property.LENGTH)); }; /** * Insert an element at the provided index. * @param {number} index Index. * @param {T} elem Element. * @api */ Collection.prototype.insertAt = function(index, elem) { if (this.unique_) { this.assertUnique_(elem); } this.array_.splice(index, 0, elem); this.updateLength_(); this.dispatchEvent( new CollectionEvent(CollectionEventType.ADD, elem)); }; /** * Remove the last element of the collection and return it. * Return `undefined` if the collection is empty. * @return {T|undefined} Element. * @api */ Collection.prototype.pop = function() { return this.removeAt(this.getLength() - 1); }; /** * Insert the provided element at the end of the collection. * @param {T} elem Element. * @return {number} New length of the collection. * @api */ Collection.prototype.push = function(elem) { if (this.unique_) { this.assertUnique_(elem); } const n = this.getLength(); this.insertAt(n, elem); return this.getLength(); }; /** * Remove the first occurrence of an element from the collection. * @param {T} elem Element. * @return {T|undefined} The removed element or undefined if none found. * @api */ Collection.prototype.remove = function(elem) { const arr = this.array_; for (let i = 0, ii = arr.length; i < ii; ++i) { if (arr[i] === elem) { return this.removeAt(i); } } return undefined; }; /** * Remove the element at the provided index and return it. * Return `undefined` if the collection does not contain this index. * @param {number} index Index. * @return {T|undefined} Value. * @api */ Collection.prototype.removeAt = function(index) { const prev = this.array_[index]; this.array_.splice(index, 1); this.updateLength_(); this.dispatchEvent(new CollectionEvent(CollectionEventType.REMOVE, prev)); return prev; }; /** * Set the element at the provided index. * @param {number} index Index. * @param {T} elem Element. * @api */ Collection.prototype.setAt = function(index, elem) { const n = this.getLength(); if (index < n) { if (this.unique_) { this.assertUnique_(elem, index); } const prev = this.array_[index]; this.array_[index] = elem; this.dispatchEvent( new CollectionEvent(CollectionEventType.REMOVE, prev)); this.dispatchEvent( new CollectionEvent(CollectionEventType.ADD, elem)); } else { for (let j = n; j < index; ++j) { this.insertAt(j, undefined); } this.insertAt(index, elem); } }; /** * @private */ Collection.prototype.updateLength_ = function() { this.set(Property.LENGTH, this.array_.length); }; /** * @private * @param {T} elem Element. * @param {number=} opt_except Optional index to ignore. */ Collection.prototype.assertUnique_ = function(elem, opt_except) { for (let i = 0, ii = this.array_.length; i < ii; ++i) { if (this.array_[i] === elem && i !== opt_except) { throw new AssertionError(58); } } }; export default Collection;