import {InputElement} from "./InputElement"; import {UIEventSource} from "../../Logic/UIEventSource"; import {Utils} from "../../Utils"; export class RadioButton extends InputElement { private static _nextId = 0; private readonly value: UIEventSource; private _elements: InputElement[]; private _selectFirstAsDefault: boolean; private _dontStyle: boolean constructor( elements: InputElement[], options?: { selectFirstAsDefault?: true | boolean, dontStyle?: boolean } ) { super(); options = options ?? {} this._selectFirstAsDefault = options.selectFirstAsDefault ?? true; this._elements = Utils.NoNull(elements); this.value = new UIEventSource(undefined); this._dontStyle = options.dontStyle ?? false } IsValid(t: T): boolean { for (const inputElement of this._elements) { if (inputElement.IsValid(t)) { return true; } } return false; } GetValue(): UIEventSource { return this.value; } protected InnerConstructElement(): HTMLElement { const elements = this._elements; const selectFirstAsDefault = this._selectFirstAsDefault; const selectedElementIndex: UIEventSource = new UIEventSource(null); const value = UIEventSource.flatten( selectedElementIndex.map((selectedIndex) => { if (selectedIndex !== undefined && selectedIndex !== null) { return elements[selectedIndex].GetValue(); } }), elements.map((e) => e?.GetValue()) ); value.syncWith(this.value); if (selectFirstAsDefault) { value.addCallbackAndRun((selected) => { if (selected === undefined) { for (const element of elements) { const v = element.GetValue().data; if (v !== undefined) { value.setData(v); break; } } } }); } for (let i = 0; i < elements.length; i++) { // If an element is clicked, the radio button corresponding with it should be selected as well elements[i]?.onClick(() => { selectedElementIndex.setData(i); }); elements[i].GetValue().addCallback(() => { selectedElementIndex.setData(i); }); } const groupId = "radiogroup" + RadioButton._nextId; RadioButton._nextId++; const form = document.createElement("form"); const inputs = []; const wrappers: HTMLElement[] = []; for (let i1 = 0; i1 < elements.length; i1++) { let element = elements[i1]; const labelHtml = element.ConstructElement(); if (labelHtml === undefined) { continue; } const input = document.createElement("input"); input.id = "radio" + groupId + "-" + i1; input.name = groupId; input.type = "radio"; input.classList.add( "cursor-pointer", "p-1", "mr-2" ); if (!this._dontStyle) { input.classList.add( "p-1", "ml-2", "pl-2", "pr-0", "m-3", "mr-0" ); } input.onchange = () => { if (input.checked) { selectedElementIndex.setData(i1); } }; inputs.push(input); const label = document.createElement("label"); label.appendChild(labelHtml); label.htmlFor = input.id; label.classList.add("flex", "w-full", "cursor-pointer", "bg-red"); if (!this._dontStyle) { labelHtml.classList.add("p-2") } const block = document.createElement("div"); block.appendChild(input); block.appendChild(label); block.classList.add( "flex", "w-full", ); if (!this._dontStyle) { block.classList.add( "m-1", "border", "border-gray-400" ) } block.style.borderRadius = "1.5rem" wrappers.push(block); form.appendChild(block); } value.addCallbackAndRun((selected:T) => { let somethingChecked = false; for (let i = 0; i < inputs.length; i++) { let input = inputs[i]; input.checked = !somethingChecked && elements[i].IsValid(selected); somethingChecked = somethingChecked || input.checked; if (input.checked) { wrappers[i].classList.remove("border-gray-400"); wrappers[i].classList.add("border-attention"); } else { wrappers[i].classList.add("border-gray-400"); wrappers[i].classList.remove("border-attention"); } } }); this.SetClass("flex flex-col"); return form; } }