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 } }