mirror of
https://github.com/photonstorm/phaser
synced 2025-01-12 05:08:54 +00:00
208 lines
5.4 KiB
JavaScript
208 lines
5.4 KiB
JavaScript
/**
|
|
* @author Richard Davey <rich@photonstorm.com>
|
|
* @copyright 2020 Photon Storm Ltd.
|
|
* @license {@link https://opensource.org/licenses/MIT|MIT License}
|
|
*/
|
|
|
|
var Class = require('../utils/Class');
|
|
var NOOP = require('../utils/NOOP');
|
|
|
|
/**
|
|
* @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
|
|
* @memberof Phaser.DOM
|
|
* @constructor
|
|
* @since 3.0.0
|
|
*/
|
|
var RequestAnimationFrame = new Class({
|
|
|
|
initialize:
|
|
|
|
function RequestAnimationFrame ()
|
|
{
|
|
/**
|
|
* True if RequestAnimationFrame is running, otherwise false.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#isRunning
|
|
* @type {boolean}
|
|
* @default false
|
|
* @since 3.0.0
|
|
*/
|
|
this.isRunning = false;
|
|
|
|
/**
|
|
* The callback to be invoked each step.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#callback
|
|
* @type {FrameRequestCallback}
|
|
* @since 3.0.0
|
|
*/
|
|
this.callback = NOOP;
|
|
|
|
/**
|
|
* The most recent timestamp. Either a DOMHighResTimeStamp under RAF or `Date.now` under SetTimeout.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#tick
|
|
* @type {number}
|
|
* @default 0
|
|
* @since 3.0.0
|
|
*/
|
|
this.tick = 0;
|
|
|
|
/**
|
|
* True if the step is using setTimeout instead of RAF.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#isSetTimeOut
|
|
* @type {boolean}
|
|
* @default false
|
|
* @since 3.0.0
|
|
*/
|
|
this.isSetTimeOut = false;
|
|
|
|
/**
|
|
* The setTimeout or RAF callback ID used when canceling them.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#timeOutID
|
|
* @type {?number}
|
|
* @default null
|
|
* @since 3.0.0
|
|
*/
|
|
this.timeOutID = null;
|
|
|
|
/**
|
|
* The previous time the step was called.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#lastTime
|
|
* @type {number}
|
|
* @default 0
|
|
* @since 3.0.0
|
|
*/
|
|
this.lastTime = 0;
|
|
|
|
/**
|
|
* 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;
|
|
|
|
var _this = this;
|
|
|
|
/**
|
|
* The RAF step function.
|
|
* Updates the local tick value, invokes the callback and schedules another call to requestAnimationFrame.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#step
|
|
* @type {FrameRequestCallback}
|
|
* @since 3.0.0
|
|
*/
|
|
this.step = function step ()
|
|
{
|
|
// 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
|
|
_this.lastTime = _this.tick;
|
|
|
|
_this.tick = timestamp;
|
|
|
|
_this.callback(timestamp);
|
|
|
|
_this.timeOutID = window.requestAnimationFrame(step);
|
|
};
|
|
|
|
/**
|
|
* The SetTimeout step function.
|
|
* Updates the local tick value, invokes the callback and schedules another call to setTimeout.
|
|
*
|
|
* @name Phaser.DOM.RequestAnimationFrame#stepTimeout
|
|
* @type {function}
|
|
* @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);
|
|
|
|
_this.lastTime = _this.tick;
|
|
|
|
_this.tick = d;
|
|
|
|
_this.callback(d);
|
|
|
|
_this.timeOutID = window.setTimeout(stepTimeout, delay);
|
|
};
|
|
},
|
|
|
|
/**
|
|
* Starts the requestAnimationFrame or setTimeout process running.
|
|
*
|
|
* @method Phaser.DOM.RequestAnimationFrame#start
|
|
* @since 3.0.0
|
|
*
|
|
* @param {FrameRequestCallback} callback - The callback to invoke each step.
|
|
* @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.
|
|
*/
|
|
start: function (callback, forceSetTimeOut, targetFPS)
|
|
{
|
|
if (this.isRunning)
|
|
{
|
|
return;
|
|
}
|
|
|
|
this.callback = callback;
|
|
|
|
this.isSetTimeOut = forceSetTimeOut;
|
|
|
|
this.target = targetFPS;
|
|
|
|
this.isRunning = true;
|
|
|
|
this.timeOutID = (forceSetTimeOut) ? window.setTimeout(this.stepTimeout, 0) : window.requestAnimationFrame(this.step);
|
|
},
|
|
|
|
/**
|
|
* Stops the requestAnimationFrame or setTimeout from running.
|
|
*
|
|
* @method Phaser.DOM.RequestAnimationFrame#stop
|
|
* @since 3.0.0
|
|
*/
|
|
stop: function ()
|
|
{
|
|
this.isRunning = false;
|
|
|
|
if (this.isSetTimeOut)
|
|
{
|
|
clearTimeout(this.timeOutID);
|
|
}
|
|
else
|
|
{
|
|
window.cancelAnimationFrame(this.timeOutID);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Stops the step from running and clears the callback reference.
|
|
*
|
|
* @method Phaser.DOM.RequestAnimationFrame#destroy
|
|
* @since 3.0.0
|
|
*/
|
|
destroy: function ()
|
|
{
|
|
this.stop();
|
|
|
|
this.callback = NOOP;
|
|
}
|
|
|
|
});
|
|
|
|
module.exports = RequestAnimationFrame;
|