paper.js/src/path/Path.Constructors.js

344 lines
10 KiB
JavaScript
Raw Normal View History

2011-03-06 19:50:44 -05:00
/*
* Paper.js
*
2011-03-06 19:50:44 -05:00
* This file is part of Paper.js, a JavaScript Vector Graphics Library,
* based on Scriptographer.org and designed to be largely API compatible.
2011-03-07 20:41:50 -05:00
* http://paperjs.org/
2011-03-06 19:50:44 -05:00
* http://scriptographer.org/
*
2011-03-06 19:50:44 -05:00
* Copyright (c) 2011, Juerg Lehni & Jonathan Puckey
* http://lehni.org/ & http://jonathanpuckey.com/
*
2011-07-01 06:17:45 -04:00
* Distributed under the MIT license. See LICENSE file for details.
*
2011-03-07 20:41:50 -05:00
* All rights reserved.
2011-03-06 19:50:44 -05:00
*/
Path.inject({ statics: new function() {
function readRectangle(list) {
var rect;
if (Base.hasNamed(list, 'from')) {
rect = new Rectangle(Point.readNamed(list, 'from'),
Point.readNamed(list, 'to'));
} else if (Base.hasNamed(list, 'size')) {
rect = new Rectangle(Point.readNamed(list, 'point'),
Size.readNamed(list, 'size'));
if (Base.hasNamed(list, 'center'))
rect.setCenter(Point.readNamed(list, 'center'));
} else {
rect = Rectangle.readNamed(list, 'rectangle');
}
return rect;
}
function createRectangle() {
var rect = readRectangle(arguments),
left = rect.getLeft(),
top = rect.getTop(),
right = rect.getRight(),
bottom = rect.getBottom(),
path = new Path(arguments._filtered);
path._add([
new Segment(Point.create(left, bottom)),
new Segment(Point.create(left, top)),
new Segment(Point.create(right, top)),
new Segment(Point.create(right, bottom))
]);
path._closed = true;
return path;
}
// Kappa, see: http://www.whizkidtech.redprince.net/bezier/circle/kappa/
var kappa = 2 * (Math.sqrt(2) - 1) / 3;
var ellipseSegments = [
new Segment([0, 0.5], [0, kappa ], [0, -kappa]),
new Segment([0.5, 0], [-kappa, 0], [kappa, 0 ]),
new Segment([1, 0.5], [0, -kappa], [0, kappa ]),
new Segment([0.5, 1], [kappa, 0 ], [-kappa, 0])
];
function createEllipse() {
var rect = readRectangle(arguments),
path = new Path(arguments._filtered),
point = rect.getPoint(true),
size = rect.getSize(true),
segments = new Array(4);
for (var i = 0; i < 4; i++) {
var segment = ellipseSegments[i];
segments[i] = new Segment(
segment._point.multiply(size).add(point),
segment._handleIn.multiply(size),
segment._handleOut.multiply(size)
);
}
path._add(segments);
path._closed = true;
return path;
}
2011-06-22 18:58:50 -04:00
return /** @lends Path */{
2011-05-23 08:33:22 -04:00
/**
* {@grouptitle Shaped Paths}
2011-06-27 06:31:39 -04:00
*
2011-05-23 08:33:22 -04:00
* Creates a Path Item with two anchor points forming a line.
*
* @param {Point} from the first anchor point of the path
* @param {Point} to the second anchor point of the path
2011-05-23 08:33:22 -04:00
* @return {Path} the newly created path
*
* @example
* var from = new Point(20, 20);
* var to = new Point(100, 100);
* var path = new Path.Line(from, to);
* path.strokeColor = 'black';
2011-05-23 08:33:22 -04:00
*/
Line: function() {
return new Path(
Point.readNamed(arguments, 'from'),
Point.readNamed(arguments, 'to')
).set(arguments._filtered);
},
2011-05-23 08:33:22 -04:00
/**
* Creates a rectangle shaped Path Item from the passed point and size.
*
* @name Path.Rectangle
2011-05-23 08:33:22 -04:00
* @param {Point} point
* @param {Size} size
* @return {Path} the newly created path
*
* @example
* var point = new Point(100, 100);
* var size = new Size(100, 100);
* var rectangle = new Rectangle(point, size);
* var path = new Path.Rectangle(rectangle);
* path.strokeColor = 'black';
2011-05-23 08:33:22 -04:00
*/
/**
* Creates a rectangle shaped Path Item from the passed points. These do not
* necessarily need to be the top left and bottom right corners, the
* constructor figures out how to fit a rectangle between them.
*
* @name Path.Rectangle
* @param {Point} from The first point defining the rectangle
* @param {Point} to The second point defining the rectangle
2011-05-23 08:33:22 -04:00
* @return {Path} the newly created path
*
* @example
* var point = new Point(100, 100);
* var point2 = new Point(200, 300);
* var path = new Path.Rectangle(point, point2);
* path.strokeColor = 'black';
2011-05-23 08:33:22 -04:00
*/
/**
* Creates a rectangle shaped Path Item from the passed abstract
* {@link Rectangle}.
*
* @param {Rectangle} rectangle
* @return {Path} the newly created path
*
2011-05-23 08:33:22 -04:00
* @example
* var point = new Point(100, 100);
* var size = new Size(100, 100);
* var rectangle = new Rectangle(point, size);
2011-05-23 08:33:22 -04:00
* var path = new Path.Rectangle(rectangle);
* path.strokeColor = 'black';
*/
Rectangle: createRectangle,
2011-05-23 08:33:22 -04:00
/**
* Creates a rectangular Path Item with rounded corners.
*
* @param {Rectangle} rectangle
* @param {Size} radius the size of the rounded corners
2011-05-23 08:33:22 -04:00
* @return {Path} the newly created path
*
* @example
* var point = new Point(100, 100);
* var size = new Size(100, 100);
* var rectangle = new Rectangle(point, size);
* var cornerSize = new Size(30, 30);
* var path = new Path.RoundRectangle(rectangle, cornerSize);
2011-05-23 08:33:22 -04:00
*/
RoundRectangle: function(rect, radius) {
var _rect = Rectangle.readNamed(arguments, 'rectangle'),
_radius = Size.readNamed(arguments, 'radius');
if (_radius.isZero())
return createRectangle(rect);
_radius = Size.min(_radius, _rect.getSize(true).divide(2));
var bl = _rect.getBottomLeft(true),
tl = _rect.getTopLeft(true),
tr = _rect.getTopRight(true),
br = _rect.getBottomRight(true),
h = _radius.multiply(kappa * 2), // handle vector
path = new Path();
path._add([
new Segment(bl.add(_radius.width, 0), null, [-h.width, 0]),
new Segment(bl.subtract(0, _radius.height), [0, h.height], null),
new Segment(tl.add(0, _radius.height), null, [0, -h.height]),
new Segment(tl.add(_radius.width, 0), [-h.width, 0], null),
new Segment(tr.subtract(_radius.width, 0), null, [h.width, 0]),
new Segment(tr.add(0, _radius.height), [0, -h.height], null),
new Segment(br.subtract(0, _radius.height), null, [0, h.height]),
new Segment(br.subtract(_radius.width, 0), [h.width, 0], null)
]);
path._closed = true;
return path;
},
2011-05-23 08:33:22 -04:00
/**
* Creates an ellipse shaped Path Item.
*
* @param {Rectangle} rectangle
* @param {Boolean} [circumscribed=false] when set to {@code true} the
* ellipse shaped path will be created so the rectangle fits into
* it. When set to {@code false} the ellipse path will fit within
* the rectangle.
2011-05-23 08:33:22 -04:00
* @return {Path} the newly created path
*
* @example
* var topLeft = new Point(100, 100);
* var size = new Size(150, 100);
* var rectangle = new Rectangle(topLeft, size);
* var path = new Path.Ellipse(rectangle);
* path.fillColor = 'black';
2011-05-23 08:33:22 -04:00
*/
Ellipse: createEllipse,
/**
* @deprecated use {@link #Path.Ellipse(rect)} instead.
*/
Oval: createEllipse,
2011-05-23 08:33:22 -04:00
/**
* Creates a circle shaped Path Item.
*
2011-05-23 08:33:22 -04:00
* @param {Point} center the center point of the circle
* @param {Number} radius the radius of the circle
2011-05-23 08:33:22 -04:00
* @return {Path} the newly created path
*
* @example
* var path = new Path.Circle(new Point(100, 100), 50);
2011-05-23 08:33:22 -04:00
*/
Circle: function(center, radius) {
var _center = Point.readNamed(arguments, 'center'),
_radius = Base.readNamed(arguments, 'radius');
return createEllipse(new Rectangle(_center.subtract(_radius),
Size.create(_radius * 2, _radius * 2)))
.set(arguments._filtered);
},
2011-05-23 08:33:22 -04:00
/**
* Creates a circular arc shaped Path Item.
*
2011-05-23 08:33:22 -04:00
* @param {Point} from the starting point of the circular arc
* @param {Point} through the point the arc passes through
* @param {Point} to the end point of the arc
* @return {Path} the newly created path
*
* @example
* var start = new Point(0, 0);
* var through = new Point(100, 100);
* var to = new Point(200, 150);
* var path = new Path.Arc(start, through, to);
* path.strokeColor = 'black';
2011-05-23 08:33:22 -04:00
*/
Arc: function(from, through, to) {
var _from = Point.readNamed(arguments, 'from'),
_through = Point.readNamed(arguments, 'through'),
_to = Point.readNamed(arguments, 'to'),
path = new Path(arguments._filtered);
path.moveTo(_from);
path.arcTo(_through, _to);
return path;
},
2011-05-23 08:33:22 -04:00
/**
* Creates a regular polygon shaped Path Item.
*
2011-05-23 08:33:22 -04:00
* @param {Point} center the center point of the polygon
* @param {Number} numSides the number of sides of the polygon
* @param {Number} radius the radius of the polygon
2011-05-23 08:33:22 -04:00
* @return {Path} the newly created path
*
* @example
* // Create a triangle shaped path
* var center = new Point(100, 100);
* var sides = 3;
* var radius = 50;
* var triangle = new Path.RegularPolygon(center, sides, radius);
* triangle.fillColor = 'black';
*
* @example
* // Create a decahedron shaped path
* var center = new Point(100, 100);
* var sides = 10;
* var radius = 50;
* var decahedron = new Path.RegularPolygon(center, sides, radius);
* decahedron.fillColor = 'black';
2011-05-23 08:33:22 -04:00
*/
RegularPolygon: function(center, numSides, radius) {
var _center = Point.readNamed(arguments, 'center'),
_numSides = Base.readNamed(arguments, 'numSides'),
_radius = Base.readNamed(arguments, 'radius'),
path = new Path(arguments._filtered),
step = 360 / _numSides,
three = !(_numSides % 3),
vector = new Point(0, three ? -_radius : _radius),
offset = three ? -1 : 0.5,
segments = new Array(_numSides);
for (var i = 0; i < _numSides; i++) {
segments[i] = new Segment(_center.add(
vector.rotate((i + offset) * step)));
}
path._add(segments);
path._closed = true;
return path;
},
2011-05-23 08:33:22 -04:00
/**
* Creates a star shaped Path Item.
*
2011-05-23 08:33:22 -04:00
* The largest of {@code radius1} and {@code radius2} will be the outer
* radius of the star. The smallest of radius1 and radius2 will be the
* inner radius.
*
2011-05-23 08:33:22 -04:00
* @param {Point} center the center point of the star
* @param {Number} numPoints the number of points of the star
* @param {Number} radius1
* @param {Number} radius2
2011-05-23 08:33:22 -04:00
* @return {Path} the newly created path
*
* @example
* var center = new Point(100, 100);
* var points = 6;
* var radius1 = 20;
* var radius2 = 50;
* var path = new Path.Star(center, points, radius1, radius2);
* path.fillColor = 'black';
*/
Star: function(center, numPoints, radius1, radius2) {
var _center = Point.readNamed(arguments, 'center'),
_numPoints = Base.readNamed(arguments, 'numPoints') * 2,
_radius1 = Base.readNamed(arguments, 'radius1'),
_radius2 = Base.readNamed(arguments, 'radius2'),
path = new Path(arguments._filtered),
step = 360 / _numPoints,
vector = new Point(0, -1),
segments = new Array(_numPoints);
for (var i = 0; i < _numPoints; i++) {
segments[i] = new Segment(_center.add(
vector.rotate(step * i).multiply(i % 2 ? _radius2 : _radius1)));
}
path._add(segments);
path._closed = true;
return path;
}
};
}});