Source: math/Vector2.js

import {
    vec2
} from 'gl-matrix';
import Class from '../core/Class';

/**
 * 二维向量
 * @class
 */
const Vector2 = Class.create(/** @lends Vector2.prototype */ {
    /**
     * 类名
     * @type {String}
     * @default Vector2
     */
    className: 'Vector2',
    /**
     * @type {Boolean}
     * @default true
     */
    isVector2: true,
    /**
     * Creates a new empty vec2
     * @param {Number} [x=0] X component
     * @param {Number} [y=0] Y component
     * @constructs
     */
    constructor(x = 0, y = 0) {
        /**
         * 数据
         * @type {Float32Array}
         */
        this.elements = vec2.fromValues(x, y);
    },
    /**
     * Copy the values from one vec2 to this
     * @param  {Vector2} m the source vector
     * @return {Vector2} this
     */
    copy(v) {
        vec2.copy(this.elements, v.elements);
        return this;
    },
    /**
     * Creates a new vec2 initialized with values from this vector
     * @return {Vector2} a new Vector2
     */
    clone() {
        const elements = this.elements;
        return new this.constructor(elements[0], elements[1]);
    },
    /**
     * 转换到数组
     * @param  {number[]|TypedArray}  [array=[]] 数组
     * @param  {Number} [offset=0] 数组偏移值
     * @return {Array}
     */
    toArray(array = [], offset = 0) {
        const elements = this.elements;
        array[0 + offset] = elements[0];
        array[1 + offset] = elements[1];
        return array;
    },
    /**
     * 从数组赋值
     * @param  {number[]|TypedArray} array  数组
     * @param  {Number} [offset=0] 数组偏移值
     * @return {Vector2} this
     */
    fromArray(array, offset = 0) {
        const elements = this.elements;
        elements[0] = array[offset + 0];
        elements[1] = array[offset + 1];
        return this;
    },
    /**
     * Set the components of a vec4 to the given values
     * @param {Number} x X component
     * @param {Number} y Y component
     * @returns {Vector2} this
     */
    set(x, y) {
        vec2.set(this.elements, x, y);
        return this;
    },
    /**
     * Adds two vec2's
     * @param {Vector2} a
     * @param {Vector2} [b] 如果不传,计算 this 和 a 的和
     * @returns {Vector2} this
     */
    add(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.add(this.elements, a.elements, b.elements);
        return this;
    },
    /**
     * Subtracts vector b from vector a
     * @param {Vector2} a
     * @param {Vector2} [b] 如果不传,计算 this 和 a 的差
     * @returns {Vector2} this
     */
    subtract(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.subtract(this.elements, a.elements, b.elements);
        return this;
    },
    /**
     * Multiplies two vec2's
     * @param {Vector2} a
     * @param {Vector2} [b] 如果不传,计算 this 和 a 的积
     * @returns {Vector2} this
     */
    multiply(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.multiply(this.elements, a.elements, b.elements);
        return this;
    },
    /**
     * Divides two vec2's
     * @param {Vector2} a
     * @param {Vector2} [b] 如果不传,计算 this 和 a 的商
     * @returns {Vector2} this
     */
    divide(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.divide(this.elements, a.elements, b.elements);
        return this;
    },
    /**
     * Math.ceil the components of this
     * @returns {Vector2} this
     */
    ceil() {
        vec2.ceil(this.elements, this.elements);
        return this;
    },
    /**
     * Math.floor the components of this
     * @returns {Vector2} this
     */
    floor() {
        vec2.floor(this.elements, this.elements);
        return this;
    },
    /**
     * Returns the minimum of two vec2's
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @returns {Vector2} this
     */
    min(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.min(this.elements, a.elements, b.elements);
        return this;
    },
    /**
     * Returns the maximum of two vec2's
     * @param  {Vector2} a
     * @param  {Vector2} [b]  如果不传,计算 this 和 a 的结果
     * @returns {Vector2} this
     */
    max(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.max(this.elements, a.elements, b.elements);
        return this;
    },
    /**
     * Math.round the components of this
     * @returns {Vector2} this
     */
    round() {
        vec2.round(this.elements, this.elements);
        return this;
    },
    /**
     * Scales this by a scalar number
     * @param  {Number} scale amount to scale the vector by
     * @returns {Vector2} this
     */
    scale(scale) {
        vec2.scale(this.elements, this.elements, scale);
        return this;
    },
    /**
     * Adds two vec2's after scaling the second vector by a scalar value
     * @param  {Number} scale the amount to scale the second vector by before adding
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @returns {Vector2} this
     */
    scaleAndAdd(scale, a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.scaleAndAdd(this.elements, a.elements, b.elements, scale);
        return this;
    },
    /**
     * Calculates the euclidian distance between two vec2's
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @return {Number} distance between a and b
     */
    distance(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        return vec2.distance(a.elements, b.elements);
    },
    /**
     * Calculates the squared euclidian distance between two vec2's
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @return {Number} squared distance between a and b
     */
    squaredDistance(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        return vec2.squaredDistance(a.elements, b.elements);
    },
    /**
     * Calculates the length of this
     * @return {Number} length of this
     */
    length() {
        return vec2.length(this.elements);
    },
    /**
     * Calculates the squared length of this
     * @return {Number} squared length of this
     */
    squaredLength() {
        return vec2.squaredLength(this.elements);
    },
    /**
     * Negates the components of this
     * @returns {Vector2} this
     */
    negate() {
        vec2.negate(this.elements, this.elements);
        return this;
    },
    /**
     * Returns the inverse of the components of a vec2
     * @param  {Vector2} [a=this]
     * @returns {Vector2} this
     */
    inverse(a) {
        if (!a) {
            a = this;
        }
        vec2.inverse(this.elements, a.elements);
        return this;
    },
    /**
     * Normalize this
     * @returns {Vector2} this
     */
    normalize() {
        vec2.normalize(this.elements, this.elements);
        return this;
    },
    /**
     * Calculates the dot product of two vec2's
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @return {Number}  product of a and b
     */
    dot(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        return vec2.dot(a.elements, b.elements);
    },
    /**
     * Computes the cross product of two vec2's
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @return {Number}  cross product of a and b
     */
    cross(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        vec2.cross(this.elements, a.elements, b.elements);
        return this;
    },
    /**
     * Performs a linear interpolation between two vec2's
     * @param  {Vector2} v
     * @param  {Number} t interpolation amount between the two vectors
     * @returns {Vector2} this
     */
    lerp(v, t) {
        vec2.lerp(this.elements, this.elements, v.elements, t);
        return this;
    },
    /**
     * Generates a random vector with the given scale
     * @param  {Number} [scale=1] Length of the resulting vector. If ommitted, a unit vector will be returned
     * @returns {Vector2} this
     */
    random(scale) {
        vec2.random(this.elements, scale);
        return this;
    },
    /**
     * Transforms the vec2 with a mat3
     * @param  {Matrix3} m matrix to transform with
     * @returns {Vector2} this
     */
    transformMat3(m) {
        vec2.transformMat3(this.elements, this.elements, m.elements);
        return this;
    },
    /**
     * Transforms the vec2 with a mat4
     * @param  {Matrix4} m matrix to transform with
     * @returns {Vector2} this
     */
    transformMat4(m) {
        vec2.transformMat4(this.elements, this.elements, m.elements);
        return this;
    },
    /**
     * Returns whether or not the vectors have exactly the same elements in the same position (when compared with ===)
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @return {Boolean} True if the vectors are equal, false otherwise.
     */
    exactEquals(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        return vec2.exactEquals(a.elements, b.elements);
    },
    /**
     * Returns whether or not the vectors have approximately the same elements in the same position.
     * @param  {Vector2} a
     * @param  {Vector2} [b] 如果不传,计算 this 和 a 的结果
     * @return {Boolean} True if the vectors are equal, false otherwise.
     */
    equals(a, b) {
        if (!b) {
            b = a;
            a = this;
        }
        return vec2.equals(a.elements, b.elements);
    },
    /**
     * X component
     * @type {Number}
     */
    x: {
        get() {
            return this.elements[0];
        },
        set(value) {
            this.elements[0] = value;
        }
    },
    /**
     * Y component
     * @type {Number}
     */
    y: {
        get() {
            return this.elements[1];
        },
        set(value) {
            this.elements[1] = value;
        }
    }
});

/**
 * Alias for {@link Vector2#subtract}
 * @function
 */
Vector2.prototype.sub = Vector2.prototype.subtract;

/**
 * Alias for {@link Vector2#multiply}
 * @function
 */
Vector2.prototype.mul = Vector2.prototype.multiply;

/**
 * Alias for {@link Vector2#divide}
 * @function
 */
Vector2.prototype.div = Vector2.prototype.divide;

/**
 * Alias for {@link Vector2#distance}
 * @function
 */
Vector2.prototype.dist = Vector2.prototype.distance;

/**
 * Alias for {@link Vector2#squaredDistance}
 * @function
 */
Vector2.prototype.sqrDist = Vector2.prototype.squaredDistance;

/**
 * Alias for {@link Vector2#length}
 * @function
 */
Vector2.prototype.len = Vector2.prototype.length;

/**
 * Alias for {@link Vector2#squaredLength}
 * @function
 */
Vector2.prototype.sqrLen = Vector2.prototype.squaredLength;

export default Vector2;