1*6dbdd20aSAndroid Build Coastguard Worker// Copyright (C) 2024 The Android Open Source Project 2*6dbdd20aSAndroid Build Coastguard Worker// 3*6dbdd20aSAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License"); 4*6dbdd20aSAndroid Build Coastguard Worker// you may not use this file except in compliance with the License. 5*6dbdd20aSAndroid Build Coastguard Worker// You may obtain a copy of the License at 6*6dbdd20aSAndroid Build Coastguard Worker// 7*6dbdd20aSAndroid Build Coastguard Worker// http://www.apache.org/licenses/LICENSE-2.0 8*6dbdd20aSAndroid Build Coastguard Worker// 9*6dbdd20aSAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software 10*6dbdd20aSAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS, 11*6dbdd20aSAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*6dbdd20aSAndroid Build Coastguard Worker// See the License for the specific language governing permissions and 13*6dbdd20aSAndroid Build Coastguard Worker// limitations under the License. 14*6dbdd20aSAndroid Build Coastguard Worker 15*6dbdd20aSAndroid Build Coastguard Worker// We limit ourselves to listeners that have only one argument (or zero, if 16*6dbdd20aSAndroid Build Coastguard Worker// using void). API-wise it's more robust to wrap arguments in an interface, 17*6dbdd20aSAndroid Build Coastguard Worker// rather than passing them positionally. 18*6dbdd20aSAndroid Build Coastguard Workerexport type EvtListener<T> = (args: T) => unknown | Promise<unknown>; 19*6dbdd20aSAndroid Build Coastguard Worker 20*6dbdd20aSAndroid Build Coastguard Worker// For use in interfaces, when we want to expose only the listen() method and 21*6dbdd20aSAndroid Build Coastguard Worker// not the emit(). 22*6dbdd20aSAndroid Build Coastguard Workerexport interface Evt<T> { 23*6dbdd20aSAndroid Build Coastguard Worker addListener(listener: EvtListener<T>): Disposable; 24*6dbdd20aSAndroid Build Coastguard Worker} 25*6dbdd20aSAndroid Build Coastguard Worker 26*6dbdd20aSAndroid Build Coastguard Worker/** 27*6dbdd20aSAndroid Build Coastguard Worker * Example usage: 28*6dbdd20aSAndroid Build Coastguard Worker * 29*6dbdd20aSAndroid Build Coastguard Worker * interface OnLoadArgs {loadTime: number}; 30*6dbdd20aSAndroid Build Coastguard Worker * 31*6dbdd20aSAndroid Build Coastguard Worker * class MyClass { 32*6dbdd20aSAndroid Build Coastguard Worker * readonly onLoad = new EvtSource<OnLoadArgs>(); 33*6dbdd20aSAndroid Build Coastguard Worker * 34*6dbdd20aSAndroid Build Coastguard Worker * private doLoad() { 35*6dbdd20aSAndroid Build Coastguard Worker * this.onLoad.notify({loadTime: 42}); 36*6dbdd20aSAndroid Build Coastguard Worker * } 37*6dbdd20aSAndroid Build Coastguard Worker * } 38*6dbdd20aSAndroid Build Coastguard Worker * 39*6dbdd20aSAndroid Build Coastguard Worker * const myClass = new MyClass(); 40*6dbdd20aSAndroid Build Coastguard Worker * const listener = (args) => console.log('Load time', args.loadTime); 41*6dbdd20aSAndroid Build Coastguard Worker * trash = new DisposableStack(); 42*6dbdd20aSAndroid Build Coastguard Worker * trash.use(myClass.onLoad.listen(listener)); 43*6dbdd20aSAndroid Build Coastguard Worker * ... 44*6dbdd20aSAndroid Build Coastguard Worker * trash.dispose(); 45*6dbdd20aSAndroid Build Coastguard Worker */ 46*6dbdd20aSAndroid Build Coastguard Workerexport class EvtSource<T> implements Evt<T> { 47*6dbdd20aSAndroid Build Coastguard Worker private listeners: EvtListener<T>[] = []; 48*6dbdd20aSAndroid Build Coastguard Worker 49*6dbdd20aSAndroid Build Coastguard Worker /** 50*6dbdd20aSAndroid Build Coastguard Worker * Registers a new event listener. 51*6dbdd20aSAndroid Build Coastguard Worker * @param listener The listener to be called when the event is fired. 52*6dbdd20aSAndroid Build Coastguard Worker * @returns a Disposable object that will remove the listener on dispose. 53*6dbdd20aSAndroid Build Coastguard Worker */ 54*6dbdd20aSAndroid Build Coastguard Worker addListener(listener: EvtListener<T>): Disposable { 55*6dbdd20aSAndroid Build Coastguard Worker const listeners = this.listeners; 56*6dbdd20aSAndroid Build Coastguard Worker listeners.push(listener); 57*6dbdd20aSAndroid Build Coastguard Worker return { 58*6dbdd20aSAndroid Build Coastguard Worker [Symbol.dispose]() { 59*6dbdd20aSAndroid Build Coastguard Worker // Erase the handler from the array. (splice(length, 1) is a no-op). 60*6dbdd20aSAndroid Build Coastguard Worker const pos = listeners.indexOf(listener); 61*6dbdd20aSAndroid Build Coastguard Worker listeners.splice(pos >= 0 ? pos : listeners.length, 1); 62*6dbdd20aSAndroid Build Coastguard Worker }, 63*6dbdd20aSAndroid Build Coastguard Worker }; 64*6dbdd20aSAndroid Build Coastguard Worker } 65*6dbdd20aSAndroid Build Coastguard Worker 66*6dbdd20aSAndroid Build Coastguard Worker /** 67*6dbdd20aSAndroid Build Coastguard Worker * Fires the event, invoking all registered listeners with the provided data. 68*6dbdd20aSAndroid Build Coastguard Worker * @param args The data to be passed to the listeners. 69*6dbdd20aSAndroid Build Coastguard Worker * @returns a promise that resolves when all the listeners have fulfilled 70*6dbdd20aSAndroid Build Coastguard Worker * their promise - if they returned one - otherwise resolves immediately. 71*6dbdd20aSAndroid Build Coastguard Worker */ 72*6dbdd20aSAndroid Build Coastguard Worker async notify(args: T): Promise<void> { 73*6dbdd20aSAndroid Build Coastguard Worker const promises: unknown[] = []; 74*6dbdd20aSAndroid Build Coastguard Worker for (const listener of this.listeners) { 75*6dbdd20aSAndroid Build Coastguard Worker promises.push(Promise.resolve(listener(args))); 76*6dbdd20aSAndroid Build Coastguard Worker } 77*6dbdd20aSAndroid Build Coastguard Worker await Promise.allSettled(promises); 78*6dbdd20aSAndroid Build Coastguard Worker } 79*6dbdd20aSAndroid Build Coastguard Worker} 80