chromium/third_party/blink/web_tests/animations/timing/timing-model.html

<!doctype html>
<meta charset="utf-8">
<body>
<script>
onload = function() {

log('Sanity tests:');
assertIterationTime('1s linear', [
  {is: null, at: -1},
  {is: 0, at: 0},
  {is: 0.5, at: 0.5},
  {is: 0.2, at: 0.2},
  {is: 0.8, at: 0.8},
  {is: null, at: 2},
]);


log('\nFill-mode:');
assertIterationTime('1s none', [
  {is: null, at: -1},
  {is: null, at: 2},
]);

assertIterationTime('1s both', [
  {is: 0, at: -1},
  {is: 1, at: 2},
]);

assertIterationTime('1s forwards', [
  {is: null, at: -1},
  {is: 1, at: 2},
]);

assertIterationTime('1s backwards', [
  {is: 0, at: -1},
  {is: null, at: 2},
]);


log('\nFill-mode + reverse direction:');
assertIterationTime('1s none reverse', [
  {is: null, at: -1},
  {is: null, at: 2},
]);

assertIterationTime('1s both reverse', [
  {is: 1, at: -1},
  {is: 0, at: 2},
]);

assertIterationTime('1s forwards reverse', [
  {is: null, at: -1},
  {is: 0, at: 2},
]);

assertIterationTime('1s backwards reverse', [
  {is: 1, at: -1},
  {is: null, at: 2},
]);


log('\nEnd exclusive timing:')
assertIterationTime('1s', [
  {is: null, at: 1},
]);


log('\nZero duration:')
assertIterationTime('none', [
  {is: null, at: -1},
  {is: null, at: 0},
  {is: null, at: 1},
]);

assertIterationTime('both', [
  {is: 0, at: -1},
  {is: 1, at: 0},
  {is: 1, at: 1},
]);

assertIterationTime('forwards', [
  {is: null, at: -1},
  {is: 1, at: 0},
  {is: 1, at: 1},
]);

assertIterationTime('backwards', [
  {is: 0, at: -1},
  {is: null, at: 0},
  {is: null, at: 1},
]);


log('\nZero duration + reverse direction:')
assertIterationTime('none reverse', [
  {is: null, at: -1},
  {is: null, at: 0},
  {is: null, at: 1},
]);

assertIterationTime('both reverse', [
  {is: 1, at: -1},
  {is: 0, at: 0},
  {is: 0, at: 1},
]);

assertIterationTime('forwards reverse', [
  {is: null, at: -1},
  {is: 0, at: 0},
  {is: 0, at: 1},
]);

assertIterationTime('backwards reverse', [
  {is: 1, at: -1},
  {is: null, at: 0},
  {is: null, at: 1},
]);


log('\nZero iterations:')
assertIterationTime('1s 0s 0 none', [
  {is: null, at: -1},
  {is: null, at: 0},
  {is: null, at: 0.5},
  {is: null, at: 2},
]);

assertIterationTime('1s 0s 0 both', [
  {is: 0, at: -1},
  {is: 0, at: 0},
  {is: 0, at: 2},
]);

assertIterationTime('1s 0s 0 forwards', [
  {is: null, at: -1},
  {is: 0, at: 0},
  {is: 0, at: 2},
]);

assertIterationTime('1s 0s 0 backwards', [
  {is: 0, at: -1},
  {is: null, at: 0},
  {is: null, at: 2},
]);


log('\nZero iterations, zero duration:')
assertIterationTime('0s 0s 0 none', [
  {is: null, at: -1},
  {is: null, at: 0},
  {is: null, at: 2},
]);

assertIterationTime('0s 0s 0 both', [
  {is: 0, at: -1},
  {is: 0, at: 0},
  {is: 0, at: 1},
]);

assertIterationTime('0s 0s 0 forwards', [
  {is: null, at: -1},
  {is: 0, at: 0},
  {is: 0, at: 1},
]);

assertIterationTime('0s 0s 0 backwards', [
  {is: 0, at: -1},
  {is: null, at: 0},
  {is: null, at: 1},
]);


log('\nMultiple iterations:')
assertIterationTime('1s 2 linear', [
  {is: null, at: -1},
  {is: 0, at: 0},
  {is: 0.4, at: 0.4},
  {is: 0, at: 1},
  {is: 0.6, at: 1.6},
  {is: null, at: 2},
]);

assertIterationTime('1s 2 linear reverse', [
  {is: 1, at: 0},
  {is: 0.6, at: 0.4},
  {is: 1, at: 1},
  {is: 0.4, at: 1.6},
]);

assertIterationTime('1s 2 linear alternate', [
  {is: 0, at: 0},
  {is: 0.4, at: 0.4},
  {is: 1, at: 1},
  {is: 0.4, at: 1.6},
]);

assertIterationTime('1s 2 linear alternate-reverse', [
  {is: 1, at: 0},
  {is: 0.6, at: 0.4},
  {is: 0, at: 1},
  {is: 0.6, at: 1.6},
]);


log('\nFractional iterations:')
assertIterationTime('1s 0.5 linear', [
  {is: null, at: -1},
  {is: 0, at: 0},
  {is: 0.2, at: 0.2},
  {is: null, at: 0.5},
]);

assertIterationTime('1s 1.5 linear', [
  {is: null, at: -1},
  {is: 0, at: 0},
  {is: 0.5, at: 0.5},
  {is: 0, at: 1},
  {is: 0.4, at: 1.4},
  {is: null, at: 1.5},
]);

assertIterationTime('1s 1.5 linear reverse', [
  {is: null, at: -1},
  {is: 1, at: 0},
  {is: 0.5, at: 0.5},
  {is: 1, at: 1},
  {is: 0.6, at: 1.4},
  {is: null, at: 1.5},
]);

assertIterationTime('1s 1.6 linear forwards', [
  {is: 0.6, at: 2},
]);

assertIterationTime('1s 1.6 linear forwards reverse ', [
  {is: 0.4, at: 2},
]);

assertIterationTime('1s 0.6 linear backwards reverse', [
  {is: 1, at: -1},
]);


log('\nInfinite iterations:')
assertIterationTime('1s infinite linear', [
  {is: 0, at: 1},
  {is: 0, at: 10},
  {is: 0, at: 1000},
  {is: 0.4, at: 1000.4},
]);


log('\nInfinite iterations, zero duration:')
assertIterationTime('0s infinite linear', [
  {is: null, at: -1},
  {is: null, at: 0},
  {is: null, at: 2},
]);

};

if (window.testRunner) {
  testRunner.dumpAsText();
}

var testElement = document.createElement('div');
document.body.appendChild(testElement);
testElement.style.left = 'auto';

function update() {
 document.body.offsetTop;
}
function log(message, klass) {
 message += '\n';
 var child;
 if (klass) {
   child = document.createElement('span');
   child.classList.add(klass);
   child.textContent = message;
 } else {
   child = document.createTextNode(message);
 }
 document.body.appendChild(child);
}

function assertIterationTime(animation, expectations) {
 testElement.style.setProperty('animation', 'invalid ' + animation);
 update();

 expectations.forEach(function(expectation) {
   if (expectation.is !== null && expectation.is != Math.round(expectation.is * 100) / 100) {
     console.log('ERROR: Test requires too much precision. ' + JSON.stringify(expectation));
     return;
   }
   testElement.style.setProperty('animation-name', 'invalid');
   update();

   var delay = expectation.at * -1;
   testElement.style.setProperty('animation-delay', delay + 's');
   testElement.style.setProperty('animation-name', 'test');
   update();

   var result = getComputedStyle(testElement).left;
   if (result === 'auto') {
     result = null;
   } else {
     result = Math.round(Number(result.replace(/px$/, '')) * 100) / 100;
   }
   if (result === expectation.is) {
     log('PASS: [' + animation + '] iteration time was [' + expectation.is + '] at ' + expectation.at + 's');
   } else {
     log('FAIL: [' + animation + '] iteration time was [' + result + '] at ' + expectation.at + 's' + ' expected [' + expectation.is + ']', 'fail');
   }
 });
}
</script>
<style>
body {
   white-space: pre;
   font-family: monospace;
}
.fail {
   font-weight: bold;
   color: red;
}
@keyframes test {
   0% {
       left: 0px;
   }
   100% {
       left: 1px;
   }
}
</style>