2021-01-12 02:13:41 +09:00
|
|
|
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
|
2020-07-19 19:49:44 +02:00
|
|
|
|
|
|
|
((window) => {
|
2021-01-05 22:43:25 +01:00
|
|
|
const { setIsTrusted } = window.__bootstrap.event;
|
|
|
|
|
2020-07-19 19:49:44 +02:00
|
|
|
const add = Symbol("add");
|
|
|
|
const signalAbort = Symbol("signalAbort");
|
|
|
|
const remove = Symbol("remove");
|
|
|
|
|
2020-09-19 22:30:59 +01:00
|
|
|
const illegalConstructorKey = Symbol("illegalConstructorKey");
|
|
|
|
|
2020-07-19 19:49:44 +02:00
|
|
|
class AbortSignal extends EventTarget {
|
|
|
|
#aborted = false;
|
|
|
|
#abortAlgorithms = new Set();
|
|
|
|
|
|
|
|
[add](algorithm) {
|
|
|
|
this.#abortAlgorithms.add(algorithm);
|
|
|
|
}
|
|
|
|
|
|
|
|
[signalAbort]() {
|
|
|
|
if (this.#aborted) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.#aborted = true;
|
|
|
|
for (const algorithm of this.#abortAlgorithms) {
|
|
|
|
algorithm();
|
|
|
|
}
|
|
|
|
this.#abortAlgorithms.clear();
|
2021-01-05 22:43:25 +01:00
|
|
|
const event = new Event("abort");
|
|
|
|
setIsTrusted(event, true);
|
|
|
|
this.dispatchEvent(event);
|
2020-07-19 19:49:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
[remove](algorithm) {
|
|
|
|
this.#abortAlgorithms.delete(algorithm);
|
|
|
|
}
|
|
|
|
|
2020-11-14 14:10:23 +02:00
|
|
|
constructor(key = null) {
|
2020-09-19 22:30:59 +01:00
|
|
|
if (key != illegalConstructorKey) {
|
|
|
|
throw new TypeError("Illegal constructor.");
|
|
|
|
}
|
2020-07-19 19:49:44 +02:00
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
get aborted() {
|
|
|
|
return Boolean(this.#aborted);
|
|
|
|
}
|
|
|
|
|
|
|
|
get [Symbol.toStringTag]() {
|
|
|
|
return "AbortSignal";
|
|
|
|
}
|
|
|
|
}
|
2020-11-06 03:40:36 +02:00
|
|
|
defineEventHandler(AbortSignal.prototype, "abort");
|
2020-07-19 19:49:44 +02:00
|
|
|
class AbortController {
|
2020-09-19 22:30:59 +01:00
|
|
|
#signal = new AbortSignal(illegalConstructorKey);
|
2020-07-19 19:49:44 +02:00
|
|
|
|
|
|
|
get signal() {
|
|
|
|
return this.#signal;
|
|
|
|
}
|
|
|
|
|
|
|
|
abort() {
|
|
|
|
this.#signal[signalAbort]();
|
|
|
|
}
|
|
|
|
|
|
|
|
get [Symbol.toStringTag]() {
|
|
|
|
return "AbortController";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-06 03:40:36 +02:00
|
|
|
const handlerSymbol = Symbol("eventHandlers");
|
2020-11-09 18:49:19 +02:00
|
|
|
|
|
|
|
function makeWrappedHandler(handler) {
|
|
|
|
function wrappedHandler(...args) {
|
|
|
|
if (typeof wrappedHandler.handler !== "function") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
return wrappedHandler.handler.call(this, ...args);
|
|
|
|
}
|
|
|
|
wrappedHandler.handler = handler;
|
|
|
|
return wrappedHandler;
|
|
|
|
}
|
2020-11-10 05:34:42 +02:00
|
|
|
// TODO(benjamingr) reuse this here and websocket where possible
|
2020-11-06 03:40:36 +02:00
|
|
|
function defineEventHandler(emitter, name) {
|
|
|
|
// HTML specification section 8.1.5.1
|
|
|
|
Object.defineProperty(emitter, `on${name}`, {
|
|
|
|
get() {
|
2020-11-09 18:49:19 +02:00
|
|
|
return this[handlerSymbol]?.get(name)?.handler;
|
2020-11-06 03:40:36 +02:00
|
|
|
},
|
|
|
|
set(value) {
|
|
|
|
if (!this[handlerSymbol]) {
|
|
|
|
this[handlerSymbol] = new Map();
|
|
|
|
}
|
2020-11-09 18:49:19 +02:00
|
|
|
let handlerWrapper = this[handlerSymbol]?.get(name);
|
|
|
|
if (handlerWrapper) {
|
|
|
|
handlerWrapper.handler = value;
|
|
|
|
} else {
|
|
|
|
handlerWrapper = makeWrappedHandler(value);
|
|
|
|
this.addEventListener(name, handlerWrapper);
|
|
|
|
}
|
|
|
|
this[handlerSymbol].set(name, handlerWrapper);
|
2020-11-06 03:40:36 +02:00
|
|
|
},
|
|
|
|
configurable: true,
|
|
|
|
enumerable: true,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-08-19 14:43:20 +02:00
|
|
|
window.AbortSignal = AbortSignal;
|
|
|
|
window.AbortController = AbortController;
|
|
|
|
window.__bootstrap = window.__bootstrap || {};
|
2020-07-19 19:49:44 +02:00
|
|
|
window.__bootstrap.abortSignal = {
|
|
|
|
add,
|
|
|
|
signalAbort,
|
|
|
|
remove,
|
|
|
|
};
|
|
|
|
})(this);
|