forked from MapComplete/MapComplete
91 lines
2.6 KiB
TypeScript
91 lines
2.6 KiB
TypeScript
import {UIEventSource} from "../Logic/UIEventSource";
|
|
import BaseUIElement from "./BaseUIElement";
|
|
|
|
export abstract class UIElement extends BaseUIElement{
|
|
|
|
private static nextId: number = 0;
|
|
public readonly id: string;
|
|
public readonly _source: UIEventSource<any>;
|
|
|
|
private lastInnerRender: string;
|
|
|
|
protected constructor(source: UIEventSource<any> = undefined) {
|
|
super()
|
|
this.id = `ui-${this.constructor.name}-${UIElement.nextId}`;
|
|
this._source = source;
|
|
UIElement.nextId++;
|
|
this.ListenTo(source);
|
|
}
|
|
|
|
public ListenTo(source: UIEventSource<any>) {
|
|
if (source === undefined) {
|
|
return this;
|
|
}
|
|
//console.trace("Got a listenTo in ", this.constructor.name)
|
|
const self = this;
|
|
source.addCallback(() => {
|
|
self.lastInnerRender = undefined;
|
|
if(self._constructedHtmlElement !== undefined){
|
|
self.UpdateElement(self._constructedHtmlElement);
|
|
}
|
|
|
|
})
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Should be overridden for specific HTML functionality
|
|
*/
|
|
protected InnerConstructElement(): HTMLElement {
|
|
// Uses the old fashioned way to construct an element using 'InnerRender'
|
|
const innerRender = this.InnerRender();
|
|
if (innerRender === undefined || innerRender === "") {
|
|
return undefined;
|
|
}
|
|
const el = document.createElement("span")
|
|
if (typeof innerRender === "string") {
|
|
el.innerHTML = innerRender
|
|
} else {
|
|
const subElement = innerRender.ConstructElement();
|
|
if (subElement === undefined) {
|
|
return undefined;
|
|
}
|
|
el.appendChild(subElement)
|
|
}
|
|
return el;
|
|
}
|
|
|
|
protected UpdateElement(el: HTMLElement) : void{
|
|
const innerRender = this.InnerRender();
|
|
|
|
if (typeof innerRender === "string") {
|
|
if(el.innerHTML !== innerRender){
|
|
el.innerHTML = innerRender
|
|
}
|
|
} else {
|
|
const subElement = innerRender.ConstructElement();
|
|
if(el.children.length === 1 && el.children[0] === subElement){
|
|
return; // Nothing changed
|
|
}
|
|
|
|
while (el.firstChild) {
|
|
el.removeChild(el.firstChild);
|
|
}
|
|
|
|
if (subElement === undefined) {
|
|
return;
|
|
}
|
|
el.appendChild(subElement)
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* @deprecated The method should not be used
|
|
*/
|
|
protected abstract InnerRender(): string | BaseUIElement;
|
|
|
|
}
|
|
|
|
|
|
|