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;