import IHTMLCollection from './IHTMLCollection.js'; import * as PropertySymbol from '../../PropertySymbol.js'; /** * HTML collection. */ export default class HTMLCollection extends Array implements IHTMLCollection { protected [PropertySymbol.namedItems]: { [k: string]: T[] } = {}; /** * Returns item by index. * * @param index Index. */ public item(index: number): T | null { return index >= 0 && this[index] ? this[index] : null; } /** * Returns named item. * * @param name Name. * @returns Node. */ public namedItem(name: string): T | null { return this[PropertySymbol.namedItems][name] && this[PropertySymbol.namedItems][name].length ? this[PropertySymbol.namedItems][name][0] : null; } /** * Appends named item. * * @param node Node. * @param name Name. */ public [PropertySymbol.appendNamedItem](node: T, name: string): void { if (name) { this[PropertySymbol.namedItems][name] = this[PropertySymbol.namedItems][name] || []; if (!this[PropertySymbol.namedItems][name].includes(node)) { this[PropertySymbol.namedItems][name].push(node); } if (!this.hasOwnProperty(name) && this[PropertySymbol.isValidPropertyName](name)) { this[name] = this[PropertySymbol.namedItems][name][0]; } } } /** * Appends named item. * * @param node Node. * @param name Name. */ public [PropertySymbol.removeNamedItem](node: T, name: string): void { if (name && this[PropertySymbol.namedItems][name]) { const index = this[PropertySymbol.namedItems][name].indexOf(node); if (index > -1) { this[PropertySymbol.namedItems][name].splice(index, 1); if (this[PropertySymbol.namedItems][name].length === 0) { delete this[PropertySymbol.namedItems][name]; if (this.hasOwnProperty(name) && this[PropertySymbol.isValidPropertyName](name)) { delete this[name]; } } else if (this[PropertySymbol.isValidPropertyName](name)) { this[name] = this[PropertySymbol.namedItems][name][0]; } } } } /** * Returns "true" if the property name is valid. * * @param name Name. * @returns True if the property name is valid. */ protected [PropertySymbol.isValidPropertyName](name: string): boolean { return ( !this.constructor.prototype.hasOwnProperty(name) && !Array.prototype.hasOwnProperty(name) && (isNaN(Number(name)) || name.includes('.')) ); } }