You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3125 lines
66 KiB
3125 lines
66 KiB
4 years ago
|
/*==LICENSE==*
|
||
|
|
||
|
CyanWorlds.com Engine - MMOG client, server and tools
|
||
|
Copyright (C) 2011 Cyan Worlds, Inc.
|
||
|
|
||
|
This program is free software: you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License as published by
|
||
|
the Free Software Foundation, either version 3 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
This program is distributed in the hope that it will be useful,
|
||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
GNU General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU General Public License
|
||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
Additional permissions under GNU GPL version 3 section 7
|
||
|
|
||
|
If you modify this Program, or any covered work, by linking or
|
||
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
|
||
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
|
||
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
|
||
|
(or a modified version of those libraries),
|
||
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
|
||
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
|
||
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
|
||
|
licensors of this Program grant you additional
|
||
|
permission to convey the resulting work. Corresponding Source for a
|
||
|
non-source form of such a combination shall include the source code for
|
||
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
|
||
|
work.
|
||
|
|
||
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
||
|
or by snail mail at:
|
||
|
Cyan Worlds, Inc.
|
||
|
14617 N Newport Hwy
|
||
|
Mead, WA 99021
|
||
|
|
||
|
*==LICENSE==*/
|
||
|
|
||
|
#include "hsTypes.h"
|
||
|
#include "hsBounds.h"
|
||
|
#include "hsStream.h"
|
||
|
|
||
|
#include "hsFastMath.h"
|
||
|
|
||
|
#if defined(__MWERKS__) && !defined(HS_DEBUGGING)
|
||
|
#pragma optimization_level 2
|
||
|
#endif
|
||
|
|
||
|
const hsScalar hsBounds::kRealSmall = 1.0e-5f;
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// hsBounds
|
||
|
//
|
||
|
/////////////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
void hsBounds::Read(hsStream *s)
|
||
|
{
|
||
|
fType =(hsBoundsType) s->ReadSwap32();
|
||
|
}
|
||
|
|
||
|
void hsBounds::Write(hsStream *s)
|
||
|
{
|
||
|
s->WriteSwap32((Int32)fType);
|
||
|
}
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// hsBounds3
|
||
|
//
|
||
|
/////////////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#if 0 // MESH_GEN_DEFER
|
||
|
void hsBounds3::Draw(hsGView3* v, hsG3DDevice* d,
|
||
|
hsScalar r, hsScalar g, hsScalar b, hsScalar a,
|
||
|
hsBool spheric)
|
||
|
{
|
||
|
|
||
|
hsGViewClipState* clipState = v->SaveClipDisabled();
|
||
|
if( hsGClip3::kClipCulled & v->ClipTestBounds(this) )
|
||
|
{
|
||
|
v->RestoreClipDisabled(clipState);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Setup Material
|
||
|
hsGMaterial *mat = TRACKED_NEW hsGMaterial;
|
||
|
hsGLayer* lay = mat->MakeBaseLayer();
|
||
|
lay->SetAmbientColor(r,g,b,a);
|
||
|
lay->SetMiscFlags(hsGMatState::kMiscWireFrame | hsGMatState::kMiscTwoSided);
|
||
|
lay->SetShadeFlags(hsGMatState::kShadeNoShade);
|
||
|
mat->SetLayer(lay, 0);
|
||
|
|
||
|
// Setup tMesh
|
||
|
hsGTriMesh tMesh;
|
||
|
if( spheric )
|
||
|
MakeTriMeshSphere(&tMesh);
|
||
|
else
|
||
|
MakeTriMesh(&tMesh, hsTriangle3::kTwoSided);
|
||
|
|
||
|
|
||
|
tMesh.SetMaterial(mat);
|
||
|
hsRefCnt_SafeUnRef(mat);
|
||
|
|
||
|
tMesh.Render(v,d);
|
||
|
|
||
|
v->RestoreClipDisabled(clipState);
|
||
|
}
|
||
|
#endif // MESH_GEN_DEFER
|
||
|
|
||
|
void hsBounds3::Transform(const hsMatrix44 *mat)
|
||
|
{
|
||
|
#if 0 // IDENT
|
||
|
if( mat->fFlags & hsMatrix44::kIsIdent )
|
||
|
return;
|
||
|
#endif // IDENT
|
||
|
|
||
|
hsAssert(fType != kBoundsUninitialized, "Can't transform an unitialized bound");
|
||
|
if(fType == kBoundsNormal)
|
||
|
{
|
||
|
hsPoint3 corners[8];
|
||
|
this->GetCorners(corners);
|
||
|
|
||
|
mat->MapPoints(8, corners);
|
||
|
this->Reset(8,corners);
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBounds3::Reset(const hsPoint3 *p)
|
||
|
{
|
||
|
fType = kBoundsNormal;
|
||
|
fMins = fMaxs = *p;
|
||
|
fBounds3Flags |= kCenterValid;
|
||
|
fCenter = *p;
|
||
|
}
|
||
|
|
||
|
void hsBounds3::Reset(const hsBounds3 *b)
|
||
|
{
|
||
|
if( kBoundsNormal == b->fType )
|
||
|
{
|
||
|
fType = kBoundsNormal;
|
||
|
fMins = b->fMins;
|
||
|
fMaxs = b->fMaxs;
|
||
|
if( b->fBounds3Flags & kCenterValid )
|
||
|
{
|
||
|
fBounds3Flags |= kCenterValid;
|
||
|
fCenter = b->fCenter;
|
||
|
}
|
||
|
else
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
else
|
||
|
fType = b->fType;
|
||
|
}
|
||
|
|
||
|
void hsBounds3::Reset(int n, const hsPoint3 *p)
|
||
|
{
|
||
|
fType = kBoundsNormal;
|
||
|
fMins = fMaxs = *p;
|
||
|
for(int i = 1; i < n ; i++)
|
||
|
this->Union(&p[i]);
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
|
||
|
void hsBounds3::Union(const hsPoint3 *p)
|
||
|
{
|
||
|
if(fType == kBoundsNormal) // Add this point if bounds is normal
|
||
|
{
|
||
|
for (int i = 0; i < 3; i++)
|
||
|
{
|
||
|
if ((*p)[i] > fMaxs[i])
|
||
|
fMaxs[i] =(*p)[i];
|
||
|
else if ((*p)[i] < fMins[i])
|
||
|
fMins[i] =(*p)[i];
|
||
|
}
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if(fType != kBoundsFull) // Otherwise re-init unless bounds is full already
|
||
|
this->Reset(p);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBounds3::Union(const hsVector3 *v)
|
||
|
{
|
||
|
if(fType == kBoundsNormal) // Add this point if bounds is normal
|
||
|
{
|
||
|
for (int i = 0; i < 3; i++)
|
||
|
{
|
||
|
if( (*v)[i] > 0 )
|
||
|
fMaxs[i] += (*v)[i];
|
||
|
else
|
||
|
fMins[i] += (*v)[i];
|
||
|
}
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void hsBounds3::Union(const hsBounds3 *p)
|
||
|
{
|
||
|
if(fType == kBoundsNormal && p->GetType() == kBoundsNormal) // Add this point if bounds is normal
|
||
|
{
|
||
|
for (int i = 0; i < 3; i++)
|
||
|
{
|
||
|
if (p->fMaxs[i] > fMaxs[i])
|
||
|
fMaxs[i] = p->fMaxs[i];
|
||
|
if (p->fMins[i] < fMins[i])
|
||
|
fMins[i] = p->fMins[i];
|
||
|
}
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
else if(fType == kBoundsEmpty || fType == kBoundsUninitialized)
|
||
|
{
|
||
|
*this = *p;
|
||
|
}
|
||
|
// If fType is kBoundsFull don't do anything
|
||
|
}
|
||
|
|
||
|
void hsBounds3::MakeSymmetric(const hsPoint3* p)
|
||
|
{
|
||
|
if( fType != kBoundsNormal )
|
||
|
return;
|
||
|
|
||
|
hsScalar delMax = 0;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar delUp;
|
||
|
|
||
|
delUp = fMaxs[i] - (*p)[i];
|
||
|
delMax = hsMaximum(delMax, delUp);
|
||
|
delUp = (*p)[i] - fMins[i];
|
||
|
delMax = hsMaximum(delMax, delUp);
|
||
|
}
|
||
|
const hsScalar sqrtTwo = 1.41421f;
|
||
|
delMax *= sqrtTwo;
|
||
|
hsAssert((delMax > -1.e6f)&&(delMax < 1.e6f), "MakeSymmetric going out to sea");
|
||
|
fCenter = *p;
|
||
|
fMaxs.Set(delMax, delMax, delMax);
|
||
|
fMaxs += fCenter;
|
||
|
fMins.Set(-delMax, -delMax, -delMax);
|
||
|
fMins += fCenter;
|
||
|
fBounds3Flags |= kCenterValid;
|
||
|
}
|
||
|
|
||
|
void hsBounds3::InscribeSphere()
|
||
|
{
|
||
|
if( fType != kBoundsNormal )
|
||
|
return;
|
||
|
|
||
|
const hsScalar ooSix = hsScalarInvert(2.f * 3.f);
|
||
|
hsScalar a = GetMaxDim() * ooSix;
|
||
|
hsPoint3 p = GetCenter();
|
||
|
p.fX += a;
|
||
|
p.fY += a;
|
||
|
p.fZ += a;
|
||
|
fMaxs = p;
|
||
|
a *= -2.f;
|
||
|
p.fX += a;
|
||
|
p.fY += a;
|
||
|
p.fZ += a;
|
||
|
fMins = p;
|
||
|
|
||
|
// Center still valid, type still normal
|
||
|
}
|
||
|
|
||
|
// neg, pos, zero == disjoint, I contain other, overlap
|
||
|
Int32 hsBounds3::TestBound(const hsBounds3& other) const
|
||
|
{
|
||
|
Int32 retVal = 1;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if( GetMins()[i] > other.GetMaxs()[i] )
|
||
|
return -1;
|
||
|
if( GetMaxs()[i] < other.GetMins()[i] )
|
||
|
return -1;
|
||
|
|
||
|
if( GetMaxs()[i] < other.GetMaxs()[i] )
|
||
|
retVal = 0;
|
||
|
if( GetMins()[i] > other.GetMins()[i] )
|
||
|
retVal = 0;
|
||
|
}
|
||
|
return retVal;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3::IsInside(const hsPoint3* pos) const
|
||
|
{
|
||
|
hsAssert(fType != kBoundsUninitialized, "Invalid bounds type for hsBounds3::IsInside() ");
|
||
|
if(fType == kBoundsEmpty)
|
||
|
return false;
|
||
|
if(fType == kBoundsFull)
|
||
|
return true;
|
||
|
return !(pos->fX>fMaxs.fX || pos->fY>fMaxs.fY || pos->fZ>fMaxs.fZ ||
|
||
|
pos->fX<fMins.fX || pos->fY<fMins.fY || pos->fZ<fMins.fZ);
|
||
|
}
|
||
|
|
||
|
#if 0 // MESH_GEN_DEFER
|
||
|
void hsBounds3::MakeTriMeshSphere(hsGTriMesh* tMesh, hsPoint3* cornersIn) const
|
||
|
{
|
||
|
hsPoint3 center = (*GetMaxs() + *GetMins()) * 0.5f;
|
||
|
hsScalar radius = GetMaxDim() * 0.5f;
|
||
|
|
||
|
const int nLong = 9;
|
||
|
const int nLati = 5;
|
||
|
|
||
|
int nPts = nLong * nLati + 3;
|
||
|
int nFaces = nLong * 2 + nLong * (nLati - 1) * 2; // == nLong * nLati * 2
|
||
|
tMesh->AllocatePointers(nFaces /*faces*/, nPts /*pts*/, 0 /*uvs*/, 0 /*colors*/);
|
||
|
tMesh->SetNumTriVertex(nPts);
|
||
|
|
||
|
int iCenter = nPts - 3;
|
||
|
int iNorthPole = nPts - 2;
|
||
|
int iSouthPole = nPts - 1;
|
||
|
hsPoint3 pt;
|
||
|
pt = center;
|
||
|
tMesh->SetPoint(iCenter, &pt);
|
||
|
pt.fZ += radius;
|
||
|
tMesh->SetPoint(iNorthPole, &pt);
|
||
|
pt.fZ -= 2.f * radius;
|
||
|
tMesh->SetPoint(iSouthPole, &pt);
|
||
|
|
||
|
int i, j;
|
||
|
for( i = 0; i < nLong; i++ )
|
||
|
{
|
||
|
for( j = 0; j < nLati; j++ )
|
||
|
{
|
||
|
hsScalar theta = (hsScalar(i) / nLong) * 2.f * hsScalarPI;
|
||
|
hsScalar cosTheta = hsCosine(theta);
|
||
|
hsScalar sinTheta = hsSine(theta);
|
||
|
|
||
|
hsScalar phi = (hsScalar(j+1) / (nLati+1)) * hsScalarPI;
|
||
|
hsScalar cosPhi = hsCosine(phi);
|
||
|
hsScalar sinPhi = hsSine(phi);
|
||
|
|
||
|
pt.fX = center.fX + radius * sinPhi * cosTheta;
|
||
|
pt.fY = center.fY + radius * sinPhi * sinTheta;
|
||
|
pt.fZ = center.fZ + radius * cosPhi;
|
||
|
tMesh->SetPoint(j + i * nLati, &pt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hsTriangle3* tri;
|
||
|
int nTris = 0;
|
||
|
int iNext;
|
||
|
for( i = 0; i < nLong; i++ )
|
||
|
{
|
||
|
if( (iNext = i + 1) >= nLong )
|
||
|
iNext = 0;
|
||
|
|
||
|
tri = tMesh->GetTriFromPool(nTris);
|
||
|
tri->Zero();
|
||
|
tri->fFlags |= hsTriangle3::kTwoSided;
|
||
|
tMesh->SetTriangle(nTris++, tri);
|
||
|
tri->SetQuickMeshVerts(i * nLati, iNext * nLati, iNorthPole);
|
||
|
|
||
|
tri = tMesh->GetTriFromPool(i);
|
||
|
tri->Zero();
|
||
|
tri->fFlags |= hsTriangle3::kTwoSided;
|
||
|
tMesh->SetTriangle(nTris++, tri);
|
||
|
tri->SetQuickMeshVerts(nLati-1 + iNext * nLati, nLati-1 + i * nLati, iSouthPole);
|
||
|
|
||
|
int jNext;
|
||
|
for( j = 0; j < nLati-1; j++ )
|
||
|
{
|
||
|
jNext = j + 1;
|
||
|
|
||
|
tri = tMesh->GetTriFromPool(nTris);
|
||
|
tri->Zero();
|
||
|
tri->fFlags |= hsTriangle3::kTwoSided;
|
||
|
tMesh->SetTriangle(nTris++, tri);
|
||
|
tri->SetQuickMeshVerts(j + i * nLati, j + iNext * nLati, jNext + i * nLati);
|
||
|
|
||
|
tri = tMesh->GetTriFromPool(nTris);
|
||
|
tri->Zero();
|
||
|
tri->fFlags |= hsTriangle3::kTwoSided;
|
||
|
tMesh->SetTriangle(nTris++, tri);
|
||
|
tri->SetQuickMeshVerts(jNext + iNext * nLati, jNext + i * nLati, j + iNext * nLati);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Allocate and create mesh from bounding box
|
||
|
//
|
||
|
void hsBounds3::MakeTriMesh(hsGTriMesh* tMesh, UInt32 triFlags, hsPoint3* cornersIn) const
|
||
|
{
|
||
|
hsAssert(cornersIn || fType == kBoundsNormal,
|
||
|
"Invalid bounds type for hsBounds3::MakeTriMesh ");
|
||
|
|
||
|
const int maxNew= 12;
|
||
|
// Setup tMesh
|
||
|
tMesh->AllocatePointers(maxNew /*faces*/, 8 /*pts*/, 0 /*uvs*/, 0 /*colors*/);
|
||
|
tMesh->SetNumTriVertex(8);
|
||
|
int i;
|
||
|
hsPoint3 corners[8];
|
||
|
// Set Points
|
||
|
if( !cornersIn )
|
||
|
{
|
||
|
GetCorners(corners);
|
||
|
cornersIn = corners;
|
||
|
}
|
||
|
for(i=0; i<8; i++)
|
||
|
{
|
||
|
tMesh->SetPoint(i, &cornersIn[i]);
|
||
|
}
|
||
|
tMesh->GetVertexPool()->SetCount(8);
|
||
|
|
||
|
// Set faces
|
||
|
hsTriangle3 *tri;
|
||
|
int triNum=0;
|
||
|
|
||
|
|
||
|
static int verts[maxNew * 3] = {
|
||
|
/* -Y */ 6,2,3,
|
||
|
/* -Y */ 6,3,7,
|
||
|
/* Y */ 5,1,0,
|
||
|
/* Y */ 5,0,4,
|
||
|
/* -X */ 7,3,1,
|
||
|
/* -X */ 7,1,5,
|
||
|
/* X */ 4,0,2,
|
||
|
/* X */ 4,2,6,
|
||
|
/* Z */ 3,0,1,
|
||
|
/* Z */ 3,2,0,
|
||
|
/* -Z */ 7,4,6,
|
||
|
/* -Z */ 7,5,4
|
||
|
};
|
||
|
int v=0;
|
||
|
for (;triNum < maxNew;triNum++)
|
||
|
{
|
||
|
tri = tMesh->GetTriFromPool(triNum);
|
||
|
tri->Zero();
|
||
|
tri->fFlags |= triFlags;
|
||
|
tMesh->SetTriangle(triNum, tri);
|
||
|
tri->SetQuickMeshVerts(verts[v + 0],verts[v + 1],verts[v + 2]);
|
||
|
v += 3;
|
||
|
}
|
||
|
tMesh->SetTrianglePointers();
|
||
|
}
|
||
|
#endif // MESH_GEN_DEFER
|
||
|
|
||
|
void hsBounds3::TestPlane(const hsPlane3 *p, hsPoint2 &depth) const
|
||
|
{
|
||
|
TestPlane(p->fN, depth);
|
||
|
}
|
||
|
void hsBounds3::TestPlane(const hsVector3 &n, hsPoint2 &depth) const
|
||
|
{
|
||
|
hsAssert(fType == kBoundsNormal, "TestPlane only valid for kBoundsNormal filled bounds");
|
||
|
|
||
|
hsScalar dmax = fMins.InnerProduct(n);
|
||
|
hsScalar dmin = dmax;
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar dd;
|
||
|
dd = fMaxs[i] - fMins[i];
|
||
|
dd *= n[i];
|
||
|
|
||
|
if( dd < 0 )
|
||
|
dmin += dd;
|
||
|
else
|
||
|
dmax += dd;
|
||
|
}
|
||
|
|
||
|
depth.fX = dmin;
|
||
|
depth.fY = dmax;
|
||
|
}
|
||
|
|
||
|
hsScalar hsBounds3::ClosestPointToLine(const hsPoint3 *p, const hsPoint3 *v0, const hsPoint3 *v1, hsPoint3 *out)
|
||
|
{
|
||
|
hsVector3 del(v1, v0);
|
||
|
hsScalar magSq = del.MagnitudeSquared();
|
||
|
hsScalar t = 0.f;
|
||
|
if( magSq < hsBounds::kRealSmall )
|
||
|
{
|
||
|
*out = *v0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
t = del.InnerProduct(hsVector3(p, v0)) * hsScalarInvert(magSq);
|
||
|
if( t >= hsScalar1 )
|
||
|
*out = *v1;
|
||
|
else if( t <= 0 )
|
||
|
*out = *v0;
|
||
|
else
|
||
|
*out = *v0 + del * t;
|
||
|
}
|
||
|
return t;
|
||
|
}
|
||
|
|
||
|
hsScalar hsBounds3::ClosestPointToInfiniteLine(const hsPoint3* p, const hsVector3* v, hsPoint3* out)
|
||
|
{
|
||
|
hsScalar magSq = v->MagnitudeSquared();
|
||
|
hsScalar t = 0.f;
|
||
|
hsPoint3 origin(0,0,0);
|
||
|
if( magSq < hsBounds::kRealSmall )
|
||
|
{
|
||
|
*out = origin;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
t = v->InnerProduct(hsVector3(*p)) * hsScalarInvert(magSq);
|
||
|
*out = hsPoint3(*v * t);
|
||
|
}
|
||
|
return t;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3::ClosestPoint(const hsPoint3& p, hsPoint3& inner, hsPoint3& outer) const
|
||
|
{
|
||
|
// Look for axis intervals p is within
|
||
|
int nSect = 0;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if( p[i] < fMins[i] )
|
||
|
{
|
||
|
inner[i] = fMins[i];
|
||
|
outer[i] = fMaxs[i];
|
||
|
}
|
||
|
else if( p[i] > fMaxs[i] )
|
||
|
{
|
||
|
inner[i] = fMaxs[i];
|
||
|
outer[i] = fMins[i];
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
inner[i] = p[i];
|
||
|
outer[i] = (p[i] - fMins[i] > fMaxs[i] - p[i]) ? fMins[i] : fMaxs[i];
|
||
|
nSect++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nSect == 3;
|
||
|
}
|
||
|
|
||
|
void hsBounds3::Read(hsStream *stream)
|
||
|
{
|
||
|
hsBounds::Read(stream);
|
||
|
fMins.Read(stream);
|
||
|
fMaxs.Read(stream);
|
||
|
fBounds3Flags = 0;
|
||
|
}
|
||
|
|
||
|
void hsBounds3::Write(hsStream *stream)
|
||
|
{
|
||
|
hsBounds::Write(stream);
|
||
|
fMins.Write(stream);
|
||
|
fMaxs.Write(stream);
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////
|
||
|
//////////////////////////////////////////////////
|
||
|
// Plane Bounds util class
|
||
|
//////////////////////////////////////////////////
|
||
|
|
||
|
hsPoint3 hsBoundsOriented::GetCenter() const
|
||
|
{
|
||
|
hsAssert(fCenterValid==true, "Unset center for hsBoundsOriented::GetCenter()");
|
||
|
return fCenter;
|
||
|
}
|
||
|
|
||
|
void hsBoundsOriented::TestPlane(const hsVector3 &n, hsPoint2 &depth) const
|
||
|
{
|
||
|
hsAssert(false, "TestPlane not a valid operation for hsBounsOriented");
|
||
|
}
|
||
|
//
|
||
|
// Return true if inside all the planes
|
||
|
//
|
||
|
hsBool hsBoundsOriented::IsInside(const hsPoint3* pos) const
|
||
|
{
|
||
|
hsAssert(fType == kBoundsNormal, "Invalid bounds type for hsBounds3::IsInside() ");
|
||
|
if(fType == kBoundsEmpty)
|
||
|
return false;
|
||
|
if(fType == kBoundsFull)
|
||
|
return true;
|
||
|
int i;
|
||
|
for( i = 0; i < fNumPlanes; i++ )
|
||
|
{
|
||
|
hsScalar dis = fPlanes[i].fN.InnerProduct(pos);
|
||
|
dis += fPlanes[i].fD;
|
||
|
if( dis > 0.f )
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void hsBoundsOriented::SetNumberPlanes(UInt32 n)
|
||
|
{
|
||
|
delete [] fPlanes;
|
||
|
fPlanes = TRACKED_NEW hsPlane3[fNumPlanes = n];
|
||
|
}
|
||
|
|
||
|
void hsBoundsOriented::SetPlane(UInt32 i, hsPlane3 *pln)
|
||
|
{
|
||
|
fType = kBoundsNormal;
|
||
|
if( i >= fNumPlanes )
|
||
|
{
|
||
|
hsPlane3 *newPlanes = TRACKED_NEW hsPlane3[i+1];
|
||
|
if( fPlanes )
|
||
|
{
|
||
|
int k;
|
||
|
for( k = 0; k < fNumPlanes; k++ )
|
||
|
*newPlanes++ = *fPlanes++;
|
||
|
delete [] fPlanes;
|
||
|
}
|
||
|
fPlanes = newPlanes;
|
||
|
fNumPlanes = i+1;
|
||
|
}
|
||
|
fPlanes[i] = *pln;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Make mesh from bounds3. Make boundsOriented from mesh tris.
|
||
|
//
|
||
|
void hsBoundsOriented::Reset(const hsBounds3* bounds)
|
||
|
{
|
||
|
#if 0 // MESH_GEN_DEFER
|
||
|
hsGTriMesh tMesh;
|
||
|
bounds->MakeTriMesh(&tMesh, 0 /* triFlags */);
|
||
|
Reset(&tMesh);
|
||
|
#endif // MESH_GEN_DEFER
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
//
|
||
|
// Make mesh from bounds3. Make boundsOriented from mesh tris.
|
||
|
//
|
||
|
void hsBoundsOriented::Union(const hsBounds3 *b)
|
||
|
{
|
||
|
#if 0 // MESH_GEN_DEFER
|
||
|
hsGTriMesh tMesh;
|
||
|
bounds->MakeTriMesh(&tMesh);
|
||
|
int i;
|
||
|
hsTriangle3 tri;
|
||
|
for (i=0; i<tMesh.GetNumTriangles(); i++)
|
||
|
{
|
||
|
tMesh.GetTriangle(i, &tri);
|
||
|
Union(&tri);
|
||
|
}
|
||
|
#endif // MESH_GEN_DEFER
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
void hsBoundsOriented::Reset(hsGTriMesh *tMesh)
|
||
|
{
|
||
|
#if 0 // MESH_GEN_DEFER
|
||
|
const float OBJCVT_ABOUT_ZERO = 1.0e-4f;
|
||
|
const float OBJCVT_ABOUT_ONE = 1.0f - OBJCVT_ABOUT_ZERO;
|
||
|
|
||
|
hsPlane3 *planes = TRACKED_NEW hsPlane3[tMesh->GetNumTriangles()];
|
||
|
UInt32 nPlanes = 0;
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < tMesh->GetNumTriangles(); i++ )
|
||
|
{
|
||
|
hsTriangle3 *tri;
|
||
|
tri = tMesh->GetTriangle(i);
|
||
|
hsPlane3 pln;
|
||
|
tri->ComputePlane(&pln);
|
||
|
|
||
|
hsScalar norm = hsFastMath::InvSqrRoot(pln.fN.MagnitudeSquared());
|
||
|
pln.fN *= norm;
|
||
|
|
||
|
int j;
|
||
|
for( j = 0; j < nPlanes; j++ )
|
||
|
{
|
||
|
if( (pln.fN.InnerProduct(planes[j].fN)> OBJCVT_ABOUT_ONE)
|
||
|
&&((pln.fD/planes[j].fD) >= 1.0-OBJCVT_ABOUT_ZERO)
|
||
|
&&((pln.fD/planes[j].fD) <= 1.0+OBJCVT_ABOUT_ZERO) )
|
||
|
break;
|
||
|
}
|
||
|
if( j == nPlanes )
|
||
|
planes[nPlanes++] = pln;
|
||
|
}
|
||
|
|
||
|
SetNumberPlanes(nPlanes);
|
||
|
for( i = 0; i < nPlanes; i++ )
|
||
|
SetPlane(i, planes+i);
|
||
|
|
||
|
delete [] planes;
|
||
|
|
||
|
// Compute center
|
||
|
hsPoint3 centroid(0,0,0);
|
||
|
for(i=0; i<tMesh->GetNumPoints(); i++)
|
||
|
{
|
||
|
centroid = centroid + *tMesh->GetPoint(i);
|
||
|
}
|
||
|
centroid = centroid / (hsScalar)tMesh->GetNumPoints();
|
||
|
SetCenter(¢roid);
|
||
|
#endif // MESH_GEN_DEFER
|
||
|
}
|
||
|
|
||
|
|
||
|
void hsBoundsOriented::Write(hsStream *stream)
|
||
|
{
|
||
|
hsBounds::Write(stream);
|
||
|
fCenter.Write(stream);
|
||
|
stream->WriteSwap32(fCenterValid);
|
||
|
stream->WriteSwap32(fNumPlanes);
|
||
|
int i;
|
||
|
for( i = 0; i < fNumPlanes; i++ )
|
||
|
{
|
||
|
fPlanes[i].Write(stream);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBoundsOriented::Read(hsStream *stream)
|
||
|
{
|
||
|
hsBounds::Read(stream);
|
||
|
fCenter.Read(stream);
|
||
|
fCenterValid = (hsBool)stream->ReadSwap32();
|
||
|
fNumPlanes = stream->ReadSwap32();
|
||
|
if (fPlanes)
|
||
|
delete [] fPlanes;
|
||
|
fPlanes = TRACKED_NEW hsPlane3[fNumPlanes];
|
||
|
int i;
|
||
|
for( i = 0; i < fNumPlanes; i++ )
|
||
|
{
|
||
|
fPlanes[i].Read(stream);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
hsBounds3Ext::hsBounds3Ext(const hsBounds3 &b)
|
||
|
{
|
||
|
Reset(&b);
|
||
|
}
|
||
|
hsBounds3Ext &hsBounds3Ext::operator=(const hsBounds3 &b)
|
||
|
{
|
||
|
Reset(&b);
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::IMakeMinsMaxs()
|
||
|
{
|
||
|
hsAssert(!(fExtFlags & kAxisAligned), "Axis aligned box defined by min and max");
|
||
|
fMins = fMaxs = fCorner;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if(!IAxisIsZero(i) )
|
||
|
{
|
||
|
int j;
|
||
|
for( j = 0; j < 3; j++ )
|
||
|
{
|
||
|
if( fAxes[i][j] < 0 )
|
||
|
fMins[j] += fAxes[i][j];
|
||
|
else
|
||
|
fMaxs[j] += fAxes[i][j];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::IMakeDists() const
|
||
|
{
|
||
|
hsAssert(!(fExtFlags & kAxisAligned), "Dists only useful for transformed BB");
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
fDists[i].fX = fCorner.InnerProduct(fAxes[i]);
|
||
|
if( !IAxisIsZero(i) )
|
||
|
{
|
||
|
fDists[i].fY = fDists[i].fX + fAxes[i].InnerProduct(fAxes[i]);
|
||
|
if( fDists[i].fX > fDists[i].fY )
|
||
|
{
|
||
|
hsScalar t = fDists[i].fX;
|
||
|
fDists[i].fX = fDists[i].fY;
|
||
|
fDists[i].fY = t;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
fDists[i].fY = fDists[i].fX;
|
||
|
}
|
||
|
fExtFlags |= kDistsSet;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::IMakeSphere() const
|
||
|
{
|
||
|
if(!(fBounds3Flags & kCenterValid) )
|
||
|
ICalcCenter();
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
if( fBounds3Flags & kIsSphere )
|
||
|
{
|
||
|
fRadius = fMaxs[0] - fMins[0];
|
||
|
int i;
|
||
|
for( i = 1; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar dist = fMaxs[i] - fMins[i];
|
||
|
if( dist < fRadius )
|
||
|
fRadius = dist;
|
||
|
}
|
||
|
fRadius *= 0.5f;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fRadius = hsSquareRoot(hsVector3(&fMaxs, &fCenter).MagnitudeSquared());
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if( fBounds3Flags & kIsSphere )
|
||
|
{
|
||
|
hsScalar minMagSq = fAxes[0].MagnitudeSquared();
|
||
|
hsScalar magSq = fAxes[1].MagnitudeSquared();
|
||
|
if( magSq < minMagSq )
|
||
|
magSq = minMagSq;
|
||
|
magSq = fAxes[2].MagnitudeSquared();
|
||
|
if( magSq < minMagSq )
|
||
|
magSq = minMagSq;
|
||
|
fRadius = hsSquareRoot(magSq);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hsVector3 accum;
|
||
|
accum.Set(0,0,0);
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if( !IAxisIsZero(i) )
|
||
|
accum += fAxes[i];
|
||
|
}
|
||
|
fRadius = hsSquareRoot((accum * 0.5f).MagnitudeSquared());
|
||
|
}
|
||
|
}
|
||
|
fExtFlags |= kSphereSet;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Reset(const hsBounds3 *b)
|
||
|
{
|
||
|
fExtFlags = kAxisAligned;
|
||
|
hsBounds3::Reset(b);
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Reset(const hsPoint3 *p)
|
||
|
{
|
||
|
fExtFlags = kAxisAligned | kSphereSet;
|
||
|
hsBounds3::Reset(p);
|
||
|
fRadius = 0;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Reset(const hsBounds3Ext *b)
|
||
|
{
|
||
|
hsBounds3::Reset(b);
|
||
|
fExtFlags = b->fExtFlags;
|
||
|
if (!(fExtFlags & kAxisAligned))
|
||
|
{
|
||
|
fCorner = b->fCorner;
|
||
|
fAxes[0] = b->fAxes[0];
|
||
|
fAxes[1] = b->fAxes[1];
|
||
|
fAxes[2] = b->fAxes[2];
|
||
|
}
|
||
|
if (fExtFlags & kDistsSet)
|
||
|
{
|
||
|
fDists[0] = b->fDists[0];
|
||
|
fDists[1] = b->fDists[1];
|
||
|
fDists[2] = b->fDists[2];
|
||
|
}
|
||
|
if (fExtFlags & kSphereSet)
|
||
|
fRadius = b->fRadius;
|
||
|
}
|
||
|
|
||
|
|
||
|
void hsBounds3Ext::GetCorners(hsPoint3 *b) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsBounds3::GetCorners(b);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
int i;
|
||
|
for( i = 0; i < 8; i++ )
|
||
|
{
|
||
|
b[i] = fCorner;
|
||
|
if( !(i & 0x1) && !(fExtFlags & kAxisZeroZero) )b[i] += fAxes[0];
|
||
|
if( !(i & 0x2) && !(fExtFlags & kAxisOneZero) )b[i] += fAxes[1];
|
||
|
if( !(i & 0x4) && !(fExtFlags & kAxisTwoZero) )b[i] += fAxes[2];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::GetAxes(hsVector3 *fAxis0, hsVector3 *fAxis1, hsVector3 *fAxis2) const
|
||
|
{
|
||
|
if( !(fExtFlags & kAxisAligned) )
|
||
|
{
|
||
|
*fAxis0 = fAxes[0];
|
||
|
*fAxis1 = fAxes[1];
|
||
|
*fAxis2 = fAxes[2];
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fAxis0->Set(fMaxs.fX - fMins.fX, 0, 0);
|
||
|
fAxis1->Set(0, fMaxs.fY - fMins.fY, 0);
|
||
|
fAxis2->Set(0, 0, fMaxs.fZ - fMins.fZ);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Reset(int n, const hsPoint3 *p)
|
||
|
{
|
||
|
fExtFlags = kAxisAligned;
|
||
|
hsBounds3::Reset(n, p);
|
||
|
}
|
||
|
|
||
|
// mf horse - could union in a point preserving axes...
|
||
|
void hsBounds3Ext::Union(const hsPoint3 *p)
|
||
|
{
|
||
|
fExtFlags = kAxisAligned;
|
||
|
hsBounds3::Union(p);
|
||
|
}
|
||
|
void hsBounds3Ext::Union(const hsVector3 *v)
|
||
|
{
|
||
|
#if 0 // smarter union
|
||
|
fExtFlags = kAxisAligned;
|
||
|
hsBounds3::Union(v);
|
||
|
#else // smarter union
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsBounds3::Union(v);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar dot = fAxes[i].InnerProduct(v);
|
||
|
dot /= fAxes[i].MagnitudeSquared();
|
||
|
if( dot > 0 )
|
||
|
{
|
||
|
fAxes[i] += dot * fAxes[i];
|
||
|
fExtFlags &= ~(1 << (20+i)); // axis not zero no more
|
||
|
}
|
||
|
else if( dot < 0 )
|
||
|
{
|
||
|
hsVector3 del = dot * fAxes[i];
|
||
|
fCorner += del;
|
||
|
del = -del;
|
||
|
fAxes[i] += del;
|
||
|
fExtFlags &= ~(1 << (20+i)); // axis not zero no more
|
||
|
}
|
||
|
}
|
||
|
fExtFlags &= ~(kSphereSet | kDistsSet);
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
#endif // smarter union
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Union(const hsBounds3 *b)
|
||
|
{
|
||
|
fExtFlags = kAxisAligned;
|
||
|
hsBounds3::Union(b);
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::MakeSymmetric(const hsPoint3* p)
|
||
|
{
|
||
|
if( fType != kBoundsNormal )
|
||
|
return;
|
||
|
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
fExtFlags = kAxisAligned;
|
||
|
hsBounds3::MakeSymmetric(p);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Can do this preserving axes, but may not be worth it.
|
||
|
fExtFlags = kAxisAligned;
|
||
|
hsBounds3::MakeSymmetric(p);
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::InscribeSphere()
|
||
|
{
|
||
|
fBounds3Flags |= kIsSphere;
|
||
|
fExtFlags |= kAxisAligned;
|
||
|
IMakeSphere();
|
||
|
return;
|
||
|
#if 0
|
||
|
if( fType != kBoundsNormal )
|
||
|
return;
|
||
|
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsBounds3::InscribeSphere();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const hsScalar oneThird = hsScalarInvert(3.f);
|
||
|
// hsScalar a = GetMaxDim() * hsScalarInvert(6.f);
|
||
|
hsScalar a = GetRadius() * oneThird;
|
||
|
hsPoint3 p = GetCenter();
|
||
|
p.fX += a;
|
||
|
p.fY += a;
|
||
|
p.fZ += a;
|
||
|
fMaxs = p;
|
||
|
a *= -2.f;
|
||
|
p.fX += a;
|
||
|
p.fY += a;
|
||
|
p.fZ += a;
|
||
|
fMins = p;
|
||
|
|
||
|
// Center still valid, type still normal
|
||
|
fExtFlags = kAxisAligned;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Transform(const hsMatrix44 *m)
|
||
|
{
|
||
|
if( fType != kBoundsNormal )
|
||
|
return;
|
||
|
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
fExtFlags = 0;
|
||
|
|
||
|
fCorner = *m * fMins;
|
||
|
hsVector3 v;
|
||
|
hsScalar span;
|
||
|
span = fMaxs.fX - fMins.fX;
|
||
|
if( span < kRealSmall )
|
||
|
{
|
||
|
fExtFlags |= kAxisZeroZero;
|
||
|
span = hsScalar1;
|
||
|
}
|
||
|
v.Set(span, 0, 0);
|
||
|
fAxes[0] = *m * v;
|
||
|
span = fMaxs.fY - fMins.fY;
|
||
|
if( span < kRealSmall )
|
||
|
{
|
||
|
fExtFlags |= kAxisOneZero;
|
||
|
span = hsScalar1;
|
||
|
}
|
||
|
v.Set(0, span, 0);
|
||
|
fAxes[1] = *m * v;
|
||
|
span = fMaxs.fZ - fMins.fZ;
|
||
|
if( span < kRealSmall )
|
||
|
{
|
||
|
fExtFlags |= kAxisTwoZero;
|
||
|
span = hsScalar1;
|
||
|
}
|
||
|
v.Set(0, 0, span);
|
||
|
fAxes[2] = *m * v;
|
||
|
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
#if 0 // IDENT
|
||
|
if( m->fFlags & hsMatrix44::kIsIdent )
|
||
|
return;
|
||
|
#endif // IDENT
|
||
|
|
||
|
fCorner = *m * fCorner;
|
||
|
fAxes[0] = *m * fAxes[0];
|
||
|
fAxes[1] = *m * fAxes[1];
|
||
|
fAxes[2] = *m * fAxes[2];
|
||
|
|
||
|
fExtFlags &= kAxisZeroZero|kAxisOneZero|kAxisTwoZero;
|
||
|
}
|
||
|
IMakeMinsMaxs();
|
||
|
fBounds3Flags &= ~kCenterValid;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Translate(const hsVector3 &v)
|
||
|
{
|
||
|
if( fType != kBoundsNormal )
|
||
|
return;
|
||
|
|
||
|
fMins += v;
|
||
|
fMaxs += v;
|
||
|
if( fBounds3Flags & kCenterValid )
|
||
|
fCenter += v;
|
||
|
if( !(fExtFlags & kAxisAligned) )
|
||
|
{
|
||
|
fCorner += v;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar d;
|
||
|
d = fAxes[i].InnerProduct(v);
|
||
|
fDists[i].fX += d;
|
||
|
fDists[i].fY += d;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::IsInside(const hsPoint3 *p) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
return hsBounds3::IsInside(p);
|
||
|
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar diss = p->InnerProduct(fAxes[i]);
|
||
|
if( (diss < fDists[i].fX)
|
||
|
||(diss > fDists[i].fY) )
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
|
||
|
}
|
||
|
|
||
|
// neg, pos, zero == disjoint, I contain other, overlap
|
||
|
Int32 hsBounds3Ext::TestBound(const hsBounds3Ext& other) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
return hsBounds3::TestBound(other);
|
||
|
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
|
||
|
Int32 retVal = 1;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsPoint2 depth;
|
||
|
other.TestPlane(fAxes[i], depth);
|
||
|
|
||
|
if( fDists[i].fX > depth.fY )
|
||
|
return -1;
|
||
|
if( fDists[i].fY < depth.fX )
|
||
|
return -1;
|
||
|
|
||
|
if( fDists[i].fY < depth.fY )
|
||
|
retVal = 0;
|
||
|
if( fDists[i].fX > depth.fX )
|
||
|
retVal = 0;
|
||
|
}
|
||
|
return retVal;
|
||
|
}
|
||
|
|
||
|
|
||
|
void hsBounds3Ext::TestPlane(const hsVector3 &n, hsPoint2 &depth) const
|
||
|
{
|
||
|
hsAssert(fType == kBoundsNormal, "TestPlane only valid for kBoundsNormal filled bounds");
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsBounds3::TestPlane(n, depth);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hsScalar dmax = fCorner.InnerProduct(n);
|
||
|
hsScalar dmin = dmax;
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if( !IAxisIsZero(i) )
|
||
|
{
|
||
|
hsScalar d;
|
||
|
d = fAxes[i].InnerProduct(n);
|
||
|
if( d < 0 )
|
||
|
dmin += d;
|
||
|
else
|
||
|
dmax += d;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
depth.fX = dmin;
|
||
|
depth.fY = dmax;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::TestPlane(const hsPlane3 *p, const hsVector3 &myVel, hsPoint2 &depth) const
|
||
|
{
|
||
|
TestPlane(p->fN, myVel, depth);
|
||
|
}
|
||
|
void hsBounds3Ext::TestPlane(const hsVector3 &n, const hsVector3 &myVel, hsPoint2 &depth) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsScalar dmax = fMins.InnerProduct(n);
|
||
|
hsScalar dmin = dmax;
|
||
|
hsScalar dvel = myVel.InnerProduct(n);
|
||
|
if( dvel < 0 )
|
||
|
dmin += dvel;
|
||
|
else
|
||
|
dmax += dvel;
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar dd;
|
||
|
dd = fMaxs[i] - fMins[i];
|
||
|
dd *= n[i];
|
||
|
|
||
|
if( dd < 0 )
|
||
|
dmin += dd;
|
||
|
else
|
||
|
dmax += dd;
|
||
|
}
|
||
|
|
||
|
depth.fX = dmin;
|
||
|
depth.fY = dmax;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hsScalar dmax = fCorner.InnerProduct(n);
|
||
|
hsScalar dmin = dmax;
|
||
|
hsScalar dvel = myVel.InnerProduct(n);
|
||
|
if( dvel < 0 )
|
||
|
dmin += dvel;
|
||
|
else
|
||
|
dmax += dvel;
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if( !IAxisIsZero(i) )
|
||
|
{
|
||
|
hsScalar d;
|
||
|
d = fAxes[i].InnerProduct(n);
|
||
|
if( d < 0 )
|
||
|
dmin += d;
|
||
|
else
|
||
|
dmax += d;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
depth.fX = dmin;
|
||
|
depth.fY = dmax;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Int32 hsBounds3Ext::TestPoints(int n, const hsPoint3 *pList, const hsVector3 &ptVel) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
Int32 retVal = -1;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar effMax = fMaxs[i];
|
||
|
hsScalar effMin = fMins[i];
|
||
|
if( ptVel[i] < 0 )
|
||
|
effMax -= ptVel[i];
|
||
|
else
|
||
|
effMin -= ptVel[i];
|
||
|
|
||
|
int j;
|
||
|
const UInt32 low = 0x1, hi = 0x2;
|
||
|
UInt32 mask = low | hi;
|
||
|
for( j = 0; j < n; j++ )
|
||
|
{
|
||
|
if( pList[j][i] > effMin )
|
||
|
mask &= ~low;
|
||
|
if( pList[j][i] < effMax )
|
||
|
mask &= ~hi;
|
||
|
if( mask )
|
||
|
retVal = 0;
|
||
|
}
|
||
|
if( mask )
|
||
|
return 1;
|
||
|
}
|
||
|
return retVal;
|
||
|
}
|
||
|
else // non-axis aligned case
|
||
|
{
|
||
|
Int32 retVal = -1; // all inside
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar diff = fAxes[i].InnerProduct(ptVel);
|
||
|
hsBool someLow = false;
|
||
|
hsBool someHi = false;
|
||
|
hsBool someIn = false;
|
||
|
int j;
|
||
|
for( j = 0; j < n; j++ )
|
||
|
{
|
||
|
hsScalar d = fAxes[i].InnerProduct(pList[j]);
|
||
|
hsScalar ddiff = d + diff;
|
||
|
if( d < fDists[i].fX )
|
||
|
someLow = true;
|
||
|
else if( d > fDists[i].fY )
|
||
|
someHi = true;
|
||
|
else
|
||
|
someIn = true;
|
||
|
|
||
|
if( ddiff < fDists[i].fX )
|
||
|
someLow = true;
|
||
|
else if( ddiff > fDists[i].fY )
|
||
|
someHi = true;
|
||
|
else
|
||
|
someIn = true;
|
||
|
|
||
|
if( someIn &&(someHi || someLow) )
|
||
|
break;
|
||
|
}
|
||
|
if( someHi && !(someLow || someIn) )
|
||
|
return 1;
|
||
|
if( someLow && !(someHi || someIn) )
|
||
|
return 1;
|
||
|
if( someHi || someLow )
|
||
|
retVal = 0;
|
||
|
}
|
||
|
return retVal;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Int32 hsBounds3Ext::TestPoints(int n, const hsPoint3 *pList) const
|
||
|
{
|
||
|
hsBool someIn = false;
|
||
|
hsBool someOut = false;
|
||
|
int i;
|
||
|
for( i = 0; i < n; i++ )
|
||
|
{
|
||
|
if( IsInside(pList+i) )
|
||
|
someIn = true;
|
||
|
else
|
||
|
someOut = true;
|
||
|
if( someIn && someOut )
|
||
|
return 0;
|
||
|
}
|
||
|
if( someIn )
|
||
|
return -1;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ClosestPoint(const hsPoint3& p, hsPoint3& inner, hsPoint3& outer) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
return hsBounds3::ClosestPoint(p, inner, outer);
|
||
|
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
|
||
|
int nSect = 0;
|
||
|
inner = outer = fCorner;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar dist = fAxes[i].InnerProduct(p);
|
||
|
if( dist < fDists[i].fX )
|
||
|
{
|
||
|
outer += fAxes[i];
|
||
|
}
|
||
|
else if( dist > fDists[i].fY )
|
||
|
{
|
||
|
inner += fAxes[i];
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hsScalar t = (dist - fDists[i].fX) / (fDists[i].fY - fDists[i].fX);
|
||
|
inner += t * fAxes[i];
|
||
|
if( t > 0.5f )
|
||
|
outer += fAxes[i];
|
||
|
|
||
|
nSect++;
|
||
|
}
|
||
|
}
|
||
|
return nSect == 3;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectBB(const hsBounds3Ext &other, const hsVector3 &myVel) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
if( other.fExtFlags & kAxisAligned )
|
||
|
return ISectABB(other, myVel);
|
||
|
|
||
|
return other.ISectBB(*this, -myVel);
|
||
|
}
|
||
|
hsAssert(!(fExtFlags & kAxisAligned), "Other can be axis-aligned, but not me!");
|
||
|
hsPoint2 depth;
|
||
|
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
if( !(other.fExtFlags & (kDistsSet|kAxisAligned)) )
|
||
|
other.IMakeDists();
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
other.TestPlane(fAxes[i], -myVel, depth);
|
||
|
if( (depth.fX > fDists[i].fY)
|
||
|
||(depth.fY < fDists[i].fX) )
|
||
|
return false;
|
||
|
|
||
|
if( other.fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsScalar myMin = fMins[i];
|
||
|
hsScalar myMax = fMaxs[i];
|
||
|
if( myVel[i] < 0 )
|
||
|
myMin += myVel[i];
|
||
|
else
|
||
|
myMax += myVel[i];
|
||
|
if( (other.fMins[i] > myMax)
|
||
|
||(other.fMaxs[i] < myMin) )
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
TestPlane(other.fAxes[i], myVel, depth);
|
||
|
if( (depth.fX > other.fDists[i].fY)
|
||
|
||(depth.fY < other.fDists[i].fX) )
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// still leaves the 3 axes of origAxis.Cross(myVel)
|
||
|
hsVector3 ax = fAxes[i] % myVel;
|
||
|
hsScalar dmax = fCorner.InnerProduct(ax);
|
||
|
hsScalar dmin = dmax;
|
||
|
int j = i+1;
|
||
|
if( 3 == j )j = 0;
|
||
|
hsScalar d;
|
||
|
d = fAxes[j].InnerProduct(ax);
|
||
|
if( d < 0 )
|
||
|
dmin += d;
|
||
|
else
|
||
|
dmax += d;
|
||
|
j = ( j == 2 ? 0 : j+1 );
|
||
|
d = fAxes[j].InnerProduct(ax);
|
||
|
if( d < 0 )
|
||
|
dmin += d;
|
||
|
else
|
||
|
dmax += d;
|
||
|
other.TestPlane(ax, depth);
|
||
|
if( (depth.fX > dmax)
|
||
|
||(depth.fY < dmin) )
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
|
||
|
static hsBool ISectInterval(const hsPoint2& other, const hsPoint2& mine)
|
||
|
{
|
||
|
if( other.fY - mine.fX <= 0 )
|
||
|
return false;
|
||
|
if( mine.fY - other.fX <= 0 )
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
static hsBool ITestDepth(const hsPoint2& other, const hsPoint2& mine,
|
||
|
const hsVector3& inAx,
|
||
|
hsVector3 &outAx, hsScalar& depth)
|
||
|
{
|
||
|
depth = 0;
|
||
|
hsScalar d0, d1;
|
||
|
d0 = other.fY - mine.fX;
|
||
|
if( d0 <= 0 )
|
||
|
return false;
|
||
|
d1 = mine.fY - other.fX;
|
||
|
if( d1 <= 0 )
|
||
|
return false;
|
||
|
|
||
|
// if one interval is proper subset of other, skip
|
||
|
if( (mine.fX < other.fX)^(mine.fY < other.fY) )
|
||
|
{
|
||
|
depth = 0;
|
||
|
return true;
|
||
|
}
|
||
|
if( d0 < d1 )
|
||
|
{
|
||
|
outAx = inAx;
|
||
|
depth = d0;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
outAx = -inAx;
|
||
|
depth = d1;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
Int32 hsBounds3Ext::IClosestISect(const hsBounds3Ext& other, const hsVector3& myVel,
|
||
|
hsScalar* tClose, hsScalar* tImpact) const
|
||
|
{
|
||
|
// Should assert both have their spheres set.
|
||
|
|
||
|
hsVector3 meToOt(&other.GetCenter(), &GetCenter());
|
||
|
|
||
|
// cTerm = (myCenter - otCenter)^2 - (myRad + otRad)^2
|
||
|
hsScalar cTerm;
|
||
|
|
||
|
cTerm = GetRadius() + other.GetRadius();
|
||
|
cTerm *= -cTerm;
|
||
|
|
||
|
hsScalar meToOtLen = meToOt.MagnitudeSquared();
|
||
|
cTerm += meToOtLen;
|
||
|
if( cTerm <= 0 )
|
||
|
{
|
||
|
*tClose = *tImpact = 0;
|
||
|
return -1; // started off in contact
|
||
|
}
|
||
|
|
||
|
hsScalar ooATerm = myVel.InnerProduct(myVel);
|
||
|
if( ooATerm < hsBounds::kRealSmall )
|
||
|
{
|
||
|
*tClose = *tImpact = 0;
|
||
|
return 0;
|
||
|
}
|
||
|
ooATerm = hsScalarInvert(ooATerm);
|
||
|
|
||
|
hsScalar bTerm = myVel.InnerProduct(meToOt);
|
||
|
bTerm *= ooATerm;
|
||
|
hsScalar bSqTerm = bTerm * bTerm;
|
||
|
// bTerm is t for closest point to line
|
||
|
|
||
|
hsScalar det = bSqTerm - ooATerm * cTerm;
|
||
|
if( det < 0 )
|
||
|
{
|
||
|
*tClose = *tImpact = bTerm;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
det = hsSquareRoot(det);
|
||
|
*tClose = bTerm;
|
||
|
*tImpact = bTerm - det;
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Unalign()
|
||
|
{
|
||
|
fExtFlags = 0;
|
||
|
|
||
|
fCorner = fMins;
|
||
|
hsVector3 v;
|
||
|
hsScalar span;
|
||
|
span = fMaxs.fX - fMins.fX;
|
||
|
if( span < kRealSmall )
|
||
|
{
|
||
|
fExtFlags |= kAxisZeroZero;
|
||
|
span = hsScalar1;
|
||
|
}
|
||
|
fAxes[0].Set(span, 0, 0);
|
||
|
span = fMaxs.fY - fMins.fY;
|
||
|
if( span < kRealSmall )
|
||
|
{
|
||
|
fExtFlags |= kAxisOneZero;
|
||
|
span = hsScalar1;
|
||
|
}
|
||
|
fAxes[1].Set(0, span, 0);
|
||
|
span = fMaxs.fZ - fMins.fZ;
|
||
|
if( span < kRealSmall )
|
||
|
{
|
||
|
fExtFlags |= kAxisTwoZero;
|
||
|
span = hsScalar1;
|
||
|
}
|
||
|
fAxes[2].Set(0, 0, span);
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectBB(const hsBounds3Ext &other, const hsVector3 &myVel, hsHitInfoExt *hit) const
|
||
|
{
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsBounds3Ext meUnalign(*this);
|
||
|
meUnalign.Unalign();
|
||
|
return meUnalign.ISectBB(other, myVel, hit);
|
||
|
}
|
||
|
hsAssert(!(fExtFlags & kAxisAligned), "Other can be axis-aligned, but not me!");
|
||
|
hsPoint2 depth;
|
||
|
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
if( !(other.fExtFlags & (kDistsSet|kAxisAligned)) )
|
||
|
other.IMakeDists();
|
||
|
|
||
|
const hsScalar kRealBig = 1.e30f;
|
||
|
hsScalar tstDepths[9];
|
||
|
hsVector3 tstAxes[9];
|
||
|
hsScalar totDepth = 0;
|
||
|
int nDeep = 0;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
const hsScalar kFavorConstant = 0.01f; // smaller is favored
|
||
|
|
||
|
other.TestPlane(fAxes[i], -myVel, depth);
|
||
|
|
||
|
if( !ITestDepth(depth, fDists[i], fAxes[i], tstAxes[i], tstDepths[i]) )
|
||
|
return false;
|
||
|
|
||
|
other.TestPlane(fAxes[i], depth);
|
||
|
if( !ISectInterval(depth, fDists[i]) )
|
||
|
tstDepths[i] *= kFavorConstant;
|
||
|
|
||
|
if( tstDepths[i] > 0 )
|
||
|
{
|
||
|
totDepth += tstDepths[i];
|
||
|
nDeep++;
|
||
|
}
|
||
|
|
||
|
if( other.fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
hsPoint2 mine;
|
||
|
mine.fX = fMins[i];
|
||
|
mine.fY = fMaxs[i];
|
||
|
if( myVel[i] > 0 )mine.fY += myVel[i];
|
||
|
else mine.fX += myVel[i];
|
||
|
depth.fX = other.fMins[i];
|
||
|
depth.fY = other.fMaxs[i];
|
||
|
|
||
|
hsVector3 ax;
|
||
|
ax.Set( 0 == i ? hsScalar1 : 0,
|
||
|
1 == i ? hsScalar1 : 0,
|
||
|
2 == i ? hsScalar1 : 0);
|
||
|
|
||
|
if( !ITestDepth(depth, mine, ax, tstAxes[i+3], tstDepths[i+3]) )
|
||
|
return false;
|
||
|
|
||
|
mine.fX = fMins[i];
|
||
|
mine.fY = fMaxs[i];
|
||
|
if( !ISectInterval(depth, mine) )
|
||
|
tstDepths[i+3] *= kFavorConstant;
|
||
|
|
||
|
if( tstDepths[i+3] )
|
||
|
{
|
||
|
totDepth += tstDepths[i+3];
|
||
|
nDeep++;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
TestPlane(other.fAxes[i], myVel, depth);
|
||
|
|
||
|
if( !ITestDepth(other.fDists[i], depth, other.fAxes[i], tstAxes[i+3], tstDepths[i+3]) )
|
||
|
return false;
|
||
|
|
||
|
TestPlane(other.fAxes[i], depth);
|
||
|
|
||
|
if( !ISectInterval(other.fDists[i], depth) )
|
||
|
tstDepths[i+3] *= kFavorConstant;
|
||
|
|
||
|
if( tstDepths[i+3] )
|
||
|
{
|
||
|
totDepth += tstDepths[i+3];
|
||
|
nDeep++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
// still leaves the 3 axes of origAxis.Cross(myVel)
|
||
|
hsVector3 ax = fAxes[i] % myVel;
|
||
|
if( ax.MagnitudeSquared() > kRealSmall )
|
||
|
{
|
||
|
hsPoint2 myDepth;
|
||
|
myDepth.fX = myDepth.fY = fCorner.InnerProduct(ax);
|
||
|
hsScalar d;
|
||
|
int j = i == 2 ? 0 : i+1;
|
||
|
if( !IAxisIsZero(j) )
|
||
|
{
|
||
|
d = fAxes[j].InnerProduct(ax);
|
||
|
if( d < 0 )
|
||
|
myDepth.fX += d;
|
||
|
else
|
||
|
myDepth.fY += d;
|
||
|
}
|
||
|
j = ( j == 2 ? 0 : j+1 );
|
||
|
if( !IAxisIsZero(j) )
|
||
|
{
|
||
|
d = fAxes[j].InnerProduct(ax);
|
||
|
if( d < 0 )
|
||
|
myDepth.fX += d;
|
||
|
else
|
||
|
myDepth.fY += d;
|
||
|
}
|
||
|
other.TestPlane(ax, depth);
|
||
|
|
||
|
if( !ITestDepth(depth, myDepth, ax, tstAxes[i+6], tstDepths[i+6]) )
|
||
|
return false;
|
||
|
totDepth += tstDepths[i+6];
|
||
|
}
|
||
|
else
|
||
|
tstDepths[i+6] = 0;
|
||
|
#endif;
|
||
|
}
|
||
|
|
||
|
hsVector3 norm;
|
||
|
if( totDepth <= 0 )
|
||
|
{
|
||
|
hsScalar t, tIgnore;
|
||
|
IClosestISect(other, myVel, &tIgnore, &t);
|
||
|
if( t < 0 )
|
||
|
t = 0;
|
||
|
else if( t > 1.f )
|
||
|
t = 1.f;
|
||
|
hsPoint3 hitPt = GetCenter() + myVel * t;
|
||
|
norm.Set(&hitPt, &other.GetCenter());
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// now do a weighted average of the axes
|
||
|
hsAssert(totDepth > 0, "nobody home");
|
||
|
norm.Set(0,0,0);
|
||
|
for( i =0; i < 6; i++ )
|
||
|
{
|
||
|
if( tstDepths[i] > 0 )
|
||
|
norm += tstAxes[i] / tstDepths[i];
|
||
|
// norm += tstAxes[i] * (1.f - tstDepths[i] / totDepth);
|
||
|
}
|
||
|
}
|
||
|
hsPoint2 otherDepth;
|
||
|
norm.Normalize();
|
||
|
other.TestPlane(norm, otherDepth);
|
||
|
TestPlane(norm, myVel, depth);
|
||
|
|
||
|
hit->Set(this, &other, norm, otherDepth.fY - depth.fX);
|
||
|
|
||
|
// mf horse hack test
|
||
|
if( hit->fDepth < 0 )
|
||
|
return false;
|
||
|
hsAssert(hit->fDepth >= 0, "Negative Depth");
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectABB(const hsBounds3Ext &other, const hsVector3 &myVel) const
|
||
|
{
|
||
|
hsPoint3 effMaxs = fMaxs;
|
||
|
hsPoint3 effMins = fMins;
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar effMax = fMaxs[i];
|
||
|
hsScalar effMin = fMins[i];
|
||
|
if( myVel[i] > 0 )
|
||
|
effMax += myVel[i];
|
||
|
else
|
||
|
effMin += myVel[i];
|
||
|
|
||
|
if( (effMax < other.fMins[i])
|
||
|
||(effMin > other.fMaxs[i]) )
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectBS(const hsBounds3Ext &other, const hsVector3 &myVel) const
|
||
|
{
|
||
|
if( !(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
if( !(other.fExtFlags & kSphereSet) )
|
||
|
other.IMakeSphere();
|
||
|
|
||
|
hsPoint3 closestPt = GetCenter();
|
||
|
// we should know whether we have a useful velocity or not...
|
||
|
// having the speed cached away would get rid of several
|
||
|
// such uglies...
|
||
|
if( myVel.MagnitudeSquared() > 0 )
|
||
|
{
|
||
|
hsScalar parm = hsVector3(&other.GetCenter(), &fCenter).InnerProduct(myVel)
|
||
|
/ myVel.InnerProduct(myVel);
|
||
|
if( parm > 0 )
|
||
|
{
|
||
|
if( parm > hsScalar1 )
|
||
|
parm = hsScalar1;
|
||
|
closestPt += myVel * parm;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hsScalar combRad = fRadius + other.fRadius;
|
||
|
|
||
|
return hsVector3(&closestPt, &other.GetCenter()).MagnitudeSquared() < combRad*combRad;
|
||
|
}
|
||
|
|
||
|
#if 0 // Commenting out this which will be made redundant and/or obsolete by Havok integration
|
||
|
hsBool hsBounds3Ext::ISectTriABB(hsBounds3Tri &tri, const hsVector3 &myVel) const
|
||
|
{
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar effMax = fMaxs[i];
|
||
|
hsScalar effMin = fMins[i];
|
||
|
if( myVel[i] < 0 )
|
||
|
effMin += myVel[i];
|
||
|
else
|
||
|
effMax += myVel[i];
|
||
|
|
||
|
int j;
|
||
|
const UInt32 low = 0x1, hi = 0x2;
|
||
|
UInt32 mask = low | hi;
|
||
|
for( j = 0; j < 3; j++ )
|
||
|
{
|
||
|
if( tri.fVerts[j][i] > effMin )
|
||
|
mask &= ~low;
|
||
|
if( tri.fVerts[j][i] < effMax )
|
||
|
mask &= ~hi;
|
||
|
}
|
||
|
if( mask )
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::TriBSHitInfo(hsBounds3Tri& tri, const hsVector3& myVel, hsHitInfoExt* hit) const
|
||
|
{
|
||
|
hsPoint3 myPt = GetCenter();
|
||
|
myPt += myVel;
|
||
|
|
||
|
hsPoint3 closePt;
|
||
|
hsBool onTri = tri.ClosestTriPoint(&myPt, &closePt);
|
||
|
|
||
|
hsVector3 repel;
|
||
|
repel.Set(&myPt, &closePt);
|
||
|
|
||
|
hsScalar myDepth;
|
||
|
hsScalar repelMagSq = repel.MagnitudeSquared();
|
||
|
if( repelMagSq < hsBounds::kRealSmall )
|
||
|
{
|
||
|
repel = tri.fNormal;
|
||
|
myDepth = GetRadius();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
myDepth = hsFastMath::InvSqrt(repelMagSq);
|
||
|
repel *= myDepth;
|
||
|
myDepth = 1.f / myDepth;
|
||
|
myDepth = GetRadius() - myDepth;
|
||
|
if( myDepth < 0 )
|
||
|
myDepth = 0;
|
||
|
}
|
||
|
|
||
|
if( tri.fNormal.InnerProduct(myPt) < tri.fDist )
|
||
|
{
|
||
|
repel += tri.fNormal * (-2.f * repel.InnerProduct(tri.fNormal));
|
||
|
myDepth = GetRadius() * 2.f - myDepth;
|
||
|
if( myDepth < 0 )
|
||
|
myDepth = 0;
|
||
|
}
|
||
|
|
||
|
hit->Set(this, &tri, &repel, myDepth);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#if 0 // TOCENTER
|
||
|
hsBool hsBounds3Ext::TriBBHitInfo(hsBounds3Tri& tri, const hsVector3& myVel, hsHitInfoExt* hit) const
|
||
|
{
|
||
|
// Find our closest point (after movement)
|
||
|
hsPoint3 myPt = fCorner;
|
||
|
myPt += myVel;
|
||
|
|
||
|
const hsScalar kMinDist = 1.f; // Huge min dist because world is really big right now. mf horse
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar axDot = fAxes[i].InnerProduct(tri.fNormal);
|
||
|
if( axDot < -kMinDist )
|
||
|
{
|
||
|
// moving towards
|
||
|
myPt += fAxes[i];
|
||
|
}
|
||
|
else if( axDot < kMinDist )
|
||
|
{
|
||
|
// need to interp
|
||
|
axDot /= -(kMinDist*2.f);
|
||
|
axDot += 0.5f;
|
||
|
myPt += fAxes[i] * axDot;
|
||
|
}
|
||
|
// else moving away, skip it
|
||
|
}
|
||
|
|
||
|
// Find closest point on tri to our closest corner
|
||
|
hsPoint3 closePt;
|
||
|
hsBool onTri = tri.ClosestTriPoint(&myPt, &closePt);
|
||
|
|
||
|
// Repel vector is from closest corner to closest point on tri
|
||
|
hsVector3 repel;
|
||
|
repel.Set(&myPt, &closePt);
|
||
|
repel += (-2.f * repel.InnerProduct(tri.fNormal)) * tri.fNormal;
|
||
|
|
||
|
hsScalar repelMag = hsFastMath::InvSqrt(repel.MagnitudeSquared());
|
||
|
|
||
|
if( repelMag < hsBounds::kRealSmall )
|
||
|
{
|
||
|
hsPoint2 faceDepth;
|
||
|
TestPlane(tri.fNormal, myVel, faceDepth);
|
||
|
hit->Set(this, &tri, &tri.fNormal, tri.fDist - faceDepth.fX);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
repel *= repelMag;
|
||
|
repelMag = 1.f / repelMag;
|
||
|
|
||
|
hit->Set(this, &tri, &repel, repelMag);
|
||
|
|
||
|
// Return true of our closest corner projects on to tri (along normal or myVel?)
|
||
|
return onTri;
|
||
|
}
|
||
|
#else // TOCENTER
|
||
|
|
||
|
hsBool hsBounds3Ext::TriBBHitInfo(hsBounds3Tri& tri, const hsVector3& myVel, hsHitInfoExt* hit) const
|
||
|
{
|
||
|
hsPoint3 myPt = GetCenter();
|
||
|
myPt += myVel;
|
||
|
|
||
|
hsPoint3 closePt;
|
||
|
hsBool onTri = tri.ClosestTriPoint(&myPt, &closePt);
|
||
|
|
||
|
hsVector3 repel;
|
||
|
repel.Set(&myPt, &closePt);
|
||
|
hsScalar repelDotNorm = repel.InnerProduct(tri.fNormal);
|
||
|
if( repelDotNorm < 0 )
|
||
|
{
|
||
|
repel += (-2.f * repelDotNorm) * tri.fNormal;
|
||
|
}
|
||
|
|
||
|
hsScalar repelMagSq = repel.MagnitudeSquared();
|
||
|
if( repelMagSq < hsBounds::kRealSmall )
|
||
|
repel = tri.fNormal;
|
||
|
else
|
||
|
{
|
||
|
hsScalar repelMag = hsFastMath::InvSqrt(repelMagSq);
|
||
|
repel *= repelMag;
|
||
|
}
|
||
|
|
||
|
|
||
|
hsPoint2 triDepth;
|
||
|
tri.TestPlane(repel, triDepth);
|
||
|
|
||
|
hsPoint2 myDepth;
|
||
|
TestPlane(repel, myVel, myDepth);
|
||
|
|
||
|
hit->Set(this, &tri, &repel, triDepth.fY - myDepth.fX);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#endif // TOCENTER
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectTriBB(hsBounds3Tri &tri, const hsVector3 &myVel) const
|
||
|
{
|
||
|
hsPoint2 faceDepth;
|
||
|
// first test box against the triangle plane
|
||
|
TestPlane(tri.fNormal, myVel, faceDepth);
|
||
|
|
||
|
if( (tri.fDist > faceDepth.fY)
|
||
|
||(tri.fDist < faceDepth.fX) )
|
||
|
return false;
|
||
|
|
||
|
// now test tri against box planes
|
||
|
if( TestPoints(3, tri.fVerts, -myVel) > 0 )
|
||
|
return false;
|
||
|
|
||
|
if( !(tri.fTriFlags & hsBounds3Tri::kAxesSet) )
|
||
|
tri.SetAxes();
|
||
|
|
||
|
hsScalar depth = tri.fDist - faceDepth.fX;
|
||
|
hsVector3 norm = tri.fNormal;
|
||
|
|
||
|
// that only leaves the planes of triEdge.Cross(vel)
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsPoint2 depths;
|
||
|
TestPlane(tri.fPerpAxes[i], myVel, depths);
|
||
|
if( (tri.fPerpDists[i].fY < depths.fX)
|
||
|
||(tri.fPerpDists[i].fX > depths.fY) )
|
||
|
return false;
|
||
|
|
||
|
#if 0
|
||
|
hsScalar testDepth = tri.fPerpDists[i].fY - depths.fX;
|
||
|
if( testDepth < depth )
|
||
|
{
|
||
|
depth = testDepth;
|
||
|
norm = tri.fPerpAxes[i];
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
hsScalar vDotN = myVel.InnerProduct(tri.fNormal);
|
||
|
if( vDotN > 0 )
|
||
|
depth -= vDotN;
|
||
|
|
||
|
if( depth <= 0 )
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectTriBB(hsBounds3Tri &tri, const hsVector3 &myVel, hsHitInfoExt *hit) const
|
||
|
{
|
||
|
hsPoint2 faceDepth;
|
||
|
// first test box against the triangle plane
|
||
|
TestPlane(tri.fNormal, myVel, faceDepth);
|
||
|
|
||
|
if( (tri.fDist > faceDepth.fY)
|
||
|
||(tri.fDist < faceDepth.fX) )
|
||
|
return false;
|
||
|
|
||
|
hsScalar centDist = tri.fNormal.InnerProduct(hit->fRootCenter);
|
||
|
if( centDist < tri.fDist )
|
||
|
return false;
|
||
|
|
||
|
// now test tri against box planes
|
||
|
if( TestPoints(3, tri.fVerts, -myVel) > 0 )
|
||
|
return false;
|
||
|
|
||
|
if( !(tri.fTriFlags & hsBounds3Tri::kAxesSet) )
|
||
|
tri.SetAxes();
|
||
|
|
||
|
hsScalar depth = tri.fDist - faceDepth.fX;
|
||
|
hsVector3 norm = tri.fNormal;
|
||
|
|
||
|
// that only leaves the planes of triEdge.Cross(vel)
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsPoint2 depths;
|
||
|
TestPlane(tri.fPerpAxes[i], myVel, depths);
|
||
|
if( (tri.fPerpDists[i].fY < depths.fX)
|
||
|
||(tri.fPerpDists[i].fX > depths.fY) )
|
||
|
return false;
|
||
|
|
||
|
#if 0
|
||
|
hsScalar testDepth = tri.fPerpDists[i].fY - depths.fX;
|
||
|
if( testDepth < depth )
|
||
|
{
|
||
|
depth = testDepth;
|
||
|
norm = tri.fPerpAxes[i];
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
hsScalar vDotN = myVel.InnerProduct(tri.fNormal);
|
||
|
if( vDotN > 0 )
|
||
|
depth -= vDotN;
|
||
|
|
||
|
if( (tri.fTriFlags & hsBounds3Tri::kDoubleSide) )
|
||
|
{
|
||
|
if( tri.fNormal.InnerProduct(hit->fRootCenter) - tri.fDist < 0 )
|
||
|
{
|
||
|
depth = -tri.fDist + faceDepth.fY;
|
||
|
if( vDotN < 0 )
|
||
|
depth += vDotN;
|
||
|
|
||
|
tri.fNormal = -tri.fNormal;
|
||
|
tri.fDist = -tri.fDist;
|
||
|
}
|
||
|
}
|
||
|
if( depth <= 0 )
|
||
|
return false;
|
||
|
|
||
|
// printf("ATTRIBUTE triBnd addr %x\n",&tri.fNormal); /* Takashi Nakata TEST Add */
|
||
|
hit->Set(this, &tri, &norm, depth);
|
||
|
|
||
|
return hit->fDepth > hsBounds::kRealSmall;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectTriBS(hsBounds3Tri &tri, const hsVector3 &myVel) const
|
||
|
{
|
||
|
if( !(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
|
||
|
hsAssert(fBounds3Flags & kCenterValid, "Sphere set but not center (TriBS)");
|
||
|
hsScalar radScaled = fRadius * tri.fNormal.Magnitude();
|
||
|
hsScalar centerDist = tri.fNormal.InnerProduct(fCenter);
|
||
|
hsScalar velDist = tri.fNormal.InnerProduct(myVel);
|
||
|
hsScalar effMin = centerDist;
|
||
|
hsScalar effMax = centerDist;
|
||
|
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
effMax += radScaled;
|
||
|
effMin -= radScaled;
|
||
|
|
||
|
if( tri.fDist <= effMin )
|
||
|
return false;
|
||
|
if( tri.fDist >= effMax )
|
||
|
return false;
|
||
|
|
||
|
// mf horse
|
||
|
hsScalar normDepth = tri.fDist - (centerDist - radScaled + velDist);
|
||
|
if( normDepth <= 0 )
|
||
|
{
|
||
|
// we'll report a depth of zero to (hopefully) neutralize any effects
|
||
|
if( tri.fTriFlags & hsBounds3Tri::kDoubleSide )
|
||
|
{
|
||
|
normDepth = -tri.fDist + (centerDist + radScaled + velDist);
|
||
|
if( normDepth > 0 )
|
||
|
{
|
||
|
tri.fDist = -tri.fDist;
|
||
|
tri.fNormal = -tri.fNormal;
|
||
|
}
|
||
|
else
|
||
|
normDepth = 0;
|
||
|
}
|
||
|
else
|
||
|
normDepth = 0;
|
||
|
}
|
||
|
hsAssert(normDepth >= 0, "NegativeDepth");
|
||
|
|
||
|
if( !(tri.fTriFlags & hsBounds3Tri::kAxesSet) )
|
||
|
tri.SetAxes();
|
||
|
|
||
|
hsAssert(fBounds3Flags & kCenterValid, "Sphere set but not center (TriBS)");
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
centerDist = tri.fPerpAxes[i].InnerProduct(fCenter);
|
||
|
velDist = tri.fPerpAxes[i].InnerProduct(myVel);
|
||
|
effMin = centerDist;
|
||
|
effMax = centerDist;
|
||
|
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
|
||
|
hsScalar radScale = fRadius * tri.fPerpAxes[i].Magnitude();
|
||
|
effMax += radScale;
|
||
|
effMin -= radScale;
|
||
|
if( tri.fPerpDists[i].fY <= effMin )
|
||
|
return false;
|
||
|
if( tri.fPerpDists[i].fX >= effMax )
|
||
|
return false;
|
||
|
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectTriBS(hsBounds3Tri &tri, const hsVector3 &myVel, hsHitInfoExt *hit) const
|
||
|
{
|
||
|
if( !(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
|
||
|
hsAssert(fBounds3Flags & kCenterValid, "Sphere set but not center (TriBS)");
|
||
|
hsScalar radScaled = fRadius * tri.fNormal.Magnitude();
|
||
|
hsScalar centerDist = tri.fNormal.InnerProduct(fCenter);
|
||
|
hsScalar velDist = tri.fNormal.InnerProduct(myVel);
|
||
|
hsScalar effMin = centerDist;
|
||
|
hsScalar effMax = centerDist;
|
||
|
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
effMax += radScaled;
|
||
|
effMin -= radScaled;
|
||
|
|
||
|
if( tri.fDist <= effMin )
|
||
|
return false;
|
||
|
if( tri.fDist >= effMax )
|
||
|
return false;
|
||
|
|
||
|
// mf horse
|
||
|
hsScalar normDepth = tri.fDist - (centerDist - radScaled + velDist);
|
||
|
if( normDepth <= 0 )
|
||
|
{
|
||
|
#if 0 // need to report the collision even if the object is leaving the tri
|
||
|
// we'll report a depth of zero to (hopefully) neutralize any effects
|
||
|
if(!(tri.fTriFlags & hsBounds3Tri::kDoubleSide) )
|
||
|
return false;
|
||
|
normDepth = -tri.fDist + (centerDist + radScaled + velDist);
|
||
|
if( normDepth <= 0 )
|
||
|
return false;
|
||
|
tri.fDist = -tri.fDist;
|
||
|
tri.fNormal = -tri.fNormal;
|
||
|
#else
|
||
|
// we'll report a depth of zero to (hopefully) neutralize any effects
|
||
|
if( tri.fTriFlags & hsBounds3Tri::kDoubleSide )
|
||
|
{
|
||
|
normDepth = -tri.fDist + (centerDist + radScaled + velDist);
|
||
|
if( normDepth > 0 )
|
||
|
{
|
||
|
tri.fDist = -tri.fDist;
|
||
|
tri.fNormal = -tri.fNormal;
|
||
|
}
|
||
|
else
|
||
|
normDepth = 0;
|
||
|
}
|
||
|
else
|
||
|
normDepth = 0;
|
||
|
#endif
|
||
|
}
|
||
|
hsAssert(normDepth >= 0, "NegativeDepth");
|
||
|
|
||
|
if( !(tri.fTriFlags & hsBounds3Tri::kAxesSet) )
|
||
|
tri.SetAxes();
|
||
|
|
||
|
hsAssert(fBounds3Flags & kCenterValid, "Sphere set but not center (TriBS)");
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
centerDist = tri.fPerpAxes[i].InnerProduct(fCenter);
|
||
|
velDist = tri.fPerpAxes[i].InnerProduct(myVel);
|
||
|
effMin = centerDist;
|
||
|
effMax = centerDist;
|
||
|
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
|
||
|
hsScalar radScale = fRadius * tri.fPerpAxes[i].Magnitude();
|
||
|
effMax += radScale;
|
||
|
effMin -= radScale;
|
||
|
if( tri.fPerpDists[i].fY <= effMin )
|
||
|
return false;
|
||
|
if( tri.fPerpDists[i].fX >= effMax )
|
||
|
return false;
|
||
|
|
||
|
}
|
||
|
|
||
|
hsScalar invLen = hsScalarInvert(tri.fNormal.Magnitude());
|
||
|
hit->Set(this, &tri, &tri.fNormal, normDepth);
|
||
|
|
||
|
// mf horse - move this into Set()?
|
||
|
hit->fNormal *= invLen;
|
||
|
hit->fDepth *= invLen;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#endif // Commenting out this which will be made redundant and/or obsolete by Havok integration
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectBSBS(const hsBounds3Ext& other, const hsVector3& myVel, hsHitInfoExt *hit) const
|
||
|
{
|
||
|
if(!(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
if(!(other.fExtFlags & kSphereSet) )
|
||
|
other.IMakeSphere();
|
||
|
|
||
|
hsScalar tClose, tImpact;
|
||
|
if( !IClosestISect(other, myVel, &tClose, &tImpact) )
|
||
|
return false;
|
||
|
if( (tImpact < 0) || (tImpact > 1.f) )
|
||
|
return false;
|
||
|
if( tClose < 0 )
|
||
|
tClose = 0;
|
||
|
if( tClose > 1.f )
|
||
|
tClose = 1.f;
|
||
|
|
||
|
hsPoint3 closePt = GetCenter() + myVel * tClose;
|
||
|
hsVector3 del;
|
||
|
del.Set(&closePt, &other.GetCenter());
|
||
|
|
||
|
hsScalar mag = del.Magnitude();
|
||
|
hsScalar depth = GetRadius() + other.GetRadius() - mag;
|
||
|
if( depth <= 0 )
|
||
|
return false;
|
||
|
|
||
|
hsPoint3 hitPt = GetCenter() + myVel * tImpact;
|
||
|
hsVector3 norm;
|
||
|
norm.Set(&hitPt, &other.GetCenter());
|
||
|
norm.Normalize();
|
||
|
|
||
|
hit->Set(this, &other, norm, depth);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectBSBox(const hsBounds3Ext &other, const hsVector3 &myVel, hsHitInfoExt *hit) const
|
||
|
{
|
||
|
hit->fDelPos = -myVel;
|
||
|
if( other.ISectBoxBS(*this, hit->fDelPos, hit) )
|
||
|
{
|
||
|
hit->fNormal = -hit->fNormal;
|
||
|
hit->fBoxBnd = this;
|
||
|
hit->fOtherBoxBnd = &other;
|
||
|
hit->fDelPos = myVel;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hit->fDelPos = myVel;
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectBoxBS(const hsBounds3Ext &other, const hsVector3 &myVel, hsHitInfoExt *hit) const
|
||
|
{
|
||
|
if(!(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
hsAssert(fBounds3Flags & kCenterValid, "Sphere set but not center (BoxBS(vel))");
|
||
|
|
||
|
hsVector3 minAxis;
|
||
|
hsScalar minDepth;
|
||
|
hsBool haveAxis = false;
|
||
|
hsVector3 tstAxis;
|
||
|
hsScalar tstDepth;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsBool tryAxis;
|
||
|
if( other.fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
// first try the other box axes
|
||
|
hsScalar effMin = fCenter[i];
|
||
|
hsScalar effMax = effMin;
|
||
|
hsScalar velDist = myVel[i];
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
effMax += fRadius;
|
||
|
effMin -= fRadius;
|
||
|
|
||
|
if( effMax < other.fMins[i] )
|
||
|
return false;
|
||
|
if( effMin > other.fMaxs[i] )
|
||
|
return false;
|
||
|
|
||
|
if( (other.fMins[i] <= effMin)
|
||
|
&&(other.fMaxs[i] <= effMax) )
|
||
|
{
|
||
|
tstDepth = other.fMaxs[i] - effMin;
|
||
|
hsAssert(tstDepth > -kRealSmall, "Late to be finding sep axis");
|
||
|
tstAxis.Set(i == 0 ? hsScalar1 : 0, i & 1 ? hsScalar1 : 0, i & 2 ? hsScalar1 : 0);
|
||
|
tryAxis = true;
|
||
|
}
|
||
|
else
|
||
|
if( (other.fMins[i] >= effMin)
|
||
|
&&(other.fMaxs[i] >= effMax) )
|
||
|
{
|
||
|
tstDepth = effMax - other.fMins[i];
|
||
|
hsAssert(tstDepth > -kRealSmall, "Late to be finding sep axis");
|
||
|
tstAxis.Set(i == 0 ? -hsScalar1 : 0, i & 1 ? -hsScalar1 : 0, i & 2 ? -hsScalar1 : 0);
|
||
|
tryAxis = true;
|
||
|
}
|
||
|
else
|
||
|
tryAxis = false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// first try the other box axes
|
||
|
hsScalar radScaled = fRadius * other.fAxes[i].Magnitude();
|
||
|
hsScalar centerDist = other.fAxes[i].InnerProduct(fCenter);
|
||
|
hsScalar effMin = centerDist;
|
||
|
hsScalar effMax = centerDist;
|
||
|
hsScalar velDist = other.fAxes[i].InnerProduct(myVel);
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
effMax += radScaled;
|
||
|
effMin -= radScaled;
|
||
|
|
||
|
if( !(other.fExtFlags & kDistsSet) )
|
||
|
other.IMakeDists();
|
||
|
|
||
|
if( effMax < other.fDists[i].fX )
|
||
|
return false;
|
||
|
if( effMin > other.fDists[i].fY )
|
||
|
return false;
|
||
|
|
||
|
if( centerDist <= other.fDists[i].fX )
|
||
|
{
|
||
|
tstDepth = effMax - other.fDists[i].fX;
|
||
|
tstAxis = -other.fAxes[i];
|
||
|
hsAssert(tstDepth > -kRealSmall, "Late to be finding sep axis");
|
||
|
}
|
||
|
else
|
||
|
if( centerDist >= other.fDists[i].fY )
|
||
|
{
|
||
|
tstDepth = other.fDists[i].fY - effMin;
|
||
|
tstAxis = other.fAxes[i];
|
||
|
hsAssert(tstDepth > -kRealSmall, "Late to be finding sep axis");
|
||
|
}
|
||
|
else
|
||
|
tryAxis = false;
|
||
|
|
||
|
}
|
||
|
if( tryAxis )
|
||
|
{
|
||
|
hsScalar magSq = tstAxis.MagnitudeSquared();
|
||
|
if( magSq > kRealSmall )
|
||
|
{
|
||
|
tstDepth *= tstDepth * hsScalarInvert(magSq);
|
||
|
if( !haveAxis||(tstDepth < minDepth) )
|
||
|
{
|
||
|
minDepth = tstDepth;
|
||
|
minAxis = tstAxis;
|
||
|
haveAxis = true;
|
||
|
}
|
||
|
hsAssert(!haveAxis || (minAxis.MagnitudeSquared() > kRealSmall), "Bogus");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
// now try the axis between the center of sphere and center of other box
|
||
|
hsVector3 diag(&fCenter, &other.GetCenter());
|
||
|
if( !haveAxis && (diag.MagnitudeSquared() < kRealSmall) )
|
||
|
diag.Set(1.f, 0, 0);
|
||
|
hsScalar effMin = diag.InnerProduct(fCenter);
|
||
|
hsScalar effMax = effMin;
|
||
|
hsScalar velDist = diag.InnerProduct(myVel);
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
hsScalar radDist = fRadius * diag.Magnitude();
|
||
|
effMax += radDist;
|
||
|
effMin -= radDist;
|
||
|
hsPoint2 otherDepth;
|
||
|
other.TestPlane(diag, otherDepth);
|
||
|
if( effMax < otherDepth.fX )
|
||
|
return false;
|
||
|
if( effMin > otherDepth.fY )
|
||
|
return false;
|
||
|
|
||
|
tstAxis = diag;
|
||
|
tstDepth = otherDepth.fY - effMin;
|
||
|
hsScalar magSq = tstAxis.MagnitudeSquared();
|
||
|
if( magSq > 0 )
|
||
|
{
|
||
|
tstDepth *= tstDepth * hsScalarInvert(magSq);
|
||
|
if( !haveAxis ||(tstDepth < minDepth) )
|
||
|
{
|
||
|
minDepth = tstDepth;
|
||
|
minAxis = tstAxis;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hsScalar invMag = hsScalarInvert(minAxis.Magnitude());
|
||
|
minAxis *= invMag;
|
||
|
hsAssert(minDepth >= 0, "Late to find sep plane");
|
||
|
minDepth = hsSquareRoot(minDepth);
|
||
|
hit->Set(this, &other, minAxis, minDepth);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectBoxBS(const hsBounds3Ext &other, const hsVector3 &myVel) const
|
||
|
{
|
||
|
if( !(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
hsAssert(fBounds3Flags & kCenterValid, "Sphere set but not center (BoxBS)");
|
||
|
|
||
|
if( other.fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
// first try the other box axes
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar effMin = fCenter[i];
|
||
|
hsScalar effMax = effMin;
|
||
|
hsScalar velDist = myVel[i];
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
effMax += fRadius;
|
||
|
effMin -= fRadius;
|
||
|
|
||
|
if( effMax < other.fMins[i] )
|
||
|
return false;
|
||
|
if( effMin > other.fMaxs[i] )
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// first try the other box axes
|
||
|
if( !(other.fExtFlags & kDistsSet) )
|
||
|
other.IMakeDists();
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar effMin = other.fAxes[i].InnerProduct(fCenter);
|
||
|
hsScalar effMax = effMin;
|
||
|
hsScalar velDist = other.fAxes[i].InnerProduct(myVel);
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
hsScalar radScaled = fRadius * other.fAxes[i].Magnitude();
|
||
|
effMax += radScaled;
|
||
|
effMin -= radScaled;
|
||
|
|
||
|
if( effMax < other.fDists[i].fX )
|
||
|
return false;
|
||
|
if( effMin > other.fDists[i].fY )
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// now try the axis between the center of sphere and center of other box
|
||
|
hsVector3 diag(&fCenter, &other.GetCenter());
|
||
|
hsScalar effMin = diag.InnerProduct(fCenter);
|
||
|
hsScalar effMax = effMin;
|
||
|
hsScalar velDist = diag.InnerProduct(myVel);
|
||
|
if( velDist > 0 )
|
||
|
effMax += velDist;
|
||
|
else
|
||
|
effMin += velDist;
|
||
|
hsScalar radDist = fRadius * diag.Magnitude();
|
||
|
effMax += radDist;
|
||
|
effMin -= radDist;
|
||
|
hsPoint2 otherDepth;
|
||
|
other.TestPlane(diag, otherDepth);
|
||
|
if( effMax < otherDepth.fX )
|
||
|
return false;
|
||
|
if( effMin > otherDepth.fY )
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectLine(const hsPoint3* from, const hsPoint3* at) const
|
||
|
{
|
||
|
if( !(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
|
||
|
hsPoint3 onLine;
|
||
|
hsScalar z = ClosestPointToLine(&fCenter, from, at, &onLine);
|
||
|
|
||
|
hsScalar distSq = hsVector3(&onLine, &fCenter).MagnitudeSquared();
|
||
|
if( distSq >= fRadius*fRadius )
|
||
|
return false;
|
||
|
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if( ((*from)[i] < fMins[i])&&((*at)[i] < fMins[i]) )
|
||
|
return false;
|
||
|
if( ((*from)[i] > fMaxs[i])&&((*at)[i] > fMaxs[i]) )
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar d0 = fAxes[i].InnerProduct(from);
|
||
|
hsScalar d1 = fAxes[i].InnerProduct(at);
|
||
|
if( d0 < d1 )
|
||
|
{
|
||
|
if( d1 < fDists[i].fX )
|
||
|
return false;
|
||
|
if( d0 > fDists[i].fY )
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if( d0 < fDists[i].fX )
|
||
|
return false;
|
||
|
if( d1 > fDists[i].fY )
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectCone(const hsPoint3* from, const hsPoint3* at, hsScalar radius) const
|
||
|
{
|
||
|
if( !(fExtFlags & kSphereSet) )
|
||
|
IMakeSphere();
|
||
|
|
||
|
// expensive
|
||
|
hsPoint3 onLine;
|
||
|
ClosestPointToLine(&fCenter, from, at, &onLine);
|
||
|
|
||
|
hsScalar distSq = hsVector3(&onLine, &fCenter).MagnitudeSquared();
|
||
|
hsScalar radiusSq = fRadius * fRadius;
|
||
|
if (distSq - radius*radius >= radiusSq)
|
||
|
return false;
|
||
|
|
||
|
hsScalar dist = hsVector3(from, &onLine).Magnitude();
|
||
|
hsScalar len = hsVector3(from, at).Magnitude();
|
||
|
hsScalar partRadius = radius/len * dist;
|
||
|
if (distSq - fRadius*fRadius - partRadius*partRadius >= 0)
|
||
|
{
|
||
|
hsVector3 rayToCenter(&fCenter,&onLine);
|
||
|
rayToCenter.Normalize();
|
||
|
|
||
|
hsPoint3 atEdge = *at + rayToCenter*radius;
|
||
|
|
||
|
ClosestPointToLine(&fCenter, from, &atEdge, &onLine);
|
||
|
|
||
|
distSq = hsVector3(&onLine, &fCenter).MagnitudeSquared();
|
||
|
if( distSq >= radiusSq )
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// incorrect
|
||
|
if( fExtFlags & kAxisAligned )
|
||
|
{
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
if( ((*from)[i] < fMins[i])&&((*at)[i]+radius < fMins[i]) )
|
||
|
return false;
|
||
|
if( ((*from)[i] > fMaxs[i])&&((*at)[i]-radius > fMaxs[i]) )
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if( !(fExtFlags & kDistsSet) )
|
||
|
IMakeDists();
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
ClosestPointToInfiniteLine(at, &fAxes[i], &onLine);
|
||
|
hsVector3 atLine(&onLine,at);
|
||
|
atLine.Normalize();
|
||
|
hsPoint3 atEdge = *at + atLine * radius;
|
||
|
|
||
|
hsScalar d0 = fAxes[i].InnerProduct(*from);
|
||
|
hsScalar d1 = fAxes[i].InnerProduct(atEdge);
|
||
|
if( d0 < d1 )
|
||
|
{
|
||
|
if( d1 < fDists[i].fX )
|
||
|
return false;
|
||
|
if( d0 > fDists[i].fY )
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if( d0 < fDists[i].fX )
|
||
|
return false;
|
||
|
if( d1 > fDists[i].fY )
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
|
||
|
hsBool hsBounds3Ext::ISectRayBS(const hsPoint3& from, const hsPoint3& to, hsPoint3& at) const
|
||
|
{
|
||
|
hsVector3 c2f(&from,&GetCenter());
|
||
|
hsVector3 f2t(&to,&from);
|
||
|
hsScalar a = f2t.MagnitudeSquared();
|
||
|
hsScalar b = 2 * (c2f.InnerProduct(f2t));
|
||
|
hsScalar c = c2f.MagnitudeSquared() - GetRadius()*GetRadius();
|
||
|
|
||
|
hsScalar disc = b*b - 4*a*c;
|
||
|
if (disc < 0)
|
||
|
return false;
|
||
|
else
|
||
|
{
|
||
|
hsScalar discSqrt = hsSquareRoot(disc);
|
||
|
hsScalar denom = 1.f/(2*a);
|
||
|
hsScalar t = (-b - discSqrt) * denom;
|
||
|
|
||
|
if (t<1 && t>0)
|
||
|
at = from + (f2t * t);
|
||
|
else
|
||
|
return false;
|
||
|
#if 0
|
||
|
{
|
||
|
t = (-b + discSqrt) * denom;
|
||
|
if (t > 1)
|
||
|
return false;
|
||
|
at = from + (f2t * t);
|
||
|
}
|
||
|
#endif
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void hsBounds3Ext::Read(hsStream *s)
|
||
|
{
|
||
|
fExtFlags = s->ReadSwap32();
|
||
|
hsBounds3::Read(s);
|
||
|
if( !(fExtFlags & kAxisAligned) )
|
||
|
{
|
||
|
fCorner.Read(s);
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
fAxes[i].Read(s);
|
||
|
fDists[i].fX = s->ReadSwapScalar();
|
||
|
fDists[i].fY = s->ReadSwapScalar();
|
||
|
}
|
||
|
IMakeMinsMaxs();
|
||
|
IMakeDists();
|
||
|
}
|
||
|
IMakeSphere();
|
||
|
}
|
||
|
void hsBounds3Ext::Write(hsStream *s)
|
||
|
{
|
||
|
s->WriteSwap32(fExtFlags);
|
||
|
hsBounds3::Write(s);
|
||
|
if( !(fExtFlags & kAxisAligned) )
|
||
|
{
|
||
|
fCorner.Write(s);
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
fAxes[i].Write(s);
|
||
|
if( fExtFlags & kDistsSet )
|
||
|
{
|
||
|
s->WriteSwapScalar(fDists[i].fX);
|
||
|
s->WriteSwapScalar(fDists[i].fY);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Playing nice with binary patches--writing uninited values BAD!
|
||
|
s->WriteSwapScalar( 0.f );
|
||
|
s->WriteSwapScalar( 0.f );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if 0 // Commenting out this which will be made redundant and/or obsolete by Havok integration
|
||
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
void hsBounds3Tri::TestPlane(const hsVector3 &n, hsPoint2 &depth) const
|
||
|
{
|
||
|
depth.fX = depth.fY = n.InnerProduct(fVerts[0]);
|
||
|
|
||
|
hsScalar d1, d2;
|
||
|
|
||
|
d1 = n.InnerProduct(fVerts[1]);
|
||
|
d2 = n.InnerProduct(fVerts[2]);
|
||
|
|
||
|
if( d1 > d2 )
|
||
|
{
|
||
|
if( d1 > depth.fY )
|
||
|
depth.fY = d1;
|
||
|
if( d2 < depth.fX )
|
||
|
depth.fX = d2;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if( d2 > depth.fY )
|
||
|
depth.fY = d2;
|
||
|
if( d1 < depth.fX )
|
||
|
depth.fX = d1;
|
||
|
}
|
||
|
}
|
||
|
hsBool hsBounds3Tri::ClosestTriPoint(const hsPoint3 *p, hsPoint3 *out, const hsVector3 *ax) const
|
||
|
{
|
||
|
// project point onto tri plane
|
||
|
hsPoint3 pPln;
|
||
|
if( ax )
|
||
|
{
|
||
|
hsScalar t;
|
||
|
|
||
|
t = fNormal.InnerProduct(fVerts[0] - *p);
|
||
|
hsScalar s = fNormal.InnerProduct(ax);
|
||
|
if( (s > hsBounds::kRealSmall)||(s < -hsBounds::kRealSmall) )
|
||
|
{
|
||
|
t /= s;
|
||
|
|
||
|
pPln = *p;
|
||
|
pPln += *ax * t;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return ClosestTriPoint(p, out);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hsScalar t;
|
||
|
|
||
|
t = fNormal.InnerProduct(fVerts[0] - *p);
|
||
|
t /= fNormal.MagnitudeSquared();
|
||
|
|
||
|
pPln = *p;
|
||
|
pPln += fNormal * t;
|
||
|
}
|
||
|
|
||
|
if( !(fTriFlags & kAxesSet) )
|
||
|
SetAxes();
|
||
|
|
||
|
int nIn = 0;
|
||
|
int firstIn, secondIn;
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
hsScalar tst = fPerpAxes[i].InnerProduct(pPln);
|
||
|
hsBool in = false;
|
||
|
if( fOnIsMax & (1 << i) )
|
||
|
{
|
||
|
if( tst <= fPerpDists[i].fY )
|
||
|
in = true;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if( tst >= fPerpDists[i].fX )
|
||
|
in = true;
|
||
|
}
|
||
|
if( in )
|
||
|
{
|
||
|
if( nIn++ )
|
||
|
secondIn = i;
|
||
|
else
|
||
|
firstIn = i;
|
||
|
}
|
||
|
}
|
||
|
switch( nIn )
|
||
|
{
|
||
|
case 3:
|
||
|
*out = pPln;
|
||
|
break;
|
||
|
case 1:
|
||
|
{
|
||
|
int k, kPlus;
|
||
|
k = firstIn == 2 ? 0 : firstIn+1;
|
||
|
kPlus = k == 2 ? 0 : k+1;
|
||
|
|
||
|
hsPoint3 pTmp;
|
||
|
hsScalar z;
|
||
|
z = hsBounds3::ClosestPointToLine(&pPln, fVerts+k, fVerts+kPlus, &pTmp);
|
||
|
if( z <= hsScalar1 )
|
||
|
*out = pTmp;
|
||
|
else
|
||
|
{
|
||
|
k = kPlus;
|
||
|
kPlus = k == 2 ? 0 : k+1;
|
||
|
z = hsBounds3::ClosestPointToLine(&pPln, fVerts+k, fVerts+kPlus, out);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case 2:
|
||
|
{
|
||
|
int k, kPlus;
|
||
|
k = secondIn == 2 ? 0 : secondIn+1;
|
||
|
if( k == firstIn )
|
||
|
k++;
|
||
|
kPlus = k == 2 ? 0 : k+1;
|
||
|
|
||
|
hsBounds3::ClosestPointToLine(&pPln, fVerts+k, fVerts+kPlus, out);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case 0:
|
||
|
hsAssert(false, "Extreme bogosity, inverted tri?!?");
|
||
|
*out = pPln;
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
#ifdef HS_DEBUGGING // mf horse testing
|
||
|
#if 0
|
||
|
if( 0 )
|
||
|
{
|
||
|
hsVector3 ndeb = hsVector3(fVerts+1, fVerts) % hsVector3(fVerts+2, fVerts);
|
||
|
hsScalar dis;
|
||
|
dis = fNormal.InnerProduct(pPln) - fDist;
|
||
|
if( (fDist > hsBounds::kRealSmall)||(fDist < -hsBounds::kRealSmall) )
|
||
|
dis /= fDist;
|
||
|
hsAssert((dis < hsBounds::kRealSmall)&&(dis > -hsBounds::kRealSmall), "Non-planar pPln");
|
||
|
dis = hsVector3(&pPln, out).MagnitudeSquared();
|
||
|
hsScalar vDis;
|
||
|
vDis = hsVector3(&pPln, fVerts+0).MagnitudeSquared();
|
||
|
hsAssert( vDis - dis > -hsBounds::kRealSmall, "Bad closest point");
|
||
|
vDis = hsVector3(&pPln, fVerts+1).MagnitudeSquared();
|
||
|
hsAssert( vDis - dis > -hsBounds::kRealSmall, "Bad closest point");
|
||
|
vDis = hsVector3(&pPln, fVerts+2).MagnitudeSquared();
|
||
|
hsAssert( vDis - dis > -hsBounds::kRealSmall, "Bad closest point");
|
||
|
hsBool dork = false;
|
||
|
if( dork )
|
||
|
{
|
||
|
hsScalar zn[3];
|
||
|
hsScalar zf[3];
|
||
|
hsScalar z[3];
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
z[i] = fPerpAxes[i].InnerProduct(fVerts[i]);
|
||
|
int j;
|
||
|
j = i == 0 ? 2 : i-1;
|
||
|
zf[i] = fPerpAxes[i].InnerProduct(fVerts[j]);
|
||
|
j = i == 2 ? 0 : i+1;
|
||
|
zn[i] = fPerpAxes[i].InnerProduct(fVerts[j]);
|
||
|
}
|
||
|
return ClosestTriPoint(p, out, ax);
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
return 3 == nIn;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Tri::SetAxes() const
|
||
|
{
|
||
|
fOnIsMax = 0;
|
||
|
|
||
|
hsVector3 edge[3];
|
||
|
edge[0].Set(fVerts, fVerts+1);
|
||
|
edge[1].Set(fVerts+1, fVerts+2);
|
||
|
edge[2].Set(fVerts+2, fVerts);
|
||
|
hsVector3 perp = edge[2] % edge[0];
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
int j = i == 2 ? 0 : i+1;
|
||
|
int k = j == 2 ? 0 : j+1;
|
||
|
fPerpAxes[i] = edge[i] % perp;
|
||
|
fPerpAxes[i].Normalize();
|
||
|
|
||
|
fPerpDists[i].fX = fPerpAxes[i].InnerProduct(fVerts[i]);
|
||
|
fPerpDists[i].fY = fPerpAxes[i].InnerProduct(fVerts[k]);
|
||
|
if( fPerpDists[i].fX > fPerpDists[i].fY )
|
||
|
{
|
||
|
fOnIsMax |= 1 << i;
|
||
|
hsScalar d = fPerpDists[i].fX;
|
||
|
fPerpDists[i].fX = fPerpDists[i].fY;
|
||
|
fPerpDists[i].fY = d;
|
||
|
}
|
||
|
}
|
||
|
fTriFlags |= kAxesSet;
|
||
|
}
|
||
|
|
||
|
hsBounds3Tri* hsBounds3Tri::Transform(const hsMatrix44& x)
|
||
|
{
|
||
|
#if 0 // IDENT
|
||
|
if( x.fFlags & hsMatrix44::kIsIdent )
|
||
|
return this;
|
||
|
#endif // IDENT
|
||
|
|
||
|
fVerts[0] = x * fVerts[0];
|
||
|
fVerts[1] = x * fVerts[1];
|
||
|
fVerts[2] = x * fVerts[2];
|
||
|
|
||
|
hsVector3 v1, v2;
|
||
|
v1.Set(&fVerts[1], &fVerts[0]);
|
||
|
v2.Set(&fVerts[2], &fVerts[0]);
|
||
|
fNormal = v1 % v2;
|
||
|
// mf horse - do we need to normalize here?
|
||
|
// fNormal.Normalize();
|
||
|
|
||
|
fDist = fNormal.InnerProduct(fVerts[0]);
|
||
|
|
||
|
fTriFlags &= ~kAxesSet;
|
||
|
|
||
|
SetAxes();
|
||
|
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
hsBounds3Tri* hsBounds3Tri::Translate(const hsVector3& v)
|
||
|
{
|
||
|
fVerts[0] += v;
|
||
|
fVerts[1] += v;
|
||
|
fVerts[2] += v;
|
||
|
|
||
|
fDist = fNormal.InnerProduct(fVerts[0]);
|
||
|
|
||
|
int i;
|
||
|
for( i = 0; i < 3; i++ )
|
||
|
{
|
||
|
int j = i == 2 ? 0 : i+1;
|
||
|
int k = j == 2 ? 0 : j+1;
|
||
|
|
||
|
hsScalar del = fPerpAxes[i].InnerProduct(v);
|
||
|
fPerpDists[i].fX += del;
|
||
|
fPerpDists[i].fY += del;
|
||
|
}
|
||
|
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
void hsBounds3Tri::Set(const hsPoint3& v0,
|
||
|
const hsPoint3& v1,
|
||
|
const hsPoint3& v2,
|
||
|
hsTriangle3* t,
|
||
|
const hsMatrix44& x)
|
||
|
{
|
||
|
fVerts[0] = v0;
|
||
|
fVerts[1] = v1;
|
||
|
fVerts[2] = v2;
|
||
|
|
||
|
fOnIsMax = 0;
|
||
|
fTriangle = t;
|
||
|
|
||
|
if( t->fFlags & hsTriangle3::kTwoSided )
|
||
|
fTriFlags |= kDoubleSide;
|
||
|
|
||
|
#if 0 // IDENT
|
||
|
if( x.fFlags & hsMatrix44::kIsIdent )
|
||
|
{
|
||
|
hsVector3 v1, v2;
|
||
|
v1.Set(&fVerts[1], &fVerts[0]);
|
||
|
v2.Set(&fVerts[2], &fVerts[0]);
|
||
|
fNormal = v1 % v2;
|
||
|
// mf horse - do we need to normalize here?
|
||
|
// fNormal.Normalize();
|
||
|
|
||
|
fDist = fNormal.InnerProduct(fVerts[0]);
|
||
|
|
||
|
fTriFlags &= ~kAxesSet;
|
||
|
|
||
|
SetAxes();
|
||
|
}
|
||
|
else
|
||
|
#endif // IDENT
|
||
|
Transform(x);
|
||
|
}
|
||
|
|
||
|
hsBounds3Tri::hsBounds3Tri(const hsPoint3& v0,
|
||
|
const hsPoint3& v1,
|
||
|
const hsPoint3& v2,
|
||
|
hsTriangle3* t,
|
||
|
const hsMatrix44& x)
|
||
|
{
|
||
|
Set(v0, v1, v2, t, x);
|
||
|
}
|
||
|
|
||
|
hsBounds3Tri::hsBounds3Tri(hsTriangle3* t, const hsMatrix44& x)
|
||
|
{
|
||
|
Set(t->fVert[0]->fVtx->fLocalPos,
|
||
|
t->fVert[2]->fVtx->fLocalPos,
|
||
|
t->fVert[2]->fVtx->fLocalPos,
|
||
|
t, x);
|
||
|
}
|
||
|
|
||
|
void hsBounds3Tri::Set(hsPoint3 *v0, hsPoint3 *v1, hsPoint3 *v2, hsVector3 *n, UInt32 triFlags, hsTriangle3 *t)
|
||
|
{
|
||
|
fTriFlags = 0;
|
||
|
|
||
|
if( triFlags & hsTriangle3::kTwoSided )
|
||
|
fTriFlags |= kDoubleSide;
|
||
|
|
||
|
fNormal = *n;
|
||
|
fVerts[0] = *v0;
|
||
|
fVerts[1] = *v1;
|
||
|
fVerts[2] = *v2;
|
||
|
|
||
|
fOnIsMax = 0;
|
||
|
fTriangle = t;
|
||
|
|
||
|
fDist = fNormal.InnerProduct(fVerts[0]);
|
||
|
}
|
||
|
|
||
|
hsBounds3Tri::hsBounds3Tri(hsPoint3 *v0, hsPoint3 *v1, hsPoint3 *v2, hsVector3 *n, UInt32 triFlags, hsTriangle3 *t)
|
||
|
{
|
||
|
Set(v0, v1, v2, n, triFlags, t);
|
||
|
}
|
||
|
|
||
|
hsBounds3Tri::hsBounds3Tri(hsTriangle3* t)
|
||
|
{
|
||
|
Set(&t->fVert[0]->fVtx->fLocalPos,
|
||
|
&t->fVert[1]->fVtx->fLocalPos,
|
||
|
&t->fVert[2]->fVtx->fLocalPos,
|
||
|
&t->fNormal, t->fFlags, t);
|
||
|
}
|
||
|
|
||
|
hsBounds3Tri::~hsBounds3Tri()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
|
||
|
// Finds closest intersection vertex or triangle/center-line intersection
|
||
|
hsBool hsBounds3Tri::ISectCone(const hsPoint3& from, const hsPoint3& to, hsScalar cosThetaSq, hsBool ignoreFacing, hsPoint3& at, hsBool& backSide) const
|
||
|
{
|
||
|
hsScalar d0 = from.InnerProduct(fNormal);
|
||
|
hsScalar d1 = at.InnerProduct(fNormal);
|
||
|
hsScalar dt = fNormal.InnerProduct(fVerts[0]);
|
||
|
backSide = d0 < dt;
|
||
|
if( !ignoreFacing && backSide )
|
||
|
return false;
|
||
|
if ( (d0 < dt || d1 < dt) &&
|
||
|
(d0 > dt || d1 > dt) &&
|
||
|
ClosestTriPoint(&from, &at, &hsVector3(&to,&from)) )
|
||
|
return true;
|
||
|
|
||
|
hsVector3 av(&to,&from);
|
||
|
hsScalar distASq = av.MagnitudeSquared();
|
||
|
hsScalar radiusSq = distASq * (1-cosThetaSq)/cosThetaSq;
|
||
|
|
||
|
hsScalar minDistSq = 0;
|
||
|
Int32 minVert = 0;
|
||
|
hsBool sect = false;
|
||
|
for (Int32 i=0; i<3; i++)
|
||
|
{
|
||
|
hsPoint3 onLine;
|
||
|
hsScalar t = hsBounds3::ClosestPointToLine(&fVerts[i], &from, &to, &onLine);
|
||
|
|
||
|
// outside the cap of the cylinder
|
||
|
if (t<0 || t>1)
|
||
|
continue;
|
||
|
|
||
|
// outside the edge of the cylinder
|
||
|
if (hsVector3(&onLine, &fVerts[i]).MagnitudeSquared() >= radiusSq)
|
||
|
continue;
|
||
|
|
||
|
hsVector3 bv(&fVerts[i],&from);
|
||
|
|
||
|
hsScalar distBSq = bv.MagnitudeSquared();
|
||
|
|
||
|
hsScalar cosMuSquared = (av * bv) / (distASq * distBSq);
|
||
|
|
||
|
// outside the angle of the cone
|
||
|
if (cosMuSquared > cosThetaSq)
|
||
|
continue;
|
||
|
|
||
|
if (!sect || distBSq < minDistSq)
|
||
|
{
|
||
|
minVert = i;
|
||
|
minDistSq = distBSq;
|
||
|
sect = true;
|
||
|
}
|
||
|
}
|
||
|
at = fVerts[minVert];
|
||
|
return sect;
|
||
|
}
|
||
|
#endif // Commenting out this which will be made redundant and/or obsolete by Havok integration
|