Lightningbeam/tests/specs/timeline-animation.test.js

318 lines
12 KiB
JavaScript

/**
* Timeline animation tests for Lightningbeam
* Tests shape and object animations across keyframes
*/
import { describe, it, before } from 'mocha';
import { expect } from '@wdio/globals';
import { waitForAppReady } from '../helpers/app.js';
import {
drawRectangle,
drawEllipse,
clickCanvas,
dragCanvas,
selectMultipleShapes,
useKeyboardShortcut,
setPlayheadTime,
getPlayheadTime,
addKeyframe,
getPixelColor
} from '../helpers/canvas.js';
import { assertShapeExists } from '../helpers/assertions.js';
import { verifyManually, logStep } from '../helpers/manual.js';
describe('Timeline Animation', () => {
before(async () => {
await waitForAppReady();
});
describe('Shape Keyframe Animation', () => {
it('should animate shape position across keyframes', async () => {
// Draw a rectangle at frame 1 (time 0)
await drawRectangle(100, 100, 100, 100);
// Select the shape by dragging a selection box over it
await selectMultipleShapes([{ x: 150, y: 150 }]);
await browser.pause(200);
// Verify it exists at original position
await assertShapeExists(150, 150, 'Shape should exist at frame 1');
// Move to frame 10 (time in seconds, assuming 30fps: frame 10 = 10/30 ≈ 0.333s)
await setPlayheadTime(0.333);
// Add a keyframe at this position
await addKeyframe();
await browser.pause(200);
await logStep('About to drag selected shape - dragging selected shapes does not move them yet');
// Shape is selected, so dragging from its center will move it
await dragCanvas(150, 150, 250, 150);
await browser.pause(300);
await verifyManually(
'VERIFY: Did the shape move to x=250?\n' +
'Expected: Shape at x=250\n' +
'Note: Dragging selected shapes is not implemented yet\n\n' +
'Click OK if at x=250, Cancel if not'
);
// At frame 10, shape should be at the new position (moved 100px to the right)
await assertShapeExists(250, 150, 'Shape should be at new position at frame 10');
// Go back to frame 1
await setPlayheadTime(0);
await browser.pause(200);
await verifyManually(
'VERIFY: Did the shape return to original position (x=150)?\n\n' +
'Click OK if yes, Cancel if no'
);
// Shape should be at original position
await assertShapeExists(150, 150, 'Shape should be at original position at frame 1');
// Go to middle frame (frame 5, time ≈ 0.166s)
await setPlayheadTime(0.166);
await browser.pause(200);
await verifyManually(
'VERIFY: Is the shape interpolated at x=200 (halfway)?\n\n' +
'Click OK if yes, Cancel if no'
);
// Shape should be interpolated between the two positions
// At frame 5 (halfway), shape should be around x=200 (halfway between 150 and 250)
await assertShapeExists(200, 150, 'Shape should be interpolated at frame 5');
});
it('should modify shape edges when dragging edge of unselected shape', async () => {
// Draw a rectangle
await drawRectangle(400, 100, 100, 100);
// Get color at center
const centerColorBefore = await getPixelColor(450, 150);
// WITHOUT selecting the shape, drag the right edge
// The right edge is at x=500, so drag from there
await dragCanvas(500, 150, 550, 150);
await browser.pause(300);
// The shape should now be modified - it's been curved/stretched
// The original center should still have the shape
await assertShapeExists(450, 150, 'Center should still have shape');
// And there should be shape data extended to the right
const rightColorAfter = await getPixelColor(525, 150);
// This should have color (not white) since we dragged the edge there
expect(rightColorAfter.toLowerCase()).not.toBe('#ffffff');
});
it('should handle multiple keyframes on the same shape', async () => {
// Draw a shape
await drawRectangle(100, 100, 80, 80);
// Select it
await selectMultipleShapes([{ x: 140, y: 140 }]);
await browser.pause(200);
// Keyframe 1: time 0 (original position at x=140, y=140)
// Keyframe 2: time 0.333 (move right)
await setPlayheadTime(0.333);
await addKeyframe();
await browser.pause(200);
await logStep('Dragging selected shape (not implemented yet)');
// Shape should still be selected, drag to move
await dragCanvas(140, 140, 200, 140);
await browser.pause(300);
await verifyManually('VERIFY: Did shape move to x=200? (probably not)\nClick OK if at x=200, Cancel if not');
// Keyframe 3: time 0.666 (move down)
await setPlayheadTime(0.666);
await addKeyframe();
await browser.pause(200);
// Drag to move down
await dragCanvas(200, 140, 200, 180);
await browser.pause(300);
await verifyManually('VERIFY: Did shape move to y=180?\nClick OK if yes, Cancel if no');
// Verify positions at each keyframe
await setPlayheadTime(0);
await browser.pause(200);
await verifyManually('VERIFY: Shape at original position (x=140, y=140)?\nClick OK if yes, Cancel if no');
await assertShapeExists(140, 140, 'Shape at keyframe 1 (x=140, y=140)');
await setPlayheadTime(0.333);
await browser.pause(200);
await verifyManually('VERIFY: Shape at x=200, y=140?\nClick OK if yes, Cancel if no');
await assertShapeExists(200, 140, 'Shape at keyframe 2 (x=200, y=140)');
await setPlayheadTime(0.666);
await browser.pause(200);
await verifyManually('VERIFY: Shape at x=200, y=180?\nClick OK if yes, Cancel if no');
await assertShapeExists(200, 180, 'Shape at keyframe 3 (x=200, y=180)');
// Check interpolation between keyframe 1 and 2 (at t=0.166, halfway)
await setPlayheadTime(0.166);
await browser.pause(200);
await verifyManually('VERIFY: Shape interpolated at x=170, y=140?\nClick OK if yes, Cancel if no');
await assertShapeExists(170, 140, 'Shape interpolated between kf1 and kf2');
// Check interpolation between keyframe 2 and 3 (at t=0.5, halfway)
await setPlayheadTime(0.5);
await browser.pause(200);
await verifyManually('VERIFY: Shape interpolated at x=200, y=160?\nClick OK if yes, Cancel if no');
await assertShapeExists(200, 160, 'Shape interpolated between kf2 and kf3');
});
});
describe('Group/Object Animation', () => {
it('should animate group position across keyframes', async () => {
// Create a group with two shapes
await drawRectangle(300, 100, 60, 60);
await drawRectangle(380, 100, 60, 60);
await selectMultipleShapes([
{ x: 330, y: 130 },
{ x: 410, y: 130 }
]);
await useKeyboardShortcut('g', true);
await browser.pause(300);
// Verify both shapes exist at frame 1
await assertShapeExists(330, 130, 'First shape at frame 1');
await assertShapeExists(410, 130, 'Second shape at frame 1');
// Select the group by dragging a selection box over it
await selectMultipleShapes([{ x: 370, y: 130 }]);
await browser.pause(200);
// Move to frame 10 and add keyframe
await setPlayheadTime(0.333);
await addKeyframe();
await browser.pause(200);
await logStep('Dragging group down');
// Group is selected, so dragging will move it
// Drag from center of group down
await dragCanvas(370, 130, 370, 200);
await browser.pause(300);
await verifyManually('VERIFY: Did the group move down to y=200?\nClick OK if yes, Cancel if no');
// At frame 10, group should be at new position (moved down)
await assertShapeExists(330, 200, 'First shape at new position at frame 10');
await assertShapeExists(410, 200, 'Second shape at new position at frame 10');
// Go to frame 1
await setPlayheadTime(0);
await browser.pause(200);
await verifyManually('VERIFY: Did group return to original position (y=130)?\nClick OK if yes, Cancel if no');
// Group should be at original position
await assertShapeExists(330, 130, 'First shape at original position at frame 1');
await assertShapeExists(410, 130, 'Second shape at original position at frame 1');
// Go to frame 5 (middle, t=0.166)
await setPlayheadTime(0.166);
await browser.pause(200);
await verifyManually('VERIFY: Is group interpolated at y=165 (halfway)?\nClick OK if yes, Cancel if no');
// Group should be interpolated (halfway between y=130 and y=200, so y=165)
await assertShapeExists(330, 165, 'First shape interpolated at frame 5');
await assertShapeExists(410, 165, 'Second shape interpolated at frame 5');
});
it('should maintain relative positions of shapes within animated group', async () => {
// Create a group (using safer y coordinates)
await drawRectangle(100, 250, 50, 50);
await drawRectangle(170, 250, 50, 50);
await selectMultipleShapes([
{ x: 125, y: 275 },
{ x: 195, y: 275 }
]);
await useKeyboardShortcut('g', true);
await browser.pause(300);
// Select group
await selectMultipleShapes([{ x: 160, y: 275 }]);
await browser.pause(200);
// Add keyframe and move
await setPlayheadTime(0.333);
await addKeyframe();
await browser.pause(200);
await dragCanvas(160, 275, 260, 275);
await browser.pause(300);
// At both keyframes, shapes should maintain 70px horizontal distance
await setPlayheadTime(0);
await browser.pause(200);
await assertShapeExists(125, 275, 'First shape at frame 1');
await assertShapeExists(195, 275, 'Second shape at frame 1 (70px apart)');
await setPlayheadTime(0.333);
await browser.pause(200);
// Both shapes moved 100px to the right
await assertShapeExists(225, 275, 'First shape at frame 10');
await assertShapeExists(295, 275, 'Second shape at frame 10 (still 70px apart)');
});
});
describe('Interpolation', () => {
it('should smoothly interpolate between keyframes', async () => {
// Draw a simple shape
await drawRectangle(100, 100, 50, 50);
// Select it
await selectMultipleShapes([{ x: 125, y: 125 }]);
await browser.pause(200);
// Keyframe at start (x=125)
await setPlayheadTime(0);
await browser.pause(100);
// Keyframe at end (1 second = frame 30, move to x=325)
await setPlayheadTime(1.0);
await addKeyframe();
await browser.pause(200);
await logStep('Dragging shape (selected shapes cannot be dragged yet)');
await dragCanvas(125, 125, 325, 125);
await browser.pause(300);
await verifyManually('VERIFY: Did shape move to x=325? (probably not)\nClick OK if at x=325, Cancel if not');
// Check multiple intermediate frames for smooth interpolation
// Total movement: 200px over 1 second
// At 25% (0.25s), x should be 125 + 50 = 175
await setPlayheadTime(0.25);
await browser.pause(200);
await verifyManually('VERIFY: Shape at x=175 (25% interpolation)?\nClick OK if yes, Cancel if no');
await assertShapeExists(175, 125, 'Shape at 25% interpolation');
// At 50% (0.5s), x should be 125 + 100 = 225
await setPlayheadTime(0.5);
await browser.pause(200);
await verifyManually('VERIFY: Shape at x=225 (50% interpolation)?\nClick OK if yes, Cancel if no');
await assertShapeExists(225, 125, 'Shape at 50% interpolation');
// At 75% (0.75s), x should be 125 + 150 = 275
await setPlayheadTime(0.75);
await browser.pause(200);
await verifyManually('VERIFY: Shape at x=275 (75% interpolation)?\nClick OK if yes, Cancel if no');
await assertShapeExists(275, 125, 'Shape at 75% interpolation');
});
});
});