2019-01-11 19:33:44 +00:00
|
|
|
import { IDisposable } from "@coder/disposable";
|
|
|
|
|
|
|
|
export interface Event<T> {
|
2019-03-26 18:01:25 +00:00
|
|
|
(listener: (value: T) => void): IDisposable;
|
|
|
|
(id: number | string, listener: (value: T) => void): IDisposable;
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-03-26 18:01:25 +00:00
|
|
|
* Emitter typecasts for a single event type. You can optionally use IDs, but
|
|
|
|
* using undefined with IDs will not work. If you emit without an ID, *all*
|
|
|
|
* listeners regardless of their ID (or lack thereof) will receive the event.
|
|
|
|
* Similarly, if you listen without an ID you will get *all* events for any or
|
|
|
|
* no ID.
|
2019-01-11 19:33:44 +00:00
|
|
|
*/
|
|
|
|
export class Emitter<T> {
|
2019-03-26 18:01:25 +00:00
|
|
|
private listeners = <Array<(value: T) => void>>[];
|
|
|
|
private readonly idListeners = new Map<number | string, Array<(value: T) => void>>();
|
2019-01-11 19:33:44 +00:00
|
|
|
|
|
|
|
public get event(): Event<T> {
|
2019-03-26 18:01:25 +00:00
|
|
|
return (id: number | string | ((value: T) => void), cb?: (value: T) => void): IDisposable => {
|
|
|
|
if (typeof id !== "function") {
|
|
|
|
if (this.idListeners.has(id)) {
|
|
|
|
this.idListeners.get(id)!.push(cb!);
|
|
|
|
} else {
|
|
|
|
this.idListeners.set(id, [cb!]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
dispose: (): void => {
|
|
|
|
if (this.idListeners.has(id)) {
|
|
|
|
const cbs = this.idListeners.get(id)!;
|
|
|
|
const i = cbs.indexOf(cb!);
|
|
|
|
if (i !== -1) {
|
|
|
|
cbs.splice(i, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
|
|
|
|
2019-03-26 18:01:25 +00:00
|
|
|
cb = id;
|
|
|
|
this.listeners.push(cb);
|
|
|
|
|
2019-01-11 19:33:44 +00:00
|
|
|
return {
|
|
|
|
dispose: (): void => {
|
2019-03-26 18:01:25 +00:00
|
|
|
const i = this.listeners.indexOf(cb!);
|
|
|
|
if (i !== -1) {
|
|
|
|
this.listeners.splice(i, 1);
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-02-06 16:16:43 +00:00
|
|
|
* Emit an event with a value.
|
2019-01-11 19:33:44 +00:00
|
|
|
*/
|
2019-03-26 18:01:25 +00:00
|
|
|
public emit(value: T): void;
|
|
|
|
public emit(id: number | string, value: T): void;
|
|
|
|
public emit(id: number | string | T, value?: T): void {
|
|
|
|
if ((typeof id === "number" || typeof id === "string") && typeof value !== "undefined") {
|
|
|
|
if (this.idListeners.has(id)) {
|
|
|
|
this.idListeners.get(id)!.forEach((cb) => cb(value!));
|
|
|
|
}
|
|
|
|
this.listeners.forEach((cb) => cb(value!));
|
|
|
|
} else {
|
|
|
|
this.idListeners.forEach((cbs) => cbs.forEach((cb) => cb((id as T)!)));
|
|
|
|
this.listeners.forEach((cb) => cb((id as T)!));
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-02-06 16:16:43 +00:00
|
|
|
* Dispose the current events.
|
2019-01-11 19:33:44 +00:00
|
|
|
*/
|
2019-03-26 18:01:25 +00:00
|
|
|
public dispose(): void;
|
|
|
|
public dispose(id: number | string): void;
|
|
|
|
public dispose(id?: number | string): void {
|
|
|
|
if (typeof id !== "undefined") {
|
|
|
|
this.idListeners.delete(id);
|
|
|
|
} else {
|
|
|
|
this.listeners = [];
|
|
|
|
this.idListeners.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public get counts(): { [key: string]: number } {
|
|
|
|
const counts = <{ [key: string]: number }>{};
|
|
|
|
if (this.listeners.length > 0) {
|
|
|
|
counts["n/a"] = this.listeners.length;
|
|
|
|
}
|
|
|
|
this.idListeners.forEach((cbs, id) => {
|
|
|
|
if (cbs.length > 0) {
|
|
|
|
counts[`${id}`] = cbs.length;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return counts;
|
2019-01-11 19:33:44 +00:00
|
|
|
}
|
2019-02-06 00:08:48 +00:00
|
|
|
}
|