UNPKG

flatten-js

Version:

Javascript library for 2d geometry

247 lines (218 loc) 7.35 kB
/** * Created by Alex Bol on 2/19/2017. */ "use strict"; module.exports = function(Flatten) { /** * Class representing a vector * @type {Vector} */ Flatten.Vector = class Vector { /** * Vector may be constructed by two points, or by two float numbers, * or by array of two numbers * @param {Point} ps - start point * @param {Point} pe - end point */ constructor(...args) { /** * x-coordinate of a vector (float number) * @type {number} */ this.x = 0; /** * y-coordinate of a vector (float number) * @type {number} */ this.y = 0; /* return zero vector */ if (args.length === 0) { return; } if (args.length === 1 && args[0] instanceof Array && args[0].length === 2) { let arr = args[0]; if (typeof(arr[0]) == "number" && typeof(arr[1]) == "number") { this.x = arr[0]; this.y = arr[1]; return; } } if (args.length === 1 && args[0] instanceof Object && args[0].name === "vector") { let {x, y} = args[0]; this.x = x; this.y = y; return; } if (args.length === 2) { let a1 = args[0]; let a2 = args[1]; if (typeof(a1) == "number" && typeof(a2) == "number") { this.x = a1; this.y = a2; return; } if (a1 instanceof Flatten.Point && a2 instanceof Flatten.Point) { this.x = a2.x - a1.x; this.y = a2.y - a1.y; return; } } throw Flatten.Errors.ILLEGAL_PARAMETERS; } /** * Method clone returns new instance of Vector * @returns {Vector} */ clone() { return new Vector(this.x, this.y); } /** * Slope of the vector in radians from 0 to 2PI * @returns {number} */ get slope() { let angle = Math.atan2(this.y, this.x); if (angle<0) angle = 2*Math.PI + angle; return angle; } /** * Length of vector * @returns {number} */ get length() { return Math.sqrt(this.dot(this)); } /** * Returns true if vectors are equal up to [DP_TOL]{@link http://localhost:63342/flatten-js/docs/global.html#DP_TOL} * tolerance * @param {Vector} v * @returns {boolean} */ equalTo(v) { return Flatten.Utils.EQ(this.x, v.x) && Flatten.Utils.EQ(this.y, v.y); } /** * Returns new vector multiplied by scalar * @param {number} scalar * @returns {Vector} */ multiply(scalar) { return ( new Vector(scalar * this.x, scalar * this.y) ); } /** * Returns scalar product (dot product) of two vectors <br/> * <code>dot_product = (this * v)</code> * @param {Vector} v Other vector * @returns {number} */ dot(v) { return ( this.x * v.x + this.y * v.y ); } /** * Returns vector product (cross product) of two vectors <br/> * <code>cross_product = (this x v)</code> * @param {Vector} v Other vector * @returns {number} */ cross(v) { return ( this.x * v.y - this.y * v.x ); } /** * Returns unit vector.<br/> * Throw error if given vector has zero length * @returns {Vector} */ normalize() { if (!Flatten.Utils.EQ_0(this.length)) { return ( new Vector(this.x / this.length, this.y / this.length) ); } throw Flatten.Errors.ZERO_DIVISION; } /** * Returns new vector rotated by given angle, * positive angle defines rotation in counter clockwise direction, * negative - in clockwise direction * @param {number} angle - Angle in radians * @returns {Vector} */ rotate(angle) { let point = new Flatten.Point(this.x, this.y); let rpoint = point.rotate(angle); return new Flatten.Vector(rpoint.x, rpoint.y); } /** * Returns vector rotated 90 degrees counter clockwise * @returns {Vector} */ rotate90CCW() { return new Flatten.Vector(-this.y, this.x); }; /** * Returns vector rotated 90 degrees clockwise * @returns {Vector} */ rotate90CW() { return new Flatten.Vector(this.y, -this.x); }; /** * Return inverted vector * @returns {Vector} */ invert() { return new Flatten.Vector(-this.x, -this.y); } /** * Return result of addition of other vector to this vector as a new vector * @param {Vector} v Other vector * @returns {Vector} */ add(v) { return new Flatten.Vector(this.x + v.x, this.y + v.y); } /** * Return result of subtraction of other vector from current vector as a new vector * @param {Vector} v Another vector * @returns {Vector} */ subtract(v) { return new Flatten.Vector(this.x - v.x, this.y - v.y); } /** * Return angle between this vector and other vector. <br/> * Angle is measured from 0 to 2*PI in the counter clockwise direction * from current vector to other. * @param {Vector} v Another vector * @returns {number} */ angleTo(v) { let norm1 = this.normalize(); let norm2 = v.normalize(); let angle = Math.atan2(norm1.cross(norm2), norm1.dot(norm2)); if (angle<0) angle += 2*Math.PI; return angle; } /** * Return vector projection of the current vector on another vector * @param {Vector} v Another vector * @returns {Vector} */ projectionOn(v){ let n = v.normalize(); let d = this.dot(n); return n.multiply(d); } /** * This method returns an object that defines how data will be * serialized when called JSON.stringify() method * @returns {Object} */ toJSON() { return Object.assign({},this,{name:"vector"}); } }; /** * Function to create vector equivalent to "new" constructor * @param args */ Flatten.vector = (...args) => new Flatten.Vector(...args); };