Game Physics JS: Hard constraints

Simple collision resolution

Collisions refer to situations where two objects are touching, with or without closing velocity.

The closing velocity vc is the total speed at which two object move together. The separating velocity vs is the opposite:

vs = - vc = (̇pa - ̇pb)·(normalize(pa - pb))

The coefficient of restitution c controls the speed at which two objects separate after a collision:

v's = -cvs

The contact normal between two colliding particles a and b, seen from object a's perspective, is:

n̂ = normalize(pa - pb)

In other terms:

vs = (̇pa - ̇pb)·n̂

The collision resolution is not a force added to the acceleration but an impulse g (an instant change in velocity).

g = ṁp

Impulses aren't accumulated like forces, but applied one at a time, to the object's position, as soon as they occur.

Collision processing

JavaScript implementation

class ParticleContact { particles; restitution; contactNormal; penetration; constructor(particles, restitution, contactNormal, penetration){ this.particles = particles; this.restitution = restitution; this.contactNormal = contactNormal; this.penetration = penetration; } resolve(duration){ this.resolveVelocity(duration); this.resolveInterpenetration(duration); } calculateSeparatingVelocity(){ var relativeVelocity = this.particles[0].velocity.clone(); if(this.particles[1]) relativeVelocity.sub(particle[1].velocity); return dot(relativeVelocity, this.contactNormal); // scalar value } resolveVelocity(duration){ // Find the velocity in the direction of the contact var separatingVelocity = this.calculateSeparatingVelocity(); // Check if it needs to be resolved if(separatingVelocity > 0){ // If the contact is separating or stationary, no impulse is required return; } // Calculate the new separating velocity var newSepVelocity = -separatingVelocity * this.restitution; // TODO: handle resting contact (see below) // Compute velocity difference var deltaVelocity = newSepVelocity - separatingVelocity; // Compute total inverse mass var totalInverseMass = this.particles[0].inverseMass; if(this.particles[1]) totalInverseMass += this.particles[1].inverseMass; // If all particles have infinite mass, then impulses have no effect if (totalInverseMass <= 0) return; // Calculate the impulse to apply var impulse = deltaVelocity / totalInverseMass; // Find the amount of impulse per unit of inverse mass var impulsePerIMass = this.contactNormal.scale(impulse); // Apply impulses: in the direction of the contact, proportional to the inverse mass this.particles[0].velocity.addScaledVector(impulsePerIMass, this.particles[0].inverseMass); if(this.particles[1]){ // Particle 1 goes in the opposite direction this.particles[1].velocity.addScaledVector(impulsePerIMass, -this.particles[1].inverseMass); } } }

Collision detection will be detailed later.

Resolving collisions consists in moving interpenetrating objects apart, proportionally to their inverse mass, to separate them.

Δpa = (mb / (ma + mb))dn Δpb = -(ma / (ma + mb))dn

Where Δp is the change of position, m the mass, n the contact normal in object a's perspective and d the interpenetration depth.

class ParticleContact{ // ... resolveInterpenetration(duration){ // If we don’t have any penetration, skip this step if(this.penetration <= 0) return; // The movement of each object is based on their inverse mass, so total that var totalInverseMass = this.particles[0].inverseMass; if(this.particles[1]){ totalInverseMass += this.particles[1].inverseMass; } // If all particles have infinite mass, then we do nothing if(totalInverseMass <= 0) return; // Find the amount of penetration resolution per unit of inverse mass var movePerIMass = scale(this.contactNormal, (this.penetration / totalInverseMass)); // Calculate the movement amounts var particleMovement = []; particleMovement[0] = movePerIMass * this.particles[0].inverseMass; if(particle[1]){ particleMovement[1] = movePerIMass * -this.particles[1].inverseMass; } else{ particleMovement[1] = new Vector3(0,0,0); } // Apply the penetration resolution this.particles[0].position.add(particleMovement[0]); if(this.particles[1]){ this.particles[1].position.add(particleMovement[1]); } } }

Resting contacts (ex: a particle resting on a plane) can cause objects to vibrate due to rounding errors accumulating at each frame.

To solve that, we can cancel the restitution as soon as the velocity buildup is equal to the acceleration.

resolveVelocity(duration){ // ... // Calculate the new separating velocity // ... // Check the velocity buildup due to acceleration only var accCausedVelocity = this.particles[0].acceleration.clone(); if(this.particles[1]){ accCausedVelocity -= this.particles[1].acceleration; } var accCausedSepVelocity = accCausedVelocity.addScaledVector(this.contactNormal, duration); // If closing velocity is due to acceleration buildup, remove it from separating velocity if(accCausedSepVelocity < 0){ newSepVelocity += restitution * accCausedSepVelocity; // Make sure we haven’t removed more than was there to remove if(newSepVelocity < 0){ newSepVelocity = 0; } } // ... }

Resolution order: solve the most severe contacts first (the ones with the biggest closing velocity).

TODO (p 164)

Collision-like things

1) Cables can force two objects to be no more than a specific distance apart.

They're similar to regular collisions but with inverse normals, to pull the separating objects together.

TODO (p 168)

2) Rods enforce a fixed distance between two objects.

TODO (p 171)



Next