chrome-devtools-frontend
Version:
Chrome DevTools UI
152 lines (133 loc) • 5.86 kB
text/typescript
// Copyright 2021 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
/*
* Copyright (C) 2008 Apple Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
import type * as Platform from '../platform/platform.js';
import {
type EventDescriptor,
type EventListener,
type EventTarget,
type EventTargetEvent,
type EventPayloadToRestParameters,
} from './EventTarget.js';
export interface ListenerCallbackTuple<Events, T extends keyof Events> {
thisObject?: Object;
listener: EventListener<Events, T>;
disposed?: boolean;
}
export class ObjectWrapper<Events> implements EventTarget<Events> {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
listeners?: Map<keyof Events, Set<ListenerCallbackTuple<Events, any>>>;
addEventListener<T extends keyof Events>(eventType: T, listener: EventListener<Events, T>, thisObject?: Object):
EventDescriptor<Events, T> {
if (!this.listeners) {
this.listeners = new Map();
}
let listenersForEventType = this.listeners.get(eventType);
if (!listenersForEventType) {
listenersForEventType = new Set();
this.listeners.set(eventType, listenersForEventType);
}
listenersForEventType.add({thisObject, listener});
return {eventTarget: this, eventType, thisObject, listener};
}
once<T extends keyof Events>(eventType: T): Promise<Events[T]> {
return new Promise(resolve => {
const descriptor = this.addEventListener(eventType, event => {
this.removeEventListener(eventType, descriptor.listener);
resolve(event.data);
});
});
}
removeEventListener<T extends keyof Events>(eventType: T, listener: EventListener<Events, T>, thisObject?: Object):
void {
const listeners = this.listeners?.get(eventType);
if (!listeners) {
return;
}
for (const listenerTuple of listeners) {
if (listenerTuple.listener === listener && listenerTuple.thisObject === thisObject) {
listenerTuple.disposed = true;
listeners.delete(listenerTuple);
}
}
if (!listeners.size) {
this.listeners?.delete(eventType);
}
}
hasEventListeners(eventType: keyof Events): boolean {
return Boolean(this.listeners && this.listeners.has(eventType));
}
dispatchEventToListeners<T extends keyof Events>(
eventType: Platform.TypeScriptUtilities.NoUnion<T>,
...[eventData]: EventPayloadToRestParameters<Events, T>): void {
const listeners = this.listeners?.get(eventType);
if (!listeners) {
return;
}
// `eventData` is typed as `Events[T] | undefined`:
// - `undefined` when `Events[T]` is void.
// - `Events[T]` otherwise.
// We cast it to `Events[T]` which is the correct type in all instances, as
// `void` will be cast and used as `undefined`.
const event = {data: eventData as Events[T], source: this};
// Work on a snapshot of the current listeners, callbacks might remove/add
// new listeners.
for (const listener of [...listeners]) {
if (!listener.disposed) {
listener.listener.call(listener.thisObject, event);
}
}
}
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any
type Constructor = new (...args: any[]) => {};
// eslint-disable-next-line @typescript-eslint/explicit-function-return-type
export function eventMixin<Events, Base extends Constructor>(base: Base) {
return class EventHandling extends base implements EventTarget<Events> {
#events = new ObjectWrapper<Events>();
addEventListener<T extends keyof Events>(
eventType: T, listener: (arg0: EventTargetEvent<Events[T]>) => void,
thisObject?: Object): EventDescriptor<Events, T> {
return this.#events.addEventListener(eventType, listener, thisObject);
}
once<T extends keyof Events>(eventType: T): Promise<Events[T]> {
return this.#events.once(eventType);
}
removeEventListener<T extends keyof Events>(
eventType: T, listener: (arg0: EventTargetEvent<Events[T]>) => void, thisObject?: Object): void {
this.#events.removeEventListener(eventType, listener, thisObject);
}
hasEventListeners(eventType: keyof Events): boolean {
return this.#events.hasEventListeners(eventType);
}
dispatchEventToListeners<T extends keyof Events>(
eventType: Platform.TypeScriptUtilities.NoUnion<T>,
...eventData: EventPayloadToRestParameters<Events, T>): void {
this.#events.dispatchEventToListeners(eventType, ...eventData);
}
};
}