Implement Curve#isLinear() to check for parametrical linearity.

Along with some unit tests for it.
This commit is contained in:
Jürg Lehni 2015-09-06 17:56:12 +02:00
parent 9d12a0a82c
commit 31d9e1cd6e
2 changed files with 84 additions and 56 deletions

View file

@ -797,8 +797,8 @@ statics: {
* @type Rectangle * @type Rectangle
* @ignore * @ignore
*/ */
}), new function() { // Injection scope for tests }), Base.each({ // Injection scope for tests both as instance and static methods
function isStraight(l, h1, h2) { isStraight: function(l, h1, h2) {
if (h1.isZero() && h2.isZero()) { if (h1.isZero() && h2.isZero()) {
// No handles. // No handles.
return true; return true;
@ -814,9 +814,30 @@ statics: {
return p1 >= 0 && p1 <= 1 && p2 <= 0 && p2 >= -1; return p1 >= 0 && p1 <= 1 && p2 <= 0 && p2 >= -1;
} }
return false; return false;
} },
isLinear: function(l, h1, h2) {
var third = l.divide(3);
return h1.equals(third) && h2.negate().equals(third);
}
}, function(test, name) {
this[name] = function() {
var seg1 = this._segment1,
seg2 = this._segment2;
return test(seg2._point.subtract(seg1._point),
seg1._handleOut, seg2._handleIn);
};
this.statics[name] = function(v) {
var p1x = v[0], p1y = v[1],
p2x = v[6], p2y = v[7];
return test(new Point(p2x - p1x, p2y - p1y),
new Point(v[2] - p1x, v[3] - p1y),
new Point(v[4] - p2x, v[5] - p2y));
};
}, /** @lends Curve# */{
statics: {}, // Filled in the loop above
return /** @lends Curve# */{
/** /**
* {@grouptitle Tests} * {@grouptitle Tests}
* *
@ -839,14 +860,20 @@ statics: {
* line that connects the curve's start and end point, not falling * line that connects the curve's start and end point, not falling
* outside of the line. * outside of the line.
* *
* @name Curve#isStraight
* @function
* @return {Boolean} {@true if the curve is straight} * @return {Boolean} {@true if the curve is straight}
*/ */
isStraight: function() {
var seg1 = this._segment1, /**
seg2 = this._segment2; * Checks if this curve is parametrically linear, meaning that its
return isStraight(seg2._point.subtract(seg1._point), * handles are positioned at 1/3 and 2/3 of the total length of the
seg1._handleOut, seg2._handleIn); * straight curve.
}, *
* @name Curve#isLinear
* @function
* @return {Boolean} {@true if the curve is parametrically linear}
*/
/** /**
* Checks if the the two curves describe straight lines that are * Checks if the the two curves describe straight lines that are
@ -858,19 +885,8 @@ statics: {
isCollinear: function(curve) { isCollinear: function(curve) {
return this.isStraight() && curve.isStraight() return this.isStraight() && curve.isStraight()
&& this.getVector().isCollinear(curve.getVector()); && this.getVector().isCollinear(curve.getVector());
},
statics: {
isStraight: function(v) {
var p1x = v[0], p1y = v[1],
p2x = v[6], p2y = v[7];
return isStraight(new Point(p2x - p1x, p2y - p1y),
new Point(v[2] - p1x, v[3] - p1y),
new Point(v[4] - p2x, v[5] - p2y));
} }
} }), /** @lends Curve# */{
}
}, /** @lends Curve# */{
// Explicitly deactivate the creation of beans, as we have functions here // Explicitly deactivate the creation of beans, as we have functions here
// that look like bean getters but actually read arguments. // that look like bean getters but actually read arguments.
// See #getParameterOf(), #getLocationOf(), #getNearestLocation(), ... // See #getParameterOf(), #getLocationOf(), #getNearestLocation(), ...

View file

@ -211,3 +211,15 @@ test('Curve#isStraight()', function() {
return new Curve([100, 100], null, [-50, -50], [100, 100]).isStraight(); return new Curve([100, 100], null, [-50, -50], [100, 100]).isStraight();
}, false); }, false);
}); });
test('Curve#isLinear()', function() {
equals(function() {
return new Curve([100, 100], [100 / 3, 100 / 3], [-100 / 3, -100 / 3], [200, 200]).isLinear();
}, true);
equals(function() {
return new Curve([100, 100], null, null, [100, 100]).isLinear();
}, true);
equals(function() {
return new Curve([100, 100], null, null, [200, 200]).isLinear();
}, false);
});