mirror of
https://github.com/hiloteam/Hilo.git
synced 2025-12-08 20:35:59 +00:00
431 lines
14 KiB
JavaScript
431 lines
14 KiB
JavaScript
/**
|
|
* Hilo
|
|
* Copyright 2015 alibaba.com
|
|
* Licensed under the MIT License
|
|
*/
|
|
|
|
/**
|
|
* <iframe src='../../../examples/Tween.html?noHeader' width = '550' height = '130' scrolling='no'></iframe>
|
|
* <br/>
|
|
* Demo:
|
|
* <pre>
|
|
* ticker.addTick(Hilo.Tween);//Tween works after being added to ticker
|
|
*
|
|
* var view = new View({x:5, y:10});
|
|
* Hilo.Tween.to(view, {
|
|
* x:100,
|
|
* y:20,
|
|
* alpha:0
|
|
* }, {
|
|
* duration:1000,
|
|
* delay:500,
|
|
* ease:Hilo.Ease.Quad.EaseIn,
|
|
* onComplete:function(){
|
|
* console.log('complete');
|
|
* }
|
|
* });
|
|
* </pre>
|
|
* @class Tween class makes tweening (easing, slow motion).
|
|
* @param {Object} target Tween target object.
|
|
* @param {Object} fromProps Beginning properties of target tweening object.
|
|
* @param {Object} toProps Ending properties of target tweening object.
|
|
* @param {Object} params Tweening parameters, include all writable Tween class properties.
|
|
* @module hilo/tween/Tween
|
|
* @requires hilo/core/Class
|
|
* @property {Object} target Tween target object, readonly!
|
|
* @property {Int} duration Tweening duration, measure in ms.
|
|
* @property {Int} delay Tweenning delay time, measure in ms.
|
|
* @property {Boolean} paused Is tweening paused, default value is false.
|
|
* @property {Boolean} loop Does tweening loop, default value is false.
|
|
* @property {Boolean} reverse Does tweening reverse, default value is false.
|
|
* @property {Int} repeat Repeat times of tweening, default value is 0.
|
|
* @property {Int} repeatDelay Delay time of repeating tweening, measure in ms.
|
|
* @property {Function} ease Tweening transform function, default value is null.
|
|
* @property {Int} time Time that tweening taken, measure in ms, readonly!
|
|
* @property {Function} onStart Function invoked on the beginning of tweening. Require 1 parameter: tween. default value is null.
|
|
* @property {Function} onUpdate Function invoked on tweening update. Require 2 parameters: ratio, tween. default value is null.
|
|
* @property {Function} onComplete Function invoked on the end of tweening. Require 1 parameter: tween. default value is null.
|
|
*/
|
|
var Tween = (function(){
|
|
|
|
function now(){
|
|
return +new Date();
|
|
}
|
|
|
|
return Class.create(/** @lends Tween.prototype */{
|
|
constructor: function(target, fromProps, toProps, params){
|
|
var me = this;
|
|
|
|
me.target = target;
|
|
me._startTime = 0;
|
|
me._seekTime = 0;
|
|
me._pausedTime = 0;
|
|
me._pausedStartTime = 0;
|
|
me._reverseFlag = 1;
|
|
me._repeatCount = 0;
|
|
|
|
//no fromProps if pass 3 arguments
|
|
if(arguments.length == 3){
|
|
params = toProps;
|
|
toProps = fromProps;
|
|
fromProps = null;
|
|
}
|
|
|
|
for(var p in params) me[p] = params[p];
|
|
me._fromProps = fromProps;
|
|
me._toProps = toProps;
|
|
|
|
//for old version compatiblity
|
|
if(!params.duration && params.time){
|
|
me.duration = params.time || 0;
|
|
me.time = 0;
|
|
}
|
|
},
|
|
|
|
target: null,
|
|
duration: 1000,
|
|
delay: 0,
|
|
paused: false,
|
|
loop: false,
|
|
reverse: false,
|
|
repeat: 0,
|
|
repeatDelay: 0,
|
|
ease: null,
|
|
time: 0, //ready only
|
|
|
|
isStart:false,
|
|
isComplete:false,
|
|
onStart: null,
|
|
onUpdate: null,
|
|
onComplete: null,
|
|
|
|
/**
|
|
* Set beginning properties and ending properties of tweening object.
|
|
* @param {Object} fromProps Beginning properties of target tweening object.
|
|
* @param {Object} toProps Ending properties of target tweening object.
|
|
* @returns {Tween} Current Tween, for chain calls.
|
|
*/
|
|
setProps: function(fromProps, toProps){
|
|
var me = this, target = me.target,
|
|
propNames = fromProps || toProps,
|
|
from = me._fromProps = {}, to = me._toProps = {};
|
|
|
|
fromProps = fromProps || target;
|
|
toProps = toProps || target;
|
|
|
|
for(var p in propNames){
|
|
to[p] = toProps[p] || 0;
|
|
target[p] = from[p] = fromProps[p] || 0;
|
|
}
|
|
return me;
|
|
},
|
|
|
|
/**
|
|
* Starting the tweening.
|
|
* @returns {Tween} Current Tween, for chain calls.
|
|
*/
|
|
start: function(){
|
|
var me = this;
|
|
me._startTime = now() + me.delay;
|
|
me._seekTime = 0;
|
|
me._pausedTime = 0;
|
|
me._reverseFlag = 1;
|
|
me._repeatCount = 0;
|
|
me.paused = false;
|
|
me.isStart = false;
|
|
me.isComplete = false;
|
|
Tween.add(me);
|
|
return me;
|
|
},
|
|
|
|
/**
|
|
* Stop the tweening.
|
|
* @returns {Tween} Current Tween, for chain calls.
|
|
*/
|
|
stop: function(){
|
|
Tween.remove(this);
|
|
return this;
|
|
},
|
|
|
|
/**
|
|
* Pause the tweening.
|
|
* @returns {Tween} Current Tween, for chain calls.
|
|
*/
|
|
pause: function(){
|
|
var me = this;
|
|
me.paused = true;
|
|
me._pausedStartTime = now();
|
|
return me;
|
|
},
|
|
|
|
/**
|
|
* Continue to play the tweening.
|
|
* @returns {Tween} Current Tween, for chain calls.
|
|
*/
|
|
resume: function(){
|
|
var me = this;
|
|
me.paused = false;
|
|
if(me._pausedStartTime) me._pausedTime += now() - me._pausedStartTime;
|
|
me._pausedStartTime = 0;
|
|
return me;
|
|
},
|
|
|
|
/**
|
|
* Tween jumps to some point.
|
|
* @param {Number} time The time to jump to, range from 0 to duration.
|
|
* @param {Boolean} pause Is paused.
|
|
* @returns {Tween} Current Tween, for chain calls.
|
|
*/
|
|
seek: function(time, pause){
|
|
var me = this, current = now();
|
|
me._startTime = current;
|
|
me._seekTime = time;
|
|
me._pausedTime = 0;
|
|
if(pause !== undefined) me.paused = pause;
|
|
me._update(current, true);
|
|
Tween.add(me);
|
|
return me;
|
|
},
|
|
|
|
/**
|
|
* Link next Tween. The beginning time of next Tween depends on the delay value. If delay is a string that begins with '+' or '-', next Tween will begin at (delay) ms after or before the current tween is ended. If delay is out of previous situation, next Tween will begin at (delay) ms after the beginning point of current Tween.
|
|
* @param {Tween} tween Tween to link.
|
|
* @returns {Tween} next Tween, for chain calls.
|
|
*/
|
|
link: function(tween){
|
|
var me = this, delay = tween.delay, startTime = me._startTime;
|
|
|
|
var plus, minus;
|
|
if(typeof delay === 'string'){
|
|
plus = delay.indexOf('+') == 0;
|
|
minus = delay.indexOf('-') == 0;
|
|
delay = plus || minus ? Number(delay.substr(1)) * (plus ? 1 : -1) : Number(delay);
|
|
}
|
|
tween.delay = delay;
|
|
tween._startTime = plus || minus ? startTime + me.duration + delay : startTime + delay;
|
|
|
|
me._next = tween;
|
|
Tween.remove(tween);
|
|
return tween;
|
|
},
|
|
|
|
/**
|
|
* Private render method inside Tween class.
|
|
* @private
|
|
*/
|
|
_render: function(ratio){
|
|
var me = this, target = me.target, fromProps = me._fromProps, p;
|
|
for(p in fromProps) target[p] = fromProps[p] + (me._toProps[p] - fromProps[p]) * ratio;
|
|
},
|
|
|
|
/**
|
|
* Private update method inside Tween class.
|
|
* @private
|
|
*/
|
|
_update: function(time, forceUpdate){
|
|
var me = this;
|
|
if(me.paused && !forceUpdate) return;
|
|
if(me.isComplete) return true;
|
|
|
|
//elapsed time
|
|
var elapsed = time - me._startTime - me._pausedTime + me._seekTime;
|
|
if(elapsed < 0) return;
|
|
|
|
//elapsed ratio
|
|
var ratio = elapsed / me.duration, complete = false, callback;
|
|
ratio = ratio <= 0 ? 0 : ratio >= 1 ? 1 : ratio;
|
|
var easeRatio = me.ease ? me.ease(ratio) : ratio;
|
|
|
|
if(me.reverse && me.isStart){
|
|
//backward
|
|
if(me._reverseFlag < 0) {
|
|
ratio = 1 - ratio;
|
|
easeRatio = 1 - easeRatio;
|
|
}
|
|
//forward
|
|
if(ratio < 1e-7){
|
|
//repeat complete or not loop
|
|
if((me.repeat > 0 && me._repeatCount++ >= me.repeat) || (me.repeat == 0 && !me.loop)){
|
|
complete = true;
|
|
}else{
|
|
me._startTime = now();
|
|
me._pausedTime = 0;
|
|
me._reverseFlag *= -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
//start callback
|
|
if(!me.isStart) {
|
|
me.setProps(me._fromProps, me._toProps);
|
|
me.isStart = true;
|
|
if(me.onStart){
|
|
me.onStart.call(me, me);
|
|
}
|
|
}
|
|
me.time = elapsed;
|
|
|
|
//render & update callback
|
|
me._render(easeRatio);
|
|
(callback = me.onUpdate) && callback.call(me, ratio, me);
|
|
|
|
//check if complete
|
|
if(ratio >= 1){
|
|
if(me.reverse){
|
|
me._startTime = now();
|
|
me._pausedTime = 0;
|
|
me._reverseFlag *= -1;
|
|
}else if(me.loop || me.repeat > 0 && me._repeatCount++ < me.repeat){
|
|
me._startTime = now() + me.repeatDelay;
|
|
me._pausedTime = 0;
|
|
}else{
|
|
me.isComplete = true;
|
|
}
|
|
}
|
|
|
|
//next tween
|
|
var next = me._next;
|
|
if(next && next.time <= 0){
|
|
var nextStartTime = next._startTime;
|
|
if(nextStartTime > 0 && nextStartTime <= time){
|
|
//parallel tween
|
|
next._render(ratio);
|
|
next.time = elapsed;
|
|
Tween.add(next);
|
|
}else if(me.isComplete && (nextStartTime < 0 || nextStartTime > time)){
|
|
//next tween
|
|
next.start();
|
|
}
|
|
}
|
|
|
|
//complete
|
|
if(me.isComplete){
|
|
(callback = me.onComplete) && callback.call(me, me);
|
|
return true;
|
|
}
|
|
},
|
|
|
|
Statics: /** @lends Tween */ {
|
|
/**
|
|
* @private
|
|
*/
|
|
_tweens: [],
|
|
|
|
/**
|
|
* Update all Tween instances.
|
|
* @returns {Object} Tween。
|
|
*/
|
|
tick: function(){
|
|
var tweens = Tween._tweens, tween, i, len = tweens.length;
|
|
|
|
for(i = 0; i < len; i++){
|
|
tween = tweens[i];
|
|
if(tween && tween._update(now())){
|
|
tweens.splice(i, 1);
|
|
i--;
|
|
}
|
|
}
|
|
return Tween;
|
|
},
|
|
|
|
/**
|
|
* Add a Tween instance.
|
|
* @param {Tween} tween Tween object to add.
|
|
* @returns {Object} Tween。
|
|
*/
|
|
add: function(tween){
|
|
var tweens = Tween._tweens;
|
|
if(tweens.indexOf(tween) == -1) tweens.push(tween);
|
|
return Tween;
|
|
},
|
|
|
|
/**
|
|
* Remove one Tween target.
|
|
* @param {Tween|Object|Array} tweenOrTarget Tween object, target object or an array of object to remove
|
|
* @returns {Object} Tween。
|
|
*/
|
|
remove: function(tweenOrTarget){
|
|
var i, l;
|
|
if(tweenOrTarget instanceof Array){
|
|
for(i = 0, l = tweenOrTarget.length;i < l;i ++){
|
|
Tween.remove(tweenOrTarget[i]);
|
|
}
|
|
return Tween;
|
|
}
|
|
|
|
var tweens = Tween._tweens;
|
|
if(tweenOrTarget instanceof Tween){
|
|
i = tweens.indexOf(tweenOrTarget);
|
|
if(i > -1) tweens.splice(i, 1);
|
|
}else{
|
|
for(i = 0; i < tweens.length; i++){
|
|
if(tweens[i].target === tweenOrTarget){
|
|
tweens.splice(i, 1);
|
|
i--;
|
|
}
|
|
}
|
|
}
|
|
|
|
return Tween;
|
|
},
|
|
|
|
/**
|
|
* Remove all Tween instances.
|
|
* @returns {Object} Tween。
|
|
*/
|
|
removeAll: function(){
|
|
Tween._tweens.length = 0;
|
|
return Tween;
|
|
},
|
|
|
|
/**
|
|
* Create a tween, make target object easing from beginning properties to ending properties.
|
|
* @param {Object|Array} target Tweening target or tweening target array.
|
|
* @param fromProps Beginning properties of target tweening object.
|
|
* @param toProps Ending properties of target tweening object.
|
|
* @param params Tweening parameters.
|
|
* @returns {Tween|Array} An tween instance or an array of tween instance.
|
|
*/
|
|
fromTo: function(target, fromProps, toProps, params){
|
|
params = params || {};
|
|
var isArray = target instanceof Array;
|
|
target = isArray ? target : [target];
|
|
|
|
var tween, i, stagger = params.stagger, tweens = [];
|
|
for(i = 0; i < target.length; i++){
|
|
tween = new Tween(target[i], fromProps, toProps, params);
|
|
if(stagger) tween.delay = (params.delay || 0) + (i * stagger || 0);
|
|
tween.start();
|
|
tweens.push(tween);
|
|
}
|
|
|
|
return isArray?tweens:tween;
|
|
},
|
|
|
|
/**
|
|
* Create a tween, make target object easing from current properties to ending properties.
|
|
* @param {Object|Array} target Tweening target or tweening target array.
|
|
* @param toProps Ending properties of target tweening object.
|
|
* @param params Tweening parameters.
|
|
* @returns {Tween|Array} An tween instance or an array of tween instance.
|
|
*/
|
|
to: function(target, toProps, params){
|
|
return Tween.fromTo(target, null, toProps, params);
|
|
},
|
|
|
|
/**
|
|
* Create a tween, make target object easing from beginning properties to current properties.
|
|
* @param {Object|Array} target Tweening target or tweening target array.
|
|
* @param fromProps Beginning properties of target tweening object.
|
|
* @param params Tweening parameters.
|
|
* @returns {Tween|Array} An tween instance or an array of tween instance.
|
|
*/
|
|
from: function(target, fromProps, params){
|
|
return Tween.fromTo(target, fromProps, null, params);
|
|
}
|
|
}
|
|
|
|
});
|
|
|
|
})();
|