mirror of
https://github.com/ionic-team/ionic-framework.git
synced 2025-08-20 20:33:32 +08:00
web animations update
This commit is contained in:
171
scripts/resources/web-animations-js/src/tick.js
Normal file
171
scripts/resources/web-animations-js/src/tick.js
Normal file
@ -0,0 +1,171 @@
|
||||
// Copyright 2014 Google Inc. All rights reserved.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
|
||||
(function(shared, scope, testing) {
|
||||
var originalRequestAnimationFrame = window.requestAnimationFrame;
|
||||
var rafCallbacks = [];
|
||||
var rafId = 0;
|
||||
window.requestAnimationFrame = function(f) {
|
||||
var id = rafId++;
|
||||
if (rafCallbacks.length == 0 && !WEB_ANIMATIONS_TESTING) {
|
||||
originalRequestAnimationFrame(processRafCallbacks);
|
||||
}
|
||||
rafCallbacks.push([id, f]);
|
||||
return id;
|
||||
};
|
||||
|
||||
window.cancelAnimationFrame = function(id) {
|
||||
rafCallbacks.forEach(function(entry) {
|
||||
if (entry[0] == id) {
|
||||
entry[1] = function() {};
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
function processRafCallbacks(t) {
|
||||
var processing = rafCallbacks;
|
||||
rafCallbacks = [];
|
||||
if (t < timeline.currentTime)
|
||||
t = timeline.currentTime;
|
||||
tick(t);
|
||||
processing.forEach(function(entry) { entry[1](t); });
|
||||
if (needsRetick)
|
||||
tick(t);
|
||||
applyPendingEffects();
|
||||
_now = undefined;
|
||||
}
|
||||
|
||||
function compareAnimations(leftAnimation, rightAnimation) {
|
||||
return leftAnimation._sequenceNumber - rightAnimation._sequenceNumber;
|
||||
}
|
||||
|
||||
function InternalTimeline() {
|
||||
this._animations = [];
|
||||
// Android 4.3 browser has window.performance, but not window.performance.now
|
||||
this.currentTime = window.performance && performance.now ? performance.now() : 0;
|
||||
};
|
||||
|
||||
InternalTimeline.prototype = {
|
||||
_play: function(effect) {
|
||||
effect._timing = shared.normalizeTimingInput(effect.timing);
|
||||
var animation = new scope.Animation(effect);
|
||||
animation._idle = false;
|
||||
animation._timeline = this;
|
||||
this._animations.push(animation);
|
||||
scope.restart();
|
||||
scope.invalidateEffects();
|
||||
return animation;
|
||||
}
|
||||
};
|
||||
|
||||
var _now = undefined;
|
||||
|
||||
if (WEB_ANIMATIONS_TESTING) {
|
||||
var now = function() { return timeline.currentTime; };
|
||||
} else {
|
||||
var now = function() {
|
||||
if (_now == undefined)
|
||||
_now = performance.now();
|
||||
return _now;
|
||||
};
|
||||
}
|
||||
|
||||
var ticking = false;
|
||||
var hasRestartedThisFrame = false;
|
||||
|
||||
scope.restart = function() {
|
||||
if (!ticking) {
|
||||
ticking = true;
|
||||
requestAnimationFrame(function() {});
|
||||
hasRestartedThisFrame = true;
|
||||
}
|
||||
return hasRestartedThisFrame;
|
||||
};
|
||||
|
||||
var needsRetick = false;
|
||||
scope.invalidateEffects = function() {
|
||||
needsRetick = true;
|
||||
};
|
||||
|
||||
var pendingEffects = [];
|
||||
function applyPendingEffects() {
|
||||
pendingEffects.forEach(function(f) { f(); });
|
||||
pendingEffects.length = 0;
|
||||
}
|
||||
|
||||
var t60hz = 1000 / 60;
|
||||
|
||||
var originalGetComputedStyle = window.getComputedStyle;
|
||||
Object.defineProperty(window, 'getComputedStyle', {
|
||||
configurable: true,
|
||||
enumerable: true,
|
||||
value: function() {
|
||||
if (needsRetick) {
|
||||
var time = now();
|
||||
if (time - timeline.currentTime > 0)
|
||||
timeline.currentTime += t60hz * (Math.floor((time - timeline.currentTime) / t60hz) + 1);
|
||||
tick(timeline.currentTime);
|
||||
}
|
||||
applyPendingEffects();
|
||||
return originalGetComputedStyle.apply(this, arguments);
|
||||
},
|
||||
});
|
||||
|
||||
function tick(t) {
|
||||
hasRestartedThisFrame = false;
|
||||
var timeline = scope.timeline;
|
||||
timeline.currentTime = t;
|
||||
timeline._animations.sort(compareAnimations);
|
||||
ticking = false;
|
||||
var updatingAnimations = timeline._animations;
|
||||
timeline._animations = [];
|
||||
|
||||
var newPendingClears = [];
|
||||
var newPendingEffects = [];
|
||||
updatingAnimations = updatingAnimations.filter(function(animation) {
|
||||
animation._inTimeline = animation._tick(t);
|
||||
|
||||
if (!animation._inEffect)
|
||||
newPendingClears.push(animation._effect);
|
||||
else
|
||||
newPendingEffects.push(animation._effect);
|
||||
|
||||
if (!animation._isFinished && !animation._paused && !animation._idle)
|
||||
ticking = true;
|
||||
|
||||
return animation._inTimeline;
|
||||
});
|
||||
|
||||
// FIXME: Should remove dupliactes from pendingEffects.
|
||||
pendingEffects.push.apply(pendingEffects, newPendingClears);
|
||||
pendingEffects.push.apply(pendingEffects, newPendingEffects);
|
||||
|
||||
timeline._animations.push.apply(timeline._animations, updatingAnimations);
|
||||
needsRetick = false;
|
||||
|
||||
if (ticking)
|
||||
requestAnimationFrame(function() {});
|
||||
};
|
||||
|
||||
if (WEB_ANIMATIONS_TESTING) {
|
||||
testing.tick = function(t) { timeline.currentTime = t; processRafCallbacks(t); };
|
||||
testing.isTicking = function() { return ticking; };
|
||||
testing.setTicking = function(newVal) { ticking = newVal; };
|
||||
}
|
||||
|
||||
var timeline = new InternalTimeline();
|
||||
scope.timeline = timeline;
|
||||
|
||||
})(webAnimationsShared, webAnimations1, webAnimationsTesting);
|
Reference in New Issue
Block a user