mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-09 14:22:08 -05:00
0894e625b0
Closes #537.
2889 lines
107 KiB
JavaScript
2889 lines
107 KiB
JavaScript
/*
|
|
* Paper.js - The Swiss Army Knife of Vector Graphics Scripting.
|
|
* http://paperjs.org/
|
|
*
|
|
* Copyright (c) 2011 - 2014, Juerg Lehni & Jonathan Puckey
|
|
* http://scratchdisk.com/ & http://jonathanpuckey.com/
|
|
*
|
|
* Distributed under the MIT license. See LICENSE file for details.
|
|
*
|
|
* All rights reserved.
|
|
*/
|
|
|
|
/**
|
|
* @name Path
|
|
*
|
|
* @class The path item represents a path in a Paper.js project.
|
|
*
|
|
* @extends PathItem
|
|
*/
|
|
// DOCS: Explain that path matrix is always applied with each transformation.
|
|
var Path = PathItem.extend(/** @lends Path# */{
|
|
_class: 'Path',
|
|
_serializeFields: {
|
|
segments: [],
|
|
closed: false
|
|
},
|
|
|
|
/**
|
|
* Creates a new path item and places it at the top of the active layer.
|
|
*
|
|
* @name Path#initialize
|
|
* @param {Segment[]} [segments] An array of segments (or points to be
|
|
* converted to segments) that will be added to the path
|
|
* @return {Path} the newly created path
|
|
*
|
|
* @example
|
|
* // Create an empty path and add segments to it:
|
|
* var path = new Path();
|
|
* path.strokeColor = 'black';
|
|
* path.add(new Point(30, 30));
|
|
* path.add(new Point(100, 100));
|
|
*
|
|
* @example
|
|
* // Create a path with two segments:
|
|
* var segments = [new Point(30, 30), new Point(100, 100)];
|
|
* var path = new Path(segments);
|
|
* path.strokeColor = 'black';
|
|
*/
|
|
/**
|
|
* Creates a new path item from an object description and places it at the
|
|
* top of the active layer.
|
|
*
|
|
* @name Path#initialize
|
|
* @param {Object} object an object literal containing properties to
|
|
* be set on the path
|
|
* @return {Path} the newly created path
|
|
*
|
|
* @example {@paperscript}
|
|
* var path = new Path({
|
|
* segments: [[20, 20], [80, 80], [140, 20]],
|
|
* fillColor: 'black',
|
|
* closed: true
|
|
* });
|
|
*
|
|
* @example {@paperscript}
|
|
* var path = new Path({
|
|
* segments: [[20, 20], [80, 80], [140, 20]],
|
|
* strokeColor: 'red',
|
|
* strokeWidth: 20,
|
|
* strokeCap: 'round',
|
|
* selected: true
|
|
* });
|
|
*/
|
|
/**
|
|
* Creates a new path item from SVG path-data and places it at the top of
|
|
* the active layer.
|
|
*
|
|
* @name Path#initialize
|
|
* @param {String} pathData the SVG path-data that describes the geometry
|
|
* of this path.
|
|
* @return {Path} the newly created path
|
|
*
|
|
* @example {@paperscript}
|
|
* var pathData = 'M100,50c0,27.614-22.386,50-50,50S0,77.614,0,50S22.386,0,50,0S100,22.386,100,50';
|
|
* var path = new Path(pathData);
|
|
* path.fillColor = 'red';
|
|
*/
|
|
initialize: function Path(arg) {
|
|
this._closed = false;
|
|
this._segments = [];
|
|
// arg can either be an object literal containing properties to be set
|
|
// on the path, a list of segments to be set, or the first of multiple
|
|
// arguments describing separate segments.
|
|
// If it is an array, it can also be a description of a point, so
|
|
// check its first entry for object as well.
|
|
// But first see if segments are directly passed at all. If not, try
|
|
// _set(arg).
|
|
var segments = Array.isArray(arg)
|
|
? typeof arg[0] === 'object'
|
|
? arg
|
|
: arguments
|
|
// See if it behaves like a segment or a point, but filter out
|
|
// rectangles, as accepted by some Path.Constructor constructors.
|
|
: arg && (arg.size === undefined && (arg.x !== undefined
|
|
|| arg.point !== undefined))
|
|
? arguments
|
|
: null;
|
|
// Always call setSegments() to initialize a few related variables.
|
|
if (segments && segments.length > 0) {
|
|
// This sets _curves and _selectedSegmentState too!
|
|
this.setSegments(segments);
|
|
} else {
|
|
this._curves = undefined; // For hidden class optimization
|
|
this._selectedSegmentState = 0;
|
|
if (!segments && typeof arg === 'string') {
|
|
this.setPathData(arg);
|
|
// Erase for _initialize() call below.
|
|
arg = null;
|
|
}
|
|
}
|
|
// Only pass on arg as props if it wasn't consumed for segments already.
|
|
this._initialize(!segments && arg);
|
|
},
|
|
|
|
_equals: function(item) {
|
|
return Base.equals(this._segments, item._segments);
|
|
},
|
|
|
|
clone: function(insert) {
|
|
var copy = new Path(Item.NO_INSERT);
|
|
copy.setSegments(this._segments);
|
|
copy._closed = this._closed;
|
|
if (this._clockwise !== undefined)
|
|
copy._clockwise = this._clockwise;
|
|
return this._clone(copy, insert);
|
|
},
|
|
|
|
_changed: function _changed(flags) {
|
|
_changed.base.call(this, flags);
|
|
if (flags & /*#=*/ChangeFlag.GEOMETRY) {
|
|
// The _currentPath is already cleared in Item, but clear it on the
|
|
// parent too, for children of CompoundPaths, and Groups (ab)used as
|
|
// clipping paths.
|
|
var parent = this._parent;
|
|
if (parent)
|
|
parent._currentPath = undefined;
|
|
// Clockwise state becomes undefined as soon as geometry changes.
|
|
this._length = this._clockwise = undefined;
|
|
// Only notify all curves if we're not told that only one Segment
|
|
// has changed and took already care of notifications.
|
|
if (this._curves && !(flags & /*#=*/ChangeFlag.SEGMENTS)) {
|
|
for (var i = 0, l = this._curves.length; i < l; i++)
|
|
this._curves[i]._changed();
|
|
}
|
|
// Clear cached curves used for winding direction and containment
|
|
// calculation.
|
|
// NOTE: This is only needed with __options.booleanOperations
|
|
this._monoCurves = undefined;
|
|
} else if (flags & /*#=*/ChangeFlag.STROKE) {
|
|
// TODO: We could preserve the purely geometric bounds that are not
|
|
// affected by stroke: _bounds.bounds and _bounds.handleBounds
|
|
this._bounds = undefined;
|
|
}
|
|
},
|
|
|
|
getStyle: function() {
|
|
// If this path is part of a compound-path, return the parent's style.
|
|
var parent = this._parent;
|
|
return (parent instanceof CompoundPath ? parent : this)._style;
|
|
},
|
|
|
|
/**
|
|
* The segments contained within the path.
|
|
*
|
|
* @type Segment[]
|
|
* @bean
|
|
*/
|
|
getSegments: function() {
|
|
return this._segments;
|
|
},
|
|
|
|
setSegments: function(segments) {
|
|
var fullySelected = this.isFullySelected();
|
|
this._segments.length = 0;
|
|
this._selectedSegmentState = 0;
|
|
// Calculate new curves next time we call getCurves()
|
|
this._curves = undefined;
|
|
if (segments && segments.length > 0)
|
|
this._add(Segment.readAll(segments));
|
|
// Preserve fullySelected state.
|
|
// TODO: Do we still need this?
|
|
if (fullySelected)
|
|
this.setFullySelected(true);
|
|
},
|
|
|
|
/**
|
|
* The first Segment contained within the path.
|
|
*
|
|
* @type Segment
|
|
* @bean
|
|
*/
|
|
getFirstSegment: function() {
|
|
return this._segments[0];
|
|
},
|
|
|
|
/**
|
|
* The last Segment contained within the path.
|
|
*
|
|
* @type Segment
|
|
* @bean
|
|
*/
|
|
getLastSegment: function() {
|
|
return this._segments[this._segments.length - 1];
|
|
},
|
|
|
|
/**
|
|
* The curves contained within the path.
|
|
*
|
|
* @type Curve[]
|
|
* @bean
|
|
*/
|
|
getCurves: function() {
|
|
var curves = this._curves,
|
|
segments = this._segments;
|
|
if (!curves) {
|
|
var length = this._countCurves();
|
|
curves = this._curves = new Array(length);
|
|
for (var i = 0; i < length; i++)
|
|
curves[i] = new Curve(this, segments[i],
|
|
// Use first segment for segment2 of closing curve
|
|
segments[i + 1] || segments[0]);
|
|
}
|
|
return curves;
|
|
},
|
|
|
|
/**
|
|
* The first Curve contained within the path.
|
|
*
|
|
* @type Curve
|
|
* @bean
|
|
*/
|
|
getFirstCurve: function() {
|
|
return this.getCurves()[0];
|
|
},
|
|
|
|
/**
|
|
* The last Curve contained within the path.
|
|
*
|
|
* @type Curve
|
|
* @bean
|
|
*/
|
|
getLastCurve: function() {
|
|
var curves = this.getCurves();
|
|
return curves[curves.length - 1];
|
|
},
|
|
|
|
/**
|
|
* Specifies whether the path is closed. If it is closed, Paper.js connects
|
|
* the first and last segments.
|
|
*
|
|
* @type Boolean
|
|
* @bean
|
|
*
|
|
* @example {@paperscript}
|
|
* var myPath = new Path();
|
|
* myPath.strokeColor = 'black';
|
|
* myPath.add(new Point(50, 75));
|
|
* myPath.add(new Point(100, 25));
|
|
* myPath.add(new Point(150, 75));
|
|
*
|
|
* // Close the path:
|
|
* myPath.closed = true;
|
|
*/
|
|
isClosed: function() {
|
|
return this._closed;
|
|
},
|
|
|
|
setClosed: function(closed) {
|
|
// On-the-fly conversion to boolean:
|
|
if (this._closed != (closed = !!closed)) {
|
|
this._closed = closed;
|
|
// Update _curves length
|
|
if (this._curves) {
|
|
var length = this._curves.length = this._countCurves();
|
|
// If we were closing this path, we need to add a new curve now
|
|
if (closed)
|
|
this._curves[length - 1] = new Curve(this,
|
|
this._segments[length - 1], this._segments[0]);
|
|
}
|
|
// Use SEGMENTS notification instead of GEOMETRY since curves are
|
|
// up-to-date and don't need notification.
|
|
this._changed(/*#=*/Change.SEGMENTS);
|
|
}
|
|
}
|
|
}, /** @lends Path# */{
|
|
// Enforce bean creation for getPathData(), as it has hidden parameters.
|
|
beans: true,
|
|
|
|
getPathData: function(_matrix, _precision) {
|
|
// NOTE: #setPathData() is defined in PathItem.
|
|
var segments = this._segments,
|
|
length = segments.length,
|
|
f = new Formatter(_precision),
|
|
coords = new Array(6),
|
|
first = true,
|
|
curX, curY,
|
|
prevX, prevY,
|
|
inX, inY,
|
|
outX, outY,
|
|
parts = [];
|
|
|
|
function addSegment(segment, skipLine) {
|
|
segment._transformCoordinates(_matrix, coords, false);
|
|
curX = coords[0];
|
|
curY = coords[1];
|
|
if (first) {
|
|
parts.push('M' + f.pair(curX, curY));
|
|
first = false;
|
|
} else {
|
|
inX = coords[2];
|
|
inY = coords[3];
|
|
// TODO: Add support for H/V and/or relative commands, where
|
|
// appropriate and resulting in shorter strings.
|
|
if (inX === curX && inY === curY
|
|
&& outX === prevX && outY === prevY) {
|
|
// l = relative lineto:
|
|
if (!skipLine)
|
|
parts.push('l' + f.pair(curX - prevX, curY - prevY));
|
|
} else {
|
|
// c = relative curveto:
|
|
parts.push('c' + f.pair(outX - prevX, outY - prevY)
|
|
+ ' ' + f.pair(inX - prevX, inY - prevY)
|
|
+ ' ' + f.pair(curX - prevX, curY - prevY));
|
|
}
|
|
}
|
|
prevX = curX;
|
|
prevY = curY;
|
|
outX = coords[4];
|
|
outY = coords[5];
|
|
}
|
|
|
|
if (length === 0)
|
|
return '';
|
|
|
|
for (var i = 0; i < length; i++)
|
|
addSegment(segments[i]);
|
|
// Close path by drawing first segment again
|
|
if (this._closed && length > 0) {
|
|
addSegment(segments[0], true);
|
|
parts.push('z');
|
|
}
|
|
return parts.join('');
|
|
}
|
|
}, /** @lends Path# */{
|
|
|
|
// TODO: Consider adding getSubPath(a, b), returning a part of the current
|
|
// path, with the added benefit that b can be < a, and closed looping is
|
|
// taken into account.
|
|
|
|
isEmpty: function() {
|
|
return this._segments.length === 0;
|
|
},
|
|
|
|
isPolygon: function() {
|
|
for (var i = 0, l = this._segments.length; i < l; i++) {
|
|
if (!this._segments[i].isLinear())
|
|
return false;
|
|
}
|
|
return true;
|
|
},
|
|
|
|
_transformContent: function(matrix) {
|
|
var coords = new Array(6);
|
|
for (var i = 0, l = this._segments.length; i < l; i++)
|
|
this._segments[i]._transformCoordinates(matrix, coords, true);
|
|
return true;
|
|
},
|
|
|
|
/**
|
|
* Private method that adds a segment to the segment list. It assumes that
|
|
* the passed object is a segment already and does not perform any checks.
|
|
* If a curves list was requested, it will kept in sync with the segments
|
|
* list automatically.
|
|
*/
|
|
_add: function(segs, index) {
|
|
// Local short-cuts:
|
|
var segments = this._segments,
|
|
curves = this._curves,
|
|
amount = segs.length,
|
|
append = index == null,
|
|
index = append ? segments.length : index;
|
|
// Scan through segments to add first, convert if necessary and set
|
|
// _path and _index references on them.
|
|
for (var i = 0; i < amount; i++) {
|
|
var segment = segs[i];
|
|
// If the segments belong to another path already, clone them before
|
|
// adding:
|
|
if (segment._path)
|
|
segment = segs[i] = segment.clone();
|
|
segment._path = this;
|
|
segment._index = index + i;
|
|
// If parts of this segment are selected, adjust the internal
|
|
// _selectedSegmentState now
|
|
if (segment._selectionState)
|
|
this._updateSelection(segment, 0, segment._selectionState);
|
|
}
|
|
if (append) {
|
|
// Append them all at the end by using push
|
|
segments.push.apply(segments, segs);
|
|
} else {
|
|
// Insert somewhere else
|
|
segments.splice.apply(segments, [index, 0].concat(segs));
|
|
// Adjust the indices of the segments above.
|
|
for (var i = index + amount, l = segments.length; i < l; i++)
|
|
segments[i]._index = i;
|
|
}
|
|
// Keep the curves list in sync all the time in case it was requested
|
|
// already.
|
|
if (curves || segs._curves) {
|
|
if (!curves)
|
|
curves = this._curves = [];
|
|
// We need to step one index down from the inserted segment to
|
|
// get its curve, except for the first segment.
|
|
var from = index > 0 ? index - 1 : index,
|
|
start = from,
|
|
to = Math.min(from + amount, this._countCurves());
|
|
if (segs._curves) {
|
|
// Reuse removed curves.
|
|
curves.splice.apply(curves, [from, 0].concat(segs._curves));
|
|
start += segs._curves.length;
|
|
}
|
|
// Insert new curves, but do not initialize their segments yet,
|
|
// since #_adjustCurves() handles all that for us.
|
|
for (var i = start; i < to; i++)
|
|
curves.splice(i, 0, new Curve(this, null, null));
|
|
// Adjust segments for the curves before and after the removed ones
|
|
this._adjustCurves(from, to);
|
|
}
|
|
// Use SEGMENTS notification instead of GEOMETRY since curves are kept
|
|
// up-to-date by _adjustCurves() and don't need notification.
|
|
this._changed(/*#=*/Change.SEGMENTS);
|
|
return segs;
|
|
},
|
|
|
|
/**
|
|
* Adjusts segments of curves before and after inserted / removed segments.
|
|
*/
|
|
_adjustCurves: function(from, to) {
|
|
var segments = this._segments,
|
|
curves = this._curves,
|
|
curve;
|
|
for (var i = from; i < to; i++) {
|
|
curve = curves[i];
|
|
curve._path = this;
|
|
curve._segment1 = segments[i];
|
|
curve._segment2 = segments[i + 1] || segments[0];
|
|
curve._changed();
|
|
}
|
|
// If it's the first segment, correct the last segment of closed
|
|
// paths too:
|
|
if (curve = curves[this._closed && from === 0 ? segments.length - 1
|
|
: from - 1]) {
|
|
curve._segment2 = segments[from] || segments[0];
|
|
curve._changed();
|
|
}
|
|
// Fix the segment after the modified range, if it exists
|
|
if (curve = curves[to]) {
|
|
curve._segment1 = segments[to];
|
|
curve._changed();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Returns the amount of curves this path item is supposed to have, based
|
|
* on its amount of #segments and #closed state.
|
|
*/
|
|
_countCurves: function() {
|
|
var length = this._segments.length;
|
|
// Reduce length by one if it's an open path:
|
|
return !this._closed && length > 0 ? length - 1 : length;
|
|
},
|
|
|
|
// DOCS: find a way to document the variable segment parameters of Path#add
|
|
/**
|
|
* Adds one or more segments to the end of the {@link #segments} array of
|
|
* this path.
|
|
*
|
|
* @param {Segment|Point} segment the segment or point to be added.
|
|
* @return {Segment} the added segment. This is not necessarily the same
|
|
* object, e.g. if the segment to be added already belongs to another path.
|
|
*
|
|
* @example {@paperscript}
|
|
* // Adding segments to a path using point objects:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Add a segment at {x: 30, y: 75}
|
|
* path.add(new Point(30, 75));
|
|
*
|
|
* // Add two segments in one go at {x: 100, y: 20}
|
|
* // and {x: 170, y: 75}:
|
|
* path.add(new Point(100, 20), new Point(170, 75));
|
|
*
|
|
* @example {@paperscript}
|
|
* // Adding segments to a path using arrays containing number pairs:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Add a segment at {x: 30, y: 75}
|
|
* path.add([30, 75]);
|
|
*
|
|
* // Add two segments in one go at {x: 100, y: 20}
|
|
* // and {x: 170, y: 75}:
|
|
* path.add([100, 20], [170, 75]);
|
|
*
|
|
* @example {@paperscript}
|
|
* // Adding segments to a path using objects:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Add a segment at {x: 30, y: 75}
|
|
* path.add({x: 30, y: 75});
|
|
*
|
|
* // Add two segments in one go at {x: 100, y: 20}
|
|
* // and {x: 170, y: 75}:
|
|
* path.add({x: 100, y: 20}, {x: 170, y: 75});
|
|
*
|
|
* @example {@paperscript}
|
|
* // Adding a segment with handles to a path:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* path.add(new Point(30, 75));
|
|
*
|
|
* // Add a segment with handles:
|
|
* var point = new Point(100, 20);
|
|
* var handleIn = new Point(-50, 0);
|
|
* var handleOut = new Point(50, 0);
|
|
* var added = path.add(new Segment(point, handleIn, handleOut));
|
|
*
|
|
* // Select the added segment, so we can see its handles:
|
|
* added.selected = true;
|
|
*
|
|
* path.add(new Point(170, 75));
|
|
*/
|
|
add: function(segment1 /*, segment2, ... */) {
|
|
return arguments.length > 1 && typeof segment1 !== 'number'
|
|
// addSegments
|
|
? this._add(Segment.readAll(arguments))
|
|
// addSegment
|
|
: this._add([ Segment.read(arguments) ])[0];
|
|
},
|
|
|
|
/**
|
|
* Inserts one or more segments at a given index in the list of this path's
|
|
* segments.
|
|
*
|
|
* @param {Number} index the index at which to insert the segment.
|
|
* @param {Segment|Point} segment the segment or point to be inserted.
|
|
* @return {Segment} the added segment. This is not necessarily the same
|
|
* object, e.g. if the segment to be added already belongs to another path.
|
|
*
|
|
* @example {@paperscript}
|
|
* // Inserting a segment:
|
|
* var myPath = new Path();
|
|
* myPath.strokeColor = 'black';
|
|
* myPath.add(new Point(50, 75));
|
|
* myPath.add(new Point(150, 75));
|
|
*
|
|
* // Insert a new segment into myPath at index 1:
|
|
* myPath.insert(1, new Point(100, 25));
|
|
*
|
|
* // Select the segment which we just inserted:
|
|
* myPath.segments[1].selected = true;
|
|
*
|
|
* @example {@paperscript}
|
|
* // Inserting multiple segments:
|
|
* var myPath = new Path();
|
|
* myPath.strokeColor = 'black';
|
|
* myPath.add(new Point(50, 75));
|
|
* myPath.add(new Point(150, 75));
|
|
*
|
|
* // Insert two segments into myPath at index 1:
|
|
* myPath.insert(1, [80, 25], [120, 25]);
|
|
*
|
|
* // Select the segments which we just inserted:
|
|
* myPath.segments[1].selected = true;
|
|
* myPath.segments[2].selected = true;
|
|
*/
|
|
insert: function(index, segment1 /*, segment2, ... */) {
|
|
return arguments.length > 2 && typeof segment1 !== 'number'
|
|
// insertSegments
|
|
? this._add(Segment.readAll(arguments, 1), index)
|
|
// insertSegment
|
|
: this._add([ Segment.read(arguments, 1) ], index)[0];
|
|
},
|
|
|
|
addSegment: function(/* segment */) {
|
|
return this._add([ Segment.read(arguments) ])[0];
|
|
},
|
|
|
|
insertSegment: function(index /*, segment */) {
|
|
return this._add([ Segment.read(arguments, 1) ], index)[0];
|
|
},
|
|
|
|
/**
|
|
* Adds an array of segments (or types that can be converted to segments)
|
|
* to the end of the {@link #segments} array.
|
|
*
|
|
* @param {Segment[]} segments
|
|
* @return {Segment[]} an array of the added segments. These segments are
|
|
* not necessarily the same objects, e.g. if the segment to be added already
|
|
* belongs to another path.
|
|
*
|
|
* @example {@paperscript}
|
|
* // Adding an array of Point objects:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
* var points = [new Point(30, 50), new Point(170, 50)];
|
|
* path.addSegments(points);
|
|
*
|
|
* @example {@paperscript}
|
|
* // Adding an array of [x, y] arrays:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
* var array = [[30, 75], [100, 20], [170, 75]];
|
|
* path.addSegments(array);
|
|
*
|
|
* @example {@paperscript}
|
|
* // Adding segments from one path to another:
|
|
*
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
* path.addSegments([[30, 75], [100, 20], [170, 75]]);
|
|
*
|
|
* var path2 = new Path();
|
|
* path2.strokeColor = 'red';
|
|
*
|
|
* // Add the second and third segments of path to path2:
|
|
* path2.add(path.segments[1], path.segments[2]);
|
|
*
|
|
* // Move path2 30pt to the right:
|
|
* path2.position.x += 30;
|
|
*/
|
|
addSegments: function(segments) {
|
|
return this._add(Segment.readAll(segments));
|
|
},
|
|
|
|
/**
|
|
* Inserts an array of segments at a given index in the path's
|
|
* {@link #segments} array.
|
|
*
|
|
* @param {Number} index the index at which to insert the segments.
|
|
* @param {Segment[]} segments the segments to be inserted.
|
|
* @return {Segment[]} an array of the added segments. These segments are
|
|
* not necessarily the same objects, e.g. if the segment to be added already
|
|
* belongs to another path.
|
|
*/
|
|
insertSegments: function(index, segments) {
|
|
return this._add(Segment.readAll(segments), index);
|
|
},
|
|
|
|
/**
|
|
* Removes the segment at the specified index of the path's
|
|
* {@link #segments} array.
|
|
*
|
|
* @param {Number} index the index of the segment to be removed
|
|
* @return {Segment} the removed segment
|
|
*
|
|
* @example {@paperscript}
|
|
* // Removing a segment from a path:
|
|
*
|
|
* // Create a circle shaped path at { x: 80, y: 50 }
|
|
* // with a radius of 35:
|
|
* var path = new Path.Circle({
|
|
* center: new Point(80, 50),
|
|
* radius: 35,
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Remove its second segment:
|
|
* path.removeSegment(1);
|
|
*
|
|
* // Select the path, so we can see its segments:
|
|
* path.selected = true;
|
|
*/
|
|
removeSegment: function(index) {
|
|
return this.removeSegments(index, index + 1)[0] || null;
|
|
},
|
|
|
|
/**
|
|
* Removes all segments from the path's {@link #segments} array.
|
|
*
|
|
* @name Path#removeSegments
|
|
* @alias Path#clear
|
|
* @function
|
|
* @return {Segment[]} an array containing the removed segments
|
|
*/
|
|
/**
|
|
* Removes the segments from the specified {@code from} index to the
|
|
* {@code to} index from the path's {@link #segments} array.
|
|
*
|
|
* @param {Number} from the beginning index, inclusive
|
|
* @param {Number} [to=segments.length] the ending index, exclusive
|
|
* @return {Segment[]} an array containing the removed segments
|
|
*
|
|
* @example {@paperscript}
|
|
* // Removing segments from a path:
|
|
*
|
|
* // Create a circle shaped path at { x: 80, y: 50 }
|
|
* // with a radius of 35:
|
|
* var path = new Path.Circle({
|
|
* center: new Point(80, 50),
|
|
* radius: 35,
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Remove the segments from index 1 till index 2:
|
|
* path.removeSegments(1, 2);
|
|
*
|
|
* // Select the path, so we can see its segments:
|
|
* path.selected = true;
|
|
*/
|
|
removeSegments: function(from, to, _includeCurves) {
|
|
from = from || 0;
|
|
to = Base.pick(to, this._segments.length);
|
|
var segments = this._segments,
|
|
curves = this._curves,
|
|
count = segments.length, // segment count before removal
|
|
removed = segments.splice(from, to - from),
|
|
amount = removed.length;
|
|
if (!amount)
|
|
return removed;
|
|
// Update selection state accordingly
|
|
for (var i = 0; i < amount; i++) {
|
|
var segment = removed[i];
|
|
if (segment._selectionState)
|
|
this._updateSelection(segment, segment._selectionState, 0);
|
|
// Clear the indices and path references of the removed segments
|
|
segment._index = segment._path = null;
|
|
}
|
|
// Adjust the indices of the segments above.
|
|
for (var i = from, l = segments.length; i < l; i++)
|
|
segments[i]._index = i;
|
|
// Keep curves in sync
|
|
if (curves) {
|
|
// If we're removing the last segment, remove the last curve (the
|
|
// one to the left of the segment, not to the right, as normally).
|
|
// Also take into account closed paths, which have one curve more
|
|
// than segments.
|
|
var index = from > 0 && to === count + (this._closed ? 1 : 0)
|
|
? from - 1
|
|
: from,
|
|
curves = curves.splice(index, amount);
|
|
// Return the removed curves as well, if we're asked to include
|
|
// them, but exclude the first curve, since that's shared with the
|
|
// previous segment and does not connect the returned segments.
|
|
if (_includeCurves)
|
|
removed._curves = curves.slice(1);
|
|
// Adjust segments for the curves before and after the removed ones
|
|
this._adjustCurves(index, index);
|
|
}
|
|
// Use SEGMENTS notification instead of GEOMETRY since curves are kept
|
|
// up-to-date by _adjustCurves() and don't need notification.
|
|
this._changed(/*#=*/Change.SEGMENTS);
|
|
return removed;
|
|
},
|
|
|
|
// DOCS Path#clear()
|
|
clear: '#removeSegments',
|
|
|
|
/**
|
|
* The approximate length of the path in points.
|
|
*
|
|
* @type Number
|
|
* @bean
|
|
*/
|
|
getLength: function() {
|
|
if (this._length == null) {
|
|
var curves = this.getCurves();
|
|
this._length = 0;
|
|
for (var i = 0, l = curves.length; i < l; i++)
|
|
this._length += curves[i].getLength();
|
|
}
|
|
return this._length;
|
|
},
|
|
|
|
/**
|
|
* The area of the path in square points. Self-intersecting paths can
|
|
* contain sub-areas that cancel each other out.
|
|
*
|
|
* @type Number
|
|
* @bean
|
|
*/
|
|
getArea: function() {
|
|
var curves = this.getCurves();
|
|
var area = 0;
|
|
for (var i = 0, l = curves.length; i < l; i++)
|
|
area += curves[i].getArea();
|
|
return area;
|
|
},
|
|
|
|
/**
|
|
* Specifies whether an path is selected and will also return {@code true}
|
|
* if the path is partially selected, i.e. one or more of its segments is
|
|
* selected.
|
|
*
|
|
* Paper.js draws the visual outlines of selected items on top of your
|
|
* project. This can be useful for debugging, as it allows you to see the
|
|
* construction of paths, position of path curves, individual segment points
|
|
* and bounding boxes of symbol and raster items.
|
|
*
|
|
* @type Boolean
|
|
* @bean
|
|
* @see Project#selectedItems
|
|
* @see Segment#selected
|
|
* @see Point#selected
|
|
*
|
|
* @example {@paperscript}
|
|
* // Selecting an item:
|
|
* var path = new Path.Circle({
|
|
* center: new Size(80, 50),
|
|
* radius: 35
|
|
* });
|
|
* path.selected = true; // Select the path
|
|
*
|
|
* @example {@paperscript}
|
|
* // A path is selected, if one or more of its segments is selected:
|
|
* var path = new Path.Circle({
|
|
* center: new Size(80, 50),
|
|
* radius: 35
|
|
* });
|
|
*
|
|
* // Select the second segment of the path:
|
|
* path.segments[1].selected = true;
|
|
*
|
|
* // If the path is selected (which it is), set its fill color to red:
|
|
* if (path.selected) {
|
|
* path.fillColor = 'red';
|
|
* }
|
|
*
|
|
*/
|
|
/**
|
|
* Specifies whether the path and all its segments are selected. Cannot be
|
|
* {@code true} on an empty path.
|
|
*
|
|
* @type Boolean
|
|
* @bean
|
|
*
|
|
* @example {@paperscript}
|
|
* // A path is fully selected, if all of its segments are selected:
|
|
* var path = new Path.Circle({
|
|
* center: new Size(80, 50),
|
|
* radius: 35
|
|
* });
|
|
* path.fullySelected = true;
|
|
*
|
|
* var path2 = new Path.Circle({
|
|
* center: new Size(180, 50),
|
|
* radius: 35
|
|
* });
|
|
*
|
|
* // Deselect the second segment of the second path:
|
|
* path2.segments[1].selected = false;
|
|
*
|
|
* // If the path is fully selected (which it is),
|
|
* // set its fill color to red:
|
|
* if (path.fullySelected) {
|
|
* path.fillColor = 'red';
|
|
* }
|
|
*
|
|
* // If the second path is fully selected (which it isn't, since we just
|
|
* // deselected its second segment),
|
|
* // set its fill color to red:
|
|
* if (path2.fullySelected) {
|
|
* path2.fillColor = 'red';
|
|
* }
|
|
*/
|
|
isFullySelected: function() {
|
|
var length = this._segments.length;
|
|
return this._selected && length > 0 && this._selectedSegmentState
|
|
=== length * /*#=*/SelectionState.SEGMENT;
|
|
},
|
|
|
|
setFullySelected: function(selected) {
|
|
// No need to call _selectSegments() when selected is false, since
|
|
// #setSelected() does that for us
|
|
if (selected)
|
|
this._selectSegments(true);
|
|
this.setSelected(selected);
|
|
},
|
|
|
|
setSelected: function setSelected(selected) {
|
|
// Deselect all segments when path is marked as not selected
|
|
if (!selected)
|
|
this._selectSegments(false);
|
|
// No need to pass true for noChildren since Path has none anyway.
|
|
setSelected.base.call(this, selected);
|
|
},
|
|
|
|
_selectSegments: function(selected) {
|
|
var length = this._segments.length;
|
|
this._selectedSegmentState = selected
|
|
? length * /*#=*/SelectionState.SEGMENT : 0;
|
|
for (var i = 0; i < length; i++)
|
|
this._segments[i]._selectionState = selected
|
|
? /*#=*/SelectionState.SEGMENT : 0;
|
|
},
|
|
|
|
_updateSelection: function(segment, oldState, newState) {
|
|
segment._selectionState = newState;
|
|
var total = this._selectedSegmentState += newState - oldState;
|
|
// Set this path as selected in case we have selected segments. Do not
|
|
// unselect if we're down to 0, as the path itself can still remain
|
|
// selected even when empty.
|
|
if (total > 0)
|
|
this.setSelected(true);
|
|
},
|
|
|
|
/**
|
|
* Converts the curves in a path to straight lines with an even distribution
|
|
* of points. The distance between the produced segments is as close as
|
|
* possible to the value specified by the {@code maxDistance} parameter.
|
|
*
|
|
* @param {Number} maxDistance the maximum distance between the points
|
|
*
|
|
* @example {@paperscript}
|
|
* // Flattening a circle shaped path:
|
|
*
|
|
* // Create a circle shaped path at { x: 80, y: 50 }
|
|
* // with a radius of 35:
|
|
* var path = new Path.Circle({
|
|
* center: new Size(80, 50),
|
|
* radius: 35
|
|
* });
|
|
*
|
|
* // Select the path, so we can inspect its segments:
|
|
* path.selected = true;
|
|
*
|
|
* // Create a copy of the path and move it 150 points to the right:
|
|
* var copy = path.clone();
|
|
* copy.position.x += 150;
|
|
*
|
|
* // Convert its curves to points, with a max distance of 20:
|
|
* copy.flatten(20);
|
|
*/
|
|
flatten: function(maxDistance) {
|
|
var iterator = new PathIterator(this, 64, 0.1),
|
|
pos = 0,
|
|
// Adapt step = maxDistance so the points distribute evenly.
|
|
step = iterator.length / Math.ceil(iterator.length / maxDistance),
|
|
// Add/remove half of step to end, so imprecisions are ok too.
|
|
// For closed paths, remove it, because we don't want to add last
|
|
// segment again
|
|
end = iterator.length + (this._closed ? -step : step) / 2;
|
|
// Iterate over path and evaluate and add points at given offsets
|
|
var segments = [];
|
|
while (pos <= end) {
|
|
segments.push(new Segment(iterator.evaluate(pos, 0)));
|
|
pos += step;
|
|
}
|
|
this.setSegments(segments);
|
|
},
|
|
|
|
/**
|
|
* Reduces the path by removing curves that have a lenght of 0.
|
|
*/
|
|
reduce: function() {
|
|
var curves = this.getCurves();
|
|
for (var i = curves.length - 1; i >= 0; i--) {
|
|
var curve = curves[i];
|
|
if (curve.isLinear() && curve.getLength() === 0)
|
|
curve.remove();
|
|
}
|
|
return this;
|
|
},
|
|
|
|
/**
|
|
* Smooths a path by simplifying it. The {@link Path#segments} array is
|
|
* analyzed and replaced by a more optimal set of segments, reducing memory
|
|
* usage and speeding up drawing.
|
|
*
|
|
* @param {Number} [tolerance=2.5]
|
|
*
|
|
* @example {@paperscript height=300}
|
|
* // Click and drag below to draw to draw a line, when you release the
|
|
* // mouse, the is made smooth using path.simplify():
|
|
*
|
|
* var path;
|
|
* function onMouseDown(event) {
|
|
* // If we already made a path before, deselect it:
|
|
* if (path) {
|
|
* path.selected = false;
|
|
* }
|
|
*
|
|
* // Create a new path and add the position of the mouse
|
|
* // as its first segment. Select it, so we can see the
|
|
* // segment points:
|
|
* path = new Path({
|
|
* segments: [event.point],
|
|
* strokeColor: 'black',
|
|
* selected: true
|
|
* });
|
|
* }
|
|
*
|
|
* function onMouseDrag(event) {
|
|
* // On every drag event, add a segment to the path
|
|
* // at the position of the mouse:
|
|
* path.add(event.point);
|
|
* }
|
|
*
|
|
* function onMouseUp(event) {
|
|
* // When the mouse is released, simplify the path:
|
|
* path.simplify();
|
|
* path.selected = true;
|
|
* }
|
|
*/
|
|
simplify: function(tolerance) {
|
|
if (this._segments.length > 2) {
|
|
var fitter = new PathFitter(this, tolerance || 2.5);
|
|
this.setSegments(fitter.fit());
|
|
}
|
|
},
|
|
|
|
// TODO: reduceSegments([flatness])
|
|
|
|
/**
|
|
* Splits the path at the given offset. After splitting, the path will be
|
|
* open. If the path was open already, splitting will result in two paths.
|
|
*
|
|
* @name Path#split
|
|
* @function
|
|
* @param {Number} offset the offset at which to split the path
|
|
* as a number between 0 and {@link Path#length}
|
|
* @return {Path} the newly created path after splitting, if any
|
|
*
|
|
* @example {@paperscript} // Splitting an open path
|
|
* var path = new Path();
|
|
* path.strokeColor = 'black';
|
|
* path.add(20, 20);
|
|
*
|
|
* // Add an arc through {x: 90, y: 80} to {x: 160, y: 20}
|
|
* path.arcTo([90, 80], [160, 20]);
|
|
*
|
|
* // Split the path at 30% of its length:
|
|
* var path2 = path.split(path.length * 0.3);
|
|
* path2.strokeColor = 'red';
|
|
*
|
|
* // Move the newly created path 40px to the right:
|
|
* path2.position.x += 40;
|
|
*
|
|
* @example {@paperscript} // Splitting a closed path
|
|
* var path = new Path.Rectangle({
|
|
* from: [20, 20],
|
|
* to: [80, 80],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Split the path at 60% of its length:
|
|
* path.split(path.length * 0.6);
|
|
*
|
|
* // Move the first segment, to show where the path
|
|
* // was split:
|
|
* path.firstSegment.point.x += 20;
|
|
*
|
|
* // Select the first segment:
|
|
* path.firstSegment.selected = true;
|
|
*/
|
|
/**
|
|
* Splits the path at the given curve location. After splitting, the path
|
|
* will be open. If the path was open already, splitting will result in two
|
|
* paths.
|
|
*
|
|
* @name Path#split
|
|
* @function
|
|
* @param {CurveLocation} location the curve location at which to split
|
|
* the path
|
|
* @return {Path} the newly created path after splitting, if any
|
|
*
|
|
* @example {@paperscript}
|
|
* var path = new Path.Circle({
|
|
* center: view.center,
|
|
* radius: 40,
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* var pointOnCircle = view.center + {
|
|
* length: 40,
|
|
* angle: 30
|
|
* };
|
|
*
|
|
* var curveLocation = path.getNearestLocation(pointOnCircle);
|
|
*
|
|
* path.split(curveLocation);
|
|
* path.lastSegment.selected = true;
|
|
*/
|
|
/**
|
|
* Splits the path at the given curve index and parameter. After splitting,
|
|
* the path will be open. If the path was open already, splitting will
|
|
* result in two paths.
|
|
*
|
|
* @example {@paperscript} // Splitting an open path
|
|
* // Draw a V shaped path:
|
|
* var path = new Path([20, 20], [50, 80], [80, 20]);
|
|
* path.strokeColor = 'black';
|
|
*
|
|
* // Split the path half-way down its second curve:
|
|
* var path2 = path.split(1, 0.5);
|
|
*
|
|
* // Give the resulting path a red stroke-color
|
|
* // and move it 20px to the right:
|
|
* path2.strokeColor = 'red';
|
|
* path2.position.x += 20;
|
|
*
|
|
* @example {@paperscript} // Splitting a closed path
|
|
* var path = new Path.Rectangle({
|
|
* from: [20, 20],
|
|
* to: [80, 80],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Split the path half-way down its second curve:
|
|
* path.split(2, 0.5);
|
|
*
|
|
* // Move the first segment, to show where the path
|
|
* // was split:
|
|
* path.firstSegment.point.x += 20;
|
|
*
|
|
* // Select the first segment:
|
|
* path.firstSegment.selected = true;
|
|
*
|
|
* @param {Number} index the index of the curve in the {@link Path#curves}
|
|
* array at which to split
|
|
* @param {Number} parameter the parameter at which the curve will be split
|
|
* @return {Path} the newly created path after splitting, if any
|
|
*/
|
|
split: function(index, parameter) {
|
|
if (parameter === null)
|
|
return;
|
|
if (arguments.length === 1) {
|
|
var arg = index;
|
|
// split(offset), convert offset to location
|
|
if (typeof arg === 'number')
|
|
arg = this.getLocationAt(arg);
|
|
// split(location)
|
|
index = arg.index;
|
|
parameter = arg.parameter;
|
|
}
|
|
var tolerance = /*#=*/Numerical.TOLERANCE;
|
|
if (parameter >= 1 - tolerance) {
|
|
// t == 1 is the same as t == 0 and index ++
|
|
index++;
|
|
parameter--;
|
|
}
|
|
var curves = this.getCurves();
|
|
if (index >= 0 && index < curves.length) {
|
|
// Only divide curves if we're not on an existing segment already.
|
|
if (parameter > tolerance) {
|
|
// Divide the curve with the index at given parameter.
|
|
// Increase because dividing adds more segments to the path.
|
|
curves[index++].divide(parameter, true);
|
|
}
|
|
// Create the new path with the segments to the right of given
|
|
// parameter, which are removed from the current path. Pass true
|
|
// for includeCurves, since we want to preserve and move them to
|
|
// the new path through _add(), allowing us to have CurveLocation
|
|
// keep the connection to the new path through moved curves.
|
|
var segs = this.removeSegments(index, this._segments.length, true),
|
|
path;
|
|
if (this._closed) {
|
|
// If the path is closed, open it and move the segments round,
|
|
// otherwise create two paths.
|
|
this.setClosed(false);
|
|
// Just have path point to this. The moving around of segments
|
|
// will happen below.
|
|
path = this;
|
|
} else if (index > 0) {
|
|
// Pass true for _preserve, in case of CompoundPath, to avoid
|
|
// reversing of path direction, which would mess with segs!
|
|
// Use _clone to copy over all other attributes, including style
|
|
path = this._clone(new Path().insertAbove(this, true));
|
|
}
|
|
path._add(segs, 0);
|
|
// Add dividing segment again. In case of a closed path, that's the
|
|
// beginning segment again at the end, since we opened it.
|
|
this.addSegment(segs[0]);
|
|
return path;
|
|
}
|
|
return null;
|
|
},
|
|
|
|
/**
|
|
* Specifies whether the path is oriented clock-wise.
|
|
*
|
|
* @type Boolean
|
|
* @bean
|
|
*/
|
|
isClockwise: function() {
|
|
if (this._clockwise !== undefined)
|
|
return this._clockwise;
|
|
return Path.isClockwise(this._segments);
|
|
},
|
|
|
|
setClockwise: function(clockwise) {
|
|
// Only revers the path if its clockwise orientation is not the same
|
|
// as what it is now demanded to be.
|
|
// On-the-fly conversion to boolean:
|
|
if (this.isClockwise() != (clockwise = !!clockwise))
|
|
this.reverse();
|
|
// Reverse only flips _clockwise state if it was already set, so let's
|
|
// always set this here now.
|
|
this._clockwise = clockwise;
|
|
},
|
|
|
|
/**
|
|
* Reverses the orientation of the path, by reversing all its segments.
|
|
*/
|
|
reverse: function() {
|
|
this._segments.reverse();
|
|
// Reverse the handles:
|
|
for (var i = 0, l = this._segments.length; i < l; i++) {
|
|
var segment = this._segments[i];
|
|
var handleIn = segment._handleIn;
|
|
segment._handleIn = segment._handleOut;
|
|
segment._handleOut = handleIn;
|
|
segment._index = i;
|
|
}
|
|
// Clear curves since it all has changed.
|
|
this._curves = null;
|
|
// Flip clockwise state if it's defined
|
|
if (this._clockwise !== undefined)
|
|
this._clockwise = !this._clockwise;
|
|
this._changed(/*#=*/Change.GEOMETRY);
|
|
},
|
|
|
|
// DOCS: document Path#join(path) in more detail.
|
|
// DOCS: document Path#join() (joining with itself)
|
|
// TODO: Consider adding a distance / tolerance parameter for merging.
|
|
/**
|
|
* Joins the path with the specified path, which will be removed in the
|
|
* process.
|
|
*
|
|
* @param {Path} path
|
|
*
|
|
* @example {@paperscript}
|
|
* // Joining two paths:
|
|
* var path = new Path({
|
|
* segments: [[30, 25], [30, 75]],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* var path2 = new Path({
|
|
* segments: [[200, 25], [200, 75]],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Join the paths:
|
|
* path.join(path2);
|
|
*
|
|
* @example {@paperscript}
|
|
* // Joining two paths that share a point at the start or end of their
|
|
* // segments array:
|
|
* var path = new Path({
|
|
* segments: [[30, 25], [30, 75]],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* var path2 = new Path({
|
|
* segments: [[30, 25], [80, 25]],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Join the paths:
|
|
* path.join(path2);
|
|
*
|
|
* // After joining, path with have 3 segments, since it
|
|
* // shared its first segment point with the first
|
|
* // segment point of path2.
|
|
*
|
|
* // Select the path to show that they have joined:
|
|
* path.selected = true;
|
|
*
|
|
* @example {@paperscript}
|
|
* // Joining two paths that connect at two points:
|
|
* var path = new Path({
|
|
* segments: [[30, 25], [80, 25], [80, 75]],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* var path2 = new Path({
|
|
* segments: [[30, 25], [30, 75], [80, 75]],
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* // Join the paths:
|
|
* path.join(path2);
|
|
*
|
|
* // Because the paths were joined at two points, the path is closed
|
|
* // and has 4 segments.
|
|
*
|
|
* // Select the path to show that they have joined:
|
|
* path.selected = true;
|
|
*/
|
|
join: function(path) {
|
|
if (path) {
|
|
var segments = path._segments,
|
|
last1 = this.getLastSegment(),
|
|
last2 = path.getLastSegment();
|
|
if (last1._point.equals(last2._point))
|
|
path.reverse();
|
|
var first1,
|
|
first2 = path.getFirstSegment();
|
|
if (last1._point.equals(first2._point)) {
|
|
last1.setHandleOut(first2._handleOut);
|
|
this._add(segments.slice(1));
|
|
} else {
|
|
first1 = this.getFirstSegment();
|
|
if (first1._point.equals(first2._point))
|
|
path.reverse();
|
|
last2 = path.getLastSegment();
|
|
if (first1._point.equals(last2._point)) {
|
|
first1.setHandleIn(last2._handleIn);
|
|
// Prepend all segments from path except the last one
|
|
this._add(segments.slice(0, segments.length - 1), 0);
|
|
} else {
|
|
this._add(segments.slice());
|
|
}
|
|
}
|
|
if (path.closed)
|
|
this._add([segments[0]]);
|
|
path.remove();
|
|
}
|
|
// Close the resulting path and merge first and last segment if they
|
|
// touch, meaning the touched at path ends. Also do this if no path
|
|
// argument was provided, in which cases the path is joined with itself
|
|
// only if its ends touch.
|
|
var first = this.getFirstSegment(),
|
|
last = this.getLastSegment();
|
|
if (first !== last && first._point.equals(last._point)) {
|
|
first.setHandleIn(last._handleIn);
|
|
last.remove();
|
|
this.setClosed(true);
|
|
}
|
|
},
|
|
|
|
|
|
// DOCS: toShape
|
|
|
|
toShape: function(insert) {
|
|
if (!this._closed)
|
|
return null;
|
|
|
|
var segments = this._segments,
|
|
type,
|
|
size,
|
|
radius,
|
|
topCenter;
|
|
|
|
function isColinear(i, j) {
|
|
return segments[i].isColinear(segments[j]);
|
|
}
|
|
|
|
function isOrthogonal(i) {
|
|
return segments[i].isOrthogonal();
|
|
}
|
|
|
|
function isArc(i) {
|
|
return segments[i].isArc();
|
|
}
|
|
|
|
function getDistance(i, j) {
|
|
return segments[i]._point.getDistance(segments[j]._point);
|
|
}
|
|
|
|
// See if actually have any curves in the path. Differentiate
|
|
// between straight objects (line, polyline, rect, and polygon) and
|
|
// objects with curves(circle, ellipse, roundedRectangle).
|
|
if (this.isPolygon() && segments.length === 4
|
|
&& isColinear(0, 2) && isColinear(1, 3) && isOrthogonal(1)) {
|
|
type = Shape.Rectangle;
|
|
size = new Size(getDistance(0, 3), getDistance(0, 1));
|
|
topCenter = segments[1]._point.add(segments[2]._point).divide(2);
|
|
} else if (segments.length === 8 && isArc(0) && isArc(2) && isArc(4)
|
|
&& isArc(6) && isColinear(1, 5) && isColinear(3, 7)) {
|
|
// It's a rounded rectangle.
|
|
type = Shape.Rectangle;
|
|
size = new Size(getDistance(1, 6), getDistance(0, 3));
|
|
// Subtract side lengths from total width and divide by 2 to get the
|
|
// corner radius size.
|
|
radius = size.subtract(new Size(getDistance(0, 7),
|
|
getDistance(1, 2))).divide(2);
|
|
topCenter = segments[3]._point.add(segments[4]._point).divide(2);
|
|
} else if (segments.length === 4
|
|
&& isArc(0) && isArc(1) && isArc(2) && isArc(3)) {
|
|
// If the distance between (point0 and point2) and (point1
|
|
// and point3) are equal, then it is a circle
|
|
if (Numerical.isZero(getDistance(0, 2) - getDistance(1, 3))) {
|
|
type = Shape.Circle;
|
|
radius = getDistance(0, 2) / 2;
|
|
} else {
|
|
type = Shape.Ellipse;
|
|
radius = new Size(getDistance(2, 0) / 2, getDistance(3, 1) / 2);
|
|
}
|
|
topCenter = segments[1]._point;
|
|
}
|
|
|
|
if (type) {
|
|
var center = this.getPosition(true),
|
|
shape = new type({
|
|
center: center,
|
|
size: size,
|
|
radius: radius,
|
|
insert: false
|
|
});
|
|
// Determine and apply the shape's angle of rotation.
|
|
shape.rotate(topCenter.subtract(center).getAngle() + 90);
|
|
shape.setStyle(this._style);
|
|
// Insert is true by default.
|
|
if (insert || insert === undefined)
|
|
shape.insertAbove(this);
|
|
return shape;
|
|
}
|
|
return null;
|
|
},
|
|
|
|
_hitTestSelf: function(point, options) {
|
|
var that = this,
|
|
style = this.getStyle(),
|
|
segments = this._segments,
|
|
numSegments = segments.length,
|
|
closed = this._closed,
|
|
// transformed tolerance padding, see Item#hitTest. We will add
|
|
// stroke padding on top if stroke is defined.
|
|
tolerancePadding = options._tolerancePadding,
|
|
strokePadding = tolerancePadding,
|
|
join, cap, miterLimit,
|
|
area, loc, res,
|
|
hitStroke = options.stroke && style.hasStroke(),
|
|
hitFill = options.fill && style.hasFill(),
|
|
hitCurves = options.curves,
|
|
radius = hitStroke
|
|
? style.getStrokeWidth() / 2
|
|
// Set radius to 0 when we're hit-testing fills with
|
|
// tolerance, to handle tolerance through stroke hit-test
|
|
// functionality. Also use 0 when hit-testing curves.
|
|
: hitFill && options.tolerance > 0 || hitCurves
|
|
? 0 : null;
|
|
if (radius !== null) {
|
|
if (radius > 0) {
|
|
join = style.getStrokeJoin();
|
|
cap = style.getStrokeCap();
|
|
miterLimit = radius * style.getMiterLimit();
|
|
// Add the stroke radius to tolerance padding.
|
|
strokePadding = tolerancePadding.add(new Point(radius, radius));
|
|
} else {
|
|
join = cap = 'round';
|
|
}
|
|
// Using tolerance padding for fill tests will also work if there is
|
|
// no stroke, in which case radius = 0 and we will test for stroke
|
|
// locations to extend the fill area by tolerance.
|
|
}
|
|
|
|
function isCloseEnough(pt, padding) {
|
|
return point.subtract(pt).divide(padding).length <= 1;
|
|
}
|
|
|
|
function checkSegmentPoint(seg, pt, name) {
|
|
if (!options.selected || pt.isSelected()) {
|
|
var anchor = seg._point;
|
|
if (pt !== anchor)
|
|
pt = pt.add(anchor);
|
|
if (isCloseEnough(pt, strokePadding)) {
|
|
return new HitResult(name, that, {
|
|
segment: seg,
|
|
point: pt
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
function checkSegmentPoints(seg, ends) {
|
|
// Note, when checking for ends, we don't also check for handles,
|
|
// since this will happen afterwards in a separate loop, see below.
|
|
return (ends || options.segments)
|
|
&& checkSegmentPoint(seg, seg._point, 'segment')
|
|
|| (!ends && options.handles) && (
|
|
checkSegmentPoint(seg, seg._handleIn, 'handle-in') ||
|
|
checkSegmentPoint(seg, seg._handleOut, 'handle-out'));
|
|
}
|
|
|
|
// Code to check stroke join / cap areas
|
|
|
|
function addToArea(point) {
|
|
area.add(point);
|
|
}
|
|
|
|
function checkSegmentStroke(segment) {
|
|
// Handle joins / caps that are not round specificelly, by
|
|
// hit-testing their polygon areas.
|
|
if (join !== 'round' || cap !== 'round') {
|
|
// Create an 'internal' path without id and outside the DOM
|
|
// to run the hit-test on it.
|
|
area = new Path({ internal: true, closed: true });
|
|
if (closed || segment._index > 0
|
|
&& segment._index < numSegments - 1) {
|
|
// It's a join. See that it's not a round one (one of
|
|
// the handles has to be zero too for this!)
|
|
if (join !== 'round' && (segment._handleIn.isZero()
|
|
|| segment._handleOut.isZero()))
|
|
// _addBevelJoin() handles both 'bevel' and 'miter'!
|
|
Path._addBevelJoin(segment, join, radius, miterLimit,
|
|
addToArea, true);
|
|
} else if (cap !== 'round') {
|
|
// It's a cap
|
|
Path._addSquareCap(segment, cap, radius, addToArea, true);
|
|
}
|
|
// See if the above produced an area to check for
|
|
if (!area.isEmpty()) {
|
|
// Also use stroke check with tolerancePadding if the point
|
|
// is not inside the area itself, to use test caps and joins
|
|
// with same tolerance.
|
|
var loc;
|
|
return area.contains(point)
|
|
|| (loc = area.getNearestLocation(point))
|
|
&& isCloseEnough(loc.getPoint(), tolerancePadding);
|
|
}
|
|
}
|
|
// Fallback scenario is a round join / cap.
|
|
return isCloseEnough(segment._point, strokePadding);
|
|
}
|
|
|
|
// If we're asked to query for segments, ends or handles, do all that
|
|
// before stroke or fill.
|
|
if (options.ends && !options.segments && !closed) {
|
|
if (res = checkSegmentPoints(segments[0], true)
|
|
|| checkSegmentPoints(segments[numSegments - 1], true))
|
|
return res;
|
|
} else if (options.segments || options.handles) {
|
|
for (var i = 0; i < numSegments; i++)
|
|
if (res = checkSegmentPoints(segments[i]))
|
|
return res;
|
|
}
|
|
// If we're querying for stroke, perform that before fill
|
|
if (radius !== null) {
|
|
loc = this.getNearestLocation(point);
|
|
// Note that paths need at least two segments to have an actual
|
|
// stroke. But we still check for segments with the radius fallback
|
|
// check if there is only one segment.
|
|
if (loc) {
|
|
// Now see if we're on a segment, and if so, check for its
|
|
// stroke join / cap first. If not, do a normal radius check
|
|
// for round strokes.
|
|
var parameter = loc.getParameter();
|
|
if (parameter === 0 || parameter === 1 && numSegments > 1) {
|
|
if (!checkSegmentStroke(loc.getSegment()))
|
|
loc = null;
|
|
} else if (!isCloseEnough(loc.getPoint(), strokePadding)) {
|
|
loc = null;
|
|
}
|
|
}
|
|
// If we have miter joins, we may not be done yet, since they can be
|
|
// longer than the radius. Check for each segment within reach now.
|
|
if (!loc && join === 'miter' && numSegments > 1) {
|
|
for (var i = 0; i < numSegments; i++) {
|
|
var segment = segments[i];
|
|
if (point.getDistance(segment._point) <= miterLimit
|
|
&& checkSegmentStroke(segment)) {
|
|
loc = segment.getLocation();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// Don't process loc yet, as we also need to query for stroke after fill
|
|
// in some cases. Simply skip fill query if we already have a matching
|
|
// stroke. If we have a loc and no stroke then it's a result for fill.
|
|
return !loc && hitFill && this._contains(point)
|
|
|| loc && !hitStroke && !hitCurves
|
|
? new HitResult('fill', this)
|
|
: loc
|
|
? new HitResult(hitStroke ? 'stroke' : 'curve', this, {
|
|
location: loc,
|
|
// It's fine performance wise to call getPoint()
|
|
// again since it was already called before.
|
|
point: loc.getPoint()
|
|
})
|
|
: null;
|
|
}
|
|
|
|
// TODO: intersects(item)
|
|
// TODO: contains(item)
|
|
}, Base.each(['getPoint', 'getTangent', 'getNormal', 'getCurvature'],
|
|
function(name) {
|
|
this[name + 'At'] = function(offset, isParameter) {
|
|
var loc = this.getLocationAt(offset, isParameter);
|
|
return loc && loc[name]();
|
|
};
|
|
},
|
|
/** @lends Path# */{
|
|
// Explicitly deactivate the creation of beans, as we have functions here
|
|
// that look like bean getters but actually read arguments.
|
|
// See #getLocationOf(), #getNearestLocation(), #getNearestPoint()
|
|
beans: false,
|
|
|
|
_getOffset: function(location) {
|
|
var index = location && location.getIndex();
|
|
if (index != null) {
|
|
var curves = this.getCurves(),
|
|
offset = 0;
|
|
for (var i = 0; i < index; i++)
|
|
offset += curves[i].getLength();
|
|
var curve = curves[index],
|
|
parameter = location.getParameter();
|
|
if (parameter > 0)
|
|
offset += curve.getPartLength(0, parameter);
|
|
return offset;
|
|
}
|
|
return null;
|
|
},
|
|
|
|
/**
|
|
* {@grouptitle Positions on Paths and Curves}
|
|
*
|
|
* Returns the curve location of the specified point if it lies on the
|
|
* path, {@code null} otherwise.
|
|
* @param {Point} point the point on the path.
|
|
* @return {CurveLocation} the curve location of the specified point.
|
|
*/
|
|
getLocationOf: function(/* point */) {
|
|
var point = Point.read(arguments),
|
|
curves = this.getCurves();
|
|
for (var i = 0, l = curves.length; i < l; i++) {
|
|
var loc = curves[i].getLocationOf(point);
|
|
if (loc)
|
|
return loc;
|
|
}
|
|
return null;
|
|
},
|
|
|
|
/**
|
|
* Returns the length of the path from its beginning up to up to the
|
|
* specified point if it lies on the path, {@code null} otherwise.
|
|
* @param {Point} point the point on the path.
|
|
* @return {Number} the length of the path up to the specified point.
|
|
*/
|
|
getOffsetOf: function(/* point */) {
|
|
var loc = this.getLocationOf.apply(this, arguments);
|
|
return loc ? loc.getOffset() : null;
|
|
},
|
|
|
|
/**
|
|
* Returns the curve location of the specified offset on the path.
|
|
*
|
|
* @param {Number} offset the offset on the path, where {@code 0} is at
|
|
* the beginning of the path and {@link Path#length} at the end.
|
|
* @param {Boolean} [isParameter=false]
|
|
* @return {CurveLocation} the curve location at the specified offset
|
|
*/
|
|
getLocationAt: function(offset, isParameter) {
|
|
var curves = this.getCurves(),
|
|
length = 0;
|
|
if (isParameter) {
|
|
// offset consists of curve index and curve parameter, before and
|
|
// after the fractional digit.
|
|
var index = ~~offset; // = Math.floor()
|
|
return curves[index].getLocationAt(offset - index, true);
|
|
}
|
|
for (var i = 0, l = curves.length; i < l; i++) {
|
|
var start = length,
|
|
curve = curves[i];
|
|
length += curve.getLength();
|
|
if (length > offset) {
|
|
// Found the segment within which the length lies
|
|
return curve.getLocationAt(offset - start);
|
|
}
|
|
}
|
|
// It may be that through imprecision of getLength, that the end of the
|
|
// last curve was missed:
|
|
if (offset <= this.getLength())
|
|
return new CurveLocation(curves[curves.length - 1], 1);
|
|
return null;
|
|
},
|
|
|
|
/**
|
|
* Calculates the point on the path at the given offset.
|
|
*
|
|
* @name Path#getPointAt
|
|
* @function
|
|
* @param {Number} offset the offset on the path, where {@code 0} is at
|
|
* the beginning of the path and {@link Path#length} at the end.
|
|
* @param {Boolean} [isParameter=false]
|
|
* @return {Point} the point at the given offset
|
|
*
|
|
* @example {@paperscript height=150}
|
|
* // Finding the point on a path at a given offset:
|
|
*
|
|
* // Create an arc shaped path:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* path.add(new Point(40, 100));
|
|
* path.arcTo(new Point(150, 100));
|
|
*
|
|
* // We're going to be working with a third of the length
|
|
* // of the path as the offset:
|
|
* var offset = path.length / 3;
|
|
*
|
|
* // Find the point on the path:
|
|
* var point = path.getPointAt(offset);
|
|
*
|
|
* // Create a small circle shaped path at the point:
|
|
* var circle = new Path.Circle({
|
|
* center: point,
|
|
* radius: 3,
|
|
* fillColor: 'red'
|
|
* });
|
|
*
|
|
* @example {@paperscript height=150}
|
|
* // Iterating over the length of a path:
|
|
*
|
|
* // Create an arc shaped path:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* path.add(new Point(40, 100));
|
|
* path.arcTo(new Point(150, 100));
|
|
*
|
|
* var amount = 5;
|
|
* var length = path.length;
|
|
* for (var i = 0; i < amount + 1; i++) {
|
|
* var offset = i / amount * length;
|
|
*
|
|
* // Find the point on the path at the given offset:
|
|
* var point = path.getPointAt(offset);
|
|
*
|
|
* // Create a small circle shaped path at the point:
|
|
* var circle = new Path.Circle({
|
|
* center: point,
|
|
* radius: 3,
|
|
* fillColor: 'red'
|
|
* });
|
|
* }
|
|
*/
|
|
|
|
/**
|
|
* Calculates the tangent vector of the path at the given offset.
|
|
*
|
|
* @name Path#getTangentAt
|
|
* @function
|
|
* @param {Number} offset the offset on the path, where {@code 0} is at
|
|
* the beginning of the path and {@link Path#length} at the end.
|
|
* @param {Boolean} [isParameter=false]
|
|
* @return {Point} the tangent vector at the given offset
|
|
*
|
|
* @example {@paperscript height=150}
|
|
* // Working with the tangent vector at a given offset:
|
|
*
|
|
* // Create an arc shaped path:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* path.add(new Point(40, 100));
|
|
* path.arcTo(new Point(150, 100));
|
|
*
|
|
* // We're going to be working with a third of the length
|
|
* // of the path as the offset:
|
|
* var offset = path.length / 3;
|
|
*
|
|
* // Find the point on the path:
|
|
* var point = path.getPointAt(offset);
|
|
*
|
|
* // Find the tangent vector at the given offset:
|
|
* var tangent = path.getTangentAt(offset);
|
|
*
|
|
* // Make the tangent vector 60pt long:
|
|
* tangent.length = 60;
|
|
*
|
|
* var line = new Path({
|
|
* segments: [point, point + tangent],
|
|
* strokeColor: 'red'
|
|
* })
|
|
*
|
|
* @example {@paperscript height=200}
|
|
* // Iterating over the length of a path:
|
|
*
|
|
* // Create an arc shaped path:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* path.add(new Point(40, 100));
|
|
* path.arcTo(new Point(150, 100));
|
|
*
|
|
* var amount = 6;
|
|
* var length = path.length;
|
|
* for (var i = 0; i < amount + 1; i++) {
|
|
* var offset = i / amount * length;
|
|
*
|
|
* // Find the point on the path at the given offset:
|
|
* var point = path.getPointAt(offset);
|
|
*
|
|
* // Find the normal vector on the path at the given offset:
|
|
* var tangent = path.getTangentAt(offset);
|
|
*
|
|
* // Make the tangent vector 60pt long:
|
|
* tangent.length = 60;
|
|
*
|
|
* var line = new Path({
|
|
* segments: [point, point + tangent],
|
|
* strokeColor: 'red'
|
|
* })
|
|
* }
|
|
*/
|
|
|
|
/**
|
|
* Calculates the normal vector of the path at the given offset.
|
|
*
|
|
* @name Path#getNormalAt
|
|
* @function
|
|
* @param {Number} offset the offset on the path, where {@code 0} is at
|
|
* the beginning of the path and {@link Path#length} at the end.
|
|
* @param {Boolean} [isParameter=false]
|
|
* @return {Point} the normal vector at the given offset
|
|
*
|
|
* @example {@paperscript height=150}
|
|
* // Working with the normal vector at a given offset:
|
|
*
|
|
* // Create an arc shaped path:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* path.add(new Point(40, 100));
|
|
* path.arcTo(new Point(150, 100));
|
|
*
|
|
* // We're going to be working with a third of the length
|
|
* // of the path as the offset:
|
|
* var offset = path.length / 3;
|
|
*
|
|
* // Find the point on the path:
|
|
* var point = path.getPointAt(offset);
|
|
*
|
|
* // Find the normal vector at the given offset:
|
|
* var normal = path.getNormalAt(offset);
|
|
*
|
|
* // Make the normal vector 30pt long:
|
|
* normal.length = 30;
|
|
*
|
|
* var line = new Path({
|
|
* segments: [point, point + normal],
|
|
* strokeColor: 'red'
|
|
* });
|
|
*
|
|
* @example {@paperscript height=200}
|
|
* // Iterating over the length of a path:
|
|
*
|
|
* // Create an arc shaped path:
|
|
* var path = new Path({
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* path.add(new Point(40, 100));
|
|
* path.arcTo(new Point(150, 100));
|
|
*
|
|
* var amount = 10;
|
|
* var length = path.length;
|
|
* for (var i = 0; i < amount + 1; i++) {
|
|
* var offset = i / amount * length;
|
|
*
|
|
* // Find the point on the path at the given offset:
|
|
* var point = path.getPointAt(offset);
|
|
*
|
|
* // Find the normal vector on the path at the given offset:
|
|
* var normal = path.getNormalAt(offset);
|
|
*
|
|
* // Make the normal vector 30pt long:
|
|
* normal.length = 30;
|
|
*
|
|
* var line = new Path({
|
|
* segments: [point, point + normal],
|
|
* strokeColor: 'red'
|
|
* });
|
|
* }
|
|
*/
|
|
|
|
/**
|
|
* Calculates the curvature of the path at the given offset. Curvatures
|
|
* indicate how sharply a path changes direction. A straight line has zero
|
|
* curvature, where as a circle has a constant curvature. The path's radius
|
|
* at the given offset is the reciprocal value of its curvature.
|
|
*
|
|
* @name Path#getCurvatureAt
|
|
* @function
|
|
* @param {Number} offset the offset on the path, where {@code 0} is at
|
|
* the beginning of the path and {@link Path#length} at the end.
|
|
* @param {Boolean} [isParameter=false]
|
|
* @return {Number} the normal vector at the given offset
|
|
*
|
|
|
|
/**
|
|
* Returns the nearest location on the path to the specified point.
|
|
*
|
|
* @function
|
|
* @param point {Point} the point for which we search the nearest location
|
|
* @return {CurveLocation} the location on the path that's the closest to
|
|
* the specified point
|
|
*/
|
|
getNearestLocation: function(/* point */) {
|
|
var point = Point.read(arguments),
|
|
curves = this.getCurves(),
|
|
minDist = Infinity,
|
|
minLoc = null;
|
|
for (var i = 0, l = curves.length; i < l; i++) {
|
|
var loc = curves[i].getNearestLocation(point);
|
|
if (loc._distance < minDist) {
|
|
minDist = loc._distance;
|
|
minLoc = loc;
|
|
}
|
|
}
|
|
return minLoc;
|
|
},
|
|
|
|
/**
|
|
* Returns the nearest point on the path to the specified point.
|
|
*
|
|
* @function
|
|
* @param point {Point} the point for which we search the nearest point
|
|
* @return {Point} the point on the path that's the closest to the specified
|
|
* point
|
|
*
|
|
* @example {@paperscript height=200}
|
|
* var star = new Path.Star({
|
|
* center: view.center,
|
|
* points: 10,
|
|
* radius1: 30,
|
|
* radius2: 60,
|
|
* strokeColor: 'black'
|
|
* });
|
|
*
|
|
* var circle = new Path.Circle({
|
|
* center: view.center,
|
|
* radius: 3,
|
|
* fillColor: 'red'
|
|
* });
|
|
*
|
|
* function onMouseMove(event) {
|
|
* // Get the nearest point from the mouse position
|
|
* // to the star shaped path:
|
|
* var nearestPoint = star.getNearestPoint(event.point);
|
|
*
|
|
* // Move the red circle to the nearest point:
|
|
* circle.position = nearestPoint;
|
|
* }
|
|
*/
|
|
getNearestPoint: function(/* point */) {
|
|
return this.getNearestLocation.apply(this, arguments).getPoint();
|
|
}
|
|
}), new function() { // Scope for drawing
|
|
|
|
// Note that in the code below we're often accessing _x and _y on point
|
|
// objects that were read from segments. This is because the SegmentPoint
|
|
// class overrides the plain x / y properties with getter / setters and
|
|
// stores the values in these private properties internally. To avoid
|
|
// calling of getter functions all the time we directly access these private
|
|
// properties here. The distinction between normal Point objects and
|
|
// SegmentPoint objects maybe seem a bit tedious but is worth the benefit in
|
|
// performance.
|
|
|
|
function drawHandles(ctx, segments, matrix, size) {
|
|
var half = size / 2;
|
|
|
|
function drawHandle(index) {
|
|
var hX = coords[index],
|
|
hY = coords[index + 1];
|
|
if (pX != hX || pY != hY) {
|
|
ctx.beginPath();
|
|
ctx.moveTo(pX, pY);
|
|
ctx.lineTo(hX, hY);
|
|
ctx.stroke();
|
|
ctx.beginPath();
|
|
ctx.arc(hX, hY, half, 0, Math.PI * 2, true);
|
|
ctx.fill();
|
|
}
|
|
}
|
|
|
|
var coords = new Array(6);
|
|
for (var i = 0, l = segments.length; i < l; i++) {
|
|
var segment = segments[i];
|
|
segment._transformCoordinates(matrix, coords, false);
|
|
var state = segment._selectionState,
|
|
pX = coords[0],
|
|
pY = coords[1];
|
|
if (state & /*#=*/SelectionState.HANDLE_IN)
|
|
drawHandle(2);
|
|
if (state & /*#=*/SelectionState.HANDLE_OUT)
|
|
drawHandle(4);
|
|
// Draw a rectangle at segment.point:
|
|
ctx.fillRect(pX - half, pY - half, size, size);
|
|
// If the point is not selected, draw a white square that is 1 px
|
|
// smaller on all sides:
|
|
if (!(state & /*#=*/SelectionState.POINT)) {
|
|
var fillStyle = ctx.fillStyle;
|
|
ctx.fillStyle = '#ffffff';
|
|
ctx.fillRect(pX - half + 1, pY - half + 1, size - 2, size - 2);
|
|
ctx.fillStyle = fillStyle;
|
|
}
|
|
}
|
|
}
|
|
|
|
function drawSegments(ctx, path, matrix) {
|
|
var segments = path._segments,
|
|
length = segments.length,
|
|
coords = new Array(6),
|
|
first = true,
|
|
curX, curY,
|
|
prevX, prevY,
|
|
inX, inY,
|
|
outX, outY;
|
|
|
|
function drawSegment(segment) {
|
|
// Optimise code when no matrix is provided by accessing segment
|
|
// points hand handles directly, since this is the default when
|
|
// drawing paths. Matrix is only used for drawing selections and
|
|
// when #strokeScaling is false.
|
|
if (matrix) {
|
|
segment._transformCoordinates(matrix, coords, false);
|
|
curX = coords[0];
|
|
curY = coords[1];
|
|
} else {
|
|
var point = segment._point;
|
|
curX = point._x;
|
|
curY = point._y;
|
|
}
|
|
if (first) {
|
|
ctx.moveTo(curX, curY);
|
|
first = false;
|
|
} else {
|
|
if (matrix) {
|
|
inX = coords[2];
|
|
inY = coords[3];
|
|
} else {
|
|
var handle = segment._handleIn;
|
|
inX = curX + handle._x;
|
|
inY = curY + handle._y;
|
|
}
|
|
if (inX === curX && inY === curY
|
|
&& outX === prevX && outY === prevY) {
|
|
ctx.lineTo(curX, curY);
|
|
} else {
|
|
ctx.bezierCurveTo(outX, outY, inX, inY, curX, curY);
|
|
}
|
|
}
|
|
prevX = curX;
|
|
prevY = curY;
|
|
if (matrix) {
|
|
outX = coords[4];
|
|
outY = coords[5];
|
|
} else {
|
|
var handle = segment._handleOut;
|
|
outX = prevX + handle._x;
|
|
outY = prevY + handle._y;
|
|
}
|
|
}
|
|
|
|
for (var i = 0; i < length; i++)
|
|
drawSegment(segments[i]);
|
|
// Close path by drawing first segment again
|
|
if (path._closed && length > 0)
|
|
drawSegment(segments[0]);
|
|
}
|
|
|
|
return {
|
|
_draw: function(ctx, param, strokeMatrix) {
|
|
var dontStart = param.dontStart,
|
|
dontPaint = param.dontFinish || param.clip,
|
|
style = this.getStyle(),
|
|
hasFill = style.hasFill(),
|
|
hasStroke = style.hasStroke(),
|
|
dashArray = style.getDashArray(),
|
|
// dashLength is only set if we can't draw dashes natively
|
|
dashLength = !paper.support.nativeDash && hasStroke
|
|
&& dashArray && dashArray.length;
|
|
|
|
if (!dontStart)
|
|
ctx.beginPath();
|
|
|
|
if (!dontStart && this._currentPath) {
|
|
ctx.currentPath = this._currentPath;
|
|
} else if (hasFill || hasStroke && !dashLength || dontPaint) {
|
|
// Prepare the canvas path if we have any situation that
|
|
// requires it to be defined.
|
|
drawSegments(ctx, this, strokeMatrix);
|
|
if (this._closed)
|
|
ctx.closePath();
|
|
// CompoundPath collects its own _currentPath
|
|
if (!dontStart)
|
|
this._currentPath = ctx.currentPath;
|
|
}
|
|
|
|
function getOffset(i) {
|
|
// Negative modulo is necessary since we're stepping back
|
|
// in the dash sequence first.
|
|
return dashArray[((i % dashLength) + dashLength) % dashLength];
|
|
}
|
|
|
|
if (!dontPaint && (hasFill || hasStroke)) {
|
|
// If the path is part of a compound path or doesn't have a fill
|
|
// or stroke, there is no need to continue.
|
|
this._setStyles(ctx);
|
|
if (hasFill) {
|
|
ctx.fill(style.getWindingRule());
|
|
// If shadowColor is defined, clear it after fill, so it
|
|
// won't be applied to both fill and stroke. If the path is
|
|
// only stroked, we don't have to clear it.
|
|
ctx.shadowColor = 'rgba(0,0,0,0)';
|
|
}
|
|
if (hasStroke) {
|
|
if (dashLength) {
|
|
// We cannot use the path created by drawSegments above
|
|
// Use PathIterator to draw dashed paths:
|
|
// NOTE: We don't cache this path in another currentPath
|
|
// since browsers that support currentPath also support
|
|
// native dashes.
|
|
if (!dontStart)
|
|
ctx.beginPath();
|
|
var iterator = new PathIterator(this, 32, 0.25,
|
|
strokeMatrix),
|
|
length = iterator.length,
|
|
from = -style.getDashOffset(), to,
|
|
i = 0;
|
|
from = from % length;
|
|
// Step backwards in the dash sequence first until the
|
|
// from parameter is below 0.
|
|
while (from > 0) {
|
|
from -= getOffset(i--) + getOffset(i--);
|
|
}
|
|
while (from < length) {
|
|
to = from + getOffset(i++);
|
|
if (from > 0 || to > 0)
|
|
iterator.drawPart(ctx,
|
|
Math.max(from, 0), Math.max(to, 0));
|
|
from = to + getOffset(i++);
|
|
}
|
|
}
|
|
ctx.stroke();
|
|
}
|
|
}
|
|
},
|
|
|
|
_drawSelected: function(ctx, matrix) {
|
|
ctx.beginPath();
|
|
drawSegments(ctx, this, matrix);
|
|
// Now stroke it and draw its handles:
|
|
ctx.stroke();
|
|
drawHandles(ctx, this._segments, matrix, paper.settings.handleSize);
|
|
}
|
|
};
|
|
}, new function() { // Path Smoothing
|
|
|
|
/**
|
|
* Solves a tri-diagonal system for one of coordinates (x or y) of first
|
|
* bezier control points.
|
|
*
|
|
* @param rhs right hand side vector.
|
|
* @return Solution vector.
|
|
*/
|
|
function getFirstControlPoints(rhs) {
|
|
var n = rhs.length,
|
|
x = [], // Solution vector.
|
|
tmp = [], // Temporary workspace.
|
|
b = 2;
|
|
x[0] = rhs[0] / b;
|
|
// Decomposition and forward substitution.
|
|
for (var i = 1; i < n; i++) {
|
|
tmp[i] = 1 / b;
|
|
b = (i < n - 1 ? 4 : 2) - tmp[i];
|
|
x[i] = (rhs[i] - x[i - 1]) / b;
|
|
}
|
|
// Back-substitution.
|
|
for (var i = 1; i < n; i++) {
|
|
x[n - i - 1] -= tmp[n - i] * x[n - i];
|
|
}
|
|
return x;
|
|
}
|
|
|
|
return {
|
|
// Note: Documentation for smooth() is in PathItem
|
|
smooth: function() {
|
|
// This code is based on the work by Oleg V. Polikarpotchkin,
|
|
// http://ov-p.spaces.live.com/blog/cns!39D56F0C7A08D703!147.entry
|
|
// It was extended to support closed paths by averaging overlapping
|
|
// beginnings and ends. The result of this approach is very close to
|
|
// Polikarpotchkin's closed curve solution, but reuses the same
|
|
// algorithm as for open paths, and is probably executing faster as
|
|
// well, so it is preferred.
|
|
var segments = this._segments,
|
|
size = segments.length,
|
|
closed = this._closed,
|
|
n = size,
|
|
// Add overlapping ends for averaging handles in closed paths
|
|
overlap = 0;
|
|
if (size <= 2)
|
|
return;
|
|
if (closed) {
|
|
// Overlap up to 4 points since averaging beziers affect the 4
|
|
// neighboring points
|
|
overlap = Math.min(size, 4);
|
|
n += Math.min(size, overlap) * 2;
|
|
}
|
|
var knots = [];
|
|
for (var i = 0; i < size; i++)
|
|
knots[i + overlap] = segments[i]._point;
|
|
if (closed) {
|
|
// If we're averaging, add the 4 last points again at the
|
|
// beginning, and the 4 first ones at the end.
|
|
for (var i = 0; i < overlap; i++) {
|
|
knots[i] = segments[i + size - overlap]._point;
|
|
knots[i + size + overlap] = segments[i]._point;
|
|
}
|
|
} else {
|
|
n--;
|
|
}
|
|
// Calculate first Bezier control points
|
|
// Right hand side vector
|
|
var rhs = [];
|
|
|
|
// Set right hand side X values
|
|
for (var i = 1; i < n - 1; i++)
|
|
rhs[i] = 4 * knots[i]._x + 2 * knots[i + 1]._x;
|
|
rhs[0] = knots[0]._x + 2 * knots[1]._x;
|
|
rhs[n - 1] = 3 * knots[n - 1]._x;
|
|
// Get first control points X-values
|
|
var x = getFirstControlPoints(rhs);
|
|
|
|
// Set right hand side Y values
|
|
for (var i = 1; i < n - 1; i++)
|
|
rhs[i] = 4 * knots[i]._y + 2 * knots[i + 1]._y;
|
|
rhs[0] = knots[0]._y + 2 * knots[1]._y;
|
|
rhs[n - 1] = 3 * knots[n - 1]._y;
|
|
// Get first control points Y-values
|
|
var y = getFirstControlPoints(rhs);
|
|
|
|
if (closed) {
|
|
// Do the actual averaging simply by linearly fading between the
|
|
// overlapping values.
|
|
for (var i = 0, j = size; i < overlap; i++, j++) {
|
|
var f1 = i / overlap,
|
|
f2 = 1 - f1,
|
|
ie = i + overlap,
|
|
je = j + overlap;
|
|
// Beginning
|
|
x[j] = x[i] * f1 + x[j] * f2;
|
|
y[j] = y[i] * f1 + y[j] * f2;
|
|
// End
|
|
x[je] = x[ie] * f2 + x[je] * f1;
|
|
y[je] = y[ie] * f2 + y[je] * f1;
|
|
}
|
|
n--;
|
|
}
|
|
var handleIn = null;
|
|
// Now set the calculated handles
|
|
for (var i = overlap; i <= n - overlap; i++) {
|
|
var segment = segments[i - overlap];
|
|
if (handleIn)
|
|
segment.setHandleIn(handleIn.subtract(segment._point));
|
|
if (i < n) {
|
|
segment.setHandleOut(
|
|
new Point(x[i], y[i]).subtract(segment._point));
|
|
handleIn = i < n - 1
|
|
? new Point(
|
|
2 * knots[i + 1]._x - x[i + 1],
|
|
2 * knots[i + 1]._y - y[i + 1])
|
|
: new Point(
|
|
(knots[n]._x + x[n - 1]) / 2,
|
|
(knots[n]._y + y[n - 1]) / 2);
|
|
}
|
|
}
|
|
if (closed && handleIn) {
|
|
var segment = this._segments[0];
|
|
segment.setHandleIn(handleIn.subtract(segment._point));
|
|
}
|
|
}
|
|
};
|
|
}, new function() { // PostScript-style drawing commands
|
|
/**
|
|
* Helper method that returns the current segment and checks if a moveTo()
|
|
* command is required first.
|
|
*/
|
|
function getCurrentSegment(that) {
|
|
var segments = that._segments;
|
|
if (segments.length === 0)
|
|
throw new Error('Use a moveTo() command first');
|
|
return segments[segments.length - 1];
|
|
}
|
|
|
|
return {
|
|
// Note: Documentation for these methods is found in PathItem, as they
|
|
// are considered abstract methods of PathItem and need to be defined in
|
|
// all implementing classes.
|
|
moveTo: function(/* point */) {
|
|
// moveTo should only be called at the beginning of paths. But it
|
|
// can ce called again if there is nothing drawn yet, in which case
|
|
// the first segment gets readjusted.
|
|
var segments = this._segments;
|
|
if (segments.length === 1)
|
|
this.removeSegment(0);
|
|
// Let's not be picky about calling moveTo() when not at the
|
|
// beginning of a path, just bail out:
|
|
if (!segments.length)
|
|
this._add([ new Segment(Point.read(arguments)) ]);
|
|
},
|
|
|
|
moveBy: function(/* point */) {
|
|
throw new Error('moveBy() is unsupported on Path items.');
|
|
},
|
|
|
|
lineTo: function(/* point */) {
|
|
// Let's not be picky about calling moveTo() first:
|
|
this._add([ new Segment(Point.read(arguments)) ]);
|
|
},
|
|
|
|
cubicCurveTo: function(/* handle1, handle2, to */) {
|
|
var handle1 = Point.read(arguments),
|
|
handle2 = Point.read(arguments),
|
|
to = Point.read(arguments),
|
|
// First modify the current segment:
|
|
current = getCurrentSegment(this);
|
|
// Convert to relative values:
|
|
current.setHandleOut(handle1.subtract(current._point));
|
|
// And add the new segment, with handleIn set to c2
|
|
this._add([ new Segment(to, handle2.subtract(to)) ]);
|
|
},
|
|
|
|
quadraticCurveTo: function(/* handle, to */) {
|
|
var handle = Point.read(arguments),
|
|
to = Point.read(arguments),
|
|
current = getCurrentSegment(this)._point;
|
|
// This is exact:
|
|
// If we have the three quad points: A E D,
|
|
// and the cubic is A B C D,
|
|
// B = E + 1/3 (A - E)
|
|
// C = E + 1/3 (D - E)
|
|
this.cubicCurveTo(
|
|
handle.add(current.subtract(handle).multiply(1 / 3)),
|
|
handle.add(to.subtract(handle).multiply(1 / 3)),
|
|
to
|
|
);
|
|
},
|
|
|
|
curveTo: function(/* through, to, parameter */) {
|
|
var through = Point.read(arguments),
|
|
to = Point.read(arguments),
|
|
t = Base.pick(Base.read(arguments), 0.5),
|
|
t1 = 1 - t,
|
|
current = getCurrentSegment(this)._point,
|
|
// handle = (through - (1 - t)^2 * current - t^2 * to) /
|
|
// (2 * (1 - t) * t)
|
|
handle = through.subtract(current.multiply(t1 * t1))
|
|
.subtract(to.multiply(t * t)).divide(2 * t * t1);
|
|
if (handle.isNaN())
|
|
throw new Error(
|
|
'Cannot put a curve through points with parameter = ' + t);
|
|
this.quadraticCurveTo(handle, to);
|
|
},
|
|
|
|
arcTo: function(/* to, clockwise | through, to
|
|
| to, radius, rotation, clockwise, large */) {
|
|
// Get the start point:
|
|
var current = getCurrentSegment(this),
|
|
from = current._point,
|
|
to = Point.read(arguments),
|
|
through,
|
|
// Peek at next value to see if it's clockwise, with true as the
|
|
// default value.
|
|
peek = Base.peek(arguments),
|
|
clockwise = Base.pick(peek, true),
|
|
center, extent, vector, matrix;
|
|
// We're handling three different approaches to drawing arcs in one
|
|
// large function:
|
|
if (typeof clockwise === 'boolean') {
|
|
// #1: arcTo(to, clockwise)
|
|
var middle = from.add(to).divide(2),
|
|
through = middle.add(middle.subtract(from).rotate(
|
|
clockwise ? -90 : 90));
|
|
} else if (Base.remain(arguments) <= 2) {
|
|
// #2: arcTo(through, to)
|
|
through = to;
|
|
to = Point.read(arguments);
|
|
} else {
|
|
// #3: arcTo(to, radius, rotation, clockwise, large)
|
|
// Drawing arcs in SVG style:
|
|
var radius = Size.read(arguments);
|
|
// If rx = 0 or ry = 0 then this arc is treated as a
|
|
// straight line joining the endpoints.
|
|
if (radius.isZero())
|
|
return this.lineTo(to);
|
|
// See for an explanation of the following calculations:
|
|
// http://www.w3.org/TR/SVG/implnote.html#ArcImplementationNotes
|
|
var rotation = Base.read(arguments),
|
|
clockwise = !!Base.read(arguments),
|
|
large = !!Base.read(arguments),
|
|
middle = from.add(to).divide(2),
|
|
pt = from.subtract(middle).rotate(-rotation),
|
|
x = pt.x,
|
|
y = pt.y,
|
|
abs = Math.abs,
|
|
EPSILON = /*#=*/Numerical.EPSILON,
|
|
rx = abs(radius.width),
|
|
ry = abs(radius.height),
|
|
rxSq = rx * rx,
|
|
rySq = ry * ry,
|
|
xSq = x * x,
|
|
ySq = y * y;
|
|
// "...ensure radii are large enough"
|
|
var factor = Math.sqrt(xSq / rxSq + ySq / rySq);
|
|
if (factor > 1) {
|
|
rx *= factor;
|
|
ry *= factor;
|
|
rxSq = rx * rx;
|
|
rySq = ry * ry;
|
|
}
|
|
factor = (rxSq * rySq - rxSq * ySq - rySq * xSq) /
|
|
(rxSq * ySq + rySq * xSq);
|
|
if (abs(factor) < EPSILON)
|
|
factor = 0;
|
|
if (factor < 0)
|
|
throw new Error(
|
|
'Cannot create an arc with the given arguments');
|
|
center = new Point(rx * y / ry, -ry * x / rx)
|
|
// "...where the + sign is chosen if fA != fS,
|
|
// and the - sign is chosen if fA = fS."
|
|
.multiply((large === clockwise ? -1 : 1)
|
|
* Math.sqrt(factor))
|
|
.rotate(rotation).add(middle);
|
|
// Now create a matrix that maps the unit circle to the ellipse,
|
|
// for easier construction below.
|
|
matrix = new Matrix().translate(center).rotate(rotation)
|
|
.scale(rx, ry);
|
|
// Transform from and to to the unit circle coordinate space
|
|
// and calculate start vector and extend from there.
|
|
vector = matrix._inverseTransform(from);
|
|
extent = vector.getDirectedAngle(matrix._inverseTransform(to));
|
|
// "...if fS = 0 and extent is > 0, then subtract 360, whereas
|
|
// if fS = 1 and extend is < 0, then add 360."
|
|
if (!clockwise && extent > 0)
|
|
extent -= 360;
|
|
else if (clockwise && extent < 0)
|
|
extent += 360;
|
|
}
|
|
if (through) {
|
|
// Calculate center, vector and extend for non SVG versions:
|
|
// Construct the two perpendicular middle lines to
|
|
// (from, through) and (through, to), and intersect them to get
|
|
// the center.
|
|
var l1 = new Line(from.add(through).divide(2),
|
|
through.subtract(from).rotate(90), true),
|
|
l2 = new Line(through.add(to).divide(2),
|
|
to.subtract(through).rotate(90), true),
|
|
line = new Line(from, to),
|
|
throughSide = line.getSide(through);
|
|
center = l1.intersect(l2, true);
|
|
// If the two lines are collinear, there cannot be an arc as the
|
|
// circle is infinitely big and has no center point. If side is
|
|
// 0, the connecting arc line of this huge circle is a line
|
|
// between the two points, so we can use #lineTo instead.
|
|
// Otherwise we bail out:
|
|
if (!center) {
|
|
if (!throughSide)
|
|
return this.lineTo(to);
|
|
throw new Error(
|
|
'Cannot create an arc with the given arguments');
|
|
}
|
|
vector = from.subtract(center);
|
|
extent = vector.getDirectedAngle(to.subtract(center));
|
|
var centerSide = line.getSide(center);
|
|
if (centerSide === 0) {
|
|
// If the center is lying on the line, we might have gotten
|
|
// the wrong sign for extent above. Use the sign of the side
|
|
// of the through point.
|
|
extent = throughSide * Math.abs(extent);
|
|
} else if (throughSide === centerSide) {
|
|
// If the center is on the same side of the line (from, to)
|
|
// as the through point, we're extending bellow 180 degrees
|
|
// and need to adapt extent.
|
|
extent += extent < 0 ? 360 : -360;
|
|
}
|
|
}
|
|
var ext = Math.abs(extent),
|
|
count = ext >= 360 ? 4 : Math.ceil(ext / 90),
|
|
inc = extent / count,
|
|
half = inc * Math.PI / 360,
|
|
z = 4 / 3 * Math.sin(half) / (1 + Math.cos(half)),
|
|
segments = [];
|
|
for (var i = 0; i <= count; i++) {
|
|
// Explicitly use to point for last segment, since depending
|
|
// on values the calculation adds imprecision:
|
|
var pt = to,
|
|
out = null;
|
|
if (i < count) {
|
|
out = vector.rotate(90).multiply(z);
|
|
if (matrix) {
|
|
pt = matrix._transformPoint(vector);
|
|
out = matrix._transformPoint(vector.add(out))
|
|
.subtract(pt);
|
|
} else {
|
|
pt = center.add(vector);
|
|
}
|
|
}
|
|
if (i === 0) {
|
|
// Modify startSegment
|
|
current.setHandleOut(out);
|
|
} else {
|
|
// Add new Segment
|
|
var _in = vector.rotate(-90).multiply(z);
|
|
if (matrix) {
|
|
_in = matrix._transformPoint(vector.add(_in))
|
|
.subtract(pt);
|
|
}
|
|
segments.push(new Segment(pt, _in, out));
|
|
}
|
|
vector = vector.rotate(inc);
|
|
}
|
|
// Add all segments at once at the end for higher performance
|
|
this._add(segments);
|
|
},
|
|
|
|
lineBy: function(/* to */) {
|
|
var to = Point.read(arguments),
|
|
current = getCurrentSegment(this)._point;
|
|
this.lineTo(current.add(to));
|
|
},
|
|
|
|
curveBy: function(/* through, to, parameter */) {
|
|
var through = Point.read(arguments),
|
|
to = Point.read(arguments),
|
|
parameter = Base.read(arguments),
|
|
current = getCurrentSegment(this)._point;
|
|
this.curveTo(current.add(through), current.add(to), parameter);
|
|
},
|
|
|
|
cubicCurveBy: function(/* handle1, handle2, to */) {
|
|
var handle1 = Point.read(arguments),
|
|
handle2 = Point.read(arguments),
|
|
to = Point.read(arguments),
|
|
current = getCurrentSegment(this)._point;
|
|
this.cubicCurveTo(current.add(handle1), current.add(handle2),
|
|
current.add(to));
|
|
},
|
|
|
|
quadraticCurveBy: function(/* handle, to */) {
|
|
var handle = Point.read(arguments),
|
|
to = Point.read(arguments),
|
|
current = getCurrentSegment(this)._point;
|
|
this.quadraticCurveTo(current.add(handle), current.add(to));
|
|
},
|
|
|
|
// TODO: Implement version for: (to, radius, rotation, clockwise, large)
|
|
arcBy: function(/* to, clockwise | through, to */) {
|
|
var current = getCurrentSegment(this)._point,
|
|
point = current.add(Point.read(arguments)),
|
|
// Peek at next value to see if it's clockwise, with true as
|
|
// default value.
|
|
clockwise = Base.pick(Base.peek(arguments), true);
|
|
if (typeof clockwise === 'boolean') {
|
|
this.arcTo(point, clockwise);
|
|
} else {
|
|
this.arcTo(point, current.add(Point.read(arguments)));
|
|
}
|
|
},
|
|
|
|
closePath: function(join) {
|
|
this.setClosed(true);
|
|
if (join)
|
|
this.join();
|
|
}
|
|
};
|
|
}, { // A dedicated scope for the tricky bounds calculations
|
|
// We define all the different getBounds functions as static methods on Path
|
|
// and have #_getBounds directly access these. All static bounds functions
|
|
// below have the same first four parameters: segments, closed, style,
|
|
// matrix, so they can be called from #_getBounds() and also be used in
|
|
// Curve. But not all of them use all these parameters, and some define
|
|
// additional ones after.
|
|
|
|
_getBounds: function(getter, matrix) {
|
|
// See #draw() for an explanation of why we can access _style
|
|
// properties directly here:
|
|
return Path[getter](this._segments, this._closed, this.getStyle(),
|
|
matrix);
|
|
},
|
|
|
|
// Mess with indentation in order to get more line-space below:
|
|
statics: {
|
|
/**
|
|
* Determines whether the segments describe a path in clockwise or counter-
|
|
* clockwise orientation.
|
|
*
|
|
* @private
|
|
*/
|
|
isClockwise: function(segments) {
|
|
var sum = 0;
|
|
// Method derived from:
|
|
// http://stackoverflow.com/questions/1165647
|
|
// We treat the curve points and handles as the outline of a polygon of
|
|
// which we determine the orientation using the method of calculating
|
|
// the sum over the edges. This will work even with non-convex polygons,
|
|
// telling you whether it's mostly clockwise
|
|
// TODO: Check if this works correctly for all open paths.
|
|
for (var i = 0, l = segments.length; i < l; i++) {
|
|
var v = Curve.getValues(
|
|
segments[i], segments[i + 1 < l ? i + 1 : 0]);
|
|
for (var j = 2; j < 8; j += 2)
|
|
sum += (v[j - 2] - v[j]) * (v[j + 1] + v[j - 1]);
|
|
}
|
|
return sum > 0;
|
|
},
|
|
|
|
/**
|
|
* Returns the bounding rectangle of the item excluding stroke width.
|
|
*
|
|
* @private
|
|
*/
|
|
getBounds: function(segments, closed, style, matrix, strokePadding) {
|
|
var first = segments[0];
|
|
// If there are no segments, return "empty" rectangle, just like groups,
|
|
// since #bounds is assumed to never return null.
|
|
if (!first)
|
|
return new Rectangle();
|
|
var coords = new Array(6),
|
|
// Make coordinates for first segment available in prevCoords.
|
|
prevCoords = first._transformCoordinates(matrix, new Array(6), false),
|
|
min = prevCoords.slice(0, 2), // Start with values of first point
|
|
max = min.slice(), // clone
|
|
roots = new Array(2);
|
|
|
|
function processSegment(segment) {
|
|
segment._transformCoordinates(matrix, coords, false);
|
|
for (var i = 0; i < 2; i++) {
|
|
Curve._addBounds(
|
|
prevCoords[i], // prev.point
|
|
prevCoords[i + 4], // prev.handleOut
|
|
coords[i + 2], // segment.handleIn
|
|
coords[i], // segment.point,
|
|
i, strokePadding ? strokePadding[i] : 0, min, max, roots);
|
|
}
|
|
// Swap coordinate buffers.
|
|
var tmp = prevCoords;
|
|
prevCoords = coords;
|
|
coords = tmp;
|
|
}
|
|
|
|
for (var i = 1, l = segments.length; i < l; i++)
|
|
processSegment(segments[i]);
|
|
if (closed)
|
|
processSegment(first);
|
|
return new Rectangle(min[0], min[1], max[0] - min[0], max[1] - min[1]);
|
|
},
|
|
|
|
/**
|
|
* Returns the bounding rectangle of the item including stroke width.
|
|
*
|
|
* @private
|
|
*/
|
|
getStrokeBounds: function(segments, closed, style, matrix) {
|
|
// TODO: Find a way to reuse 'bounds' cache instead?
|
|
if (!style.hasStroke())
|
|
return Path.getBounds(segments, closed, style, matrix);
|
|
var length = segments.length - (closed ? 0 : 1),
|
|
radius = style.getStrokeWidth() / 2,
|
|
padding = Path._getPenPadding(radius, matrix),
|
|
bounds = Path.getBounds(segments, closed, style, matrix, padding),
|
|
join = style.getStrokeJoin(),
|
|
cap = style.getStrokeCap(),
|
|
miterLimit = radius * style.getMiterLimit();
|
|
// Create a rectangle of padding size, used for union with bounds
|
|
// further down
|
|
var joinBounds = new Rectangle(new Size(padding).multiply(2));
|
|
|
|
function add(point) {
|
|
bounds = bounds.include(matrix
|
|
? matrix._transformPoint(point, point) : point);
|
|
}
|
|
|
|
function addRound(segment) {
|
|
bounds = bounds.unite(joinBounds.setCenter(matrix
|
|
? matrix._transformPoint(segment._point) : segment._point));
|
|
}
|
|
|
|
function addJoin(segment, join) {
|
|
// When both handles are set in a segment and they are collinear,
|
|
// the join setting is ignored and round is always used.
|
|
var handleIn = segment._handleIn,
|
|
handleOut = segment._handleOut;
|
|
if (join === 'round' || !handleIn.isZero() && !handleOut.isZero()
|
|
&& handleIn.isColinear(handleOut)) {
|
|
addRound(segment);
|
|
} else {
|
|
Path._addBevelJoin(segment, join, radius, miterLimit, add);
|
|
}
|
|
}
|
|
|
|
function addCap(segment, cap) {
|
|
if (cap === 'round') {
|
|
addRound(segment);
|
|
} else {
|
|
Path._addSquareCap(segment, cap, radius, add);
|
|
}
|
|
}
|
|
|
|
for (var i = 1; i < length; i++)
|
|
addJoin(segments[i], join);
|
|
if (closed) {
|
|
addJoin(segments[0], join);
|
|
} else if (length > 0) {
|
|
addCap(segments[0], cap);
|
|
addCap(segments[segments.length - 1], cap);
|
|
}
|
|
return bounds;
|
|
},
|
|
|
|
/**
|
|
* Returns the horizontal and vertical padding that a transformed round
|
|
* stroke adds to the bounding box, by calculating the dimensions of a
|
|
* rotated ellipse.
|
|
*/
|
|
_getPenPadding: function(radius, matrix) {
|
|
if (!matrix)
|
|
return [radius, radius];
|
|
// If a matrix is provided, we need to rotate the stroke circle
|
|
// and calculate the bounding box of the resulting rotated elipse:
|
|
// Get rotated hor and ver vectors, and determine rotation angle
|
|
// and elipse values from them:
|
|
var mx = matrix.shiftless(),
|
|
hor = mx.transform(new Point(radius, 0)),
|
|
ver = mx.transform(new Point(0, radius)),
|
|
phi = hor.getAngleInRadians(),
|
|
a = hor.getLength(),
|
|
b = ver.getLength();
|
|
// Formula for rotated ellipses:
|
|
// x = cx + a*cos(t)*cos(phi) - b*sin(t)*sin(phi)
|
|
// y = cy + b*sin(t)*cos(phi) + a*cos(t)*sin(phi)
|
|
// Derivates (by Wolfram Alpha):
|
|
// derivative of x = cx + a*cos(t)*cos(phi) - b*sin(t)*sin(phi)
|
|
// dx/dt = a sin(t) cos(phi) + b cos(t) sin(phi) = 0
|
|
// derivative of y = cy + b*sin(t)*cos(phi) + a*cos(t)*sin(phi)
|
|
// dy/dt = b cos(t) cos(phi) - a sin(t) sin(phi) = 0
|
|
// This can be simplified to:
|
|
// tan(t) = -b * tan(phi) / a // x
|
|
// tan(t) = b * cot(phi) / a // y
|
|
// Solving for t gives:
|
|
// t = pi * n - arctan(b * tan(phi) / a) // x
|
|
// t = pi * n + arctan(b * cot(phi) / a)
|
|
// = pi * n + arctan(b / tan(phi) / a) // y
|
|
var sin = Math.sin(phi),
|
|
cos = Math.cos(phi),
|
|
tan = Math.tan(phi),
|
|
tx = -Math.atan(b * tan / a),
|
|
ty = Math.atan(b / (tan * a));
|
|
// Due to symetry, we don't need to cycle through pi * n solutions:
|
|
return [Math.abs(a * Math.cos(tx) * cos - b * Math.sin(tx) * sin),
|
|
Math.abs(b * Math.sin(ty) * cos + a * Math.cos(ty) * sin)];
|
|
},
|
|
|
|
_addBevelJoin: function(segment, join, radius, miterLimit, addPoint, area) {
|
|
// Handles both 'bevel' and 'miter' joins, as they share a lot of code.
|
|
var curve2 = segment.getCurve(),
|
|
curve1 = curve2.getPrevious(),
|
|
point = curve2.getPointAt(0, true),
|
|
normal1 = curve1.getNormalAt(1, true),
|
|
normal2 = curve2.getNormalAt(0, true),
|
|
step = normal1.getDirectedAngle(normal2) < 0 ? -radius : radius;
|
|
normal1.setLength(step);
|
|
normal2.setLength(step);
|
|
if (area) {
|
|
addPoint(point);
|
|
addPoint(point.add(normal1));
|
|
}
|
|
if (join === 'miter') {
|
|
// Intersect the two lines
|
|
var corner = new Line(
|
|
point.add(normal1),
|
|
new Point(-normal1.y, normal1.x), true
|
|
).intersect(new Line(
|
|
point.add(normal2),
|
|
new Point(-normal2.y, normal2.x), true
|
|
), true);
|
|
// See if we actually get a bevel point and if its distance is below
|
|
// the miterLimit. If not, make a normal bevel.
|
|
if (corner && point.getDistance(corner) <= miterLimit) {
|
|
addPoint(corner);
|
|
if (!area)
|
|
return;
|
|
}
|
|
}
|
|
// Produce a normal bevel
|
|
if (!area)
|
|
addPoint(point.add(normal1));
|
|
addPoint(point.add(normal2));
|
|
},
|
|
|
|
_addSquareCap: function(segment, cap, radius, addPoint, area) {
|
|
// Handles both 'square' and 'butt' caps, as they share a lot of code.
|
|
// Calculate the corner points of butt and square caps
|
|
var point = segment._point,
|
|
loc = segment.getLocation(),
|
|
normal = loc.getNormal().normalize(radius);
|
|
if (area) {
|
|
addPoint(point.subtract(normal));
|
|
addPoint(point.add(normal));
|
|
}
|
|
// For square caps, we need to step away from point in the direction of
|
|
// the tangent, which is the rotated normal.
|
|
// Checking loc.getParameter() for 0 is to see whether this is the first
|
|
// or the last segment of the open path, in order to determine in which
|
|
// direction to move the point.
|
|
if (cap === 'square')
|
|
point = point.add(normal.rotate(loc.getParameter() === 0 ? -90 : 90));
|
|
addPoint(point.add(normal));
|
|
addPoint(point.subtract(normal));
|
|
},
|
|
|
|
/**
|
|
* Returns the bounding rectangle of the item including handles.
|
|
*
|
|
* @private
|
|
*/
|
|
getHandleBounds: function(segments, closed, style, matrix, strokePadding,
|
|
joinPadding) {
|
|
var coords = new Array(6),
|
|
x1 = Infinity,
|
|
x2 = -x1,
|
|
y1 = x1,
|
|
y2 = x2;
|
|
for (var i = 0, l = segments.length; i < l; i++) {
|
|
var segment = segments[i];
|
|
segment._transformCoordinates(matrix, coords, false);
|
|
for (var j = 0; j < 6; j += 2) {
|
|
// Use different padding for points or handles
|
|
var padding = j === 0 ? joinPadding : strokePadding,
|
|
paddingX = padding ? padding[0] : 0,
|
|
paddingY = padding ? padding[1] : 0,
|
|
x = coords[j],
|
|
y = coords[j + 1],
|
|
xn = x - paddingX,
|
|
xx = x + paddingX,
|
|
yn = y - paddingY,
|
|
yx = y + paddingY;
|
|
if (xn < x1) x1 = xn;
|
|
if (xx > x2) x2 = xx;
|
|
if (yn < y1) y1 = yn;
|
|
if (yx > y2) y2 = yx;
|
|
}
|
|
}
|
|
return new Rectangle(x1, y1, x2 - x1, y2 - y1);
|
|
},
|
|
|
|
/**
|
|
* Returns the rough bounding rectangle of the item that is sure to include
|
|
* all of the drawing, including stroke width.
|
|
*
|
|
* @private
|
|
*/
|
|
getRoughBounds: function(segments, closed, style, matrix) {
|
|
// Delegate to handleBounds, but pass on radius values for stroke and
|
|
// joins. Hanlde miter joins specially, by passing the largets radius
|
|
// possible.
|
|
var strokeRadius = style.hasStroke() ? style.getStrokeWidth() / 2 : 0,
|
|
joinRadius = strokeRadius;
|
|
if (strokeRadius > 0) {
|
|
if (style.getStrokeJoin() === 'miter')
|
|
joinRadius = strokeRadius * style.getMiterLimit();
|
|
if (style.getStrokeCap() === 'square')
|
|
joinRadius = Math.max(joinRadius, strokeRadius * Math.sqrt(2));
|
|
}
|
|
return Path.getHandleBounds(segments, closed, style, matrix,
|
|
Path._getPenPadding(strokeRadius, matrix),
|
|
Path._getPenPadding(joinRadius, matrix));
|
|
}
|
|
}});
|