import {InputElement} from "./InputElement"; import {UIEventSource} from "../../Logic/UIEventSource"; import {UIElement} from "../UIElement"; import Combine from "../Base/Combine"; import {SubtleButton} from "../Base/SubtleButton"; import {FixedUiElement} from "../Base/FixedUiElement"; import Svg from "../../Svg"; import {Img} from "../Img"; export class MultiInput extends InputElement { private readonly _value: UIEventSource; IsSelected: UIEventSource; private elements: UIElement[] = []; private inputELements: InputElement[] = []; private addTag: UIElement; private _options: { allowMovement?: boolean }; constructor( addAElement: string, newElement: (() => T), createInput: (() => InputElement), value: UIEventSource = undefined, options?: { allowMovement?: boolean }) { super(undefined); this._value = value ?? new UIEventSource([]); value = this._value; this.ListenTo(value.map((latest : T[]) => latest.length)); this._options = options ?? {}; this.addTag = new SubtleButton(Img.AsData(Svg.addSmall), addAElement) .SetClass("small-button") .onClick(() => { this.IsSelected.setData(true); value.data.push(newElement()); value.ping(); }); const self = this; value.map((tags: string[]) => tags.length).addCallback(() => self.createElements(createInput)); this.createElements(createInput); this._value.addCallback(tags => self.load(tags)); this.IsSelected = new UIEventSource(false); } private load(tags: T[]) { if (tags === undefined) { return; } for (let i = 0; i < tags.length; i++) { this.inputELements[i].GetValue().setData(tags[i]); } } private UpdateIsSelected(){ this.IsSelected.setData(this.inputELements.map(input => input.IsSelected.data).reduce((a,b) => a && b)) } private createElements(createInput: (() => InputElement)) { this.inputELements.splice(0, this.inputELements.length); this.elements = []; const self = this; for (let i = 0; i < this._value.data.length; i++) { const input = createInput(); input.GetValue().addCallback(tag => { self._value.data[i] = tag; self._value.ping(); } ); this.inputELements.push(input); input.IsSelected.addCallback(() => this.UpdateIsSelected()); const moveUpBtn = new FixedUiElement(Img.AsData(Svg.up)) .onClick(() => { const v = self._value.data[i]; self._value.data[i] = self._value.data[i - 1]; self._value.data[i - 1] = v; self._value.ping(); }); const moveDownBtn = new FixedUiElement("") .onClick(() => { const v = self._value.data[i]; self._value.data[i] = self._value.data[i + 1]; self._value.data[i + 1] = v; self._value.ping(); }); const controls = []; if (i > 0 && this._options.allowMovement) { controls.push(moveUpBtn); } if (i + 1 < this._value.data.length && this._options.allowMovement) { controls.push(moveDownBtn); } const deleteBtn = new FixedUiElement("") .onClick(() => { self._value.data.splice(i, 1); self._value.ping(); }); controls.push(deleteBtn); this.elements.push(new Combine([input.SetStyle("width: calc(100% - 2em - 5px)"), new Combine(controls).SetStyle("display:flex;flex-direction:column;width:min-content;")]).SetClass("tag-input-row")) } this.Update(); } InnerRender(): string { return new Combine([...this.elements, this.addTag]).Render(); } IsValid(t: T[]): boolean { return false; } GetValue(): UIEventSource { return this._value; } }