/** * 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'); }); }); });