Game Physics JS: Mathematics of particles

Vectors

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; } }

Handedness

Coordinate systems and vectors operations depend on a left of right handedness. Right will be used here, as it is favored by WebGL.

Vectors and directions

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|

Multiplying a scalar and a vector

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|).

Vectors addition and subtraction

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]

Vectors multiplication

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); } }

Calculus

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).



Next