| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  | import {Utils} from "../Utils"; | 
					
						
							|  |  |  | import {UIEventSource} from "../Logic/UIEventSource"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * A thin wrapper around a html element, which allows to generate a HTML-element. | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Assumes a read-only configuration, so it has no 'ListenTo' | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export default abstract class BaseUIElement { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private clss: Set<string> = new Set<string>(); | 
					
						
							|  |  |  |     private style: string; | 
					
						
							|  |  |  |     private _onClick: () => void; | 
					
						
							|  |  |  |     private _onHover: UIEventSource<boolean>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected _constructedHtmlElement: HTMLElement; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected abstract InnerConstructElement(): HTMLElement; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public onClick(f: (() => void)) { | 
					
						
							|  |  |  |         this._onClick = f; | 
					
						
							|  |  |  |         this.SetClass("clickable") | 
					
						
							|  |  |  |         if(this._constructedHtmlElement !== undefined){ | 
					
						
							|  |  |  |             this._constructedHtmlElement.onclick = f; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-15 00:28:59 +02:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |     AttachTo(divId: string) { | 
					
						
							|  |  |  |         let element = document.getElementById(divId); | 
					
						
							|  |  |  |         if (element === null) { | 
					
						
							|  |  |  |             throw "SEVERE: could not attach UIElement to " + divId; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (element.firstChild) { | 
					
						
							|  |  |  |             //The list is LIVE so it will re-index each call
 | 
					
						
							|  |  |  |             element.removeChild(element.firstChild); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const el = this.ConstructElement(); | 
					
						
							|  |  |  |         if(el !== undefined){ | 
					
						
							|  |  |  |             element.appendChild(el) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Adds all the relevant classes, space seperated | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public SetClass(clss: string) { | 
					
						
							|  |  |  |         const all = clss.split(" ").map(clsName => clsName.trim()); | 
					
						
							|  |  |  |         let recordedChange = false; | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  |         for (let c of all) { | 
					
						
							|  |  |  |             c = c.trim(); | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |             if (this.clss.has(clss)) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  |             if(c === undefined || c === ""){ | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |             this.clss.add(c); | 
					
						
							|  |  |  |             recordedChange = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (recordedChange) { | 
					
						
							|  |  |  |             this._constructedHtmlElement?.classList.add(...Array.from(this.clss)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public RemoveClass(clss: string): BaseUIElement { | 
					
						
							|  |  |  |         if (this.clss.has(clss)) { | 
					
						
							|  |  |  |             this.clss.delete(clss); | 
					
						
							|  |  |  |             this._constructedHtmlElement?.classList.remove(clss) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-15 00:28:59 +02:00
										 |  |  |      | 
					
						
							|  |  |  |     public HasClass(clss: string): boolean{ | 
					
						
							|  |  |  |         return this.clss.has(clss) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public SetStyle(style: string): BaseUIElement { | 
					
						
							|  |  |  |         this.style = style; | 
					
						
							|  |  |  |         if(this._constructedHtmlElement !== undefined){ | 
					
						
							|  |  |  |             this._constructedHtmlElement.style.cssText = style; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-15 01:24:04 +02:00
										 |  |  |      | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * The same as 'Render', but creates a HTML element instead of the HTML representation | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public ConstructElement(): HTMLElement { | 
					
						
							|  |  |  |         if (Utils.runningFromConsole) { | 
					
						
							|  |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (this._constructedHtmlElement !== undefined) { | 
					
						
							|  |  |  |             return this._constructedHtmlElement | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-14 02:39:23 +02:00
										 |  |  |         if(this.InnerConstructElement === undefined){ | 
					
						
							|  |  |  |             throw "ERROR! This is not a correct baseUIElement: "+this.constructor.name | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-16 14:23:53 +02:00
										 |  |  | try{ | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const el = this.InnerConstructElement(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if(el === undefined){ | 
					
						
							|  |  |  |             return undefined; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this._constructedHtmlElement = el; | 
					
						
							|  |  |  |         const style = this.style | 
					
						
							|  |  |  |         if (style !== undefined && style !== "") { | 
					
						
							|  |  |  |             el.style.cssText = style | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (this.clss.size > 0) { | 
					
						
							|  |  |  |             try{ | 
					
						
							|  |  |  |                 el.classList.add(...Array.from(this.clss)) | 
					
						
							|  |  |  |             }catch(e){ | 
					
						
							|  |  |  |                 console.error("Invalid class name detected in:", Array.from(this.clss).join(" "),"\nErr msg is ",e) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (this._onClick !== undefined) { | 
					
						
							|  |  |  |             const self = this; | 
					
						
							|  |  |  |             el.onclick = (e) => { | 
					
						
							|  |  |  |                 // @ts-ignore
 | 
					
						
							|  |  |  |                 if (e.consumed) { | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 self._onClick(); | 
					
						
							|  |  |  |                 // @ts-ignore
 | 
					
						
							|  |  |  |                 e.consumed = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             el.style.pointerEvents = "all"; | 
					
						
							|  |  |  |             el.style.cursor = "pointer"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (this._onHover !== undefined) { | 
					
						
							|  |  |  |             const self = this; | 
					
						
							|  |  |  |             el.addEventListener('mouseover', () => self._onHover.setData(true)); | 
					
						
							|  |  |  |             el.addEventListener('mouseout', () => self._onHover.setData(false)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (this._onHover !== undefined) { | 
					
						
							|  |  |  |             const self = this; | 
					
						
							|  |  |  |             el.addEventListener('mouseover', () => self._onHover.setData(true)); | 
					
						
							|  |  |  |             el.addEventListener('mouseout', () => self._onHover.setData(false)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-16 14:23:53 +02:00
										 |  |  |         return el}catch(e){ | 
					
						
							|  |  |  |             const domExc = e as DOMException; | 
					
						
							|  |  |  |             if(domExc){ | 
					
						
							|  |  |  |                 console.log("An exception occured", domExc.code, domExc.message, domExc.name ) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             console.error(e) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-15 00:28:59 +02:00
										 |  |  |      | 
					
						
							|  |  |  |     public AsMarkdown(): string{ | 
					
						
							|  |  |  |         throw "AsMarkdown is not implemented by "+this.constructor.name | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  | } |