mapcomplete/UI/Input/DropDown.ts

97 lines
3.1 KiB
TypeScript
Raw Normal View History

2020-06-29 01:12:44 +00:00
import {UIElement} from "../UIElement";
2020-07-20 19:39:07 +00:00
import {InputElement} from "./InputElement";
2020-07-20 22:38:03 +00:00
import Translations from "../i18n/Translations";
import {UIEventSource} from "../../Logic/UIEventSource";
2020-07-20 19:39:07 +00:00
export class DropDown<T> extends InputElement<T> {
2020-07-20 22:38:03 +00:00
private readonly _label: UIElement;
2020-07-20 19:39:07 +00:00
private readonly _values: { value: T; shown: UIElement }[];
private readonly _value: UIEventSource<T>;
public IsSelected: UIEventSource<boolean> = new UIEventSource<boolean>(false);
2021-01-18 00:48:08 +00:00
private readonly _label_class: string;
private readonly _select_class: string;
private _form_style: string;
2020-07-20 19:39:07 +00:00
2020-07-20 22:38:03 +00:00
constructor(label: string | UIElement,
2020-07-20 19:39:07 +00:00
values: { value: T, shown: string | UIElement }[],
value: UIEventSource<T> = undefined,
2021-01-18 00:48:08 +00:00
label_class: string = "",
select_class: string = "",
form_style: string = "flex") {
super(undefined);
this._form_style = form_style;
2020-07-20 19:39:07 +00:00
this._value = value ?? new UIEventSource<T>(undefined);
2020-07-20 22:38:03 +00:00
this._label = Translations.W(label);
this._label_class = label_class || '';
this._select_class = select_class || '';
2020-07-20 19:39:07 +00:00
this._values = values.map(v => {
return {
value: v.value,
2020-07-20 22:38:03 +00:00
shown: Translations.W(v.shown)
2020-07-20 19:39:07 +00:00
}
}
);
2020-07-20 22:38:03 +00:00
for (const v of this._values) {
this.ListenTo(v.shown._source);
}
2020-07-22 14:44:07 +00:00
this.ListenTo(this._value);
2020-07-20 19:39:07 +00:00
this.onClick(() => {}) // by registering a click, the click event is consumed and doesn't bubble furter to other elements, e.g. checkboxes
}
2020-07-20 19:39:07 +00:00
GetValue(): UIEventSource<T> {
return this._value;
}
IsValid(t: T): boolean {
for (const value of this._values) {
2020-07-20 19:39:07 +00:00
if (value.value === t) {
return true;
}
}
2020-07-20 19:39:07 +00:00
return false
}
InnerRender(): string {
2020-07-20 22:07:04 +00:00
if(this._values.length <=1){
return "";
}
2020-07-20 19:39:07 +00:00
let options = "";
for (let i = 0; i < this._values.length; i++) {
options += "<option value='" + i + "'>" + this._values[i].shown.InnerRender() + "</option>"
}
return `<form class="${this._form_style}">` +
`<label class='${this._label_class}' for='dropdown-${this.id}'>${this._label.Render()}</label>` +
`<select class='${this._select_class}' name='dropdown-${this.id}' id='dropdown-${this.id}'>` +
options +
`</select>` +
`</form>`;
}
2020-07-20 19:39:07 +00:00
protected InnerUpdate(element) {
var e = document.getElementById("dropdown-" + this.id);
2020-07-20 22:07:04 +00:00
if(e === null){
return;
}
const self = this;
2020-07-20 19:39:07 +00:00
e.onchange = (() => {
// @ts-ignore
2020-07-20 19:39:07 +00:00
var index = parseInt(e.selectedIndex);
self._value.setData(self._values[index].value);
});
var t = this._value.data;
for (let i = 0; i < this._values.length ; i++) {
const value = this._values[i].value;
if (value === t) {
2020-07-20 19:39:07 +00:00
// @ts-ignore
e.selectedIndex = i;
}
}
}
}