szurubooru/client/js/models/abstract_list.js

106 lines
2.6 KiB
JavaScript

"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;