📄 collisiondetector.as
字号:
/*
Copyright (c) 2006, 2007 Alec Cove
Permission is hereby granted, free of charge, to any person obtaining a copy of this
software and associated documentation files (the "Software"), to deal in the Software
without restriction, including without limitation the rights to use, copy, modify,
merge, publish, distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject to the following
conditions:
The above copyright notice and this permission notice shall be included in all copies
or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE
OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
/*
TODO:
- Get rid of all the object testing and use the double dispatch pattern
- There's some physical differences in collision response for multisampled
particles, probably due to prev/curr differences.
*/
package org.cove.ape {
internal final class CollisionDetector {
/**
* Tests the collision between two objects. If there is a collision it is passed off
* to the CollisionResolver class.
*/
internal static function test(objA:AbstractParticle, objB:AbstractParticle):void {
if (objA.fixed && objB.fixed) return;
if (objA.multisample == 0 && objB.multisample == 0) {
normVsNorm(objA, objB);
} else if (objA.multisample > 0 && objB.multisample == 0) {
sampVsNorm(objA, objB);
} else if (objB.multisample > 0 && objA.multisample == 0) {
sampVsNorm(objB, objA);
} else if (objA.multisample == objB.multisample) {
sampVsSamp(objA, objB);
} else {
normVsNorm(objA, objB);
}
}
/**
* default test for two non-multisampled particles
*/
private static function normVsNorm(objA:AbstractParticle, objB:AbstractParticle):void {
objA.samp.copy(objA.curr);
objB.samp.copy(objB.curr);
testTypes(objA, objB);
}
/**
* Tests two particles where one is multisampled and the other is not. Let objectA
* be the multisampled particle.
*/
private static function sampVsNorm(objA:AbstractParticle, objB:AbstractParticle):void {
var s:Number = 1 / (objA.multisample + 1);
var t:Number = s;
objB.samp.copy(objB.curr);
for (var i:int = 0; i <= objA.multisample; i++) {
objA.samp.setTo(objA.prev.x + t * (objA.curr.x - objA.prev.x),
objA.prev.y + t * (objA.curr.y - objA.prev.y));
if (testTypes(objA, objB)) return;
t += s;
}
}
/**
* Tests two particles where both are of equal multisample rate
*/
private static function sampVsSamp(objA:AbstractParticle, objB:AbstractParticle):void {
var s:Number = 1 / (objA.multisample + 1);
var t:Number = s;
for (var i:int = 0; i <= objA.multisample; i++) {
objA.samp.setTo(objA.prev.x + t * (objA.curr.x - objA.prev.x),
objA.prev.y + t * (objA.curr.y - objA.prev.y));
objB.samp.setTo(objB.prev.x + t * (objB.curr.x - objB.prev.x),
objB.prev.y + t * (objB.curr.y - objB.prev.y));
if (testTypes(objA, objB)) return;
t += s;
}
}
/**
*
*/
private static function testTypes(objA:AbstractParticle, objB:AbstractParticle):Boolean {
if (objA is RectangleParticle && objB is RectangleParticle) {
return testOBBvsOBB(objA as RectangleParticle, objB as RectangleParticle);
} else if (objA is CircleParticle && objB is CircleParticle) {
return testCirclevsCircle(objA as CircleParticle, objB as CircleParticle);
} else if (objA is RectangleParticle && objB is CircleParticle) {
return testOBBvsCircle(objA as RectangleParticle, objB as CircleParticle);
} else if (objA is CircleParticle && objB is RectangleParticle) {
return testOBBvsCircle(objB as RectangleParticle, objA as CircleParticle);
}
return false;
}
/**
* Tests the collision between two RectangleParticles (aka OBBs). If there is a collision it
* determines its axis and depth, and then passes it off to the CollisionResolver for handling.
*/
private static function testOBBvsOBB(ra:RectangleParticle, rb:RectangleParticle):Boolean {
var collisionNormal:Vector;
var collisionDepth:Number = Number.POSITIVE_INFINITY;
for (var i:int = 0; i < 2; i++) {
var axisA:Vector = ra.axes[i];
var depthA:Number = testIntervals(ra.getProjection(axisA), rb.getProjection(axisA));
if (depthA == 0) return false;
var axisB:Vector = rb.axes[i];
var depthB:Number = testIntervals(ra.getProjection(axisB), rb.getProjection(axisB));
if (depthB == 0) return false;
var absA:Number = Math.abs(depthA);
var absB:Number = Math.abs(depthB);
if (absA < Math.abs(collisionDepth) || absB < Math.abs(collisionDepth)) {
var altb:Boolean = absA < absB;
collisionNormal = altb ? axisA : axisB;
collisionDepth = altb ? depthA : depthB;
}
}
CollisionResolver.resolveParticleParticle(ra, rb, collisionNormal, collisionDepth);
return true;
}
/**
* Tests the collision between a RectangleParticle (aka an OBB) and a CircleParticle.
* If there is a collision it determines its axis and depth, and then passes it off
* to the CollisionResolver.
*/
private static function testOBBvsCircle(ra:RectangleParticle, ca:CircleParticle):Boolean {
var collisionNormal:Vector;
var collisionDepth:Number = Number.POSITIVE_INFINITY;
var depths:Array = new Array(2);
// first go through the axes of the rectangle
for (var i:int = 0; i < 2; i++) {
var boxAxis:Vector = ra.axes[i];
var depth:Number = testIntervals(ra.getProjection(boxAxis), ca.getProjection(boxAxis));
if (depth == 0) return false;
if (Math.abs(depth) < Math.abs(collisionDepth)) {
collisionNormal = boxAxis;
collisionDepth = depth;
}
depths[i] = depth;
}
// determine if the circle's center is in a vertex region
var r:Number = ca.radius;
if (Math.abs(depths[0]) < r && Math.abs(depths[1]) < r) {
var vertex:Vector = closestVertexOnOBB(ca.samp, ra);
// get the distance from the closest vertex on rect to circle center
collisionNormal = vertex.minus(ca.samp);
var mag:Number = collisionNormal.magnitude();
collisionDepth = r - mag;
if (collisionDepth > 0) {
// there is a collision in one of the vertex regions
collisionNormal.divEquals(mag);
} else {
// ra is in vertex region, but is not colliding
return false;
}
}
CollisionResolver.resolveParticleParticle(ra, ca, collisionNormal, collisionDepth);
return true;
}
/**
* Tests the collision between two CircleParticles. If there is a collision it
* determines its axis and depth, and then passes it off to the CollisionResolver
* for handling.
*/
private static function testCirclevsCircle(ca:CircleParticle, cb:CircleParticle):Boolean {
var depthX:Number = testIntervals(ca.getIntervalX(), cb.getIntervalX());
if (depthX == 0) return false;
var depthY:Number = testIntervals(ca.getIntervalY(), cb.getIntervalY());
if (depthY == 0) return false;
var collisionNormal:Vector = ca.samp.minus(cb.samp);
var mag:Number = collisionNormal.magnitude();
var collisionDepth:Number = (ca.radius + cb.radius) - mag;
if (collisionDepth > 0) {
collisionNormal.divEquals(mag);
CollisionResolver.resolveParticleParticle(ca, cb, collisionNormal, collisionDepth);
return true;
}
return false;
}
/**
* Returns 0 if intervals do not overlap. Returns smallest depth if they do.
*/
private static function testIntervals(intervalA:Interval, intervalB:Interval):Number {
if (intervalA.max < intervalB.min) return 0;
if (intervalB.max < intervalA.min) return 0;
var lenA:Number = intervalB.max - intervalA.min;
var lenB:Number = intervalB.min - intervalA.max;
return (Math.abs(lenA) < Math.abs(lenB)) ? lenA : lenB;
}
/**
* Returns the location of the closest vertex on r to point p
*/
private static function closestVertexOnOBB(p:Vector, r:RectangleParticle):Vector {
var d:Vector = p.minus(r.samp);
var q:Vector = new Vector(r.samp.x, r.samp.y);
for (var i:int = 0; i < 2; i++) {
var dist:Number = d.dot(r.axes[i]);
if (dist >= 0) dist = r.extents[i];
else if (dist < 0) dist = -r.extents[i];
q.plusEquals(r.axes[i].mult(dist));
}
return q;
}
}
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -