Notes/src/public/app/components/component.ts

122 lines
4.0 KiB
TypeScript
Raw Normal View History

2020-02-01 11:15:58 +01:00
import utils from '../services/utils.js';
/**
* Abstract class for all components in the Trilium's frontend.
*
* Contains also event implementation with following properties:
* - event / command distribution is synchronous which among others mean that events are well-ordered - event
* which was sent out first will also be processed first by the component
* - execution of the event / command is asynchronous - each component executes the event on its own without regard for
* other components.
* - although the execution is async, we are collecting all the promises, and therefore it is possible to wait until the
* event / command is executed in all components - by simply awaiting the `triggerEvent()`.
*/
2020-01-15 21:36:01 +01:00
export default class Component {
componentId: string;
children: Component[];
initialized: Promise<void> | null;
parent?: Component;
position!: number;
2020-02-27 00:58:10 +01:00
constructor() {
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
}
get sanitizedClassName() {
// webpack mangles names and sometimes uses unsafe characters
return this.constructor.name.replace(/[^A-Z0-9]/ig, "_");
}
setParent(parent: Component) {
2020-02-27 00:58:10 +01:00
this.parent = parent;
2020-02-27 10:03:14 +01:00
return this;
}
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;
}
handleEvent(name: string, data: unknown): Promise<unknown> | null {
2022-11-27 23:43:25 +01:00
try {
const callMethodPromise = this.initialized
? 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);
}
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
}
triggerEvent(name: string, data = {}): Promise<unknown> | undefined | null {
return this.parent?.triggerEvent(name, data);
2020-01-15 21:36:01 +01:00
}
2020-01-24 20:15:53 +01:00
handleEventInChildren(name: string, data: unknown = {}) {
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-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
triggerCommand(name: string, data = {}): Promise<unknown> | undefined | null {
const fun = (this as any)[`${name}Command`];
if (fun) {
return this.callMethod(fun, data);
} else {
if (!this.parent) {
throw new Error(`Component "${this.componentId}" does not have a parent attached to propagate a command.`);
}
return this.parent.triggerCommand(name, data);
}
}
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
}
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}`);
}
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
}
}