xref: /aosp_15_r20/external/perfetto/ui/src/base/events.ts (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
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