hyper/lib/utils/rpc.ts
2023-06-25 19:29:00 +05:30

73 lines
2.1 KiB
TypeScript

import {EventEmitter} from 'events';
import {IpcRenderer, IpcRendererEvent} from 'electron';
import electron from 'electron';
import {FilterNever, MainEvents, RendererEvents, TypedEmitter} from '../../common';
export default class Client {
emitter: TypedEmitter<RendererEvents>;
ipc: IpcRenderer;
id!: string;
constructor() {
this.emitter = new EventEmitter();
this.ipc = electron.ipcRenderer;
this.emit = this.emit.bind(this);
if (window.__rpcId) {
setTimeout(() => {
this.id = window.__rpcId;
this.ipc.on(this.id, this.ipcListener);
this.emitter.emit('ready');
}, 0);
} else {
this.ipc.on('init', (ev: IpcRendererEvent, uid: string) => {
// we cache so that if the object
// gets re-instantiated we don't
// wait for a `init` event
window.__rpcId = uid;
this.id = uid;
this.ipc.on(uid, this.ipcListener);
this.emitter.emit('ready');
});
}
}
ipcListener = <U extends keyof RendererEvents>(
event: IpcRendererEvent,
{ch, data}: {ch: U; data: RendererEvents[U]}
) => this.emitter.emit(ch, data);
on = <U extends keyof RendererEvents>(ev: U, fn: (arg0: RendererEvents[U]) => void) => {
this.emitter.on(ev, fn);
return this;
};
once = <U extends keyof RendererEvents>(ev: U, fn: (arg0: RendererEvents[U]) => void) => {
this.emitter.once(ev, fn);
return this;
};
emit<U extends Exclude<keyof MainEvents, FilterNever<MainEvents>>>(ch: U): boolean;
emit<U extends FilterNever<MainEvents>>(ch: U, data: MainEvents[U]): boolean;
emit<U extends keyof MainEvents>(ev: U, data?: MainEvents[U]) {
if (!this.id) {
throw new Error('Not ready');
}
this.ipc.send(this.id, {ev, data});
return true;
}
removeListener = <U extends keyof RendererEvents>(ev: U, fn: (arg0: RendererEvents[U]) => void) => {
this.emitter.removeListener(ev, fn);
return this;
};
removeAllListeners = () => {
this.emitter.removeAllListeners();
return this;
};
destroy = () => {
this.removeAllListeners();
this.ipc.removeAllListeners(this.id);
};
}