Skip to content
Permalink
Branch: master
Find file Copy path
Find file Copy path
Fetching contributors…
Cannot retrieve contributors at this time
564 lines (503 sloc) 20.6 KB
/**
* Copyright 2018 The AMP HTML Authors. 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.
*/
import {prepareImageAnimation} from './transform-img.js';
import {imgLoadPromise} from '../testing/utils.js';
import {
setup as setupAnimations,
tearDown as tearDownAnimations,
offset,
} from '../testing/animation-test-controller';
/**
* Used to make sure an expected position, width or height is close to the
* actual value. This is a bit larger than ideal as Chrome/Firefox have
* substantially different values part way through the animation (at any
* given time).
*/
const COMPARISON_EPSILON = 0.3;
const {expect} = chai;
const threeByFourUri = '';
describe('prepareImageAnimation', () => {
let testContainer;
let smallerImg;
let largerImg;
let transitionContainer;
let cleanupAnimation;
const curve = {x1: 0.8, y1: 0, x2: 0.2, y2: 1};
const head = document.head!;
const styles = {
'animationDuration': '1000ms',
};
function getIntermediateImg() {
return transitionContainer.firstChild.firstChild;
}
function startAnimation(srcImg, targetImg, srcCropRect?, targetCropRect?) {
const animation = prepareImageAnimation({
transitionContainer,
styleContainer: head,
srcImg,
srcCropRect,
targetImg,
targetCropRect,
styles,
curve,
});
animation.applyAnimation();
cleanupAnimation = animation.cleanupAnimation;
}
async function updateImg(img, fit, position, src) {
img.style.objectFit = fit;
img.style.objectPosition = position;
img.src = src;
await imgLoadPromise(img);
}
before(() => {
setupAnimations();
});
after(() => {
tearDownAnimations();
});
beforeEach(() => {
cleanupAnimation = () => {};
smallerImg = document.createElement('img');
smallerImg.style.position = 'fixed';
largerImg = document.createElement('img');
largerImg.style.position = 'fixed';
testContainer = document.createElement('div');
transitionContainer = document.createElement('div');
testContainer.appendChild(smallerImg);
testContainer.appendChild(largerImg);
document.body.appendChild(testContainer);
document.body.appendChild(transitionContainer);
});
afterEach(() => {
cleanupAnimation();
document.body.removeChild(transitionContainer);
document.body.removeChild(testContainer);
});
describe('smaller to larger', () => {
beforeEach(async () => {
await updateImg(smallerImg, 'contain', '', threeByFourUri);
smallerImg.style.width = '12px';
smallerImg.style.height = '12px';
smallerImg.style.top = '500px';
smallerImg.style.left = '100px';
await updateImg(largerImg, 'contain', '', threeByFourUri);
largerImg.style.width = '24px';
largerImg.style.height = '32px';
largerImg.style.top = '10px';
largerImg.style.left = '10px';
});
it('should start with the correct size and position', () => {
startAnimation(smallerImg, largerImg);
offset(0);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// smallerImg.style.top
expect(top).to.be.closeTo(500, COMPARISON_EPSILON);
// smallerImg.style.left
expect(left).to.be.closeTo(100, COMPARISON_EPSILON);
// smallerImg.style.width
expect(width).to.be.closeTo(12, COMPARISON_EPSILON);
// smallerImg.style.height
expect(height).to.be.closeTo(12, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
} = replacementImg.getBoundingClientRect();
// 3/4 * smallerImg.style.height (contain)
expect(imgWidth).to.be.closeTo(9, COMPARISON_EPSILON);
// smallerImg.style.height
expect(imgHeight).to.be.closeTo(12, COMPARISON_EPSILON);
});
it('should have the correct size and position 200ms in', async () => {
startAnimation(smallerImg, largerImg);
offset(200);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// 20% of the animation, starting from 500, going to 10
expect(top).to.be.closeTo(486.6, COMPARISON_EPSILON);
// 20% of the animation, starting from 100, going to 10
expect(left).to.be.closeTo(97.5, COMPARISON_EPSILON);
// 20% of the animation, starting from 12, going to 24
expect(width).to.be.closeTo(12.333, COMPARISON_EPSILON);
// 20% of the animation, starting from 12, going to 32
expect(height).to.be.closeTo(12.55, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
} = replacementImg.getBoundingClientRect();
// 20% of the animation, starting from 9, going to 24
expect(imgWidth).to.be.closeTo(9.4166, COMPARISON_EPSILON);
// 20% of the animation, starting from 12, going to 32
expect(imgHeight).to.be.closeTo(12.55, COMPARISON_EPSILON);
});
it('should end with the correct size and position', async () => {
startAnimation(smallerImg, largerImg);
offset(1000);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// largerImg.style.top
expect(top).to.be.closeTo(10, COMPARISON_EPSILON);
// largerImg.style.left
expect(left).to.be.closeTo(10, COMPARISON_EPSILON);
// largerImg.style.width
expect(width).to.be.closeTo(24, COMPARISON_EPSILON);
// largerImg.style.height
expect(height).to.be.closeTo(32, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
} = replacementImg.getBoundingClientRect();
// largerImg.style.width
expect(imgWidth).to.be.closeTo(24, COMPARISON_EPSILON);
// largerImg.style.height
expect(imgHeight).to.be.closeTo(32, COMPARISON_EPSILON);
});
});
describe('larger to smaller', () => {
beforeEach(async () => {
await updateImg(smallerImg, 'contain', '', threeByFourUri);
smallerImg.style.width = '12px';
smallerImg.style.height = '12px';
smallerImg.style.top = '500px';
smallerImg.style.left = '100px';
await updateImg(largerImg, 'contain', '', threeByFourUri);
largerImg.style.width = '24px';
largerImg.style.height = '32px';
largerImg.style.top = '10px';
largerImg.style.left = '10px';
});
it('should start with the correct size and position', () => {
startAnimation(largerImg, smallerImg);
offset(0);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// largerImg.style.top
expect(top).to.be.closeTo(10, COMPARISON_EPSILON);
// largerImg.style.left
expect(left).to.be.closeTo(10, COMPARISON_EPSILON);
// largerImg.style.width
expect(width).to.be.closeTo(24, COMPARISON_EPSILON);
// largerImg.style.height
expect(height).to.be.closeTo(32, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
top: imgTop,
left: imgLeft,
} = replacementImg.getBoundingClientRect();
// largerImg.style.width
expect(imgWidth).to.be.closeTo(24, COMPARISON_EPSILON);
// largerImg.style.height
expect(imgHeight).to.be.closeTo(32, COMPARISON_EPSILON);
// starting center aligned; height matches container so top is unchanged
expect(imgTop).to.be.closeTo(10, COMPARISON_EPSILON);
// starting center aligned; width matches container so left is unchanged
expect(imgLeft).to.be.closeTo(10, COMPARISON_EPSILON);
});
it('should have the correct size and position 200ms in', async () => {
startAnimation(largerImg, smallerImg);
offset(200);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// 20% of the animation, starting from 10, going to 500
expect(top).to.be.closeTo(23.4, COMPARISON_EPSILON);
// 20% of the animation, starting from 10, going to 100
expect(left).to.be.closeTo(12.5, COMPARISON_EPSILON);
// 20% of the animation, starting from 24, going to 12
expect(width).to.be.closeTo(23.666, COMPARISON_EPSILON);
// 20% of the animation, starting from 32, going to 12
expect(height).to.be.closeTo(31.45, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
} = replacementImg.getBoundingClientRect();
// 20% of the animation, starting from 24, going to 9
expect(imgWidth).to.be.closeTo(23.583, COMPARISON_EPSILON);
// 20% of the animation, starting from 32, going to 12
expect(imgHeight).to.be.closeTo(31.45, COMPARISON_EPSILON);
});
it('should end with the correct size and position', async () => {
startAnimation(largerImg, smallerImg);
offset(1000);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// smallerImg.style.top
expect(top).to.be.closeTo(500, COMPARISON_EPSILON);
// smallerImg.style.left
expect(left).to.be.closeTo(100, COMPARISON_EPSILON);
// smallerImg.style.width
expect(width).to.be.closeTo(12, COMPARISON_EPSILON);
// smallerImg.style.height
expect(height).to.be.closeTo(12, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
top: imgTop,
left: imgLeft,
} = replacementImg.getBoundingClientRect();
// 3/4 * smallerImg.style.height (contain)
expect(imgWidth).to.be.closeTo(9, COMPARISON_EPSILON);
// smallerImg.style.height
expect(imgHeight).to.be.closeTo(12, COMPARISON_EPSILON);
// ending center aligned; height matches container so top is unchanged
expect(imgTop).to.be.closeTo(500, COMPARISON_EPSILON);
// ending center aligned, so need to offseet by 1/2 width to container
// delta ((12 - 9) / 2)
expect(imgLeft).to.be.closeTo(101.5, COMPARISON_EPSILON);
});
});
describe('scrolling', () => {
let sizer;
beforeEach(async () => {
sizer = document.createElement('div');
sizer.style.height = `${window.innerHeight * 2}px`;
sizer.style.width = `${window.innerWidth * 2}px`;
document.body.appendChild(sizer);
await updateImg(smallerImg, 'contain', '', threeByFourUri);
smallerImg.style.width = '12px';
smallerImg.style.height = '12px';
smallerImg.style.top = '500px';
smallerImg.style.left = '100px';
await updateImg(largerImg, 'contain', '', threeByFourUri);
largerImg.style.width = '24px';
largerImg.style.height = '32px';
largerImg.style.top = '10px';
largerImg.style.left = '10px';
});
afterEach(() => {
document.body.removeChild(sizer);
window.scrollTo(0, 0);
});
it('should have the correct position 200ms in', async () => {
startAnimation(largerImg, smallerImg);
offset(200);
window.scrollTo(50, 75);
const replacement = getIntermediateImg();
const {top, left} = replacement.getBoundingClientRect();
// 20% of the animation, starting from 10, going to 500 - 75 from scroll
expect(top).to.be.closeTo(-51.5, COMPARISON_EPSILON);
// 20% of the animation, starting from 10, going to 100 - 50 from scroll
expect(left).to.be.closeTo(-37.5, COMPARISON_EPSILON);
});
it('should end with the correct position', async () => {
startAnimation(largerImg, smallerImg);
offset(1000);
window.scrollTo(50, 75);
const replacement = getIntermediateImg();
const {top, left} = replacement.getBoundingClientRect();
// smallerImg.style.top - 75 from scroll
expect(top).to.be.closeTo(425, COMPARISON_EPSILON);
// smallerImg.style.left - 50 from scroll
expect(left).to.be.closeTo(50, COMPARISON_EPSILON);
});
});
describe('object-position', () => {
beforeEach(async () => {
await updateImg(smallerImg, 'contain', 'bottom left', threeByFourUri);
smallerImg.style.width = '12px';
smallerImg.style.height = '12px';
smallerImg.style.top = '500px';
smallerImg.style.left = '100px';
await updateImg(largerImg, 'contain', 'top right', threeByFourUri);
largerImg.style.width = '32px';
largerImg.style.height = '32px';
largerImg.style.top = '10px';
largerImg.style.left = '10px';
});
it('should start with the correct size and position', () => {
startAnimation(largerImg, smallerImg);
offset(0);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// largerImg.style.top
expect(top).to.be.closeTo(10, COMPARISON_EPSILON);
// largerImg.style.left
expect(left).to.be.closeTo(10, COMPARISON_EPSILON);
// largerImg.style.width
expect(width).to.be.closeTo(32, COMPARISON_EPSILON);
// largerImg.style.height
expect(height).to.be.closeTo(32, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
top: imgTop,
left: imgLeft,
} = replacementImg.getBoundingClientRect();
// 3 by 4 * 32 = 24
expect(imgWidth).to.be.closeTo(24, COMPARISON_EPSILON);
// largerImg.style.height
expect(imgHeight).to.be.closeTo(32, COMPARISON_EPSILON);
// starting top aligned; height matches container so top is unchanged
expect(imgTop).to.be.closeTo(10, COMPARISON_EPSILON);
// starting right aligned; so need to move right by delta of container
// size - img size (32 - 24)
expect(imgLeft).to.be.closeTo(18, COMPARISON_EPSILON);
});
it('should end with the correct size and position', async () => {
startAnimation(largerImg, smallerImg);
offset(1000);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// smallerImg.style.top
expect(top).to.be.closeTo(500, COMPARISON_EPSILON);
// smallerImg.style.left
expect(left).to.be.closeTo(100, COMPARISON_EPSILON);
// smallerImg.style.width
expect(width).to.be.closeTo(12, COMPARISON_EPSILON);
// smallerImg.style.height
expect(height).to.be.closeTo(12, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
top: imgTop,
left: imgLeft,
} = replacementImg.getBoundingClientRect();
// 3/4 * smallerImg.style.height (contain)
expect(imgWidth).to.be.closeTo(9, COMPARISON_EPSILON);
// smallerImg.style.height
expect(imgHeight).to.be.closeTo(12, COMPARISON_EPSILON);
// ending bottom aligned; height matches container so top is unchanged
expect(imgTop).to.be.closeTo(500, COMPARISON_EPSILON);
// ending left aligned, so just line up on the left edge
expect(imgLeft).to.be.closeTo(100, COMPARISON_EPSILON);
});
});
describe('specifying the crop container', () => {
let smallerCrop;
let largerCrop;
beforeEach(async () => {
smallerCrop = document.createElement('div');
smallerCrop.style.position = 'fixed';
smallerCrop.style.width = '12px';
smallerCrop.style.height = '16px';
smallerCrop.style.top = '500px';
smallerCrop.style.left = '100px';
await updateImg(smallerImg, 'cover', 'center center', threeByFourUri);
smallerImg.style.position = 'absolute';
smallerImg.style.top = '0';
smallerImg.style.left = '0';
smallerImg.style.width = '100%';
smallerImg.style.height = '100%';
smallerImg.style.transform = 'scale(2)';
smallerImg.style.transformOrigin = 'bottom center';
largerCrop = document.createElement('div');
largerCrop.style.position = 'fixed';
largerCrop.style.width = '24px';
largerCrop.style.height = '32px';
largerCrop.style.top = '10px';
largerCrop.style.left = '10px';
await updateImg(largerImg, 'cover', 'center center', threeByFourUri);
largerImg.style.position = 'absolute';
largerImg.style.top = '0';
largerImg.style.left = '0';
largerImg.style.width = '100%';
largerImg.style.height = '100%';
largerImg.style.transform = 'scale(1.25)';
largerImg.style.transformOrigin = 'bottom center';
smallerCrop.appendChild(smallerImg);
testContainer.appendChild(smallerCrop);
largerCrop.appendChild(largerImg);
testContainer.appendChild(largerCrop);
});
it('should start with the correct size and position', () => {
startAnimation(
smallerImg,
largerImg,
smallerCrop.getBoundingClientRect(),
largerCrop.getBoundingClientRect()
);
offset(0);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// smallerCropRect.style.top
expect(top).to.be.closeTo(500, COMPARISON_EPSILON);
// smallerCropRect.style.left
expect(left).to.be.closeTo(100, COMPARISON_EPSILON);
// smallerCropRect.style.width
expect(width).to.be.closeTo(12, COMPARISON_EPSILON);
// smallerCropRect.style.height
expect(height).to.be.closeTo(16, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
bottom: imgBottom,
left: imgLeft,
} = replacementImg.getBoundingClientRect();
// 12, scaled by 2 = 24
expect(imgWidth).to.be.closeTo(24, COMPARISON_EPSILON);
// 12 * 4/3, scaled by 2 = 32
expect(imgHeight).to.be.closeTo(32, COMPARISON_EPSILON);
// The imgBottom should match the original's bottom, since we are
// using `bottom` in our transform origin.
// 500px (top) + 16px (height) = 516
expect(imgBottom).to.be.closeTo(516, COMPARISON_EPSILON);
// left of crop container is 100px, we are scaled by 2 with
// `transform-origin: bottom center`, which moves us left by 50% of the
// width of the img. The img width is 12px.
// 100px - 6px = 94
expect(imgLeft).to.be.closeTo(94, COMPARISON_EPSILON);
});
it('should end with the correct size and position', () => {
startAnimation(
smallerImg,
largerImg,
smallerCrop.getBoundingClientRect(),
largerCrop.getBoundingClientRect()
);
offset(1000);
const replacement = getIntermediateImg();
const {top, left, width, height} = replacement.getBoundingClientRect();
// largerCropRect.style.top
expect(top).to.be.closeTo(10, COMPARISON_EPSILON);
// largerCropRect.style.left
expect(left).to.be.closeTo(10, COMPARISON_EPSILON);
// largerCropRect.style.width
expect(width).to.be.closeTo(24, COMPARISON_EPSILON);
// largerCropRect.style.height
expect(height).to.be.closeTo(32, COMPARISON_EPSILON);
const replacementImg = replacement.querySelector('img');
const {
width: imgWidth,
height: imgHeight,
bottom: imgBottom,
left: imgLeft,
} = replacementImg.getBoundingClientRect();
// 24, scaled by 1.25 = 30
expect(imgWidth).to.be.closeTo(30, COMPARISON_EPSILON);
// 32, scaled by 1.25 = 30
expect(imgHeight).to.be.closeTo(40, COMPARISON_EPSILON);
// The imgBottom should match the original's bottom, since we are
// using `bottom` in our transform origin.
// 10px (top) + 32px (height) = 42
expect(imgBottom).to.be.closeTo(42, COMPARISON_EPSILON);
// left of crop container is 10px, we are scaled by 1.25 with
// `transform-origin: bottom center`, which moves us left by 12.5% of the
// width of the original img width;
// 10 - (.125 * 24) = 7
expect(imgLeft).to.be.closeTo(7, COMPARISON_EPSILON);
});
});
});
You can’t perform that action at this time.