/* * Paper.js - The Swiss Army Knife of Vector Graphics Scripting. * http://paperjs.org/ * * Copyright (c) 2011 - 2016, Juerg Lehni & Jonathan Puckey * http://scratchdisk.com/ & http://jonathanpuckey.com/ * * Distributed under the MIT license. See LICENSE file for details. * * All rights reserved. */ QUnit.module('Path'); test('path.length', function() { var path = new Path([ new Segment(new Point(121, 334), new Point(-19, 38), new Point(30.7666015625, -61.53369140625)), new Segment(new Point(248, 320), new Point(-42, -74), new Point(42, 74)) ]); var length = path.length; equals(length, 172.10112809179614, 'path.length'); var t = path.curves[0].getTimeAt(length / 4); equals(t, 0.2255849553116685, 'path.curves[0].getTimeAt(length / 4)'); }); test('curve.getTimeAt() with straight curve', function() { var path = new Path(); path.moveTo(100, 100); path.lineTo(500, 500); var curve = path.curves[0]; var length = curve.length; var t = curve.getTimeAt(length / 3); equals(t, 0.3869631475722452); }); test('path.join(path)', function() { var path = new Path(); path.add(0, 0); path.add(10, 0); var path2 = new Path(); path2.add(10, 0); path2.add(20, 10); path.join(path2); equals(path.segments.toString(), '{ point: { x: 0, y: 0 } },{ point: { x: 10, y: 0 } },{ point: { x: 20, y: 10 } }'); equals(function() { return paper.project.activeLayer.children.length; }, 1); var path = new Path(); path.add(0, 0); path.add(10, 0); var path2 = new Path(); path2.add(20, 10); path2.add(10, 0); path.join(path2); equals(path.segments.toString(), '{ point: { x: 0, y: 0 } },{ point: { x: 10, y: 0 } },{ point: { x: 20, y: 10 } }'); var path = new Path(); path.add(0, 0); path.add(10, 0); var path2 = new Path(); path2.add(30, 10); path2.add(40, 0); path.join(path2); equals(path.segments.toString(), '{ point: { x: 0, y: 0 } },{ point: { x: 10, y: 0 } },{ point: { x: 30, y: 10 } },{ point: { x: 40, y: 0 } }'); var path = new Path(); path.add(0, 0); path.add(10, 0); path.add(20, 10); var path2 = new Path(); path2.add(0, 0); path2.add(10, 5); path2.add(20, 10); path.join(path2); equals(path.segments.toString(), '{ point: { x: 0, y: 0 } },{ point: { x: 10, y: 0 } },{ point: { x: 20, y: 10 } },{ point: { x: 10, y: 5 } }'); equals(function() { return path.closed; }, true); }); test('path.remove()', function() { var path = new Path(); path.add(0, 0); path.add(10, 0); path.add(20, 0); path.add(30, 0); path.removeSegment(0); equals(function() { return path.segments.length; }, 3); path.removeSegment(0); equals(function() { return path.segments.length; }, 2); path.removeSegments(0, 1); equals(function() { return path.segments.length; }, 1); path.remove(); equals(function() { return paper.project.activeLayer.children.length; }, 0); }); test('path.removeSegments()', function() { var path = new Path(); path.add(0, 0); path.add(10, 0); path.add(20, 0); path.add(30, 0); path.removeSegments(); equals(function() { return path.segments.length; }, 0); }); test('Is the path deselected after setting a new list of segments?', function() { var path = new Path([0, 0]); path.selected = true; equals(function() { return path.selected; }, true); equals(function() { return paper.project.selectedItems.length; }, 1); path.segments = [[0, 10]]; equals(function() { return path.selected; }, true); equals(function() { return paper.project.selectedItems.length; }, 1); }); test('Setting Path#fullySelected=true on an empty path should only set path#selected=true', function() { var path = new Path(); path.fullySelected = true; equals(function() { return path.fullySelected; }, false); equals(function() { return path.selected; }, true); }); test('After removing all segments of a fully selected path, it should still be selected.', function() { var path = new Path([10, 20], [30, 40]); path.fullySelected = true; equals(function() { return path.fullySelected; }, true); path.removeSegments(); equals(function() { return path.fullySelected; }, false); equals(function() { return path.selected; }, true); }); test('After removing all segments of a selected path, it should still be selected.', function() { var path = new Path([10, 20], [30, 40]); path.selected = true; path.removeSegments(); equals(function() { return path.selected; }, true); }); test('After simplifying a path using #simplify(), the path should stay fullySelected', function() { var path = new Path(); for (var i = 0; i < 30; i++) { path.add(i * 10, 10); } path.fullySelected = true; equals(function() { return path.selected; }, true); path.simplify(); equals(function() { return path.selected; }, true); equals(function() { return path.fullySelected; }, true); }); test('After cloning a selected item, it should be added to the Project#selectedItems array', function() { var path = new Path.Circle(new Size(80, 50), 35); path.selected = true; var copy = path.clone(); equals(function() { return paper.project.selectedItems.length; }, 2); }); test('After simplifying a path using #simplify(), the path should stay selected', function() { var path = new Path(); for (var i = 0; i < 30; i++) { path.add(i * 10, (i % 2 ? 20 : 40)); } path.selected = true; path.simplify(); equals(function() { return path.selected; }, true); }); test('After smoothing a path using #smooth(), the path should stay fullySelected', function() { var path = new Path(); for (var i = 0; i < 30; i++) { path.add(i * 10, (i % 2 ? 20 : 40)); } path.fullySelected = true; path.smooth(); equals(function() { return path.fullySelected; }, true); }); test('After smoothing a path using #smooth(), the path should stay selected', function() { var path = new Path(); for (var i = 0; i < 30; i++) { path.add(i * 10, (i % 2 ? 20 : 40)); } path.selected = true; path.smooth(); equals(function() { return path.selected; }, true); }); test('After selecting a segment, Path#selected should return true', function() { var path = new Path(); path.add([10, 10]); path.firstSegment.selected = true; equals(function() { return path.selected; }, true); }); test('Path#reverse', function() { var path = new Path.Circle([100, 100], 30); path.reverse(); equals(path.segments.toString(), '{ point: { x: 100, y: 130 }, handleIn: { x: -16.56854, y: 0 }, handleOut: { x: 16.56854, y: 0 } },{ point: { x: 130, y: 100 }, handleIn: { x: 0, y: 16.56854 }, handleOut: { x: 0, y: -16.56854 } },{ point: { x: 100, y: 70 }, handleIn: { x: 16.56854, y: 0 }, handleOut: { x: -16.56854, y: 0 } },{ point: { x: 70, y: 100 }, handleIn: { x: 0, y: -16.56854 }, handleOut: { x: 0, y: 16.56854 } }'); }); test('Path#reverse should adjust segment indices', function() { var path = new Path([[0, 0], [10, 10], [20, 20]]); path.reverse(); equals(path.segments[0].index, 0); equals(path.segments[1].index, 1); equals(path.segments[2].index, 2); }); test('Path#fullySelected', function() { var path = new Path.Circle([100, 100], 10); path.fullySelected = true; path.segments[1].selected = false; equals(function() { return path.fullySelected; }, false); }); test('Simplifying a path with three segments of the same position should not throw an error', function() { var path = new Path([20, 20], [20, 20], [20, 20]); path.simplify(); equals(function() { return path.segments.length; }, 1); }); test('Path#interpolate', function() { var path = new Path(), path0 = new Path.Circle({ center: [0, 0], radius: 10 }), path1 = new Path.Ellipse({ center: [10, 20], radius: [20, 10] }), halfway = new Path.Ellipse({ center: [5, 10], radius: [15, 10] }); path.interpolate(path0, path1, 0); equals(path, path0); path.interpolate(path0, path1, 1); equals(path, path1); path.interpolate(path0, path1, 0.5); equals(path, halfway); }); QUnit.module('Path Curves'); test('path.curves synchronisation', function() { var path = new Path(); path.add(new Point(100, 100)); equals(path.segments.toString(), "{ point: { x: 100, y: 100 } }", "path.segments: path.add(new Point(100, 100));"); equals(path.curves.toString(), "", "path.curves: path.add(new Point(100, 100));"); path.insert(0, new Point(0, 100)); equals(path.segments.toString(), "{ point: { x: 0, y: 100 } },{ point: { x: 100, y: 100 } }", "path.segments: path.insert(0, new Point(0, 100));"); equals(path.curves.toString(), "{ point1: { x: 0, y: 100 }, point2: { x: 100, y: 100 } }", "path.curves: path.insert(0, new Point(0, 100));"); path.insert(1, {point:[50, 0], handleIn:[-25, 0], handleOut:[25, 0]}); equals(path.segments.toString(), "{ point: { x: 0, y: 100 } },{ point: { x: 50, y: 0 }, handleIn: { x: -25, y: 0 }, handleOut: { x: 25, y: 0 } },{ point: { x: 100, y: 100 } }", "path.segments: path.insert(1, {point:[50, 0], handleIn:[-25, 0], handleOut:[25, 0]});"); equals(path.curves.toString(), "{ point1: { x: 0, y: 100 }, handle2: { x: -25, y: 0 }, point2: { x: 50, y: 0 } },{ point1: { x: 50, y: 0 }, handle1: { x: 25, y: 0 }, point2: { x: 100, y: 100 } }", "path.curves: path.insert(1, {point:[50, 0], handleIn:[-25, 0], handleOut:[25, 0]});"); path.closed = true; equals(path.segments.toString(), "{ point: { x: 0, y: 100 } },{ point: { x: 50, y: 0 }, handleIn: { x: -25, y: 0 }, handleOut: { x: 25, y: 0 } },{ point: { x: 100, y: 100 } }", "path.segments: path.closed = true;"); equals(path.curves.toString(), "{ point1: { x: 0, y: 100 }, handle2: { x: -25, y: 0 }, point2: { x: 50, y: 0 } },{ point1: { x: 50, y: 0 }, handle1: { x: 25, y: 0 }, point2: { x: 100, y: 100 } },{ point1: { x: 100, y: 100 }, point2: { x: 0, y: 100 } }", "path.curves: path.closed = true;"); path.removeSegments(2, 3); equals(path.segments.toString(), "{ point: { x: 0, y: 100 } },{ point: { x: 50, y: 0 }, handleIn: { x: -25, y: 0 }, handleOut: { x: 25, y: 0 } }", "path.segments: path.removeSegments(2, 3);"); equals(path.curves.toString(), "{ point1: { x: 0, y: 100 }, handle2: { x: -25, y: 0 }, point2: { x: 50, y: 0 } },{ point1: { x: 50, y: 0 }, handle1: { x: 25, y: 0 }, point2: { x: 0, y: 100 } }", "path.curves: path.removeSegments(2, 3);"); path.add(new Point(100, 100)); path.removeSegments(1, 2); equals(path.segments.toString(), "{ point: { x: 0, y: 100 } },{ point: { x: 100, y: 100 } }", "path.segments: path.add(new Point(100, 100));\npath.removeSegments(1, 2);"); equals(path.curves.toString(), "{ point1: { x: 0, y: 100 }, point2: { x: 100, y: 100 } },{ point1: { x: 100, y: 100 }, point2: { x: 0, y: 100 } }", "path.curves: path.add(new Point(100, 100));\npath.removeSegments(1, 2);"); // Transform the path, and the curves length should be invalidated (first, force-cache the first segment's length by accessing it var length = path.curves[0].length; ok(path.curves[0]._length, 'Curve length does not appear to be cached'); path.scale(2, [0, 0]); equals(path.curves[0].length, 200, 'Curve length should be updated when path is transformed'); var points = []; for (var i = 0; i < 40; i++) points.push(Point.random()); var path = new Path(points); equals(path.segments.length, 40, 'segments.length'); equals(path.curves.length, 39, 'curves.length'); path.removeSegments(); equals(path.segments.length, 0, 'segments.length'); equals(path.curves.length, 0, 'curves.length'); }); test('path.curves on closed paths', function() { var path = new Path.Circle(new Point(100, 100) , 100); equals(path.curves.toString(), "{ point1: { x: 0, y: 100 }, handle1: { x: 0, y: -55.22847 }, handle2: { x: -55.22847, y: 0 }, point2: { x: 100, y: 0 } },{ point1: { x: 100, y: 0 }, handle1: { x: 55.22847, y: 0 }, handle2: { x: 0, y: -55.22847 }, point2: { x: 200, y: 100 } },{ point1: { x: 200, y: 100 }, handle1: { x: 0, y: 55.22847 }, handle2: { x: 55.22847, y: 0 }, point2: { x: 100, y: 200 } },{ point1: { x: 100, y: 200 }, handle1: { x: -55.22847, y: 0 }, handle2: { x: 0, y: 55.22847 }, point2: { x: 0, y: 100 } }"); path.removeSegments(0, 1); equals(path.curves.toString(), "{ point1: { x: 100, y: 0 }, handle1: { x: 55.22847, y: 0 }, handle2: { x: 0, y: -55.22847 }, point2: { x: 200, y: 100 } },{ point1: { x: 200, y: 100 }, handle1: { x: 0, y: 55.22847 }, handle2: { x: 55.22847, y: 0 }, point2: { x: 100, y: 200 } },{ point1: { x: 100, y: 200 }, handle1: { x: -55.22847, y: 0 }, handle2: { x: -55.22847, y: 0 }, point2: { x: 100, y: 0 } }"); }); test('path.flatten(maxDistance)', function() { var path = new Path.Circle(new Size(80, 50), 35); // Convert its curves to points, with a max distance of 20: path.flatten(20); equals(function() { return path.lastSegment.point.equals(path.firstSegment.point); }, false, 'The points of the last and first segments should not be the same.'); equals(function() { return path.lastSegment.point.toString() != path.lastSegment.previous.point.toString(); }, true, 'The points of the last and before last segments should not be so close, that calling toString on them returns the same string value.'); }); test('Curve list after removing a segment - 1', function() { var path = new paper.Path([0, 0], [1, 1], [2, 2]); var prevCurves = path.curves.slice(); equals(function() { return path.curves.length; }, 2, 'After creating a path with three segments, we should have two curves. By accessing path.curves we also make sure the curves are created internally.'); equals(function() { return path.segments[1].remove(); }, true, 'Removing the paths second segment should be successful.'); equals(function() { return path.curves.length; }, 1, 'After removing the middle segment, we should be left with one curve.'); equals(function() { return path.curves[0] === prevCurves[0]; }, true, 'The remaining curve should be the same as in the previous state of the curves array.'); equals(function() { return path.curves[0].path === path; }, true, 'The remaining curve should stay linked to the path.'); equals(function() { return prevCurves[1].path === null; }, true, 'The removed curve should not stay linked to the path.'); }); test('Curve list after removing a segment - 2', function() { var path = new paper.Path([0, 0], [1, 1], [2, 2]); equals(function() { return path.curves.length; }, 2, 'After creating a path with three segments, we should have two curves. By accessing path.curves we also make sure the curves are created internally.'); equals(function() { return path.lastSegment.remove(); }, true, 'Removing the last segment should be successful.'); equals(function() { return path.curves.length; }, 1, 'After removing the last segment, we should be left with one curve.'); equals(function() { return path.lastSegment.remove(); }, true, 'Removing last segment should be successful.'); equals(function() { return path.curves.length; }, 0, 'After removing the last segment, we should be left with no curves.'); equals(function() { return path.lastSegment.remove(); }, true, 'Removing the final remaining segment should be successful.'); equals(function() { return path.curves.length; }, 0, 'After removing all segments, we should be left with no curves.'); path.addSegment([0, 0]); path.addSegment([1, 1]); equals(function() { return path.curves.length; }, 1, 'After adding two new segments, we should have one curve again.'); path.addSegment([2, 2]); equals(function() { return path.curves.length; }, 2, 'After adding a new segment, we should have two curves again.'); equals(function() { return path.curves[1].segment1 === path.curves[0].segment2; }, true, "The newly created curve's first segment needs to be the same as the previous curve's second segment."); path.addSegments([[3, 3], [4, 4]]); equals(function() { return path.curves.length; }, 4, 'After adding two new segments at the end, we should have four curves now.'); }); test('Splitting a straight path should produce segments without handles', function() { var path1 = new Path.Line([0, 0], [50, 50]); var path2 = path1.split(0, 0.5); equals(function() { return !path1.lastSegment.hasHandles() && !path2.firstSegment.hasHandles(); }, true); }); test('Splitting a path with one curve in the middle result in two paths of the same length with one curve each', function() { var path1 = new Path.Line([0, 0], [100, 100]); var path2 = path1.split(path1.getLocationAt(path1.length / 2)); equals(function() { return path1.curves.length; }, 1); equals(function() { return path2.curves.length; }, 1); equals(function() { return path1.length === path2.length; }, true); }); QUnit.module('Path Drawing Commands'); test('path.lineTo(point);', function() { var path = new Path(); path.moveTo([50, 50]); path.lineTo([100, 100]); equals(path.segments.toString(), '{ point: { x: 50, y: 50 } },{ point: { x: 100, y: 100 } }'); }); test('path.arcTo(from, through, to);', function() { var path = new Path(); path.moveTo([0, 20]); path.arcTo([75, 75], [100, 0]); equals(path.segments.toString(), '{ point: { x: 0, y: 20 }, handleOut: { x: -2.62559, y: 23.01251 } },{ point: { x: 30.89325, y: 74.75812 }, handleIn: { x: -21.05455, y: -9.65273 }, handleOut: { x: 21.05455, y: 9.65273 } },{ point: { x: 92.54397, y: 62.42797 }, handleIn: { x: -15.72238, y: 17.00811 }, handleOut: { x: 15.72238, y: -17.00811 } },{ point: { x: 100, y: 0 }, handleIn: { x: 11.27458, y: 20.23247 } }'); }); test('path.arcTo(from, through, to); where from, through and to all share the same y position and through lies in between from and to', function() { var path = new Path(); path.strokeColor = 'black'; path.add([40, 75]); path.arcTo([50, 75], [100, 75]); equals(path.lastSegment.point.toString(), '{ x: 100, y: 75 }', 'We expect the last segment point to be at the position where we wanted to draw the arc to.'); }); test('path.arcTo(from, through, to); where from, through and to all share the same y position and through lies to the right of to', function() { var path = new Path(); path.strokeColor = 'black'; path.add([40, 75]); var error = null; try { path.arcTo([150, 75], [100, 75]); } catch (e) { error = e; } equals(error != null, true, 'We expect this arcTo() command to throw an error'); }); test('path.arcTo(from, through, to); where from, through and to all share the same y position and through lies to the left of to', function() { var path = new Path(); path.strokeColor = 'black'; path.add([40, 75]); var error = null; try { path.arcTo([10, 75], [100, 75]); } catch (e) { error = e; } equals(error != null, true, 'We expect this arcTo() command to throw an error'); });