Skipped: how to approximate a shape with an assembly of primitives (p. 323-325).
Collision detection can be done by the GJK algorithm. However, it doesn't return all the contact information we need.
The SAT algorithm returns only one contact point per collsion, which is sometimes not enough.
These algorithms can be extended with a technique called contact coherence, which is also not perfect.
Two 3D objects can be in contact in 6 different (and non-exclusive) ways.
Contacts should be detected with the following priority:
1. Vertex–face and edge–edge (nonparallel edges)
2. Edge–face and face–face
3. Vertex–edge, vertex–vertex, and edge–edge (parallel edges)
Contacts of the 3rd group are generally ignored because at the next frame, they will become a contact of higher priority.
The contact data includes:
1) A collision point (if there are many, we can pick one arbitrarily without affecting the physics too much).
2) A collision normal.
3) The penetration depth.
4) The collision restitution.
5) The friction.
class Contact {
contactPoint;
contactNormal;
penetration;
constructor(){
// ...
}
}
In vertex-face contacts, the normal will only depend on the face and the vertex can simply be chosen as the contact point.
In edge-edge contacts, the normal is at right angles to the tangent of both edges. The contact point is the closest point from an edge on the other edge.
Edge-face contacts are only used with curved surfaces, and are similar to vertex-face.
Face-face contacts occur between two curved surfaces or one curved and one flat. The normal is one of the face's normals (chosen consistently to avod misalignments). The contact point can be the pont of greatest penetration.
Setup
class CollisionData {
contacts;
contactsLeft;
constructor(){
// ...
}
}
class Primitive {
body;
offset;
constructor(){
// ...
}
}
class Sphere extends Primitive {
position;
radius;
constructor(){
// super(...);
}
}
class Plane extends Primitive {
normal;
offset;
constructor(){
// super(...);
}
}
1) Sphere-sphere collision:
two spheres collide if the distance between their centers is less than the sum of their radii.
It's a face-face contact, and there is just one contact point.
CollisionDetector {
sphereAndSphere(one, two, data){
// Make sure we have contacts
if(data.contactsLeft <= 0) return 0;
// Cache the sphere positions
var positionOne = one.position.clone();
var positionTwo = two.position.clone();
// Find the vector between the objects
var midline = positionOne.clone().sub(positionTwo);
var size = midline.magnitude();
// See if it is large enough
if(size <= 0 || size >= one.radius + two.radius){
return 0;
}
// Create the normal
var normal = midline.scale(1/size);
var contact = data.contacts;
contact.contactNormal = normal;
contact.contactPoint = positionOne.addScaled(midline, 0.5);
contact.penetration = (one.radius + two.radius - size);
contact.setBodyData(one.body, two.body, data.friction, data.restitution);
data.addContacts(1);
return 1;
}
}
DEMO
2) Sphere-plane collision
It's actually more convenient to consider a collision with a half-space instead of a plane (with no depth).
In all cases, there's one face-face contact point.
The plane is represented by a vertical offset (l) and a normal vector (n).
The distance between a point (p) and a plane is:
d = p · n − l
sphereAndHalfSpace(sphere, plane){
// Make sure we have contacts
if (data.contactsLeft <= 0) return 0;
// Cache the sphere position
var position = sphere.position.clone();
// Find the distance from the plane
var ballDistance = dot(plane.normal, position) -
sphere.radius - plane.offset;
// Exit if not touching
if (ballDistance >= 0) return 0;
// Create the contact
var contact = data.contacts;
contact.contactNormal = plane.direction;
contact.penetration = -ballDistance;
contact.contactPoint = position.addScaledVector(plane.direction, -(ballDistance + sphere.radius));
contact.setBodyData(sphere.body, NULL,data.friction, data.restitution);
data.addContacts(1);
return 1;
}
DEMO
3) Box-plane collision
Up to 4 point-face contact points.
The approach here is to check each vertex of the box against the half-space.
However, it will be simpler to implement it later, with SAT.
4) Box-sphere collision
One contact point (vertex-face, edge-face or face-face).
We need to find the point of the box that is the closest to the center of the sphere.
If the distance to the center is smaller than the sphere's radius, there's a collision.
The closest point and its distance become the contact point and penetration depth.
The collision normal points to the center of the sphere.
boxAndSphere(box, sphere){
// Transform the centre of the sphere into box coordinates
var centre = sphere.position.clone();
var relCentre = box.transform.transformInverse(centre);
// Early out check to see if we can exclude the contact
if(
Math.abs(relCentre.x) - sphere.radius > box.halfSize.x ||
Math.abs(relCentre.y) - sphere.radius > box.halfSize.y ||
Math.abs(relCentre.z) - sphere.radius > box.halfSize.z
){
return 0;
}
var closestPt = new Vector3(0,0,0);
var dist;
// Clamp each coordinate to the box
dist = relCentre.x;
if(dist > box.halfSize.x) dist = box.halfSize.x;
if(dist < -box.halfSize.x) dist = -box.halfSize.x;
closestPt.x = dist;
dist = relCentre.y;
if(dist > box.halfSize.y) dist = box.halfSize.y;
if(dist < -box.halfSize.y) dist = -box.halfSize.y;
closestPt.y = dist;
dist = relCentre.z;
if(dist > box.halfSize.z) dist = box.halfSize.z;
if(dist < -box.halfSize.z) dist = -box.halfSize.z;
closestPt.z = dist;
// Check we're in contact
dist = closestPt.sub(relCentre).squareMagnitude();
if(dist > sphere.radius ** 2) return 0;
// Compile the contact
var closestPtWorld = box.transform.transform(closestPt);
data.contacts[0].contactNormal = (closestPtWorld - centre);
data.contacts[0].contactNormal.normalise();
data.contacts[0].contactPoint = closestPtWorld;
data.contacts[0].penetration = sphere.radius - real_sqrt(dist);
return data;
}
DEMO todo
The idea is to project two shapes along a set of important axis, and see if they are separated or overlapping.
If no separation is found, the objects collide.
For two boxes, there are 15 axis to check.
penetrationOnAxis(
const CollisionBox &one,
const CollisionBox &two,
const Vector3 &axis,
const Vector3 &toCenter // vector from the center of the first box to the center of the second
)
{
real oneProject = transformToAxis(one, axis);
real twoProject = transformToAxis(two, axis);
real distance = real_abs(toCenter * axis);
// Return the overlap (i.e., positive indicates
// overlap, negative indicates separation).
return oneProject + twoProject - distance;
}
As these approaches only return a single contact point, one solution could be to fill a contact points array over several frames.
TODO