diff --git a/src/rectangle.js b/src/rectangle.js index abf1c947..25817762 100644 --- a/src/rectangle.js +++ b/src/rectangle.js @@ -36,42 +36,63 @@ /** * @class Rect - * @classdesc A Rectangle really represents a 2x2 matrix where each row represents a - * 2 dimensional vector component, the first is (x,y) and the second is - * (width, height). The latter component implies the equation of a simple - * plane. + * @classdesc A Rectangle is described by it top left coordinates (x, y), width, + * height and degrees of rotation around (x, y). + * Note that the coordinate system used is the one commonly used with images: + * x increases when going to the right + * y increases when going to the bottom + * degrees increases clockwise with 0 being the horizontal * * @memberof OpenSeadragon - * @param {Number} x The vector component 'x'. - * @param {Number} y The vector component 'y'. - * @param {Number} width The vector component 'height'. - * @param {Number} height The vector component 'width'. + * @param {Object} options + * @param {Number} options.x X coordinate of the top left corner of the rectangle. + * @param {Number} options.y Y coordinate of the top left corner of the rectangle. + * @param {Number} options.width Width of the rectangle. + * @param {Number} options.height Height of the rectangle. + * @param {Number} options.degrees Rotation of the rectangle around (x,y) in degrees. + * @param {Number} [x] Deprecated: The vector component 'x'. + * @param {Number} [y] Deprecated: The vector component 'y'. + * @param {Number} [width] Deprecated: The vector component 'width'. + * @param {Number} [height] Deprecated: The vector component 'height'. */ -$.Rect = function( x, y, width, height ) { +$.Rect = function(x, y, width, height) { + + var options = x; + if (!$.isPlainObject(options)) { + options = { + x: x, + y: y, + width: width, + height: height + }; + } + /** * The vector component 'x'. * @member {Number} x * @memberof OpenSeadragon.Rect# */ - this.x = typeof ( x ) == "number" ? x : 0; + this.x = typeof(options.x) === "number" ? options.x : 0; /** * The vector component 'y'. * @member {Number} y * @memberof OpenSeadragon.Rect# */ - this.y = typeof ( y ) == "number" ? y : 0; + this.y = typeof(options.y) === "number" ? options.y : 0; /** * The vector component 'width'. * @member {Number} width * @memberof OpenSeadragon.Rect# */ - this.width = typeof ( width ) == "number" ? width : 0; + this.width = typeof(options.width) === "number" ? options.width : 0; /** * The vector component 'height'. * @member {Number} height * @memberof OpenSeadragon.Rect# */ - this.height = typeof ( height ) == "number" ? height : 0; + this.height = typeof(options.height) === "number" ? options.height : 0; + + this.degrees = typeof(options.degrees) === "number" ? options.degrees : 0; }; $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ @@ -80,7 +101,13 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ * @returns {OpenSeadragon.Rect} a duplicate of this Rect */ clone: function() { - return new $.Rect(this.x, this.y, this.width, this.height); + return new $.Rect({ + x: this.x, + y: this.y, + width: this.width, + height: this.height, + degrees: this.degrees + }); }, /** @@ -114,10 +141,8 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ * the rectangle. */ getBottomRight: function() { - return new $.Point( - this.x + this.width, - this.y + this.height - ); + return new $.Point(this.x + this.width, this.y + this.height) + .rotate(this.degrees, this.getTopLeft()); }, /** @@ -128,10 +153,8 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ * the rectangle. */ getTopRight: function() { - return new $.Point( - this.x + this.width, - this.y - ); + return new $.Point(this.x + this.width, this.y) + .rotate(this.degrees, this.getTopLeft()); }, /** @@ -142,10 +165,8 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ * the rectangle. */ getBottomLeft: function() { - return new $.Point( - this.x, - this.y + this.height - ); + return new $.Point(this.x, this.y + this.height) + .rotate(this.degrees, this.getTopLeft()); }, /** @@ -158,7 +179,7 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ return new $.Point( this.x + this.width / 2.0, this.y + this.height / 2.0 - ); + ).rotate(this.degrees, this.getTopLeft()); }, /** @@ -177,28 +198,31 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ * @param {OpenSeadragon.Rect} rectangle The Rectangle to compare to. * @return {Boolean} 'true' if all components are equal, otherwise 'false'. */ - equals: function( other ) { - return ( other instanceof $.Rect ) && - ( this.x === other.x ) && - ( this.y === other.y ) && - ( this.width === other.width ) && - ( this.height === other.height ); + equals: function(other) { + return (other instanceof $.Rect) && + this.x === other.x && + this.y === other.y && + this.width === other.width && + this.height === other.height && + this.degrees === other.degrees; }, /** - * Multiply all dimensions in this Rect by a factor and return a new Rect. + * Multiply all dimensions (except degrees) in this Rect by a factor and + * return a new Rect. * @function * @param {Number} factor The factor to multiply vector components. * @returns {OpenSeadragon.Rect} A new rect representing the multiplication * of the vector components by the factor */ - times: function( factor ) { - return new OpenSeadragon.Rect( - this.x * factor, - this.y * factor, - this.width * factor, - this.height * factor - ); + times: function(factor) { + return new $.Rect({ + x: this.x * factor, + y: this.y * factor, + width: this.width * factor, + height: this.height * factor, + degrees: this.degrees + }); }, /** @@ -207,13 +231,14 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ * @param {OpenSeadragon.Point} delta The translation vector. * @returns {OpenSeadragon.Rect} A new rect with altered position */ - translate: function( delta ) { - return new OpenSeadragon.Rect( - this.x + delta.x, - this.y + delta.y, - this.width, - this.height - ); + translate: function(delta) { + return new $.Rect({ + x: this.x + delta.x, + y: this.y + delta.y, + width: this.width, + height: this.height, + degrees: this.degrees + }); }, /** @@ -223,67 +248,79 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ */ // ---------- union: function(rect) { + if (this.degrees !== 0 || rect.degrees !== 0) { + throw new Error('Only union of non rotated rectangles are supported.'); + } var left = Math.min(this.x, rect.x); var top = Math.min(this.y, rect.y); var right = Math.max(this.x + this.width, rect.x + rect.width); var bottom = Math.max(this.y + this.height, rect.y + rect.height); - return new OpenSeadragon.Rect(left, top, right - left, bottom - top); + return new $.Rect({ + left: left, + top: top, + width: right - left, + height: bottom - top + }); }, /** - * Rotates a rectangle around a point. Currently only 90, 180, and 270 - * degrees are supported. + * Rotates a rectangle around a point. * @function * @param {Number} degrees The angle in degrees to rotate. * @param {OpenSeadragon.Point} pivot The point about which to rotate. * Defaults to the center of the rectangle. * @return {OpenSeadragon.Rect} */ - rotate: function( degrees, pivot ) { - // TODO support arbitrary rotation - var width = this.width, - height = this.height, - newTopLeft; - - degrees = ( degrees + 360 ) % 360; - if (degrees % 90 !== 0) { - throw new Error('Currently only 0, 90, 180, and 270 degrees are supported.'); - } - - if( degrees === 0 ){ - return new $.Rect( - this.x, - this.y, - this.width, - this.height - ); + rotate: function(degrees, pivot) { + degrees = (degrees + 360) % 360; + if (degrees === 0) { + return this.clone(); } pivot = pivot || this.getCenter(); + var newTopLeft = this.getTopLeft().rotate(degrees, pivot); + var newTopRight = this.getTopRight().rotate(degrees, pivot); - switch ( degrees ) { - case 90: - newTopLeft = this.getBottomLeft(); - width = this.height; - height = this.width; - break; - case 180: - newTopLeft = this.getBottomRight(); - break; - case 270: - newTopLeft = this.getTopRight(); - width = this.height; - height = this.width; - break; - default: - newTopLeft = this.getTopLeft(); - break; + var diff = newTopRight.minus(newTopLeft); + var radians = Math.atan(diff.y / diff.x); + if (diff.x < 0) { + radians += Math.PI; + } else if (diff.y < 0) { + radians += 2 * Math.PI; } + return new $.Rect({ + x: newTopLeft.x, + y: newTopLeft.y, + width: this.width, + height: this.height, + degrees: radians / Math.PI * 180 + }); + }, - newTopLeft = newTopLeft.rotate(degrees, pivot); - - return new $.Rect(newTopLeft.x, newTopLeft.y, width, height); + /** + * Retrieves the smallest horizontal (degrees=0) rectangle which contains + * this rectangle. + * @returns {OpenSeadrayon.Rect} + */ + getBoundingBox: function() { + if (this.degrees === 0) { + return this.clone(); + } + var topLeft = this.getTopLeft(); + var topRight = this.getTopRight(); + var bottomLeft = this.getBottomLeft(); + var bottomRight = this.getBottomRight(); + var minX = Math.min(topLeft.x, topRight.x, bottomLeft.x, bottomRight.x); + var maxX = Math.max(topLeft.x, topRight.x, bottomLeft.x, bottomRight.x); + var minY = Math.min(topLeft.y, topRight.y, bottomLeft.y, bottomRight.y); + var maxY = Math.max(topLeft.y, topRight.y, bottomLeft.y, bottomRight.y); + return new $.Rect({ + x: minX, + y: minY, + width: maxX - minX, + height: maxY - minY + }); }, /** @@ -294,11 +331,12 @@ $.Rect.prototype = /** @lends OpenSeadragon.Rect.prototype */{ */ toString: function() { return "[" + - (Math.round(this.x*100) / 100) + "," + - (Math.round(this.y*100) / 100) + "," + - (Math.round(this.width*100) / 100) + "x" + - (Math.round(this.height*100) / 100) + - "]"; + (Math.round(this.x * 100) / 100) + "," + + (Math.round(this.y * 100) / 100) + "," + + (Math.round(this.width * 100) / 100) + "x" + + (Math.round(this.height * 100) / 100) + "," + + (Math.round(this.degrees * 100) / 100) + "°" + + "]"; } }; diff --git a/test/coverage.html b/test/coverage.html index dc857202..15d3d490 100644 --- a/test/coverage.html +++ b/test/coverage.html @@ -75,6 +75,7 @@ + diff --git a/test/modules/rectangle.js b/test/modules/rectangle.js new file mode 100644 index 00000000..ac6e76a9 --- /dev/null +++ b/test/modules/rectangle.js @@ -0,0 +1,260 @@ +/* global module, asyncTest, $, ok, equal, notEqual, start, test, Util, testLog */ + +(function() { + + module('Rectangle', {}); + + var precision = 0.000000001; + + function assertPointsEquals(pointA, pointB, message) { + Util.assessNumericValue(pointA.x, pointB.x, precision, message + " x: "); + Util.assessNumericValue(pointA.y, pointB.y, precision, message + " y: "); + } + + function assertRectangleEquals(rectA, rectB, message) { + Util.assessNumericValue(rectA.x, rectB.x, precision, message + " x: "); + Util.assessNumericValue(rectA.y, rectB.y, precision, message + " y: "); + Util.assessNumericValue(rectA.width, rectB.width, precision, + message + " width: "); + Util.assessNumericValue(rectA.height, rectB.height, precision, + message + " height: "); + Util.assessNumericValue(rectA.degrees, rectB.degrees, precision, + message + " degrees: "); + } + + test('Legacy constructor', function() { + var rect = new OpenSeadragon.Rect(1, 2, 3, 4); + strictEqual(rect.x, 1, 'rect.x should be 1'); + strictEqual(rect.y, 2, 'rect.y should be 2'); + strictEqual(rect.width, 3, 'rect.width should be 3'); + strictEqual(rect.height, 4, 'rect.height should be 4'); + strictEqual(rect.degrees, 0, 'rect.degrees should be 0'); + }); + + test('Constructor', function() { + var rect = new OpenSeadragon.Rect({ + x: 1, + y: 2, + width: 3, + height: 4, + degrees: 5 + }); + strictEqual(rect.x, 1, 'rect.x should be 1'); + strictEqual(rect.y, 2, 'rect.y should be 2'); + strictEqual(rect.width, 3, 'rect.width should be 3'); + strictEqual(rect.height, 4, 'rect.height should be 4'); + strictEqual(rect.degrees, 5, 'rect.degrees should be 5'); + + rect = new OpenSeadragon.Rect({}); + strictEqual(rect.x, 0, 'rect.x should be 0'); + strictEqual(rect.y, 0, 'rect.y should be 0'); + strictEqual(rect.width, 0, 'rect.width should be 0'); + strictEqual(rect.height, 0, 'rect.height should be 0'); + strictEqual(rect.degrees, 0, 'rect.degrees should be 0'); + }); + + test('getTopLeft', function() { + var rect = new OpenSeadragon.Rect({ + x: 1, + y: 2, + width: 3, + height: 4, + degrees: 5 + }); + var expected = new OpenSeadragon.Point(1, 2); + ok(expected.equals(rect.getTopLeft()), "Incorrect top left point."); + }); + + test('getTopRight', function() { + var rect = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 1, + height: 3 + }); + var expected = new OpenSeadragon.Point(1, 0); + ok(expected.equals(rect.getTopRight()), "Incorrect top right point."); + + rect.degrees = 45; + expected = new OpenSeadragon.Point(1 / Math.sqrt(2), 1 / Math.sqrt(2)); + assertPointsEquals(expected, rect.getTopRight(), + "Incorrect top right point with rotation."); + }); + + test('getBottomLeft', function() { + var rect = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 3, + height: 1 + }); + var expected = new OpenSeadragon.Point(0, 1); + ok(expected.equals(rect.getBottomLeft()), "Incorrect bottom left point."); + + rect.degrees = 45; + expected = new OpenSeadragon.Point(-1 / Math.sqrt(2), 1 / Math.sqrt(2)); + assertPointsEquals(expected, rect.getBottomLeft(), + "Incorrect bottom left point with rotation."); + }); + + test('getBottomRight', function() { + var rect = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 1, + height: 1 + }); + var expected = new OpenSeadragon.Point(1, 1); + ok(expected.equals(rect.getBottomRight()), "Incorrect bottom right point."); + + rect.degrees = 45; + expected = new OpenSeadragon.Point(0, Math.sqrt(2)); + assertPointsEquals(expected, rect.getBottomRight(), + "Incorrect bottom right point with 45 rotation."); + + rect.degrees = 90; + expected = new OpenSeadragon.Point(-1, 1); + assertPointsEquals(expected, rect.getBottomRight(), + "Incorrect bottom right point with 90 rotation."); + + rect.degrees = 135; + expected = new OpenSeadragon.Point(-Math.sqrt(2), 0); + assertPointsEquals(expected, rect.getBottomRight(), + "Incorrect bottom right point with 135 rotation."); + }); + + test('getCenter', function() { + var rect = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 1, + height: 1 + }); + var expected = new OpenSeadragon.Point(0.5, 0.5); + ok(expected.equals(rect.getCenter()), "Incorrect center point."); + + rect.degrees = 45; + expected = new OpenSeadragon.Point(0, 0.5 * Math.sqrt(2)); + assertPointsEquals(expected, rect.getCenter(), + "Incorrect bottom right point with 45 rotation."); + + rect.degrees = 90; + expected = new OpenSeadragon.Point(-0.5, 0.5); + assertPointsEquals(expected, rect.getCenter(), + "Incorrect bottom right point with 90 rotation."); + + rect.degrees = 135; + expected = new OpenSeadragon.Point(-0.5 * Math.sqrt(2), 0); + assertPointsEquals(expected, rect.getCenter(), + "Incorrect bottom right point with 135 rotation."); + }); + + test('rotate', function() { + var rect = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 2, + height: 1 + }); + + // Rotate 45deg around center. + var expected = new OpenSeadragon.Rect({ + x: 1 - 1 / (2 * Math.sqrt(2)), + y: 0.5 - 3 / (2 * Math.sqrt(2)), + width: 2, + height: 1, + degrees: 45 + }); + var actual = rect.rotate(45); + assertRectangleEquals(expected, actual, + "Incorrect rectangle after rotation of 45deg around center."); + + expected = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 2, + height: 1, + degrees: 33 + }); + actual = rect.rotate(33, rect.getTopLeft()); + assertRectangleEquals(expected, actual, + "Incorrect rectangle after rotation of 33deg around topLeft."); + + expected = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 2, + height: 1, + degrees: 101 + }); + actual = rect.rotate(101, rect.getTopLeft()); + assertRectangleEquals(expected, actual, + "Incorrect rectangle after rotation of 187deg around topLeft."); + + expected = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 2, + height: 1, + degrees: 187 + }); + actual = rect.rotate(187, rect.getTopLeft()); + assertRectangleEquals(expected, actual, + "Incorrect rectangle after rotation of 187deg around topLeft."); + + expected = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 2, + height: 1, + degrees: 300 + }); + actual = rect.rotate(300, rect.getTopLeft()); + assertRectangleEquals(expected, actual, + "Incorrect rectangle after rotation of 300deg around topLeft."); + }); + + test('getBoundingBox', function() { + var rect = new OpenSeadragon.Rect({ + x: 0, + y: 0, + width: 2, + height: 3 + }); + + var bb = rect.getBoundingBox(); + ok(rect.equals(bb), "Bounding box of horizontal rectangle should be " + + "identical to rectangle."); + + rect.degrees = 90; + var expected = new OpenSeadragon.Rect({ + x: -3, + y: 0, + width: 3, + height: 2 + }); + assertRectangleEquals(expected, rect.getBoundingBox(), + "Bounding box of rect rotated 90deg."); + + rect.degrees = 180; + var expected = new OpenSeadragon.Rect({ + x: -2, + y: -3, + width: 2, + height: 3 + }); + assertRectangleEquals(expected, rect.getBoundingBox(), + "Bounding box of rect rotated 180deg."); + + rect.degrees = 270; + var expected = new OpenSeadragon.Rect({ + x: 0, + y: -2, + width: 3, + height: 2 + }); + assertRectangleEquals(expected, rect.getBoundingBox(), + "Bounding box of rect rotated 270deg."); + }); + +})(); diff --git a/test/test.html b/test/test.html index e52eb66a..d50e53eb 100644 --- a/test/test.html +++ b/test/test.html @@ -39,6 +39,7 @@ +