phaser/src/dom/RequestAnimationFrame.js

209 lines
5.4 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
2020-01-15 12:07:09 +00:00
* @copyright 2020 Photon Storm Ltd.
2019-05-10 15:15:04 +00:00
* @license {@link https://opensource.org/licenses/MIT|MIT License}
2018-02-12 16:01:20 +00:00
*/
2018-01-26 03:40:49 +00:00
var Class = require('../utils/Class');
var NOOP = require('../utils/NOOP');
2018-02-07 15:27:21 +00:00
/**
* @classdesc
* Abstracts away the use of RAF or setTimeOut for the core game update loop.
* This is invoked automatically by the Phaser.Game instance.
*
* @class RequestAnimationFrame
2018-10-10 09:49:13 +00:00
* @memberof Phaser.DOM
2018-02-07 15:27:21 +00:00
* @constructor
* @since 3.0.0
*/
2018-01-26 03:40:49 +00:00
var RequestAnimationFrame = new Class({
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
initialize:
2018-01-26 03:40:49 +00:00
function RequestAnimationFrame ()
{
/**
* True if RequestAnimationFrame is running, otherwise false.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#isRunning
* @type {boolean}
2018-01-26 03:40:49 +00:00
* @default false
* @since 3.0.0
*/
this.isRunning = false;
2018-01-26 03:40:49 +00:00
/**
* The callback to be invoked each step.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#callback
2018-03-19 21:43:48 +00:00
* @type {FrameRequestCallback}
2018-01-26 03:40:49 +00:00
* @since 3.0.0
*/
this.callback = NOOP;
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
/**
* The most recent timestamp. Either a DOMHighResTimeStamp under RAF or `Date.now` under SetTimeout.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#tick
2018-03-19 15:29:26 +00:00
* @type {number}
2018-01-26 03:40:49 +00:00
* @default 0
* @since 3.0.0
*/
this.tick = 0;
/**
* True if the step is using setTimeout instead of RAF.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#isSetTimeOut
* @type {boolean}
2018-01-26 03:40:49 +00:00
* @default false
* @since 3.0.0
*/
this.isSetTimeOut = false;
/**
* The setTimeout or RAF callback ID used when canceling them.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#timeOutID
* @type {?number}
2018-01-26 03:40:49 +00:00
* @default null
* @since 3.0.0
*/
this.timeOutID = null;
/**
* The previous time the step was called.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#lastTime
* @type {number}
2018-01-26 03:40:49 +00:00
* @default 0
* @since 3.0.0
*/
this.lastTime = 0;
2016-11-25 00:34:37 +00:00
/**
* The target FPS rate in ms.
* Only used when setTimeout is used instead of RAF.
*
* @name Phaser.DOM.RequestAnimationFrame#target
* @type {number}
* @default 0
* @since 3.21.0
*/
this.target = 0;
2018-01-26 03:40:49 +00:00
var _this = this;
2018-01-26 03:40:49 +00:00
/**
* The RAF step function.
* Updates the local tick value, invokes the callback and schedules another call to requestAnimationFrame.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#step
2018-03-19 21:43:48 +00:00
* @type {FrameRequestCallback}
2018-01-26 03:40:49 +00:00
* @since 3.0.0
*/
this.step = function step ()
2018-01-26 03:40:49 +00:00
{
// Because we cannot trust the time passed to this callback from the browser and need it kept in sync with event times
var timestamp = window.performance.now();
// DOMHighResTimeStamp
2018-01-26 03:40:49 +00:00
_this.lastTime = _this.tick;
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
_this.tick = timestamp;
2017-04-28 02:13:32 +00:00
_this.callback(timestamp);
_this.timeOutID = window.requestAnimationFrame(step);
2018-01-26 03:40:49 +00:00
};
2018-01-26 03:40:49 +00:00
/**
* The SetTimeout step function.
* Updates the local tick value, invokes the callback and schedules another call to setTimeout.
*
2018-02-13 00:40:51 +00:00
* @name Phaser.DOM.RequestAnimationFrame#stepTimeout
* @type {function}
2018-01-26 03:40:49 +00:00
* @since 3.0.0
*/
this.stepTimeout = function stepTimeout ()
{
var d = Date.now();
var delay = Math.min(Math.max(_this.target * 2 + _this.tick - d, 0), _this.target);
2017-04-28 02:13:32 +00:00
2018-01-26 03:40:49 +00:00
_this.lastTime = _this.tick;
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
_this.tick = d;
_this.callback(d);
_this.timeOutID = window.setTimeout(stepTimeout, delay);
2018-01-26 03:40:49 +00:00
};
},
2018-01-26 03:40:49 +00:00
/**
* Starts the requestAnimationFrame or setTimeout process running.
*
* @method Phaser.DOM.RequestAnimationFrame#start
* @since 3.0.0
*
2018-03-19 21:43:48 +00:00
* @param {FrameRequestCallback} callback - The callback to invoke each step.
2018-01-26 03:40:49 +00:00
* @param {boolean} forceSetTimeOut - Should it use SetTimeout, even if RAF is available?
* @param {number} targetFPS - The target fps rate (in ms). Only used when setTimeout is used.
2018-01-26 03:40:49 +00:00
*/
start: function (callback, forceSetTimeOut, targetFPS)
2016-11-25 00:34:37 +00:00
{
2018-01-26 03:40:49 +00:00
if (this.isRunning)
{
return;
}
this.callback = callback;
2016-11-25 00:34:37 +00:00
this.isSetTimeOut = forceSetTimeOut;
2016-11-25 00:34:37 +00:00
this.target = targetFPS;
this.isRunning = true;
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
this.timeOutID = (forceSetTimeOut) ? window.setTimeout(this.stepTimeout, 0) : window.requestAnimationFrame(this.step);
},
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
/**
* Stops the requestAnimationFrame or setTimeout from running.
*
* @method Phaser.DOM.RequestAnimationFrame#stop
* @since 3.0.0
*/
stop: function ()
2016-11-25 00:34:37 +00:00
{
this.isRunning = false;
if (this.isSetTimeOut)
{
clearTimeout(this.timeOutID);
}
else
{
window.cancelAnimationFrame(this.timeOutID);
}
},
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
/**
* Stops the step from running and clears the callback reference.
*
* @method Phaser.DOM.RequestAnimationFrame#destroy
* @since 3.0.0
*/
destroy: function ()
2016-11-25 00:34:37 +00:00
{
this.stop();
this.callback = NOOP;
}
2016-11-25 00:34:37 +00:00
2018-01-26 03:40:49 +00:00
});
2016-11-25 00:34:37 +00:00
module.exports = RequestAnimationFrame;