2011-03-06 19:50:44 -05:00
|
|
|
/*
|
2013-01-28 21:03:27 -05:00
|
|
|
* Paper.js - The Swiss Army Knife of Vector Graphics Scripting.
|
2011-03-07 20:41:50 -05:00
|
|
|
* http://paperjs.org/
|
2011-06-30 06:01:51 -04:00
|
|
|
*
|
2014-01-03 19:47:16 -05:00
|
|
|
* Copyright (c) 2011 - 2014, Juerg Lehni & Jonathan Puckey
|
|
|
|
* http://scratchdisk.com/ & http://jonathanpuckey.com/
|
2011-06-30 06:01:51 -04:00
|
|
|
*
|
2011-07-01 06:17:45 -04:00
|
|
|
* Distributed under the MIT license. See LICENSE file for details.
|
|
|
|
*
|
2011-03-07 20:41:50 -05:00
|
|
|
* All rights reserved.
|
2011-03-06 19:50:44 -05:00
|
|
|
*/
|
|
|
|
|
2011-06-22 18:56:05 -04:00
|
|
|
/**
|
|
|
|
* @name Segment
|
2011-06-30 06:01:51 -04:00
|
|
|
*
|
2011-06-27 09:07:08 -04:00
|
|
|
* @class The Segment object represents the points of a path through which its
|
|
|
|
* {@link Curve} objects pass. The segments of a path can be accessed through
|
|
|
|
* its {@link Path#segments} array.
|
2011-06-30 06:01:51 -04:00
|
|
|
*
|
2011-06-27 09:07:08 -04:00
|
|
|
* Each segment consists of an anchor point ({@link Segment#point}) and
|
|
|
|
* optionaly an incoming and an outgoing handle ({@link Segment#handleIn} and
|
|
|
|
* {@link Segment#handleOut}), describing the tangents of the two {@link Curve}
|
|
|
|
* objects that are connected by this segment.
|
2011-06-22 18:56:05 -04:00
|
|
|
*/
|
2013-05-27 15:48:58 -04:00
|
|
|
var Segment = Base.extend(/** @lends Segment# */{
|
2014-08-16 13:24:54 -04:00
|
|
|
_class: 'Segment',
|
|
|
|
beans: true,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new Segment object.
|
|
|
|
*
|
|
|
|
* @name Segment#initialize
|
|
|
|
* @param {Point} [point={x: 0, y: 0}] the anchor point of the segment
|
|
|
|
* @param {Point} [handleIn={x: 0, y: 0}] the handle point relative to the
|
2015-06-16 11:50:37 -04:00
|
|
|
* anchor point of the segment that describes the in tangent of the
|
|
|
|
* segment
|
2014-08-16 13:24:54 -04:00
|
|
|
* @param {Point} [handleOut={x: 0, y: 0}] the handle point relative to the
|
2015-06-16 11:50:37 -04:00
|
|
|
* anchor point of the segment that describes the out tangent of the
|
|
|
|
* segment
|
2014-08-16 13:24:54 -04:00
|
|
|
*
|
|
|
|
* @example {@paperscript}
|
|
|
|
* var handleIn = new Point(-80, -100);
|
|
|
|
* var handleOut = new Point(80, 100);
|
|
|
|
*
|
|
|
|
* var firstPoint = new Point(100, 50);
|
|
|
|
* var firstSegment = new Segment(firstPoint, null, handleOut);
|
|
|
|
*
|
|
|
|
* var secondPoint = new Point(300, 50);
|
|
|
|
* var secondSegment = new Segment(secondPoint, handleIn, null);
|
|
|
|
*
|
|
|
|
* var path = new Path(firstSegment, secondSegment);
|
|
|
|
* path.strokeColor = 'black';
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* Creates a new Segment object.
|
|
|
|
*
|
|
|
|
* @name Segment#initialize
|
|
|
|
* @param {Object} object an object literal containing properties to
|
2015-06-16 11:50:37 -04:00
|
|
|
* be set on the segment
|
2014-08-16 13:24:54 -04:00
|
|
|
*
|
|
|
|
* @example {@paperscript}
|
|
|
|
* // Creating segments using object notation:
|
|
|
|
* var firstSegment = new Segment({
|
|
|
|
* point: [100, 50],
|
|
|
|
* handleOut: [80, 100]
|
|
|
|
* });
|
|
|
|
*
|
|
|
|
* var secondSegment = new Segment({
|
|
|
|
* point: [300, 50],
|
|
|
|
* handleIn: [-80, -100]
|
|
|
|
* });
|
|
|
|
*
|
|
|
|
* var path = new Path({
|
|
|
|
* segments: [firstSegment, secondSegment],
|
|
|
|
* strokeColor: 'black'
|
|
|
|
* });
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* Creates a new Segment object.
|
|
|
|
*
|
|
|
|
* @param {Number} x the x coordinate of the segment point
|
|
|
|
* @param {Number} y the y coordinate of the segment point
|
2015-06-16 11:50:37 -04:00
|
|
|
* @param {Number} inX the x coordinate of the the handle point relative to
|
|
|
|
* the anchor point of the segment that describes the in tangent of the
|
|
|
|
* segment
|
|
|
|
* @param {Number} inY the y coordinate of the the handle point relative to
|
|
|
|
* the anchor point of the segment that describes the in tangent of the
|
|
|
|
* segment
|
|
|
|
* @param {Number} outX the x coordinate of the the handle point relative to
|
|
|
|
* the anchor point of the segment that describes the out tangent of the
|
|
|
|
* segment
|
|
|
|
* @param {Number} outY the y coordinate of the the handle point relative to
|
|
|
|
* the anchor point of the segment that describes the out tangent of the
|
|
|
|
* segment
|
2014-08-16 13:24:54 -04:00
|
|
|
*
|
|
|
|
* @example {@paperscript}
|
|
|
|
* var inX = -80;
|
|
|
|
* var inY = -100;
|
|
|
|
* var outX = 80;
|
|
|
|
* var outY = 100;
|
|
|
|
*
|
|
|
|
* var x = 100;
|
|
|
|
* var y = 50;
|
|
|
|
* var firstSegment = new Segment(x, y, inX, inY, outX, outY);
|
|
|
|
*
|
|
|
|
* var x2 = 300;
|
|
|
|
* var y2 = 50;
|
|
|
|
* var secondSegment = new Segment( x2, y2, inX, inY, outX, outY);
|
|
|
|
*
|
|
|
|
* var path = new Path(firstSegment, secondSegment);
|
|
|
|
* path.strokeColor = 'black';
|
|
|
|
* @ignore
|
|
|
|
*/
|
|
|
|
initialize: function Segment(arg0, arg1, arg2, arg3, arg4, arg5) {
|
|
|
|
var count = arguments.length,
|
|
|
|
point, handleIn, handleOut;
|
|
|
|
// TODO: Use Point.read or Point.readNamed to read these?
|
|
|
|
if (count === 0) {
|
|
|
|
// Nothing
|
|
|
|
} else if (count === 1) {
|
|
|
|
// Note: This copies from existing segments through accessors.
|
|
|
|
if (arg0.point) {
|
|
|
|
point = arg0.point;
|
|
|
|
handleIn = arg0.handleIn;
|
|
|
|
handleOut = arg0.handleOut;
|
|
|
|
} else {
|
|
|
|
point = arg0;
|
|
|
|
}
|
|
|
|
} else if (count === 2 && typeof arg0 === 'number') {
|
|
|
|
point = arguments;
|
|
|
|
} else if (count <= 3) {
|
|
|
|
point = arg0;
|
|
|
|
// Doesn't matter if these arguments exist, SegmentPointcreate
|
|
|
|
// produces creates points with (0, 0) otherwise
|
|
|
|
handleIn = arg1;
|
|
|
|
handleOut = arg2;
|
|
|
|
} else { // Read points from the arguments list as a row of numbers
|
|
|
|
point = arg0 !== undefined ? [ arg0, arg1 ] : null;
|
|
|
|
handleIn = arg2 !== undefined ? [ arg2, arg3 ] : null;
|
|
|
|
handleOut = arg4 !== undefined ? [ arg4, arg5 ] : null;
|
|
|
|
}
|
|
|
|
new SegmentPoint(point, this, '_point');
|
|
|
|
new SegmentPoint(handleIn, this, '_handleIn');
|
|
|
|
new SegmentPoint(handleOut, this, '_handleOut');
|
|
|
|
},
|
|
|
|
|
|
|
|
_serialize: function(options) {
|
2015-08-17 08:31:23 -04:00
|
|
|
// If it is straight, only serialize point, otherwise handles too.
|
|
|
|
return Base.serialize(this.isStraight() ? this._point
|
2014-08-16 13:24:54 -04:00
|
|
|
: [this._point, this._handleIn, this._handleOut],
|
|
|
|
options, true);
|
|
|
|
},
|
|
|
|
|
|
|
|
_changed: function(point) {
|
|
|
|
var path = this._path;
|
|
|
|
if (!path)
|
|
|
|
return;
|
|
|
|
// Delegate changes to affected curves if they exist.
|
|
|
|
var curves = path._curves,
|
|
|
|
index = this._index,
|
2014-09-27 16:09:25 -04:00
|
|
|
curve;
|
2014-08-16 13:24:54 -04:00
|
|
|
if (curves) {
|
|
|
|
// Updated the neighboring affected curves, depending on which point
|
|
|
|
// is changing.
|
|
|
|
// TODO: Consider exposing these curves too, through #curveIn,
|
|
|
|
// and #curveOut, next to #curve?
|
|
|
|
if ((!point || point === this._point || point === this._handleIn)
|
2014-09-27 16:09:25 -04:00
|
|
|
&& (curve = index > 0 ? curves[index - 1] : path._closed
|
|
|
|
? curves[curves.length - 1] : null))
|
|
|
|
curve._changed();
|
|
|
|
// No wrap around needed for outgoing curve, as only closed paths
|
|
|
|
// will have one for the last segment.
|
2014-08-16 13:24:54 -04:00
|
|
|
if ((!point || point === this._point || point === this._handleOut)
|
2014-09-27 16:09:25 -04:00
|
|
|
&& (curve = curves[index]))
|
|
|
|
curve._changed();
|
2014-08-16 13:24:54 -04:00
|
|
|
}
|
|
|
|
path._changed(/*#=*/Change.SEGMENTS);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The anchor point of the segment.
|
|
|
|
*
|
|
|
|
* @type Point
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getPoint: function() {
|
|
|
|
return this._point;
|
|
|
|
},
|
|
|
|
|
|
|
|
setPoint: function(/* point */) {
|
|
|
|
var point = Point.read(arguments);
|
|
|
|
// Do not replace the internal object but update it instead, so
|
|
|
|
// references to it are kept alive.
|
|
|
|
this._point.set(point.x, point.y);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The handle point relative to the anchor point of the segment that
|
|
|
|
* describes the in tangent of the segment.
|
|
|
|
*
|
|
|
|
* @type Point
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getHandleIn: function() {
|
|
|
|
return this._handleIn;
|
|
|
|
},
|
|
|
|
|
|
|
|
setHandleIn: function(/* point */) {
|
|
|
|
var point = Point.read(arguments);
|
|
|
|
// See #setPoint:
|
|
|
|
this._handleIn.set(point.x, point.y);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The handle point relative to the anchor point of the segment that
|
|
|
|
* describes the out tangent of the segment.
|
|
|
|
*
|
|
|
|
* @type Point
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getHandleOut: function() {
|
|
|
|
return this._handleOut;
|
|
|
|
},
|
|
|
|
|
|
|
|
setHandleOut: function(/* point */) {
|
|
|
|
var point = Point.read(arguments);
|
|
|
|
// See #setPoint:
|
|
|
|
this._handleOut.set(point.x, point.y);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
2015-08-17 08:31:23 -04:00
|
|
|
* Checks whether the segment has curve handles defined, meaning it is not
|
|
|
|
* a straight segment.
|
2014-08-16 13:24:54 -04:00
|
|
|
*
|
2015-08-17 08:31:23 -04:00
|
|
|
* @return {Boolean} {@true if the segment has handles defined}
|
|
|
|
* @see Curve#hasHandles()
|
|
|
|
* @see Path#hasHandles()
|
2014-08-16 13:24:54 -04:00
|
|
|
*/
|
2015-08-17 08:31:23 -04:00
|
|
|
hasHandles: function() {
|
|
|
|
return !this.isStraight();
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
|
|
|
|
2015-08-17 08:31:23 -04:00
|
|
|
/**
|
2015-08-22 16:06:42 -04:00
|
|
|
* Checks whether the segment is straight, meaning it has no curve handles
|
|
|
|
* defined. If two straight segments follow each each other in a path, the
|
|
|
|
* curve between them will appear as a straight line.
|
|
|
|
* Note that this is not the same as {@link #isLinear()}, which performs a
|
|
|
|
* full linearity check that includes handles running collinear to the line
|
|
|
|
* direction.
|
2015-08-17 08:31:23 -04:00
|
|
|
*
|
|
|
|
* @return {Boolean} {@true if the segment is straight}
|
2015-08-22 16:06:42 -04:00
|
|
|
* @see Curve#isStraight()
|
2015-08-17 08:31:23 -04:00
|
|
|
*/
|
|
|
|
isStraight: function() {
|
|
|
|
return this._handleIn.isZero() && this._handleOut.isZero();
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
|
|
|
|
2015-08-17 08:31:23 -04:00
|
|
|
/**
|
|
|
|
* Checks if the curve that starts in this segment appears as a line. This
|
|
|
|
* can mean that it has no handles defined, or that the handles run
|
|
|
|
* collinear with the line.
|
|
|
|
*
|
|
|
|
* @return {Boolean} {@true if the curve is linear}
|
|
|
|
* @see Curve#isLinear()
|
|
|
|
* @see Path#isLinear()
|
|
|
|
*/
|
|
|
|
isLinear: function() {
|
2015-08-19 05:43:37 -04:00
|
|
|
return Segment.isLinear(this, this.getNext());
|
2015-08-17 08:31:23 -04:00
|
|
|
},
|
2014-08-16 13:24:54 -04:00
|
|
|
|
|
|
|
/**
|
2015-08-17 08:33:48 -04:00
|
|
|
* Checks if the the two segments are the beginning of two lines that are
|
|
|
|
* collinear, meaning they run in parallel.
|
|
|
|
*
|
|
|
|
* @param {Segment} the other segment to check against
|
|
|
|
* @return {Boolean} {@true if the two lines are collinear}
|
|
|
|
* @see Curve#isCollinear(curve)
|
2014-08-16 13:24:54 -04:00
|
|
|
*/
|
2015-06-16 12:39:52 -04:00
|
|
|
isCollinear: function(segment) {
|
2015-08-19 05:43:37 -04:00
|
|
|
return Segment.isCollinear(this, this.getNext(),
|
|
|
|
segment, segment.getNext());
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
|
|
|
|
2015-06-16 12:39:52 -04:00
|
|
|
// TODO: Remove version with typo after a while (deprecated June 2015)
|
|
|
|
isColinear: '#isCollinear',
|
|
|
|
|
2015-08-17 08:33:48 -04:00
|
|
|
/**
|
|
|
|
* Checks if the segment is connecting two lines that are orthogonal,
|
|
|
|
* meaning they connect at an 90° angle.
|
|
|
|
*
|
|
|
|
* @return {Boolean} {@true if the two lines connected by this segment are
|
|
|
|
* orthogonal}
|
|
|
|
*/
|
2014-08-16 13:24:54 -04:00
|
|
|
isOrthogonal: function() {
|
2015-08-19 05:43:37 -04:00
|
|
|
return Segment.isOrthogonal(this.getPrevious(), this, this.getNext());
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
2015-08-17 08:33:48 -04:00
|
|
|
* Checks if the segment is the beginning of an orthogonal arc, as used in
|
|
|
|
* the construction of circles and ellipses.
|
|
|
|
*
|
|
|
|
* @return {Boolean} {@true if the segment is the beginning of an orthogonal
|
|
|
|
* arc}
|
|
|
|
* @see Curve#isOrthogonalArc()
|
2014-08-16 13:24:54 -04:00
|
|
|
*/
|
2015-08-17 08:33:48 -04:00
|
|
|
isOrthogonalArc: function() {
|
2015-08-19 05:43:37 -04:00
|
|
|
return Segment.isOrthogonalArc(this, this.getNext());
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
|
|
|
|
2015-08-17 08:33:48 -04:00
|
|
|
// TODO: Remove a while (deprecated August 2015)
|
|
|
|
isArc: '#isOrthogonalArc',
|
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
_selectionState: 0,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Specifies whether the {@link #point} of the segment is selected.
|
|
|
|
* @type Boolean
|
|
|
|
* @bean
|
|
|
|
* @example {@paperscript}
|
|
|
|
* var path = new Path.Circle({
|
|
|
|
* center: [80, 50],
|
|
|
|
* radius: 40
|
|
|
|
* });
|
|
|
|
*
|
|
|
|
* // Select the third segment point:
|
|
|
|
* path.segments[2].selected = true;
|
|
|
|
*/
|
|
|
|
isSelected: function(_point) {
|
|
|
|
var state = this._selectionState;
|
|
|
|
return !_point ? !!(state & /*#=*/SelectionState.SEGMENT)
|
|
|
|
: _point === this._point ? !!(state & /*#=*/SelectionState.POINT)
|
|
|
|
: _point === this._handleIn ? !!(state & /*#=*/SelectionState.HANDLE_IN)
|
|
|
|
: _point === this._handleOut ? !!(state & /*#=*/SelectionState.HANDLE_OUT)
|
|
|
|
: false;
|
|
|
|
},
|
|
|
|
|
|
|
|
setSelected: function(selected, _point) {
|
|
|
|
var path = this._path,
|
|
|
|
selected = !!selected, // convert to boolean
|
|
|
|
state = this._selectionState,
|
|
|
|
oldState = state,
|
|
|
|
flag = !_point ? /*#=*/SelectionState.SEGMENT
|
|
|
|
: _point === this._point ? /*#=*/SelectionState.POINT
|
|
|
|
: _point === this._handleIn ? /*#=*/SelectionState.HANDLE_IN
|
|
|
|
: _point === this._handleOut ? /*#=*/SelectionState.HANDLE_OUT
|
|
|
|
: 0;
|
|
|
|
if (selected) {
|
|
|
|
state |= flag;
|
|
|
|
} else {
|
|
|
|
state &= ~flag;
|
|
|
|
}
|
|
|
|
// Set the selection state even if path is not defined yet, to allow
|
|
|
|
// selected segments to be inserted into paths and make JSON
|
|
|
|
// deserialization work.
|
|
|
|
this._selectionState = state;
|
|
|
|
// If the selection state of the segment has changed, we need to let
|
|
|
|
// it's path know and possibly add or remove it from
|
|
|
|
// project._selectedItems
|
|
|
|
if (path && state !== oldState) {
|
|
|
|
path._updateSelection(this, oldState, state);
|
|
|
|
// Let path know that we changed something and the view should be
|
|
|
|
// redrawn
|
|
|
|
path._changed(/*#=*/Change.ATTRIBUTE);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@grouptitle Hierarchy}
|
|
|
|
*
|
|
|
|
* The index of the segment in the {@link Path#segments} array that the
|
|
|
|
* segment belongs to.
|
|
|
|
*
|
|
|
|
* @type Number
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getIndex: function() {
|
|
|
|
return this._index !== undefined ? this._index : null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The path that the segment belongs to.
|
|
|
|
*
|
|
|
|
* @type Path
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getPath: function() {
|
|
|
|
return this._path || null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The curve that the segment belongs to. For the last segment of an open
|
|
|
|
* path, the previous segment is returned.
|
|
|
|
*
|
|
|
|
* @type Curve
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getCurve: function() {
|
|
|
|
var path = this._path,
|
|
|
|
index = this._index;
|
|
|
|
if (path) {
|
|
|
|
// The last segment of an open path belongs to the last curve.
|
|
|
|
if (index > 0 && !path._closed
|
|
|
|
&& index === path._segments.length - 1)
|
|
|
|
index--;
|
|
|
|
return path.getCurves()[index] || null;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The curve location that describes this segment's position ont the path.
|
|
|
|
*
|
|
|
|
* @type CurveLocation
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getLocation: function() {
|
|
|
|
var curve = this.getCurve();
|
|
|
|
return curve
|
|
|
|
// Determine whether the parameter for this segment is 0 or 1.
|
|
|
|
? new CurveLocation(curve, this === curve._segment1 ? 0 : 1)
|
|
|
|
: null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@grouptitle Sibling Segments}
|
|
|
|
*
|
|
|
|
* The next segment in the {@link Path#segments} array that the segment
|
|
|
|
* belongs to. If the segments belongs to a closed path, the first segment
|
|
|
|
* is returned for the last segment of the path.
|
|
|
|
*
|
|
|
|
* @type Segment
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getNext: function() {
|
|
|
|
var segments = this._path && this._path._segments;
|
|
|
|
return segments && (segments[this._index + 1]
|
|
|
|
|| this._path._closed && segments[0]) || null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The previous segment in the {@link Path#segments} array that the
|
|
|
|
* segment belongs to. If the segments belongs to a closed path, the last
|
|
|
|
* segment is returned for the first segment of the path.
|
|
|
|
*
|
|
|
|
* @type Segment
|
|
|
|
* @bean
|
|
|
|
*/
|
|
|
|
getPrevious: function() {
|
|
|
|
var segments = this._path && this._path._segments;
|
|
|
|
return segments && (segments[this._index - 1]
|
|
|
|
|| this._path._closed && segments[segments.length - 1]) || null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the reversed the segment, without modifying the segment itself.
|
|
|
|
* @return {Segment} the reversed segment
|
|
|
|
*/
|
|
|
|
reverse: function() {
|
|
|
|
return new Segment(this._point, this._handleOut, this._handleIn);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes the segment from the path that it belongs to.
|
|
|
|
* @return {Boolean} {@true if the segment was removed}
|
|
|
|
*/
|
|
|
|
remove: function() {
|
|
|
|
return this._path ? !!this._path.removeSegment(this._index) : false;
|
|
|
|
},
|
|
|
|
|
|
|
|
clone: function() {
|
|
|
|
return new Segment(this._point, this._handleIn, this._handleOut);
|
|
|
|
},
|
|
|
|
|
|
|
|
equals: function(segment) {
|
|
|
|
return segment === this || segment && this._class === segment._class
|
|
|
|
&& this._point.equals(segment._point)
|
|
|
|
&& this._handleIn.equals(segment._handleIn)
|
|
|
|
&& this._handleOut.equals(segment._handleOut)
|
|
|
|
|| false;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return {String} a string representation of the segment
|
|
|
|
*/
|
|
|
|
toString: function() {
|
|
|
|
var parts = [ 'point: ' + this._point ];
|
|
|
|
if (!this._handleIn.isZero())
|
|
|
|
parts.push('handleIn: ' + this._handleIn);
|
|
|
|
if (!this._handleOut.isZero())
|
|
|
|
parts.push('handleOut: ' + this._handleOut);
|
|
|
|
return '{ ' + parts.join(', ') + ' }';
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Transform the segment by the specified matrix.
|
|
|
|
*
|
|
|
|
* @param {Matrix} matrix the matrix to transform the segment by
|
|
|
|
*/
|
|
|
|
transform: function(matrix) {
|
|
|
|
this._transformCoordinates(matrix, new Array(6), true);
|
|
|
|
this._changed();
|
|
|
|
},
|
|
|
|
|
|
|
|
_transformCoordinates: function(matrix, coords, change) {
|
|
|
|
// Use matrix.transform version() that takes arrays of multiple
|
|
|
|
// points for largely improved performance, as no calls to
|
|
|
|
// Point.read() and Point constructors are necessary.
|
|
|
|
var point = this._point,
|
|
|
|
// If change is true, only transform handles if they are set, as
|
|
|
|
// _transformCoordinates is called only to change the segment, no
|
|
|
|
// to receive the coords.
|
|
|
|
// This saves some computation time. If change is false, always
|
|
|
|
// use the real handles, as we just want to receive a filled
|
|
|
|
// coords array for getBounds().
|
2014-08-25 06:49:14 -04:00
|
|
|
handleIn = !change || !this._handleIn.isZero()
|
2014-08-16 13:24:54 -04:00
|
|
|
? this._handleIn : null,
|
|
|
|
handleOut = !change || !this._handleOut.isZero()
|
|
|
|
? this._handleOut : null,
|
|
|
|
x = point._x,
|
|
|
|
y = point._y,
|
|
|
|
i = 2;
|
|
|
|
coords[0] = x;
|
|
|
|
coords[1] = y;
|
|
|
|
// We need to convert handles to absolute coordinates in order
|
|
|
|
// to transform them.
|
|
|
|
if (handleIn) {
|
|
|
|
coords[i++] = handleIn._x + x;
|
|
|
|
coords[i++] = handleIn._y + y;
|
|
|
|
}
|
|
|
|
if (handleOut) {
|
|
|
|
coords[i++] = handleOut._x + x;
|
|
|
|
coords[i++] = handleOut._y + y;
|
|
|
|
}
|
|
|
|
// If no matrix was previded, this was just called to get the coords and
|
|
|
|
// we are done now.
|
|
|
|
if (matrix) {
|
|
|
|
matrix._transformCoordinates(coords, coords, i / 2);
|
|
|
|
x = coords[0];
|
|
|
|
y = coords[1];
|
|
|
|
if (change) {
|
|
|
|
// If change is true, we need to set the new values back
|
|
|
|
point._x = x;
|
|
|
|
point._y = y;
|
|
|
|
i = 2;
|
|
|
|
if (handleIn) {
|
|
|
|
handleIn._x = coords[i++] - x;
|
|
|
|
handleIn._y = coords[i++] - y;
|
|
|
|
}
|
|
|
|
if (handleOut) {
|
|
|
|
handleOut._x = coords[i++] - x;
|
|
|
|
handleOut._y = coords[i++] - y;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// We want to receive the results in coords, so make sure
|
|
|
|
// handleIn and out are defined too, even if they're 0
|
|
|
|
if (!handleIn) {
|
|
|
|
coords[i++] = x;
|
|
|
|
coords[i++] = y;
|
|
|
|
}
|
|
|
|
if (!handleOut) {
|
|
|
|
coords[i++] = x;
|
|
|
|
coords[i++] = y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return coords;
|
2015-08-19 05:43:37 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
statics: {
|
|
|
|
// These statics are shared between Segment and Curve, for versions of
|
|
|
|
// these methods that are implemented in both places.
|
|
|
|
|
|
|
|
isLinear: function(seg1, seg2) {
|
|
|
|
var l = seg2._point.subtract(seg1._point);
|
|
|
|
return l.isCollinear(seg1._handleOut)
|
|
|
|
&& l.isCollinear(seg2._handleIn);
|
|
|
|
},
|
|
|
|
|
|
|
|
isCollinear: function(seg1, seg2, seg3, seg4) {
|
2015-08-24 06:49:33 -04:00
|
|
|
// TODO: This assumes isStraight(), while isLinear() allows handles!
|
2015-08-19 05:43:37 -04:00
|
|
|
return seg1._handleOut.isZero() && seg2._handleIn.isZero()
|
|
|
|
&& seg3._handleOut.isZero() && seg4._handleIn.isZero()
|
|
|
|
&& seg2._point.subtract(seg1._point).isCollinear(
|
|
|
|
seg4._point.subtract(seg3._point));
|
|
|
|
},
|
|
|
|
|
|
|
|
isOrthogonal: function(seg1, seg2, seg3) {
|
2015-08-24 06:49:33 -04:00
|
|
|
// TODO: This assumes isStraight(), while isLinear() allows handles!
|
2015-08-19 05:43:37 -04:00
|
|
|
return seg1._handleOut.isZero() && seg2._handleIn.isZero()
|
2015-08-24 06:49:33 -04:00
|
|
|
&& seg2._handleOut.isZero() && seg3._handleIn.isZero()
|
|
|
|
&& seg2._point.subtract(seg1._point).isOrthogonal(
|
2015-08-19 05:43:37 -04:00
|
|
|
seg3._point.subtract(seg2._point));
|
|
|
|
},
|
|
|
|
|
|
|
|
isOrthogonalArc: function(seg1, seg2) {
|
|
|
|
var handle1 = seg1._handleOut,
|
|
|
|
handle2 = seg2._handleIn,
|
|
|
|
kappa = /*#=*/Numerical.KAPPA;
|
|
|
|
// Look at handle length and the distance to the imaginary corner
|
|
|
|
// point and see if it their relation is kappa.
|
|
|
|
if (handle1.isOrthogonal(handle2)) {
|
|
|
|
var pt1 = seg1._point,
|
|
|
|
pt2 = seg2._point,
|
|
|
|
// Find the corner point by intersecting the lines described
|
|
|
|
// by both handles:
|
|
|
|
corner = new Line(pt1, handle1, true).intersect(
|
|
|
|
new Line(pt2, handle2, true), true);
|
|
|
|
return corner && Numerical.isZero(handle1.getLength() /
|
|
|
|
corner.subtract(pt1).getLength() - kappa)
|
|
|
|
&& Numerical.isZero(handle2.getLength() /
|
|
|
|
corner.subtract(pt2).getLength() - kappa);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
2014-08-16 13:24:54 -04:00
|
|
|
}
|
2011-02-13 11:26:24 -05:00
|
|
|
});
|