mapcomplete/UI/UIElement.ts

139 lines
3.6 KiB
TypeScript
Raw Normal View History

2020-06-23 22:35:19 +00:00
import {UIEventSource} from "./UIEventSource";
export abstract class UIElement {
2020-07-20 13:54:50 +00:00
2020-06-23 22:35:19 +00:00
private static nextId: number = 0;
public readonly id: string;
public readonly _source: UIEventSource<any>;
private _hideIfEmpty = false;
// WOrkaround as document is not defined
public static runningFromConsole = false;
2020-06-23 22:35:19 +00:00
protected constructor(source: UIEventSource<any>) {
this.id = "ui-element-" + UIElement.nextId;
this._source = source;
UIElement.nextId++;
this.ListenTo(source);
}
2020-07-08 11:12:23 +00:00
public ListenTo(source: UIEventSource<any>) {
2020-07-01 00:12:33 +00:00
if (source === undefined) {
2020-07-20 23:37:48 +00:00
return this;
2020-06-23 22:35:19 +00:00
}
const self = this;
source.addCallback(() => {
self.Update();
})
2020-07-20 23:37:48 +00:00
return this;
2020-06-23 22:35:19 +00:00
}
2020-07-01 00:12:33 +00:00
private _onClick: () => void;
public onClick(f: (() => void)) {
this._onClick = f;
this.Update();
return this;
2020-07-01 00:12:33 +00:00
}
2020-07-20 13:54:50 +00:00
2020-06-23 22:35:19 +00:00
Update(): void {
if(UIElement.runningFromConsole){
return;
2020-07-25 16:00:08 +00:00
}
2020-07-01 00:12:33 +00:00
let element = document.getElementById(this.id);
2020-07-20 22:07:04 +00:00
if (element === undefined || element === null) {
2020-06-23 22:35:19 +00:00
// The element is not painted
return;
}
element.innerHTML = this.InnerRender();
2020-07-01 00:12:33 +00:00
if (this._hideIfEmpty) {
if (element.innerHTML === "") {
element.parentElement.style.display = "none";
2020-07-01 00:12:33 +00:00
} else {
element.parentElement.style.display = undefined;
}
}
2020-07-01 00:12:33 +00:00
if (this._onClick !== undefined) {
const self = this;
2020-07-22 10:17:06 +00:00
element.onclick = (e) => {
2020-07-24 13:22:28 +00:00
// @ts-ignore
2020-07-22 10:17:06 +00:00
if(e.consumed){
return;
}
2020-07-01 00:12:33 +00:00
self._onClick();
2020-07-24 13:22:28 +00:00
// @ts-ignore
2020-07-22 10:17:06 +00:00
e.consumed = true;
2020-07-01 00:12:33 +00:00
}
2020-07-01 19:21:29 +00:00
element.style.pointerEvents = "all";
2020-07-01 00:12:33 +00:00
element.style.cursor = "pointer";
}
2020-06-23 22:35:19 +00:00
this.InnerUpdate(element);
2020-07-20 07:57:19 +00:00
for (const i in this) {
const child = this[i];
if (child instanceof UIElement) {
child.Update();
} else if (child instanceof Array) {
for (const ch of child) {
if (ch instanceof UIElement) {
ch.Update();
}
}
}
}
2020-06-23 22:35:19 +00:00
}
HideOnEmpty(hide : boolean){
this._hideIfEmpty = hide;
this.Update();
return this;
}
2020-06-23 22:35:19 +00:00
// Called after the HTML has been replaced. Can be used for css tricks
2020-07-20 19:39:07 +00:00
protected InnerUpdate(htmlElement: HTMLElement) {
}
2020-06-23 22:35:19 +00:00
Render(): string {
return "<span class='uielement' id='" + this.id + "'>" + this.InnerRender() + "</span>"
}
AttachTo(divId: string) {
let element = document.getElementById(divId);
2020-07-20 07:57:19 +00:00
if (element === null) {
throw "SEVERE: could not attach UIElement to " + divId;
}
2020-06-23 22:35:19 +00:00
element.innerHTML = this.Render();
this.Update();
return this;
2020-06-23 22:35:19 +00:00
}
2020-07-20 16:24:00 +00:00
public abstract InnerRender(): string;
2020-07-20 07:57:19 +00:00
public Activate(): void {
for (const i in this) {
const child = this[i];
if (child instanceof UIElement) {
child.Activate();
} else if (child instanceof Array) {
for (const ch of child) {
if (ch instanceof UIElement) {
ch.Activate();
}
}
}
}
};
2020-06-23 22:35:19 +00:00
public IsEmpty(): boolean {
return this.InnerRender() === "";
}
2020-07-20 13:54:50 +00:00
}
2020-06-23 22:35:19 +00:00