paper.js/src/path/Path.Constructors.js
2013-10-16 14:19:25 +02:00

455 lines
13 KiB
JavaScript

/*
* Paper.js - The Swiss Army Knife of Vector Graphics Scripting.
* http://paperjs.org/
*
* Copyright (c) 2011 - 2013, Juerg Lehni & Jonathan Puckey
* http://lehni.org/ & http://jonathanpuckey.com/
*
* Distributed under the MIT license. See LICENSE file for details.
*
* All rights reserved.
*/
Path.inject({ statics: new function() {
function createPath(args) {
return new Path(Base.getNamed(args));
}
var kappa = Numerical.KAPPA,
halfKappa = kappa / 2,
ellipseSegments = [
new Segment([0, 0.5], [0, halfKappa ], [0, -halfKappa]),
new Segment([0.5, 0], [-halfKappa, 0], [halfKappa, 0 ]),
new Segment([1, 0.5], [0, -halfKappa], [0, halfKappa ]),
new Segment([0.5, 1], [halfKappa, 0 ], [-halfKappa, 0])
];
function createEllipse(rect, args) {
var path = createPath(args),
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;
}
return /** @lends Path */{
/**
* {@grouptitle Shaped Paths}
*
* Creates a linear path item from two points describing a line.
*
* @name Path.Line
* @param {Point} from the line's starting point
* @param {Point} to the line's ending point
* @return {Path} the newly created path
*
* @example {@paperscript}
* var from = new Point(20, 20);
* var to = new Point(80, 80);
* var path = new Path.Line(from, to);
* path.strokeColor = 'black';
*/
/**
* Creates a linear path item from the properties described by an object
* literal.
*
* @name Path.Line
* @param {Object} object an object literal containing properties
* describing the path's attributes
* @return {Path} the newly created path
*
* @example {@paperscript}
* var path = new Path.Line({
* from: [20, 20],
* to: [80, 80],
* strokeColor: 'black'
* });
*/
Line: function(/* from, to */) {
return new Path(
Point.readNamed(arguments, 'from'),
Point.readNamed(arguments, 'to')
).set(Base.getNamed(arguments));
},
/**
* Creates a circular path item.
*
* @name Path.Circle
* @param {Point} center the center point of the circle
* @param {Number} radius the radius of the circle
* @return {Path} the newly created path
*
* @example {@paperscript}
* var path = new Path.Circle(new Point(80, 50), 30);
* path.strokeColor = 'black';
*/
/**
* Creates a circular path item from the properties described by an
* object literal.
*
* @name Path.Circle
* @param {Object} object an object literal containing properties
* describing the path's attributes
* @return {Path} the newly created path
*
* @example {@paperscript}
* var path = new Path.Circle({
* center: [80, 50],
* radius: 30,
* strokeColor: 'black'
* });
*/
Circle: function(/* center, radius */) {
var center = Point.readNamed(arguments, 'center'),
radius = Base.readNamed(arguments, 'radius');
return createEllipse(new Rectangle(center.subtract(radius),
new Size(radius * 2, radius * 2)), arguments);
},
/**
* Creates a rectangular path item, with optionally rounded corners.
*
* @name Path.Rectangle
* @param {Rectangle} rectangle the rectangle object describing the
* geometry of the rectangular path to be created.
* @param {Size} [radius=null] the size of the rounded corners
* @return {Path} the newly created path
*
* @example {@paperscript}
* var rectangle = new Rectangle(new Point(20, 20), new Size(60, 60));
* var path = new Path.Rectangle(rectangle);
* path.strokeColor = 'black';
*
* @example {@paperscript} // The same, with rounder corners
* var rectangle = new Rectangle(new Point(20, 20), new Size(60, 60));
* var cornerSize = new Size(10, 10);
* var path = new Path.Rectangle(rectangle, cornerSize);
* path.strokeColor = 'black';
*/
/**
* Creates a rectangular path item from a point and a size object.
*
* @name Path.Rectangle
* @param {Point} point the rectangle's top-left corner.
* @param {Size} size the rectangle's size.
* @return {Path} the newly created path
*
* @example {@paperscript}
* var point = new Point(20, 20);
* var size = new Size(60, 60);
* var path = new Path.Rectangle(point, size);
* path.strokeColor = 'black';
*/
/**
* Creates a rectangular 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
* @return {Path} the newly created path
*
* @example {@paperscript}
* var from = new Point(20, 20);
* var to = new Point(80, 80);
* var path = new Path.Rectangle(from, to);
* path.strokeColor = 'black';
*/
/**
* Creates a rectangular path item from the properties described by an
* object literal.
*
* @name Path.Rectangle
* @param {Object} object an object literal containing properties
* describing the path's attributes
* @return {Path} the newly created path
*
* @example {@paperscript}
* var path = new Path.Rectangle({
* point: [20, 20],
* size: [60, 60],
* strokeColor: 'black'
* });
*
* @example {@paperscript}
* var path = new Path.Rectangle({
* from: [20, 20],
* to: [80, 80],
* strokeColor: 'black'
* });
*
* @example {@paperscript}
* var path = new Path.Rectangle({
* rectangle: {
* topLeft: [20, 20],
* bottomRight: [80, 80]
* },
* strokeColor: 'black'
* });
*
* @example {@paperscript}
* var path = new Path.Rectangle({
* topLeft: [20, 20],
* bottomRight: [80, 80],
* radius: 10,
* strokeColor: 'black'
* });
*/
Rectangle: function(/* rectangle */) {
var rect = Rectangle.readNamed(arguments, 'rectangle'),
radius = Size.readNamed(arguments, 'radius', 0, 0,
{ readNull: true }),
bl = rect.getBottomLeft(true),
tl = rect.getTopLeft(true),
tr = rect.getTopRight(true),
br = rect.getBottomRight(true),
path = createPath(arguments);
if (!radius || radius.isZero()) {
path._add([
new Segment(bl),
new Segment(tl),
new Segment(tr),
new Segment(br)
]);
} else {
radius = Size.min(radius, rect.getSize(true).divide(2));
var rx = radius.width,
ry = radius.height,
hx = rx * kappa,
hy = ry * kappa;
path._add([
new Segment(bl.add(rx, 0), null, [-hx, 0]),
new Segment(bl.subtract(0, ry), [0, hy]),
new Segment(tl.add(0, ry), null, [0, -hy]),
new Segment(tl.add(rx, 0), [-hx, 0], null),
new Segment(tr.subtract(rx, 0), null, [hx, 0]),
new Segment(tr.add(0, ry), [0, -hy], null),
new Segment(br.subtract(0, ry), null, [0, hy]),
new Segment(br.subtract(rx, 0), [hx, 0])
]);
}
// No need to use setter for _closed since _add() called _changed().
path._closed = true;
return path;
},
/**
* @deprecated use {@link #Path.Rectangle(rectangle, size)} instead.
*/
RoundRectangle: '#Rectangle',
/**
* Creates an elliptical path item.
*
* @name Path.Ellipse
* @param {Rectangle} rectangle the rectangle circumscribing the ellipse
* @return {Path} the newly created path
*
* @example {@paperscript}
* var rectangle = new Rectangle(new Point(20, 20), new Size(180, 60));
* var path = new Path.Ellipse(rectangle);
* path.fillColor = 'black';
*/
/**
* Creates an elliptical path item from the properties described by an
* object literal.
*
* @name Path.Ellipse
* @param {Object} object an object literal containing properties
* describing the path's attributes
* @return {Path} the newly created path
*
* @example {@paperscript}
* var path = new Path.Ellipse({
* point: [20, 20],
* size: [180, 60],
* fillColor: 'black'
* });
*/
Ellipse: function(/* rectangle */) {
var rect;
if (Base.hasNamed(arguments, 'center')) {
var center = Point.readNamed(arguments, 'center'),
radius = Size.readNamed(arguments, 'radius');
rect = new Rectangle(center.subtract(radius),
radius.multiply(2));
} else {
rect = Rectangle.readNamed(arguments, 'rectangle');
}
return createEllipse(rect, arguments);
},
/**
* @deprecated use {@link #Path.Ellipse(rectangle)} instead.
*/
Oval: '#Ellipse',
/**
* Creates a circular arc path item.
*
* @name Path.Arc
* @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 {@paperscript}
* var from = new Point(20, 20);
* var through = new Point(60, 20);
* var to = new Point(80, 80);
* var path = new Path.Arc(from, through, to);
* path.strokeColor = 'black';
*
*/
/**
* Creates an circular arc path item from the properties described by an
* object literal.
*
* @name Path.Arc
* @param {Object} object an object literal containing properties
* describing the path's attributes
* @return {Path} the newly created path
*
* @example {@paperscript}
* var path = new Path.Arc({
* from: [20, 20],
* through: [60, 20],
* to: [80, 80],
* strokeColor: 'black'
* });
*/
Arc: function(/* from, through, to */) {
var from = Point.readNamed(arguments, 'from'),
through = Point.readNamed(arguments, 'through'),
to = Point.readNamed(arguments, 'to'),
path = createPath(arguments);
path.moveTo(from);
path.arcTo(through, to);
return path;
},
/**
* Creates a regular polygon shaped path item.
*
* @name Path.RegularPolygon
* @param {Point} center the center point of the polygon
* @param {Number} sides the number of sides of the polygon
* @param {Number} radius the radius of the polygon
* @return {Path} the newly created path
*
* @example {@paperscript}
* var center = new Point(50, 50);
* var sides = 3;
* var radius = 40;
* var triangle = new Path.RegularPolygon(center, sides, radius);
* triangle.fillColor = 'black';
*/
/**
* Creates a regular polygon shaped path item from the properties
* described by an object literal.
*
* @name Path.RegularPolygon
* @param {Object} object an object literal containing properties
* describing the path's attributes
* @return {Path} the newly created path
*
* @example {@paperscript}
* var triangle = new Path.RegularPolygon({
* center: [50, 50],
* sides: 10,
* radius: 40,
* fillColor: 'black'
* });
*/
RegularPolygon: function(/* center, sides, radius */) {
var center = Point.readNamed(arguments, 'center'),
sides = Base.readNamed(arguments, 'sides'),
radius = Base.readNamed(arguments, 'radius'),
path = createPath(arguments),
step = 360 / sides,
three = !(sides % 3),
vector = new Point(0, three ? -radius : radius),
offset = three ? -1 : 0.5,
segments = new Array(sides);
for (var i = 0; i < sides; i++) {
segments[i] = new Segment(center.add(
vector.rotate((i + offset) * step)));
}
path._add(segments);
path._closed = true;
return path;
},
/**
* Creates a star shaped path item.
*
* 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.
*
* @name Path.Star
* @param {Point} center the center point of the star
* @param {Number} points the number of points of the star
* @param {Number} radius1
* @param {Number} radius2
* @return {Path} the newly created path
*
* @example {@paperscript}
* var center = new Point(50, 50);
* var points = 12;
* var radius1 = 25;
* var radius2 = 40;
* var path = new Path.Star(center, points, radius1, radius2);
* path.fillColor = 'black';
*/
/**
* Creates a star shaped path item from the properties described by an
* object literal.
*
* @name Path.Star
* @param {Object} object an object literal containing properties
* describing the path's attributes
* @return {Path} the newly created path
*
* @example {@paperscript}
* var path = new Path.Star({
* center: [50, 50],
* points: 12,
* radius1: 25,
* radius2: 40,
* fillColor: 'black'
* });
*/
Star: function(/* center, points, radius1, radius2 */) {
var center = Point.readNamed(arguments, 'center'),
points = Base.readNamed(arguments, 'points') * 2,
radius1 = Base.readNamed(arguments, 'radius1'),
radius2 = Base.readNamed(arguments, 'radius2'),
path = createPath(arguments),
step = 360 / points,
vector = new Point(0, -1),
segments = new Array(points);
for (var i = 0; i < points; i++) {
segments[i] = new Segment(center.add(
vector.rotate(step * i).multiply(i % 2 ? radius2 : radius1)));
}
path._add(segments);
path._closed = true;
return path;
}
};
}});