Vectors are tuples (arrays of numbers) representing position, speed, acceleration and direction in 3D space.
JavaScript implementation:
class Vector3 {
x;
y;
z;
constructor(x,y,z){
this.x = x;
this.y = y;
this.z = z;
}
}
Coordinate systems and vectors operations depend on a left of right handedness. Right will be used here, as it is favored by WebGL.
A vector can represent a position in space, or in other terms, a displacement from the origin up to a given point a:
a = [ax, ay, az]
Similarly, it can represent a displacement between two points a and b:
dab = [bx - ax, by - ay, bz - az]
A vector can be split in two elements:
a = dâ
Where d is a scalar (number) representing the distance (magnitude) and â the direction, in the form of an unit vector (a vector of magnitude 1, or normalized vector). They can be computed in the following way:
d = |a| = sqrt(ax² + ay² + az²)
â = a/d = a/|a|
Vector scaling changes a vector's length but not its direction:
ka = kdâ = [kax, kay, kaz]
This can be used to scale a vector up (k > 1), down (0 < k < 1), flip it (k = -1) or normalize it (k = 1/|a|).
Vector addition and subtraction can be done term by term:
a + b = [ax + bx, ay + by, az + bz]
a - b = [ax - bx, ay - by, az - bz]
1) Component product (term by term):
a◦b = [ax bx, ay by, az bz]
2) Scalar product (or dot product), returns a scalar (number) value.
a·b = ax bx + ay by+ az bz
The dot product is linked to the angle between two normalized vectors:
a·b = |a||b| cos θ ⟺ θ = cos-1(a.b / |a||b|)
(The division is optional if a and b are normalized)
3) Vector product (or cross product) returns a vector perpendicular to two vectors, using the current handedness:
a×b = [ay bz - az by, az bx - ax bz, ax by - ay bx
The magnitude of the cross product is also linked to the angle between the two vectors:
|a×b| = |a||b| sin θ
Contrary to the other products, the cross product is not commutative:
a×b = -b×a
JavaScript implementation
class Vector3 {
// ...
// invert
invert(){
this.x = -this.x;
this.y = -this.y;
this.z = -this.z;
}
// magnitude
magnitude(){
return Math.hypot(this.x, this.y, this.z);
}
// magnitude squared (faster for comparing two vectors)
squareMagnitude(){
this.x ** 2 + this.y ** 2 + this.z ** 2;
}
// normalize
normalize(){
var l = this.matnitude();
if(l > 0) this.scale(1/l);
}
// scale
scale(s){
this.x *= s;
this.y *= s;
this.z *= s;
}
// add
add(v){
this.x += v.x;
this.y += v.y;
this.z += v.z;
}
// subtract
sub(v){
this.x -= v.x;
this.y -= v.y;
this.z -= v.z;
}
// add a scaled vector
addScaled(v, s){
this.x += v.x * s;
this.y += v.y * s;
this.z += v.z * s;
}
// component product
componentProduct(v){
this.x *= v.x;
this.y *= v.y;
this.z *= v.z;
}
// scalar (dot) product (operator used in the book: *)
scalarProduct(v){
return this.x * v.x + this.y*v.y + this.z*v.z;
}
// vector (cross) product (operator used in the book: %)
vectorProduct(v){
return Vector3(this.y*b.z - this.z*b.y,
this.z*b.x - this.x*b.z,
this.x*b.y - this.y*b.x);
}
// Clone
clone(){
return new Vector3(this.x, this.y, this.z);
}
}
The part of calculus that interests us here is how things change over time (position, force, velocity...).
The change itself is described with differential calculus, and its effect with integral calculus.
Differential calculus
Velocity is the rate of change of a value over time:
v = (change in position) / (time passed) = dp / dt = ̇p
Acceleration is the rate of change of velocity:
a = dv / dt = d²p / dt² = p̈
Velocity and acceleration can be applied to vectors:
̇a = [̇ax, ̇ay, ̇az]
ä = [äx, äy, äz]
(Such vectors can be added, subtracted and scaled normally. Vector products are different).
The speed s is the magnitude of the velocity vector:
̇x = sd̂
Integral calculus
Integration is the opposite of differentiation. In a game's physics engine, it refers to updating position and velocity at each frame (in regard to time t):
p' = p + ̇pt
̇p' = ̇p + p̈t
So, p' = p + ̇pt + 1/2p̈t²
Here's how to update a position with code:
position.addScaledVector(velocity, time);
// position.addScaledVector(acceleration, time * time * 0.5);
(In reality, the effect of acceleration on the position is negligeable and the last line is generally omitted).