2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2020-07-19 19:49:44 +02:00
|
|
|
|
2023-12-07 14:21:01 +01:00
|
|
|
import { core, primordials } from "ext:core/mod.js";
|
2024-03-01 11:15:18 -07:00
|
|
|
import { op_defer, op_now } from "ext:core/ops";
|
2023-02-07 20:22:46 +01:00
|
|
|
const {
|
|
|
|
Uint8Array,
|
|
|
|
Uint32Array,
|
|
|
|
PromisePrototypeThen,
|
2023-04-03 02:41:41 +09:00
|
|
|
TypedArrayPrototypeGetBuffer,
|
2023-02-07 20:22:46 +01:00
|
|
|
TypeError,
|
|
|
|
indirectEval,
|
2024-03-01 11:15:18 -07:00
|
|
|
ReflectApply,
|
2023-02-07 20:22:46 +01:00
|
|
|
} = primordials;
|
2024-01-11 07:37:25 +09:00
|
|
|
|
2023-03-08 07:44:54 -04:00
|
|
|
import * as webidl from "ext:deno_webidl/00_webidl.js";
|
2023-02-07 20:22:46 +01:00
|
|
|
|
|
|
|
const hrU8 = new Uint8Array(8);
|
2023-04-03 02:41:41 +09:00
|
|
|
const hr = new Uint32Array(TypedArrayPrototypeGetBuffer(hrU8));
|
2023-02-07 20:22:46 +01:00
|
|
|
function opNow() {
|
2024-01-11 07:37:25 +09:00
|
|
|
op_now(hrU8);
|
2023-02-07 20:22:46 +01:00
|
|
|
return (hr[0] * 1000 + hr[1] / 1e6);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
2024-03-01 11:15:18 -07:00
|
|
|
function checkThis(thisArg) {
|
|
|
|
if (thisArg !== null && thisArg !== undefined && thisArg !== globalThis) {
|
|
|
|
throw new TypeError("Illegal invocation");
|
2023-02-07 20:22:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-03-01 11:15:18 -07:00
|
|
|
* Call a callback function immediately.
|
2023-02-07 20:22:46 +01:00
|
|
|
*/
|
2024-03-01 11:15:18 -07:00
|
|
|
function setImmediate(callback, ...args) {
|
|
|
|
if (args.length > 0) {
|
|
|
|
const unboundCallback = callback;
|
|
|
|
callback = () => ReflectApply(unboundCallback, window, args);
|
2020-07-19 19:49:44 +02:00
|
|
|
}
|
|
|
|
|
2024-03-01 11:15:18 -07:00
|
|
|
return core.queueImmediate(
|
|
|
|
callback,
|
2023-02-07 20:22:46 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-03-01 11:15:18 -07:00
|
|
|
* Call a callback function after a delay.
|
2023-02-07 20:22:46 +01:00
|
|
|
*/
|
|
|
|
function setTimeout(callback, timeout = 0, ...args) {
|
|
|
|
checkThis(this);
|
2024-03-01 11:15:18 -07:00
|
|
|
// If callback is a string, replace it with a function that evals the string on every timeout
|
2023-02-07 20:22:46 +01:00
|
|
|
if (typeof callback !== "function") {
|
2024-03-01 11:15:18 -07:00
|
|
|
const unboundCallback = webidl.converters.DOMString(callback);
|
|
|
|
callback = () => indirectEval(unboundCallback);
|
|
|
|
}
|
|
|
|
if (args.length > 0) {
|
|
|
|
const unboundCallback = callback;
|
|
|
|
callback = () => ReflectApply(unboundCallback, window, args);
|
2020-07-19 19:49:44 +02:00
|
|
|
}
|
2023-02-07 20:22:46 +01:00
|
|
|
timeout = webidl.converters.long(timeout);
|
2024-03-01 11:15:18 -07:00
|
|
|
return core.queueUserTimer(
|
|
|
|
core.getTimerDepth() + 1,
|
|
|
|
false,
|
|
|
|
timeout,
|
|
|
|
callback,
|
|
|
|
);
|
2023-02-07 20:22:46 +01:00
|
|
|
}
|
2020-07-19 19:49:44 +02:00
|
|
|
|
2024-03-01 11:15:18 -07:00
|
|
|
/**
|
|
|
|
* Call a callback function after a delay.
|
|
|
|
*/
|
2023-02-07 20:22:46 +01:00
|
|
|
function setInterval(callback, timeout = 0, ...args) {
|
|
|
|
checkThis(this);
|
|
|
|
if (typeof callback !== "function") {
|
2024-03-01 11:15:18 -07:00
|
|
|
const unboundCallback = webidl.converters.DOMString(callback);
|
|
|
|
callback = () => indirectEval(unboundCallback);
|
2021-12-09 17:00:55 +09:00
|
|
|
}
|
2024-03-01 11:15:18 -07:00
|
|
|
if (args.length > 0) {
|
|
|
|
const unboundCallback = callback;
|
|
|
|
callback = () => ReflectApply(unboundCallback, window, args);
|
2023-05-23 00:19:44 +02:00
|
|
|
}
|
|
|
|
timeout = webidl.converters.long(timeout);
|
2024-03-01 11:15:18 -07:00
|
|
|
return core.queueUserTimer(
|
|
|
|
core.getTimerDepth() + 1,
|
|
|
|
true,
|
|
|
|
timeout,
|
|
|
|
callback,
|
|
|
|
);
|
2023-05-23 00:19:44 +02:00
|
|
|
}
|
|
|
|
|
2024-03-01 11:15:18 -07:00
|
|
|
/**
|
|
|
|
* Clear a timeout or interval.
|
|
|
|
*/
|
2023-02-07 20:22:46 +01:00
|
|
|
function clearTimeout(id = 0) {
|
|
|
|
checkThis(this);
|
|
|
|
id = webidl.converters.long(id);
|
2024-03-01 11:15:18 -07:00
|
|
|
core.cancelTimer(id);
|
2023-02-07 20:22:46 +01:00
|
|
|
}
|
2021-12-09 17:00:55 +09:00
|
|
|
|
2024-03-01 11:15:18 -07:00
|
|
|
/**
|
|
|
|
* Clear a timeout or interval.
|
|
|
|
*/
|
2023-02-07 20:22:46 +01:00
|
|
|
function clearInterval(id = 0) {
|
|
|
|
checkThis(this);
|
2024-03-01 11:15:18 -07:00
|
|
|
id = webidl.converters.long(id);
|
|
|
|
core.cancelTimer(id);
|
2023-02-07 20:22:46 +01:00
|
|
|
}
|
|
|
|
|
2024-03-01 11:15:18 -07:00
|
|
|
/**
|
|
|
|
* Mark a timer as not blocking event loop exit.
|
|
|
|
*/
|
|
|
|
function unrefTimer(id) {
|
|
|
|
core.unrefTimer(id);
|
2023-02-07 20:22:46 +01:00
|
|
|
}
|
|
|
|
|
2024-03-01 11:15:18 -07:00
|
|
|
/**
|
|
|
|
* Mark a timer as blocking event loop exit.
|
|
|
|
*/
|
|
|
|
function refTimer(id) {
|
|
|
|
core.refTimer(id);
|
2023-02-07 20:22:46 +01:00
|
|
|
}
|
|
|
|
|
2023-12-24 06:04:32 -07:00
|
|
|
// Defer to avoid starving the event loop. Not using queueMicrotask()
|
|
|
|
// for that reason: it lets promises make forward progress but can
|
|
|
|
// still starve other parts of the event loop.
|
|
|
|
function defer(go) {
|
2024-03-01 11:15:18 -07:00
|
|
|
PromisePrototypeThen(op_defer(), () => go());
|
2023-12-24 06:04:32 -07:00
|
|
|
}
|
|
|
|
|
2023-02-07 20:22:46 +01:00
|
|
|
export {
|
|
|
|
clearInterval,
|
|
|
|
clearTimeout,
|
2023-12-24 06:04:32 -07:00
|
|
|
defer,
|
2023-02-07 20:22:46 +01:00
|
|
|
opNow,
|
|
|
|
refTimer,
|
2024-03-01 11:15:18 -07:00
|
|
|
setImmediate,
|
2023-02-07 20:22:46 +01:00
|
|
|
setInterval,
|
|
|
|
setTimeout,
|
|
|
|
unrefTimer,
|
|
|
|
};
|