import IHTMLFormControlsCollection from './IHTMLFormControlsCollection.js'; import IHTMLInputElement from '../html-input-element/IHTMLInputElement.js'; import IHTMLTextAreaElement from '../html-text-area-element/IHTMLTextAreaElement.js'; import IHTMLSelectElement from '../html-select-element/IHTMLSelectElement.js'; import RadioNodeList from './RadioNodeList.js'; import IHTMLButtonElement from '../html-button-element/IHTMLButtonElement.js'; /** * HTMLFormControlsCollection. * * @see https://developer.mozilla.org/en-US/docs/Web/API/HTMLFormControlsCollection */ export default class HTMLFormControlsCollection extends Array implements IHTMLFormControlsCollection { public _namedItems: { [k: string]: RadioNodeList } = {}; /** * Returns item by index. * * @param index Index. */ public item( index: number ): IHTMLInputElement | IHTMLTextAreaElement | IHTMLSelectElement | IHTMLButtonElement | null { return index >= 0 && this[index] ? this[index] : null; } /** * Returns named item. * * @param name Name. * @returns Node. */ public namedItem( name: string ): | IHTMLInputElement | IHTMLTextAreaElement | IHTMLSelectElement | IHTMLButtonElement | RadioNodeList | null { if (this._namedItems[name] && this._namedItems[name].length) { if (this._namedItems[name].length === 1) { return this._namedItems[name][0]; } return this._namedItems[name]; } return null; } /** * Appends named item. * * @param node Node. * @param name Name. */ public _appendNamedItem( node: IHTMLInputElement | IHTMLTextAreaElement | IHTMLSelectElement | IHTMLButtonElement, name: string ): void { if (name) { this._namedItems[name] = this._namedItems[name] || new RadioNodeList(); if (!this._namedItems[name].includes(node)) { this._namedItems[name].push(node); } if (this._isValidPropertyName(name)) { this[name] = this._namedItems[name].length > 1 ? this._namedItems[name] : this._namedItems[name][0]; } } } /** * Appends named item. * * @param node Node. * @param name Name. */ public _removeNamedItem( node: IHTMLInputElement | IHTMLTextAreaElement | IHTMLSelectElement | IHTMLButtonElement, name: string ): void { if (name && this._namedItems[name]) { const index = this._namedItems[name].indexOf(node); if (index > -1) { this._namedItems[name].splice(index, 1); if (this._namedItems[name].length === 0) { delete this._namedItems[name]; if (this.hasOwnProperty(name) && this._isValidPropertyName(name)) { delete this[name]; } } else if (this._isValidPropertyName(name)) { this[name] = this._namedItems[name].length > 1 ? this._namedItems[name] : this._namedItems[name][0]; } } } } /** * Returns "true" if the property name is valid. * * @param name Name. * @returns True if the property name is valid. */ protected _isValidPropertyName(name: string): boolean { return ( !this.constructor.prototype.hasOwnProperty(name) && !Array.prototype.hasOwnProperty(name) && (isNaN(Number(name)) || name.includes('.')) ); } }