"use strict"; const events = require("../events.js"); class AbstractList extends events.EventTarget { constructor() { super(); this._list = []; } static fromResponse(response) { const ret = new this(); for (let item of response) { const addedItem = this._itemClass.fromResponse(item); if (addedItem.addEventListener) { addedItem.addEventListener("delete", (e) => { ret.remove(addedItem); }); addedItem.addEventListener("change", (e) => { ret.dispatchEvent( new CustomEvent("change", { detail: e.detail, }) ); }); } ret._list.push(addedItem); } return ret; } sync(plainList) { this.clear(); for (let item of plainList || []) { this.add(this.constructor._itemClass.fromResponse(item)); } } add(item) { if (item.addEventListener) { item.addEventListener("delete", (e) => { this.remove(item); }); item.addEventListener("change", (e) => { this.dispatchEvent( new CustomEvent("change", { detail: e.detail, }) ); }); } this._list.push(item); const detail = {}; detail[this.constructor._itemName] = item; this.dispatchEvent( new CustomEvent("add", { detail: detail, }) ); } clear() { for (let item of [...this._list]) { this.remove(item); } } remove(itemToRemove) { for (let [index, item] of this._list.entries()) { if (item !== itemToRemove) { continue; } this._list.splice(index, 1); const detail = {}; detail[this.constructor._itemName] = itemToRemove; this.dispatchEvent( new CustomEvent("remove", { detail: detail, }) ); return; } } get length() { return this._list.length; } at(index) { return this._list[index]; } map(...args) { return this._list.map(...args); } filter(...args) { return this._list.filter(...args); } [Symbol.iterator]() { return this._list[Symbol.iterator](); } } module.exports = AbstractList;