172 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| class Rectangle {
 | |
|     /**
 | |
|      * A utility for creating and comparing axis-aligned rectangles.
 | |
|      * Rectangles are always initialized to the "largest possible rectangle";
 | |
|      * use one of the init* methods below to set up a particular rectangle.
 | |
|      * @constructor
 | |
|      */
 | |
|     constructor () {
 | |
|         this.left = -Infinity;
 | |
|         this.right = Infinity;
 | |
|         this.bottom = -Infinity;
 | |
|         this.top = Infinity;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Initialize a Rectangle from given Scratch-coordinate bounds.
 | |
|      * @param {number} left Left bound of the rectangle.
 | |
|      * @param {number} right Right bound of the rectangle.
 | |
|      * @param {number} bottom Bottom bound of the rectangle.
 | |
|      * @param {number} top Top bound of the rectangle.
 | |
|      */
 | |
|     initFromBounds (left, right, bottom, top) {
 | |
|         this.left = left;
 | |
|         this.right = right;
 | |
|         this.bottom = bottom;
 | |
|         this.top = top;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Initialize a Rectangle to the minimum AABB around a set of points.
 | |
|      * @param {Array<Array<number>>} points Array of [x, y] points.
 | |
|      */
 | |
|     initFromPointsAABB (points) {
 | |
|         this.left = Infinity;
 | |
|         this.right = -Infinity;
 | |
|         this.top = -Infinity;
 | |
|         this.bottom = Infinity;
 | |
| 
 | |
|         for (let i = 0; i < points.length; i++) {
 | |
|             const x = points[i][0];
 | |
|             const y = points[i][1];
 | |
|             if (x < this.left) {
 | |
|                 this.left = x;
 | |
|             }
 | |
|             if (x > this.right) {
 | |
|                 this.right = x;
 | |
|             }
 | |
|             if (y > this.top) {
 | |
|                 this.top = y;
 | |
|             }
 | |
|             if (y < this.bottom) {
 | |
|                 this.bottom = y;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Determine if this Rectangle intersects some other.
 | |
|      * Note that this is a comparison assuming the Rectangle was
 | |
|      * initialized with Scratch-space bounds or points.
 | |
|      * @param {!Rectangle} other Rectangle to check if intersecting.
 | |
|      * @return {boolean} True if this Rectangle intersects other.
 | |
|      */
 | |
|     intersects (other) {
 | |
|         return (
 | |
|             this.left <= other.right &&
 | |
|             other.left <= this.right &&
 | |
|             this.top >= other.bottom &&
 | |
|             other.top >= this.bottom
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Determine if this Rectangle fully contains some other.
 | |
|      * Note that this is a comparison assuming the Rectangle was
 | |
|      * initialized with Scratch-space bounds or points.
 | |
|      * @param {!Rectangle} other Rectangle to check if fully contained.
 | |
|      * @return {boolean} True if this Rectangle fully contains other.
 | |
|      */
 | |
|     contains (other) {
 | |
|         return (
 | |
|             other.left > this.left &&
 | |
|             other.right < this.right &&
 | |
|             other.top < this.top &&
 | |
|             other.bottom > this.bottom
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Clamp a Rectangle to bounds.
 | |
|      * @param {number} left Left clamp.
 | |
|      * @param {number} right Right clamp.
 | |
|      * @param {number} bottom Bottom clamp.
 | |
|      * @param {number} top Top clamp.
 | |
|      */
 | |
|     clamp (left, right, bottom, top) {
 | |
|         this.left = Math.max(this.left, left);
 | |
|         this.right = Math.min(this.right, right);
 | |
|         this.bottom = Math.max(this.bottom, bottom);
 | |
|         this.top = Math.min(this.top, top);
 | |
|         // Ensure rectangle coordinates in order.
 | |
|         this.left = Math.min(this.left, this.right);
 | |
|         this.right = Math.max(this.right, this.left);
 | |
|         this.bottom = Math.min(this.bottom, this.top);
 | |
|         this.top = Math.max(this.top, this.bottom);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Push out the Rectangle to integer bounds.
 | |
|      */
 | |
|     snapToInt () {
 | |
|         this.left = Math.floor(this.left);
 | |
|         this.right = Math.ceil(this.right);
 | |
|         this.bottom = Math.floor(this.bottom);
 | |
|         this.top = Math.ceil(this.top);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Compute the intersection of two bounding Rectangles.
 | |
|      * Could be an impossible box if they don't intersect.
 | |
|      * @param {Rectangle} a One rectangle
 | |
|      * @param {Rectangle} b Other rectangle
 | |
|      * @param {?Rectangle} result A resulting storage rectangle  (safe to pass
 | |
|      *                            a or b if you want to overwrite one)
 | |
|      * @returns {Rectangle} resulting rectangle
 | |
|      */
 | |
|     static intersect (a, b, result = new Rectangle()) {
 | |
|         result.left = Math.max(a.left, b.left);
 | |
|         result.right = Math.min(a.right, b.right);
 | |
|         result.top = Math.min(a.top, b.top);
 | |
|         result.bottom = Math.max(a.bottom, b.bottom);
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Compute the union of two bounding Rectangles.
 | |
|      * @param {Rectangle} a One rectangle
 | |
|      * @param {Rectangle} b Other rectangle
 | |
|      * @param {?Rectangle} result A resulting storage rectangle  (safe to pass
 | |
|      *                            a or b if you want to overwrite one)
 | |
|      * @returns {Rectangle} resulting rectangle
 | |
|      */
 | |
|     static union (a, b, result = new Rectangle()) {
 | |
|         result.left = Math.min(a.left, b.left);
 | |
|         result.right = Math.max(a.right, b.right);
 | |
|         // Scratch Space - +y is up
 | |
|         result.top = Math.max(a.top, b.top);
 | |
|         result.bottom = Math.min(a.bottom, b.bottom);
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Width of the Rectangle.
 | |
|      * @return {number} Width of rectangle.
 | |
|      */
 | |
|     get width () {
 | |
|         return Math.abs(this.left - this.right);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Height of the Rectangle.
 | |
|      * @return {number} Height of rectangle.
 | |
|      */
 | |
|     get height () {
 | |
|         return Math.abs(this.top - this.bottom);
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | |
| module.exports = Rectangle;
 | 
