1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
|
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
/**
* Methods for setting up and tearing down page event listeners. These are used
* to dismiss Feature Callouts when the callout's anchor element is clicked.
*/
export class PageEventManager {
/**
* A set of parameters defining a page event listener.
* @typedef {Object} PageEventListenerParams
* @property {String} type Event type string e.g. `click`
* @property {String} selectors Target selector, e.g. `tag.class, #id[attr]`
* @property {PageEventListenerOptions} [options] addEventListener options
*
* @typedef {Object} PageEventListenerOptions
* @property {Boolean} [capture] Use event capturing phase?
* @property {Boolean} [once] Remove listener after first event?
* @property {Boolean} [preventDefault] Inverted value for `passive` option
*/
/**
* Maps event listener params to their abort controllers.
* @type {Map<PageEventListenerParams, AbortController>}
*/
_listeners = new Map();
/**
* @param {Document} doc The document to look for event targets in
*/
constructor(doc) {
this.doc = doc;
}
/**
* Adds a page event listener.
* @param {PageEventListenerParams} params
* @param {Function} callback Function to call when event is triggered
*/
on(params, callback) {
if (this._listeners.has(params)) {
return;
}
const { type, selectors, options = {} } = params;
const controller = new AbortController();
const opt = {
capture: !!options.capture,
passive: !options.preventDefault,
signal: controller.signal,
};
const targets = this.doc.querySelectorAll(selectors);
for (const target of targets) {
target.addEventListener(type, callback, opt);
}
this._listeners.set(params, controller);
}
/**
* Removes a page event listener.
* @param {PageEventListenerParams} params
*/
off(params) {
const controller = this._listeners.get(params);
if (!controller) {
return;
}
controller.abort();
this._listeners.delete(params);
}
/**
* Adds a page event listener that is removed after the first event.
* @param {PageEventListenerParams} params
* @param {Function} callback Function to call when event is triggered
*/
once(params, callback) {
const wrappedCallback = (...args) => {
this.off(params);
callback(...args);
};
this.on(params, wrappedCallback);
}
/**
* Removes all page event listeners.
*/
clear() {
for (const controller of this._listeners.values()) {
controller.abort();
}
this._listeners.clear();
}
}
|