Skip to content

Commit 29052b6

Browse files
committed
lib: add requestAnimationFrame global
Adds the browser API `requestAnimationFrame()` as an isomorphic wrapper around `setImmediate()`. Marked Legacy with a clear indication that code targeting Node.js should use `setImmediate()` instead. Signed-off-by: James M Snell <jasnell@gmail.com>
1 parent 30fe4ed commit 29052b6

File tree

5 files changed

+85
-0
lines changed

5 files changed

+85
-0
lines changed

.eslintrc.js

+2
Original file line numberDiff line numberDiff line change
@@ -331,5 +331,7 @@ module.exports = {
331331
globalThis: 'readable',
332332
btoa: 'readable',
333333
atob: 'readable',
334+
requestAnimationFrame: 'readable',
335+
cancelAnimationFrame: 'readable',
334336
},
335337
};

doc/api/globals.md

+50
Original file line numberDiff line numberDiff line change
@@ -164,6 +164,29 @@ added: REPLACEME
164164
165165
Global alias for [`buffer.btoa()`][].
166166

167+
## `cancelAnimationFrame(immediateObject)`
168+
<!-- YAML
169+
added: REPLACEME
170+
-->
171+
172+
<!--type=global-->
173+
174+
> Stability: 3 - Legacy: Use clearImmediate() instead.
175+
176+
* `immediateObject` {Object} The return value of `requestAnimationFrame()`
177+
identifying the request being canceled.
178+
179+
```js
180+
const req = requestAnimationFrame((ts) => {
181+
console.log(`called at ${ts}`);
182+
});
183+
184+
cancelAnimationFrame(req);
185+
```
186+
187+
In Node.js, `cancelAnimationFrame()` is an alias for `clearImmediate()`.
188+
Code specifically targeting Node.js should use `clearImmediate()` instead.
189+
167190
## `clearImmediate(immediateObject)`
168191
<!-- YAML
169192
added: v0.9.1
@@ -329,6 +352,33 @@ DataHandler.prototype.load = async function load(key) {
329352
};
330353
```
331354

355+
## `requestAnimationFrame(callback)`
356+
<!-- YAML
357+
added: REPLACEME
358+
-->
359+
360+
<!--type=global-->
361+
362+
> Stability: 3 - Legacy: Use the setImmediate() instead.
363+
364+
* `callback` {Function}
365+
* Returns {Object}
366+
367+
Registers a function with `setImmediate()` that is invoked with a single
368+
argument set to the value of `perf_hooks.performance.now()`.
369+
370+
```js
371+
requestAnimationFrame((ts) => {
372+
console.log(`called at ${ts}`);
373+
});
374+
```
375+
376+
Node.js does not implement the same timing and event loop frame semantics
377+
as Web Browsers and does not include any notion of rendering or "animation".
378+
The `requestAnimationFrame()` method should be considered a close approximation
379+
to enable cross-environment portable JavaScript code to be written. Code
380+
specifically targeting Node.js should use `setImmediate()` instead.
381+
332382
## `require()`
333383

334384
This variable may appear to be global but is not. See [`require()`][].

lib/internal/bootstrap/node.js

+14
Original file line numberDiff line numberDiff line change
@@ -239,6 +239,20 @@ if (!config.noBrowserGlobals) {
239239
// Non-standard extensions:
240240
defineOperation(global, 'clearImmediate', timers.clearImmediate);
241241
defineOperation(global, 'setImmediate', timers.setImmediate);
242+
243+
function requestAnimationFrame(callback) {
244+
const { performance } = require('perf_hooks');
245+
const {
246+
codes: {
247+
ERR_INVALID_ARG_TYPE,
248+
},
249+
} = require('internal/errors');
250+
if (typeof callback !== 'function')
251+
throw new ERR_INVALID_ARG_TYPE('callback', 'Function', callback);
252+
return timers.setImmediate(() => callback(performance.now()));
253+
}
254+
defineOperation(global, 'requestAnimationFrame', requestAnimationFrame);
255+
defineOperation(global, 'cancelAnimationFrame', timers.clearImmediate);
242256
}
243257

244258
// Set the per-Environment callback that will be called

test/common/index.js

+3
Original file line numberDiff line numberDiff line change
@@ -261,13 +261,16 @@ function platformTimeout(ms) {
261261
return ms; // ARMv8+
262262
}
263263

264+
const requestAnimationFrame = globalThis.requestAnimationFrame;
265+
264266
let knownGlobals = [
265267
atob,
266268
btoa,
267269
clearImmediate,
268270
clearInterval,
269271
clearTimeout,
270272
global,
273+
requestAnimationFrame,
271274
setImmediate,
272275
setInterval,
273276
setTimeout,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
const { strictEqual } = require('assert');
5+
6+
let called = false;
7+
8+
requestAnimationFrame(common.mustCall((ts) => {
9+
called = true;
10+
strictEqual(typeof ts, 'number');
11+
}));
12+
13+
strictEqual(called, false);
14+
15+
const req = requestAnimationFrame(common.mustNotCall());
16+
cancelAnimationFrame(req);

0 commit comments

Comments
 (0)