phaser/src/tweens/TweenManager.js

736 lines
20 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
2019-01-15 16:20:22 +00:00
* @copyright 2019 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
*/
var ArrayRemove = require('../utils/array/Remove');
2018-01-16 19:49:13 +00:00
var Class = require('../utils/Class');
2018-02-10 14:56:08 +00:00
var NumberTweenBuilder = require('./builders/NumberTweenBuilder');
var PluginCache = require('../plugins/PluginCache');
2019-01-18 13:41:43 +00:00
var SceneEvents = require('../scene/events');
2019-06-27 16:14:39 +00:00
var StaggerBuilder = require('./builders/StaggerBuilder');
2018-02-10 14:56:08 +00:00
var TimelineBuilder = require('./builders/TimelineBuilder');
2018-01-16 19:49:13 +00:00
var TWEEN_CONST = require('./tween/const');
2018-02-10 14:56:08 +00:00
var TweenBuilder = require('./builders/TweenBuilder');
2018-02-10 17:40:40 +00:00
/**
* @classdesc
2018-10-22 11:12:31 +00:00
* The Tween Manager is a default Scene Plugin which controls and updates Tweens and Timelines.
2018-02-10 17:40:40 +00:00
*
* @class TweenManager
2018-10-10 09:49:13 +00:00
* @memberof Phaser.Tweens
2018-02-10 17:40:40 +00:00
* @constructor
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {Phaser.Scene} scene - The Scene which owns this Tween Manager.
2018-02-10 17:40:40 +00:00
*/
var TweenManager = new Class({
initialize:
function TweenManager (scene)
{
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* The Scene which owns this Tween Manager.
2018-02-10 17:40:40 +00:00
*
* @name Phaser.Tweens.TweenManager#scene
* @type {Phaser.Scene}
* @since 3.0.0
*/
this.scene = scene;
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* The Systems object of the Scene which owns this Tween Manager.
2018-02-10 17:40:40 +00:00
*
* @name Phaser.Tweens.TweenManager#systems
2018-02-10 17:51:02 +00:00
* @type {Phaser.Scenes.Systems}
2018-02-10 17:40:40 +00:00
* @since 3.0.0
*/
2018-01-16 19:49:13 +00:00
this.systems = scene.sys;
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* The time scale of the Tween Manager.
*
* This value scales the time delta between two frames, thus influencing the speed of time for all Tweens owned by this Tween Manager.
2018-02-10 17:40:40 +00:00
*
* @name Phaser.Tweens.TweenManager#timeScale
* @type {number}
* @default 1
* @since 3.0.0
*/
this.timeScale = 1;
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* An array of Tweens and Timelines which will be added to the Tween Manager at the start of the frame.
2018-02-10 17:40:40 +00:00
*
* @name Phaser.Tweens.TweenManager#_add
* @type {array}
* @private
* @since 3.0.0
*/
this._add = [];
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* An array of Tweens and Timelines pending to be later added to the Tween Manager.
2018-02-10 17:40:40 +00:00
*
* @name Phaser.Tweens.TweenManager#_pending
* @type {array}
* @private
* @since 3.0.0
*/
this._pending = [];
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* An array of Tweens and Timelines which are still incomplete and are actively processed by the Tween Manager.
2018-02-10 17:40:40 +00:00
*
* @name Phaser.Tweens.TweenManager#_active
* @type {array}
* @private
* @since 3.0.0
*/
this._active = [];
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* An array of Tweens and Timelines which will be removed from the Tween Manager at the start of the frame.
2018-02-10 17:40:40 +00:00
*
* @name Phaser.Tweens.TweenManager#_destroy
* @type {array}
* @private
* @since 3.0.0
*/
this._destroy = [];
2018-02-10 17:51:02 +00:00
/**
2018-10-22 11:12:31 +00:00
* The number of Tweens and Timelines which need to be processed by the Tween Manager at the start of the frame.
2018-02-10 17:51:02 +00:00
*
* @name Phaser.Tweens.TweenManager#_toProcess
* @type {integer}
* @private
* @default 0
* @since 3.0.0
*/
this._toProcess = 0;
2019-01-18 13:41:43 +00:00
scene.sys.events.once(SceneEvents.BOOT, this.boot, this);
scene.sys.events.on(SceneEvents.START, this.start, this);
},
/**
* This method is called automatically, only once, when the Scene is first created.
* Do not invoke it directly.
*
* @method Phaser.Tweens.TweenManager#boot
* @private
* @since 3.5.1
*/
boot: function ()
{
2019-01-18 13:41:43 +00:00
this.systems.events.once(SceneEvents.DESTROY, this.destroy, this);
},
2018-02-10 17:40:40 +00:00
/**
* This method is called automatically by the Scene when it is starting up.
* It is responsible for creating local systems, properties and listening for Scene events.
* Do not invoke it directly.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#start
* @private
* @since 3.5.0
2018-02-10 17:40:40 +00:00
*/
start: function ()
{
var eventEmitter = this.systems.events;
2019-01-18 13:41:43 +00:00
eventEmitter.on(SceneEvents.PRE_UPDATE, this.preUpdate, this);
eventEmitter.on(SceneEvents.UPDATE, this.update, this);
eventEmitter.once(SceneEvents.SHUTDOWN, this.shutdown, this);
2018-01-16 19:49:13 +00:00
this.timeScale = 1;
},
2018-02-10 17:40:40 +00:00
/**
2018-02-10 17:51:02 +00:00
* Create a Tween Timeline and return it, but do NOT add it to the active or pending Tween lists.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#createTimeline
* @since 3.0.0
*
* @param {Phaser.Types.Tweens.TimelineBuilderConfig} config - The configuration object for the Timeline and its Tweens.
2018-02-10 17:40:40 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.Timeline} The created Timeline object.
2018-02-10 17:40:40 +00:00
*/
createTimeline: function (config)
{
return TimelineBuilder(this, config);
},
2018-02-10 17:40:40 +00:00
/**
2018-02-10 17:51:02 +00:00
* Create a Tween Timeline and add it to the active Tween list/
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#timeline
* @since 3.0.0
*
* @param {Phaser.Types.Tweens.TimelineBuilderConfig} config - The configuration object for the Timeline and its Tweens.
2018-02-10 17:40:40 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.Timeline} The created Timeline object.
2018-02-10 17:40:40 +00:00
*/
timeline: function (config)
{
var timeline = TimelineBuilder(this, config);
if (!timeline.paused)
{
this._add.push(timeline);
this._toProcess++;
}
return timeline;
},
2018-02-10 17:40:40 +00:00
/**
2018-02-10 17:51:02 +00:00
* Create a Tween and return it, but do NOT add it to the active or pending Tween lists.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#create
* @since 3.0.0
*
2019-05-29 17:42:30 +00:00
* @param {Phaser.Types.Tweens.TweenBuilderConfig|object} config - The configuration object for the Tween.
2018-02-10 17:40:40 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.Tween} The created Tween object.
2018-02-10 17:40:40 +00:00
*/
create: function (config)
{
return TweenBuilder(this, config);
},
2018-02-10 17:40:40 +00:00
/**
2018-02-10 17:51:02 +00:00
* Create a Tween and add it to the active Tween list.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#add
* @since 3.0.0
*
2019-05-29 17:42:30 +00:00
* @param {Phaser.Types.Tweens.TweenBuilderConfig|object} config - The configuration object for the Tween.
2018-02-10 17:40:40 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.Tween} The created Tween.
2018-02-10 17:40:40 +00:00
*/
add: function (config)
{
var tween = TweenBuilder(this, config);
this._add.push(tween);
this._toProcess++;
return tween;
},
2018-02-10 17:40:40 +00:00
/**
2018-02-10 17:51:02 +00:00
* Add an existing tween into the active Tween list.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#existing
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {Phaser.Tweens.Tween} tween - The Tween to add.
2018-02-10 17:40:40 +00:00
*
2018-02-10 17:51:02 +00:00
* @return {Phaser.Tweens.TweenManager} This Tween Manager object.
2018-02-10 17:40:40 +00:00
*/
existing: function (tween)
{
this._add.push(tween);
this._toProcess++;
return this;
},
2018-02-10 17:40:40 +00:00
/**
2019-05-28 20:13:39 +00:00
* Create a Number Tween and add it to the active Tween list.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#addCounter
* @since 3.0.0
*
2019-05-28 20:13:39 +00:00
* @param {Phaser.Types.Tweens.NumberTweenBuilderConfig} config - The configuration object for the Number Tween.
2018-02-10 17:40:40 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.Tween} The created Number Tween.
2018-02-10 17:40:40 +00:00
*/
addCounter: function (config)
{
var tween = NumberTweenBuilder(this, config);
this._add.push(tween);
this._toProcess++;
return tween;
},
2019-06-27 16:14:39 +00:00
/**
* Creates a stagger function and returns it.
*
* @method Phaser.Tweens.TweenManager#stagger
* @since 3.19.0
*
* @param {Phaser.Types.Tweens.StaggerBuilderConfig} config - The configuration object for the Stagger function.
*
* @return {function} The stagger function.
*/
stagger: function (config)
{
return StaggerBuilder(config);
},
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* Updates the Tween Manager's internal lists at the start of the frame.
*
* This method will return immediately if no changes have been indicated.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#preUpdate
* @since 3.0.0
*/
2018-01-16 19:49:13 +00:00
preUpdate: function ()
{
if (this._toProcess === 0)
{
// Quick bail
return;
}
var list = this._destroy;
var active = this._active;
var pending = this._pending;
var i;
var tween;
// Clear the 'destroy' list
for (i = 0; i < list.length; i++)
{
tween = list[i];
// Remove from the 'active' array
var idx = active.indexOf(tween);
if (idx === -1)
{
// Not in the active array, is it in pending instead?
idx = pending.indexOf(tween);
if (idx > -1)
{
tween.state = TWEEN_CONST.REMOVED;
pending.splice(idx, 1);
}
}
else
{
tween.state = TWEEN_CONST.REMOVED;
active.splice(idx, 1);
}
}
list.length = 0;
// Process the addition list
// This stops callbacks and out of sync events from populating the active array mid-way during the update
list = this._add;
for (i = 0; i < list.length; i++)
{
tween = list[i];
if (tween.state === TWEEN_CONST.PENDING_ADD)
{
// Return true if the Tween should be started right away, otherwise false
if (tween.init())
{
tween.play();
this._active.push(tween);
}
else
{
this._pending.push(tween);
}
}
}
list.length = 0;
this._toProcess = 0;
},
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* Updates all Tweens and Timelines of the Tween Manager.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#update
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {number} timestamp - The current time in milliseconds.
2018-09-13 07:09:44 +00:00
* @param {number} delta - The delta time in ms since the last frame. This is a smoothed and capped value based on the FPS rate.
2018-02-10 17:40:40 +00:00
*/
update: function (timestamp, delta)
{
// Process active tweens
var list = this._active;
var tween;
// Scale the delta
delta *= this.timeScale;
for (var i = 0; i < list.length; i++)
{
tween = list[i];
// If Tween.update returns 'true' then it means it has completed,
// so move it to the destroy list
if (tween.update(timestamp, delta))
{
this._destroy.push(tween);
this._toProcess++;
}
}
},
/**
* Removes the given tween from the Tween Manager, regardless of its state (pending or active).
*
* @method Phaser.Tweens.TweenManager#remove
* @since 3.17.0
*
* @param {Phaser.Tweens.Tween} tween - The Tween to be removed.
*
* @return {Phaser.Tweens.TweenManager} This Tween Manager object.
*/
remove: function (tween)
{
ArrayRemove(this._add, tween);
ArrayRemove(this._pending, tween);
ArrayRemove(this._active, tween);
ArrayRemove(this._destroy, tween);
tween.state = TWEEN_CONST.REMOVED;
return this;
},
2018-02-10 17:40:40 +00:00
/**
2018-10-22 11:12:31 +00:00
* Checks if a Tween or Timeline is active and adds it to the Tween Manager at the start of the frame if it isn't.
2018-02-10 17:40:40 +00:00
*
* @method Phaser.Tweens.TweenManager#makeActive
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {Phaser.Tweens.Tween} tween - The Tween to check.
2018-02-10 17:40:40 +00:00
*
2018-02-10 17:51:02 +00:00
* @return {Phaser.Tweens.TweenManager} This Tween Manager object.
2018-02-10 17:40:40 +00:00
*/
makeActive: function (tween)
{
if (this._add.indexOf(tween) !== -1 || this._active.indexOf(tween) !== -1)
{
return this;
}
var idx = this._pending.indexOf(tween);
if (idx !== -1)
{
this._pending.splice(idx, 1);
}
this._add.push(tween);
tween.state = TWEEN_CONST.PENDING_ADD;
this._toProcess++;
return this;
},
2018-01-16 16:37:34 +00:00
/**
2018-02-10 17:51:02 +00:00
* Passes all Tweens to the given callback.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#each
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {function} callback - The function to call.
* @param {object} [scope] - The scope (`this` object) to call the function with.
* @param {...*} [args] - The arguments to pass into the function. Its first argument will always be the Tween currently being iterated.
2018-01-16 16:37:34 +00:00
*/
2018-02-10 17:51:02 +00:00
each: function (callback, scope)
2018-01-16 16:37:34 +00:00
{
var args = [ null ];
for (var i = 1; i < arguments.length; i++)
{
args.push(arguments[i]);
}
for (var texture in this.list)
{
args[0] = this.list[texture];
2018-02-10 17:51:02 +00:00
callback.apply(scope, args);
2018-01-16 16:37:34 +00:00
}
},
/**
2018-10-22 11:12:31 +00:00
* Returns an array of all active Tweens and Timelines in the Tween Manager.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#getAllTweens
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.Tween[]} A new array containing references to all active Tweens and Timelines.
2018-01-16 16:37:34 +00:00
*/
getAllTweens: function ()
{
var list = this._active;
var output = [];
for (var i = 0; i < list.length; i++)
{
output.push(list[i]);
}
return output;
},
/**
2018-10-22 11:12:31 +00:00
* Returns the scale of the time delta for all Tweens and Timelines owned by this Tween Manager.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#getGlobalTimeScale
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @return {number} The scale of the time delta, usually 1.
2018-01-16 16:37:34 +00:00
*/
getGlobalTimeScale: function ()
{
return this.timeScale;
},
/**
2018-10-22 11:12:31 +00:00
* Returns an array of all Tweens or Timelines in the Tween Manager which affect the given target or array of targets.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#getTweensOf
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {(object|array)} target - The target to look for. Provide an array to look for multiple targets.
2018-01-16 16:37:34 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.Tween[]} A new array containing all Tweens and Timelines which affect the given target(s).
2018-01-16 16:37:34 +00:00
*/
getTweensOf: function (target)
{
var list = this._active;
var tween;
var output = [];
var i;
if (Array.isArray(target))
{
for (i = 0; i < list.length; i++)
{
tween = list[i];
for (var t = 0; t < target.length; t++)
2018-01-16 16:37:34 +00:00
{
if (tween.hasTarget(target[t]))
{
output.push(tween);
}
}
}
}
else
{
for (i = 0; i < list.length; i++)
{
tween = list[i];
if (tween.hasTarget(target))
{
output.push(tween);
}
}
}
return output;
},
/**
2018-10-22 11:12:31 +00:00
* Checks if the given object is being affected by a playing Tween.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#isTweening
* @since 3.0.0
*
2018-10-19 11:32:43 +00:00
* @param {object} target - target Phaser.Tweens.Tween object
2018-01-16 16:37:34 +00:00
*
2018-10-19 11:32:43 +00:00
* @return {boolean} returns if target tween object is active or not
2018-01-16 16:37:34 +00:00
*/
isTweening: function (target)
{
var list = this._active;
var tween;
for (var i = 0; i < list.length; i++)
{
tween = list[i];
if (tween.hasTarget(target) && tween.isPlaying())
{
return true;
}
}
return false;
},
/**
2018-10-22 11:12:31 +00:00
* Stops all Tweens in this Tween Manager. They will be removed at the start of the frame.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#killAll
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.TweenManager} This Tween Manager.
2018-01-16 16:37:34 +00:00
*/
killAll: function ()
{
var tweens = this.getAllTweens();
for (var i = 0; i < tweens.length; i++)
{
tweens[i].stop();
}
return this;
},
/**
2018-10-22 11:12:31 +00:00
* Stops all Tweens which affect the given target or array of targets. The Tweens will be removed from the Tween Manager at the start of the frame.
*
* @see {@link #getTweensOf}
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#killTweensOf
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {(object|array)} target - The target to look for. Provide an array to look for multiple targets.
2018-01-16 16:37:34 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.TweenManager} This Tween Manager.
2018-01-16 16:37:34 +00:00
*/
killTweensOf: function (target)
{
var tweens = this.getTweensOf(target);
for (var i = 0; i < tweens.length; i++)
{
tweens[i].stop();
}
return this;
},
/**
2018-10-22 11:12:31 +00:00
* Pauses all Tweens in this Tween Manager.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#pauseAll
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.TweenManager} This Tween Manager.
2018-01-16 16:37:34 +00:00
*/
pauseAll: function ()
{
var list = this._active;
for (var i = 0; i < list.length; i++)
{
list[i].pause();
}
return this;
},
/**
2018-10-22 11:12:31 +00:00
* Resumes all Tweens in this Tween Manager.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#resumeAll
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.TweenManager} This Tween Manager.
2018-01-16 16:37:34 +00:00
*/
resumeAll: function ()
{
var list = this._active;
for (var i = 0; i < list.length; i++)
{
list[i].resume();
}
return this;
},
/**
2018-10-22 11:12:31 +00:00
* Sets a new scale of the time delta for this Tween Manager.
*
* The time delta is the time elapsed between two consecutive frames and influences the speed of time for this Tween Manager and all Tweens it owns. Values higher than 1 increase the speed of time, while values smaller than 1 decrease it. A value of 0 freezes time and is effectively equivalent to pausing all Tweens.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#setGlobalTimeScale
* @since 3.0.0
*
2018-10-22 11:12:31 +00:00
* @param {number} value - The new scale of the time delta, where 1 is the normal speed.
2018-01-16 16:37:34 +00:00
*
2018-10-22 11:12:31 +00:00
* @return {Phaser.Tweens.TweenManager} This Tween Manager.
2018-01-16 16:37:34 +00:00
*/
setGlobalTimeScale: function (value)
{
this.timeScale = value;
return this;
},
/**
* The Scene that owns this plugin is shutting down.
* We need to kill and reset all internal properties as well as stop listening to Scene events.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#shutdown
* @since 3.0.0
*/
shutdown: function ()
{
this.killAll();
this._add = [];
this._pending = [];
this._active = [];
this._destroy = [];
this._toProcess = 0;
var eventEmitter = this.systems.events;
2019-01-18 13:41:43 +00:00
eventEmitter.off(SceneEvents.PRE_UPDATE, this.preUpdate, this);
eventEmitter.off(SceneEvents.UPDATE, this.update, this);
eventEmitter.off(SceneEvents.SHUTDOWN, this.shutdown, this);
2018-01-16 16:37:34 +00:00
},
/**
* The Scene that owns this plugin is being destroyed.
* We need to shutdown and then kill off all external references.
2018-01-16 16:37:34 +00:00
*
* @method Phaser.Tweens.TweenManager#destroy
* @since 3.0.0
*/
destroy: function ()
{
this.shutdown();
2019-01-18 13:41:43 +00:00
this.scene.sys.events.off(SceneEvents.START, this.start, this);
this.scene = null;
this.systems = null;
2018-01-16 16:37:34 +00:00
}
});
PluginCache.register('TweenManager', TweenManager, 'tweens');
2018-01-16 19:49:13 +00:00
module.exports = TweenManager;