nitro/libs/renderer/src/core/communication/messages/MessageClassManager.ts

116 lines
2.9 KiB
TypeScript

import {IMessageComposer, IMessageConfiguration, IMessageEvent} from "../../../api";
import {MessageEvent} from "../../../events";
export class MessageClassManager {
private _messageIdByEvent: Map<Function, number>;
private _messageIdByComposer: Map<Function, number>;
private _messageInstancesById: Map<number, IMessageEvent[]>;
constructor() {
this._messageIdByEvent = new Map();
this._messageIdByComposer = new Map();
this._messageInstancesById = new Map();
}
public dispose(): void {
this._messageIdByEvent.clear();
this._messageIdByComposer.clear();
this._messageInstancesById.clear();
}
public registerMessages(configuration: IMessageConfiguration): void {
for (const [header, handler] of configuration.events) this.registerMessageEventClass(header, handler);
for (const [header, handler] of configuration.composers) this.registerMessageComposerClass(header, handler);
}
private registerMessageEventClass(header: number, handler: Function): void {
if (!header || !handler) return;
this._messageIdByEvent.set(handler, header);
}
private registerMessageComposerClass(header: number, handler: Function): void {
if (!header || !handler) return;
this._messageIdByComposer.set(handler, header);
}
public registerMessageEvent(event: IMessageEvent): void {
if (!event) return;
const header = this.getEventId(event);
if (!header) return;
let existing = this._messageInstancesById.get(header);
if (!existing || !existing.length) {
existing = [];
this._messageInstancesById.set(header, existing);
}
existing.push(event);
}
public removeMessageEvent(event: IMessageEvent): void {
if (!event) return;
const header = this.getEventId(event);
if (!header) return;
const existing = this._messageInstancesById.get(header);
if (!existing) return;
for (const [index, message] of existing.entries()) {
if (!message) continue;
if (message !== event) continue;
existing.splice(index, 1);
if (existing.length === 0) this._messageInstancesById.delete(header);
message.dispose();
return;
}
}
public getEvents(header: number): IMessageEvent[] {
if (!header) return null;
const existing = this._messageInstancesById.get(header);
if (!existing) return null;
return existing;
}
public getEventId(event: IMessageEvent): number {
if (!event) return -1;
//@ts-ignore
const name = (event instanceof MessageEvent ? event.constructor : event) as Function;
const existing = this._messageIdByEvent.get(name);
if (!existing) return -1;
return existing;
}
public getComposerId(composer: IMessageComposer<unknown[]>): number {
if (!composer) return -1;
const existing = this._messageIdByComposer.get(composer.constructor);
if (!existing) return -1;
return existing;
}
}