phaser/src/tweens/timeline/Timeline.js

501 lines
12 KiB
JavaScript
Raw Normal View History

var Class = require('../../utils/Class');
var TweenBuilder = require('../builder/TweenBuilder');
var TWEEN_CONST = require('../tween/const');
2017-05-18 03:02:07 +00:00
2017-10-12 14:09:52 +00:00
// Phaser.Tweens.Timeline
var Timeline = new Class({
2017-05-18 03:02:07 +00:00
initialize:
2017-05-18 03:02:07 +00:00
function Timeline (manager)
{
this.manager = manager;
2017-05-18 03:02:07 +00:00
this.isTimeline = true;
// An array of Tween objects, each containing a unique property and target being tweened.
this.data = [];
2017-05-18 03:02:07 +00:00
// data array doesn't usually change, so we can cache the length
this.totalData = 0;
// If true then duration, delay, etc values are all frame totals
this.useFrames = false;
// Scales the time applied to this Tween. A value of 1 runs in real-time. A value of 0.5 runs 50% slower, and so on.
// Value isn't used when calculating total duration of the tween, it's a run-time delta adjustment only.
this.timeScale = 1;
// Loop this tween? Can be -1 for an infinite loop, or an integer.
// When enabled it will play through ALL TweenDatas again (use TweenData.repeat to loop a single TD)
this.loop = 0;
// Time in ms/frames before the tween loops.
this.loopDelay = 0;
// How many loops are left to run?
this.loopCounter = 0;
// Time in ms/frames before the 'onComplete' event fires. This never fires if loop = true (as it never completes)
this.completeDelay = 0;
// Countdown timer (used by loopDelay and completeDelay)
this.countdown = 0;
// The current state of the tween
this.state = TWEEN_CONST.PENDING_ADD;
// The state of the tween when it was paused (used by Resume)
this._pausedState = TWEEN_CONST.PENDING_ADD;
// Does the Tween start off paused? (if so it needs to be started with Tween.play)
this.paused = false;
// Elapsed time in ms/frames of this run through the Tween.
this.elapsed = 0;
// Total elapsed time in ms/frames of the entire Tween, including looping.
this.totalElapsed = 0;
// Time in ms/frames for the whole Tween to play through once, excluding loop amounts and loop delays
this.duration = 0;
// Value between 0 and 1. The amount through the Tween, excluding loops.
this.progress = 0;
// Time in ms/frames for all Tweens to complete (including looping)
this.totalDuration = 0;
// Value between 0 and 1. The amount through the entire Tween, including looping.
this.totalProgress = 0;
2017-05-18 03:02:07 +00:00
this.callbacks = {
onComplete: null,
onLoop: null,
onStart: null,
onUpdate: null,
onYoyo: null
};
2017-05-18 03:02:07 +00:00
this.callbackScope;
},
2017-05-18 03:02:07 +00:00
setTimeScale: function (value)
2017-05-18 03:02:07 +00:00
{
this.timeScale = value;
2017-05-18 03:02:07 +00:00
return this;
},
getTimeScale: function ()
{
return this.timeScale;
},
isPlaying: function ()
{
return (this.state === TWEEN_CONST.ACTIVE);
},
add: function (config)
{
return this.queue(TweenBuilder(this, config));
},
2017-05-18 03:02:07 +00:00
queue: function (tween)
{
if (!this.isPlaying())
{
tween.parent = this;
tween.parentIsTimeline = true;
2017-05-18 03:02:07 +00:00
this.data.push(tween);
this.totalData = this.data.length;
}
return this;
},
hasOffset: function (tween)
{
return (tween.offset !== null);
},
isOffsetAbsolute: function (value)
{
return (typeof(value) === 'number');
},
isOffsetRelative: function (value)
{
var t = typeof(value);
if (t === 'string')
{
var op = value[0];
if (op === '-' || op === '+')
{
return true;
}
}
return false;
},
getRelativeOffset: function (value, base)
{
var op = value[0];
var num = parseFloat(value.substr(2));
var result = base;
switch (op)
{
case '+':
result += num;
break;
case '-':
result -= num;
break;
}
// Cannot ever be < 0
return Math.max(0, result);
},
calcDuration: function ()
{
var prevEnd = 0;
var totalDuration = 0;
var offsetDuration = 0;
for (var i = 0; i < this.totalData; i++)
{
var tween = this.data[i];
tween.init();
if (this.hasOffset(tween))
{
if (this.isOffsetAbsolute(tween.offset))
{
// An actual number, so it defines the start point from the beginning of the timeline
tween.calculatedOffset = tween.offset;
if (tween.offset === 0)
{
offsetDuration = 0;
}
// console.log('Timeline.calcDuration', i, 'absolute', tween.calculatedOffset);
}
else if (this.isOffsetRelative(tween.offset))
{
// A relative offset (i.e. '-=1000', so starts at 'offset' ms relative to the PREVIOUS Tweens ending time)
tween.calculatedOffset = this.getRelativeOffset(tween.offset, prevEnd);
// console.log('Timeline.calcDuration', i, 'relative', tween.calculatedOffset);
}
}
else
{
// Sequential
tween.calculatedOffset = offsetDuration;
// console.log('Timeline.calcDuration', i, 'sequential', tween.calculatedOffset);
}
prevEnd = tween.totalDuration + tween.calculatedOffset;
// console.log('Span', i, tween.calculatedOffset, 'to', prevEnd);
totalDuration += tween.totalDuration;
offsetDuration += tween.totalDuration;
}
// Excludes loop values
this.duration = totalDuration;
this.loopCounter = (this.loop === -1) ? 999999999999 : this.loop;
if (this.loopCounter > 0)
{
this.totalDuration = this.duration + this.completeDelay + ((this.duration + this.loopDelay) * this.loopCounter);
}
else
{
this.totalDuration = this.duration + this.completeDelay;
}
},
init: function ()
{
this.calcDuration();
this.progress = 0;
this.totalProgress = 0;
2017-05-18 03:02:07 +00:00
if (this.paused)
{
this.state = TWEEN_CONST.PAUSED;
return false;
}
else
{
return true;
}
},
resetTweens: function (resetFromLoop)
{
for (var i = 0; i < this.totalData; i++)
{
var tween = this.data[i];
tween.play(resetFromLoop);
}
},
setCallback: function (type, callback, params, scope)
{
if (Timeline.TYPES.indexOf(type) !== -1)
{
this.callbacks[type] = { func: callback, scope: scope, params: params };
}
return this;
},
play: function ()
{
if (this.state === TWEEN_CONST.ACTIVE)
{
return;
}
if (this.paused)
{
this.paused = false;
this.manager.makeActive(this);
return;
}
else
{
this.resetTweens(false);
this.state = TWEEN_CONST.ACTIVE;
}
var onStart = this.callbacks.onStart;
if (onStart)
{
onStart.func.apply(onStart.scope, onStart.params);
}
},
nextState: function ()
{
if (this.loopCounter > 0)
{
// Reset the elapsed time
// TODO: Probably ought to be set to the remainder from elapsed - duration
// as the tweens nearly always over-run by a few ms due to rAf
this.elapsed = 0;
this.progress = 0;
this.loopCounter--;
var onLoop = this.callbacks.onLoop;
if (onLoop)
{
onLoop.func.apply(onLoop.scope, onLoop.params);
}
this.resetTweens(true);
if (this.loopDelay > 0)
{
this.countdown = this.loopDelay;
this.state = TWEEN_CONST.LOOP_DELAY;
}
else
{
this.state = TWEEN_CONST.ACTIVE;
}
}
else if (this.completeDelay > 0)
{
this.countdown = this.completeDelay;
this.state = TWEEN_CONST.COMPLETE_DELAY;
}
else
{
var onComplete = this.callbacks.onComplete;
if (onComplete)
{
onComplete.func.apply(onComplete.scope, onComplete.params);
}
this.state = TWEEN_CONST.PENDING_REMOVE;
}
},
// Returns 'true' if this Timeline has finished and should be removed from the Tween Manager
// Otherwise, returns false
update: function (timestamp, delta)
{
if (this.state === TWEEN_CONST.PAUSED)
{
return;
}
var rawDelta = delta;
if (this.useFrames)
{
delta = 1 * this.manager.timeScale;
}
delta *= this.timeScale;
this.elapsed += delta;
this.progress = Math.min(this.elapsed / this.duration, 1);
this.totalElapsed += delta;
this.totalProgress = Math.min(this.totalElapsed / this.totalDuration, 1);
switch (this.state)
{
case TWEEN_CONST.ACTIVE:
var stillRunning = this.totalData;
for (var i = 0; i < this.totalData; i++)
{
var tween = this.data[i];
if (tween.update(timestamp, rawDelta))
{
stillRunning--;
}
}
var onUpdate = this.callbacks.onUpdate;
if (onUpdate)
{
onUpdate.func.apply(onUpdate.scope, onUpdate.params);
}
// Anything still running? If not, we're done
if (stillRunning === 0)
{
this.nextState();
}
break;
case TWEEN_CONST.LOOP_DELAY:
this.countdown -= delta;
if (this.countdown <= 0)
{
this.state = TWEEN_CONST.ACTIVE;
}
break;
case TWEEN_CONST.COMPLETE_DELAY:
this.countdown -= delta;
if (this.countdown <= 0)
{
var onComplete = this.callbacks.onComplete;
if (onComplete)
{
onComplete.func.apply(onComplete.scope, onComplete.params);
}
this.state = TWEEN_CONST.PENDING_REMOVE;
}
break;
}
return (this.state === TWEEN_CONST.PENDING_REMOVE);
},
// Stops the Tween immediately, whatever stage of progress it is at and flags it for removal by the TweenManager
stop: function ()
{
this.state = TWEEN_CONST.PENDING_REMOVE;
},
pause: function ()
{
if (this.state === TWEEN_CONST.PAUSED)
{
return;
}
this.paused = true;
this._pausedState = this.state;
this.state = TWEEN_CONST.PAUSED;
return this;
},
resume: function ()
{
if (this.state === TWEEN_CONST.PAUSED)
{
this.paused = false;
this.state = this._pausedState;
}
return this;
},
hasTarget: function (target)
{
for (var i = 0; i < this.data.length; i++)
{
if (this.data[i].hasTarget(target))
{
return true;
}
}
return false;
},
destroy: function ()
{
for (var i = 0; i < this.data.length; i++)
{
this.data[i].destroy();
}
}
});
2017-05-18 03:02:07 +00:00
Timeline.TYPES = [ 'onStart', 'onUpdate', 'onLoop', 'onComplete', 'onYoyo' ];
2017-05-18 03:02:07 +00:00
module.exports = Timeline;