| 
									
										
										
										
											2020-02-01 11:15:58 +01:00
										 |  |  | import utils from '../services/utils.js'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-29 19:43:19 +01:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Abstract class for all components in the Trilium's frontend. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contains also event implementation with following properties: | 
					
						
							| 
									
										
										
										
											2023-01-15 21:04:17 +01:00
										 |  |  |  * - event / command distribution is synchronous which among others mean that events are well-ordered - event | 
					
						
							| 
									
										
										
										
											2020-06-10 23:43:59 +02:00
										 |  |  |  *   which was sent out first will also be processed first by the component | 
					
						
							| 
									
										
										
										
											2020-02-29 19:43:19 +01:00
										 |  |  |  * - execution of the event / command is asynchronous - each component executes the event on its own without regard for | 
					
						
							|  |  |  |  *   other components. | 
					
						
							| 
									
										
										
										
											2023-01-15 21:04:17 +01:00
										 |  |  |  * - although the execution is async, we are collecting all the promises, and therefore it is possible to wait until the | 
					
						
							| 
									
										
										
										
											2020-02-29 19:43:19 +01:00
										 |  |  |  *   event / command is executed in all components - by simply awaiting the `triggerEvent()`. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-01-15 21:36:01 +01:00
										 |  |  | export default class Component { | 
					
						
							| 
									
										
										
										
											2024-12-22 16:22:10 +02:00
										 |  |  |     $widget!: JQuery<HTMLElement>; | 
					
						
							| 
									
										
										
										
											2024-07-25 20:55:04 +03:00
										 |  |  |     componentId: string; | 
					
						
							|  |  |  |     children: Component[]; | 
					
						
							|  |  |  |     initialized: Promise<void> | null; | 
					
						
							|  |  |  |     parent?: Component; | 
					
						
							| 
									
										
										
										
											2024-12-21 15:09:52 +02:00
										 |  |  |     position!: number; | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-27 00:58:10 +01:00
										 |  |  |     constructor() { | 
					
						
							| 
									
										
										
										
											2022-12-21 15:19:05 +01:00
										 |  |  |         this.componentId = `${this.sanitizedClassName}-${utils.randomString(8)}`; | 
					
						
							| 
									
										
										
										
											2020-01-15 21:36:01 +01:00
										 |  |  |         this.children = []; | 
					
						
							| 
									
										
										
										
											2022-06-23 23:03:35 +02:00
										 |  |  |         this.initialized = null; | 
					
						
							| 
									
										
										
										
											2020-01-15 21:36:01 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-30 22:29:38 +02:00
										 |  |  |     get sanitizedClassName() { | 
					
						
							|  |  |  |         // webpack mangles names and sometimes uses unsafe characters
 | 
					
						
							|  |  |  |         return this.constructor.name.replace(/[^A-Z0-9]/ig, "_"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |     setParent(parent: Component) { | 
					
						
							| 
									
										
										
										
											2020-02-27 00:58:10 +01:00
										 |  |  |         this.parent = parent; | 
					
						
							| 
									
										
										
										
											2020-02-27 10:03:14 +01:00
										 |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |     child(...components: Component[]) { | 
					
						
							| 
									
										
										
										
											2020-03-16 21:16:09 +01:00
										 |  |  |         for (const component of components) { | 
					
						
							| 
									
										
										
										
											2020-03-16 22:14:18 +01:00
										 |  |  |             component.setParent(this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             this.children.push(component); | 
					
						
							| 
									
										
										
										
											2020-03-16 21:16:09 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |     handleEvent(name: string, data: unknown): Promise<unknown> | null { | 
					
						
							| 
									
										
										
										
											2022-11-27 23:43:25 +01:00
										 |  |  |         try { | 
					
						
							|  |  |  |             const callMethodPromise = this.initialized | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |                 ? this.initialized.then(() => this.callMethod((this as any)[`${name}Event`], data)) | 
					
						
							|  |  |  |                 : this.callMethod((this as any)[`${name}Event`], data); | 
					
						
							| 
									
										
										
										
											2022-06-23 23:03:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-27 23:43:25 +01:00
										 |  |  |             const childrenPromise = this.handleEventInChildren(name, data); | 
					
						
							| 
									
										
										
										
											2022-06-23 23:03:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-27 23:43:25 +01:00
										 |  |  |             // don't create promises if not needed (optimization)
 | 
					
						
							|  |  |  |             return callMethodPromise && childrenPromise | 
					
						
							|  |  |  |                 ? Promise.all([callMethodPromise, childrenPromise]) | 
					
						
							|  |  |  |                 : (callMethodPromise || childrenPromise); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |         catch (e: any) { | 
					
						
							| 
									
										
										
										
											2022-11-27 23:43:25 +01:00
										 |  |  |             console.error(`Handling of event '${name}' failed in ${this.constructor.name} with error ${e.message} ${e.stack}`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-01-15 21:36:01 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 20:55:04 +03:00
										 |  |  |     triggerEvent(name: string, data = {}): Promise<unknown> | undefined | null { | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |         return this.parent?.triggerEvent(name, data); | 
					
						
							| 
									
										
										
										
											2020-01-15 21:36:01 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-01-24 20:15:53 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |     handleEventInChildren(name: string, data: unknown = {}) { | 
					
						
							| 
									
										
										
										
											2020-02-12 20:07:04 +01:00
										 |  |  |         const promises = []; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-24 20:15:53 +01:00
										 |  |  |         for (const child of this.children) { | 
					
						
							| 
									
										
										
										
											2022-07-10 15:01:05 +02:00
										 |  |  |             const ret = child.handleEvent(name, data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (ret) { | 
					
						
							|  |  |  |                 promises.push(ret); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-02-12 20:07:04 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-01-24 20:15:53 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-23 23:03:35 +02:00
										 |  |  |         // don't create promises if not needed (optimization)
 | 
					
						
							| 
									
										
										
										
											2022-07-10 15:01:05 +02:00
										 |  |  |         return promises.length > 0 ? Promise.all(promises) : null; | 
					
						
							| 
									
										
										
										
											2020-01-24 20:15:53 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-15 09:43:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 20:55:04 +03:00
										 |  |  |     triggerCommand(name: string, data = {}): Promise<unknown> | undefined | null { | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |         const fun = (this as any)[`${name}Command`]; | 
					
						
							| 
									
										
										
										
											2020-02-16 19:54:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-29 19:43:19 +01:00
										 |  |  |         if (fun) { | 
					
						
							|  |  |  |             return this.callMethod(fun, data); | 
					
						
							| 
									
										
										
										
											2024-11-22 23:02:43 +02:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             if (!this.parent) { | 
					
						
							|  |  |  |                 throw new Error(`Component "${this.componentId}" does not have a parent attached to propagate a command.`); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-29 19:43:19 +01:00
										 |  |  |             return this.parent.triggerCommand(name, data); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-02-16 19:54:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 19:21:24 +03:00
										 |  |  |     callMethod(fun: (arg: unknown) => Promise<unknown>, data: unknown) { | 
					
						
							| 
									
										
										
										
											2020-02-15 10:41:21 +01:00
										 |  |  |         if (typeof fun !== 'function') { | 
					
						
							| 
									
										
										
										
											2022-07-10 15:01:05 +02:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2020-02-15 10:41:21 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-30 22:29:38 +02:00
										 |  |  |         const startTime = Date.now(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const promise = fun.call(this, data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const took = Date.now() - startTime; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (glob.isDev && took > 20) { // measuring only sync handlers
 | 
					
						
							|  |  |  |             console.log(`Call to ${fun.name} in ${this.componentId} took ${took}ms`); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 15:01:05 +02:00
										 |  |  |         if (glob.isDev && promise) { | 
					
						
							|  |  |  |             return utils.timeLimit(promise, 20000, `Time limit failed on ${this.constructor.name} with ${fun.name}`); | 
					
						
							| 
									
										
										
										
											2021-02-21 21:28:00 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-02-15 09:43:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 15:01:05 +02:00
										 |  |  |         return promise; | 
					
						
							| 
									
										
										
										
											2020-02-15 09:43:47 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-19 22:58:08 +02:00
										 |  |  | } |