Collision detection is an important part of most 3D games. Shooting enemies, avoiding (or failing to avoid) obstacles, even simply staying on the ground usually requires some form of collision-detection. It's a vast subject and there are many excellent articles written about it. However, most of them concern the more sophisticated algorithms, such as BSP and the simple bounding-sphere check receives no more than a passing mention. It is a quite simple procedure but it still could be very useful and there are a couple of subtle points involved. So I thought perhaps I should write something about it.
[size="5"]Straightforward version
The simplest possible way to do bounding-sphere test is to measure the (squared) distance between the two objects in question and to compare the result with the (squared again) sum of their radii. The reason we use the squared distances is to avoid the costly square-root calculation. The code will look something like this:
BOOL bSphereTest(CObject3D* obj1, CObject3D* obj2 )
{
D3DVECTOR relPos = obj1->prPosition - obj2->prPosition;
float dist = relPos.x * relPos.x + relPos.y * relPos.y + relPos.z * relPos.z;
float minDist = obj1->fRadius + obj2->fRadius;
return dist <= minDist * minDist;
}
Note: This and the following code snippets use the D3DVECTOR type from Microsoft's Direct3D, if you are using some other 3D library, change the code accordingly.
This procedure is easy indeed and some cases may be even adequate but what if your objects are moving at a considerable speed? Then it could happen that at the beginning of a frame the bounding spheres do not intersect yet by the end of the frame they have passed through each other. If the distance your objects travel in one frame is greater than the size of the object this scenario is quite likely. Clearly we need something more advanced to handle this case.
[size="5"]Improved version
What we need is to somehow incorporate the object velocities into the calculation. That is quite simple if you do just a little bit of vector math (you can skip this section if you have a math phobia).
Let
data:image/s3,"s3://crabby-images/20f28/20f28af76fc6d2399c4ab3f74f3c3c84326679d9" alt="article_gr_1.gif"
data:image/s3,"s3://crabby-images/464cd/464cde2acb6652a9c1c0df2e4325339f68a6d56a" alt="article_gr_2.gif"
data:image/s3,"s3://crabby-images/79778/797785ba6fdc8ca99b78f4e5d4908958b7371a29" alt="article_gr_3.gif"
data:image/s3,"s3://crabby-images/25cea/25cea74abc8aa834286dd276bad5f496437b45c8" alt="article_gr_4.gif"
data:image/s3,"s3://crabby-images/2cec0/2cec02eab0c838ac4bd19edffc70c626fffebe7c" alt="article_gr_5.gif"
data:image/s3,"s3://crabby-images/47162/47162b6a00ba382a0f348cabd61204fbd958a5f9" alt="article_gr_6.gif"
data:image/s3,"s3://crabby-images/80d7f/80d7fe04fbd6496e713f45448383be2960d27c01" alt="article_gr_7.gif"
data:image/s3,"s3://crabby-images/2f3c2/2f3c2eb5baf9f07234f3c117b1b73ee62c0622d2" alt="article_gr_8.gif"
Equation 1
and their relative position will be
data:image/s3,"s3://crabby-images/bcc3f/bcc3f8ce3269589f1382d5e44120d103221eb3ca" alt="article_gr_9.gif"
The distance between the objects will be just the magnitude of the relative-position vector or, which is the same thing, the square of the distance will be equal to the square of the vector:
data:image/s3,"s3://crabby-images/b27c8/b27c800900d2ac017598f63e44e4424a1a4a3e1d" alt="article_gr_10.gif"
Note: I use * to denote the dot product between two vectors.
Now all we need is to find if
data:image/s3,"s3://crabby-images/8d003/8d003eb03385729ac3bd3eaaea1375ec6838b68f" alt="article_gr_11.gif"
data:image/s3,"s3://crabby-images/aaaac/aaaac3aadb0c4f66e007f9370e5330d40129cb32" alt="article_gr_12.gif"
data:image/s3,"s3://crabby-images/f3cf5/f3cf58368110f9cb125140b6506cc421bc267e56" alt="article_gr_13.gif"
data:image/s3,"s3://crabby-images/95276/95276e7267cb0cbba32a1b24a4884865fcae9136" alt="article_gr_14.gif"
The way to do it is to check if the distance already less than the minimal (we might have somehow missed the collision on the previous frame, because of physics and AI corrections or whatever) and if not then we need to solve the equation
data:image/s3,"s3://crabby-images/8e086/8e086aa7b012aeecf60270382460466593d62744" alt="article_gr_15.gif"
data:image/s3,"s3://crabby-images/79778/797785ba6fdc8ca99b78f4e5d4908958b7371a29" alt="article_gr_3.gif"
data:image/s3,"s3://crabby-images/aaaac/aaaac3aadb0c4f66e007f9370e5330d40129cb32" alt="article_gr_12.gif"
data:image/s3,"s3://crabby-images/b9221/b92217eca71ac066e36491aea5864e39b6364198" alt="article_gr_18.gif"
I have omitted [0] to make the formula more readable.
This is a simple quadratic equation and we solve it in the usual way, by finding the determinant:
data:image/s3,"s3://crabby-images/9e196/9e1960580ce451ea30709f8c076b17017545d9d3" alt="article_gr_19.gif"
Equation 2
If D<0 then the equation does not have a solution, that is the spheres do not collide at all. If D>0 then the spheres collide (if D=0 then the spheres merely touch each other, you can handle this case either way, depending on the specifics of the situation). The roots of the equation are:
data:image/s3,"s3://crabby-images/cefdd/cefddb91ae9b1f25ecc936931cb915681a720dbc" alt="article_gr_20.gif"
data:image/s3,"s3://crabby-images/02054/02054a77883b64d6d2a7c69345502cf6d5c66a8a" alt="article_gr_21.gif"
Equation 3
Then we can take
data:image/s3,"s3://crabby-images/659eb/659eb21d641385afca012c9ad0ed0647b7b712d9" alt="article_gr_22.gif"
data:image/s3,"s3://crabby-images/e7ff6/e7ff687d8175d7a203bffdc46994b0bd14e339c6" alt="article_gr_23.gif"
Second, and more important we can test if the roots are between 0 and 1 even before calculating
data:image/s3,"s3://crabby-images/659eb/659eb21d641385afca012c9ad0ed0647b7b712d9" alt="article_gr_22.gif"
data:image/s3,"s3://crabby-images/14b8b/14b8b6468dd58b8e73d9a871be6791f73b477197" alt="article_gr_25.gif"
Equation 4
This is a case of what is called Viete's theorem. Let's look closer at this, specifically let's look at the signs of these values. If
data:image/s3,"s3://crabby-images/19c3e/19c3e95eb7d10e0d0745fcdcf7c3c2a69fc042e0" alt="article_gr_26.gif"
data:image/s3,"s3://crabby-images/d32a1/d32a18edae1f27c5326ce46692b7f00d0cebf1c7" alt="article_gr_27.gif"
data:image/s3,"s3://crabby-images/6b3da/6b3da98937357dcb5aa405ebb8664c7f8d4ed680" alt="article_gr_28.gif"
data:image/s3,"s3://crabby-images/4c045/4c045fad9d49bafd90decb5fdb1a2643101e651a" alt="article_gr_29.gif"
data:image/s3,"s3://crabby-images/ba606/ba606de643ace8b00e2990064a7e58da899aab36" alt="article_gr_30.gif"
data:image/s3,"s3://crabby-images/bded7/bded7d75e676cfd43b5a1a04636c98cd08a3b844" alt="article_gr_31.gif"
data:image/s3,"s3://crabby-images/cc5cc/cc5ccb9286884c0b00ab45f45cab3080fb3ab820" alt="article_gr_32.gif"
To make these cases even easier note that the denominators in both cases are greater than 0 - just like the square of any real number. (if they are equal to 0 then the relative speed of the spheres is 0, not moving at all. Then everything is just like in the previous section). Therefore the signs of these expressions are the same as the signs of their numerators. The numerator of the second expression is just the square of the distance between the spheres at t=0 minus the
data:image/s3,"s3://crabby-images/4d623/4d623886a4c5559e67b4ecd66d25cdace7542618" alt="article_gr_33.gif"
data:image/s3,"s3://crabby-images/ed786/ed786accb7b9a541d47aa0d95df922e5b3859bad" alt="article_gr_34.gif"
Ok, so we can find the cases where solution is <0 that is when the collision has already happened. Now we need also to weed out the cases when the solution is >1. That is the collision might happen but not within the next frame, so we are not interested. (if you are interested in these situations as well just skip this check). To find out we just write the condition out:
data:image/s3,"s3://crabby-images/74f3c/74f3c7a34b8a1551690632c1d37a77a8f70566b6" alt="article_gr_35.gif"
Or, after some algebraic manipulations, we arrive at these 2 conditions
data:image/s3,"s3://crabby-images/58899/588995d01bd31aed7fbdcee13f21d76c34879fcd" alt="article_gr_36.gif"
data:image/s3,"s3://crabby-images/462e0/462e0ce1684052fbcaba806ee0e573544a43484e" alt="article_gr_37.gif"
If these two inequalities hold true then
data:image/s3,"s3://crabby-images/1c337/1c3377116b2262487934cc850f9747ee76d42329" alt="article_gr_38.gif"
Now, If we passed all the tests then we have no choice but to calculate D and see if it is >0. If you wish to know the exact time (or position) of the collision then you can go on and calculate its square root, find
data:image/s3,"s3://crabby-images/a73d5/a73d528ba8013cdea178dc1c69dca75f33b91e30" alt="article_gr_39.gif"
BOOL bSphereTest(CObject3D* obj1, CObject3D* obj2 )
{
// Relative velocity
D3DVECTOR dv = obj2->prVelocity - obj1->prVelocity;
// Relative position
D3DVECTOR dp = obj2->prPosition - obj1->prPosition;
//Minimal distance squared
float r = obj1->fRadius + obj2->fRadius;
//dP^2-r^2
float pp = dp.x * dp.x + dp.y * dp.y + dp.z * dp.z - r*r;
//(1)Check if the spheres are already intersecting
if ( pp < 0 ) return true;
//dP*dV
float pv = dp.x * dv.x + dp.y * dv.y + dp.z * dv.z;
//(2)Check if the spheres are moving away from each other
if ( pv >= 0 ) return false;
//dV^2
float vv = dv.x * dv.x + dv.y * dv.y + dv.z * dv.z;
//(3)Check if the spheres can intersect within 1 frame
if ( (pv + vv) <= 0 && (vv + 2 * pv + pp) >= 0 ) return false;
//Discriminant/4
float D = pv * pv - pp * vv;
return ( D > 0 );
}
[size="5"]One more improvement
The code in snippet 2 works quite well in many circumstances but there's a little problem with it. It is not a mathematical problem but a computer-related one. Think about the very last calculation
float D = pv * pv - pp * vv;
BOOL bSphereTest(CObject3D* obj1, CObject3D* obj2)
{
//Initialize the return value
*t = 0.0f;
// Relative velocity
D3DVECTOR dv = obj2->prVelocity - obj1->prVelocity;
// Relative position
D3DVECTOR dp = obj2->prPosition - obj1->prPosition;
//Minimal distance squared
float r = obj1->fRadius + obj2->fRadius;
//dP^2-r^2
float pp = dp.x * dp.x + dp.y * dp.y + dp.z * dp.z - r*r;
//(1)Check if the spheres are already intersecting
if ( pp < 0 ) return true;
//dP*dV
float pv = dp.x * dv.x + dp.y * dv.y + dp.z * dv.z;
//(2)Check if the spheres are moving away from each other
if ( pv >= 0 ) return false;
//dV^2
float vv = dv.x * dv.x + dv.y * dv.y + dv.z * dv.z;
//(3)Check if the spheres can intersect within 1 frame
if ( (pv + vv) <= 0 && (vv + 2 * pv + pp) >= 0 ) return false;
//tmin = -dP*dV/dV*2
//the time when the distance between the spheres is minimal
float tmin = -pv/vv;
//Discriminant/(4*dV^2) = -(dp^2-r^2+dP*dV*tmin)
return ( pp + pv * tmin > 0 );
}
[size="5"]Conclusion
So here it is, the bounding-sphere collision detection algorithm. It might be useful as is (say, for a pool game) or can be a quick-and-dirty test before doing a more sophisticated check - polygon-level, for instance. You may also want to try to improve the accuracy of the collision detection by using a hierarchy of bounding spheres, breaking your object into several parts and enclosing each of them in a bounding sphere of its own.
//Discriminant/(4*dV^2) = -(dp^2-r^2+dP*dV*tmin)
return ( pp + pv * tmin [color=#ff0000][font=arial,helvetica,sans-serif][i][b]<[/b][/i][/font][/color] 0 );