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