Rev 595 | Blame | Compare with Previous | Last modification | View Log | RSS feed
/* ----------------------------------------------------------------------- *
* This file is part of GEL, http://www.imm.dtu.dk/GEL
* Copyright (C) the authors and DTU Informatics
* For license and list of authors, see ../../doc/intro.pdf
* ----------------------------------------------------------------------- */
#include "verification.h"
using namespace CGLA;
using namespace std;
namespace Geometry
{
float SqrDistance (const Vec3f& rkPoint,const Triangle& rkTri)
{
Vec3f kDiff = rkTri.get_v0() - rkPoint;
double fA00 = sqr_length(rkTri.get_edge(0));
double fA01 = dot(rkTri.get_edge(0),-rkTri.get_edge(2));
double fA11 = sqr_length(rkTri.get_edge(2));
double fB0 = dot(kDiff,rkTri.get_edge(0));
double fB1 = dot(kDiff,-rkTri.get_edge(2));
double fC = sqr_length(kDiff);
double fDet = fabs(fA00*fA11-fA01*fA01);
double fS = fA01*fB1-fA11*fB0;
double fT = fA01*fB0-fA00*fB1;
double fSqrDist;
if ( fS + fT <= fDet )
{
if ( fS < (double)0.0 )
{
if ( fT < (double)0.0 ) // region 4
{
if ( fB0 < (double)0.0 )
{
fT = (double)0.0;
if ( -fB0 >= fA00 )
{
fS = (double)1.0;
fSqrDist = fA00+((double)2.0)*fB0+fC;
}
else
{
fS = -fB0/fA00;
fSqrDist = fB0*fS+fC;
}
}
else
{
fS = (double)0.0;
if ( fB1 >= (double)0.0 )
{
fT = (double)0.0;
fSqrDist = fC;
}
else if ( -fB1 >= fA11 )
{
fT = (double)1.0;
fSqrDist = fA11+((double)2.0)*fB1+fC;
}
else
{
fT = -fB1/fA11;
fSqrDist = fB1*fT+fC;
}
}
}
else // region 3
{
fS = (double)0.0;
if ( fB1 >= (double)0.0 )
{
fT = (double)0.0;
fSqrDist = fC;
}
else if ( -fB1 >= fA11 )
{
fT = (double)1.0;
fSqrDist = fA11+((double)2.0)*fB1+fC;
}
else
{
fT = -fB1/fA11;
fSqrDist = fB1*fT+fC;
}
}
}
else if ( fT < (double)0.0 ) // region 5
{
fT = (double)0.0;
if ( fB0 >= (double)0.0 )
{
fS = (double)0.0;
fSqrDist = fC;
}
else if ( -fB0 >= fA00 )
{
fS = (double)1.0;
fSqrDist = fA00+((double)2.0)*fB0+fC;
}
else
{
fS = -fB0/fA00;
fSqrDist = fB0*fS+fC;
}
}
else // region 0
{
// minimum at interior point
double fInvDet = ((double)1.0)/fDet;
fS *= fInvDet;
fT *= fInvDet;
fSqrDist = fS*(fA00*fS+fA01*fT+((double)2.0)*fB0) +
fT*(fA01*fS+fA11*fT+((double)2.0)*fB1)+fC;
}
}
else
{
double fTmp0, fTmp1, fNumer, fDenom;
if ( fS < (double)0.0 ) // region 2
{
fTmp0 = fA01 + fB0;
fTmp1 = fA11 + fB1;
if ( fTmp1 > fTmp0 )
{
fNumer = fTmp1 - fTmp0;
fDenom = fA00-2.0f*fA01+fA11;
if ( fNumer >= fDenom )
{
fS = (double)1.0;
fT = (double)0.0;
fSqrDist = fA00+((double)2.0)*fB0+fC;
}
else
{
fS = fNumer/fDenom;
fT = (double)1.0 - fS;
fSqrDist = fS*(fA00*fS+fA01*fT+2.0f*fB0) +
fT*(fA01*fS+fA11*fT+((double)2.0)*fB1)+fC;
}
}
else
{
fS = (double)0.0;
if ( fTmp1 <= (double)0.0 )
{
fT = (double)1.0;
fSqrDist = fA11+((double)2.0)*fB1+fC;
}
else if ( fB1 >= (double)0.0 )
{
fT = (double)0.0;
fSqrDist = fC;
}
else
{
fT = -fB1/fA11;
fSqrDist = fB1*fT+fC;
}
}
}
else if ( fT < (double)0.0 ) // region 6
{
fTmp0 = fA01 + fB1;
fTmp1 = fA00 + fB0;
if ( fTmp1 > fTmp0 )
{
fNumer = fTmp1 - fTmp0;
fDenom = fA00-((double)2.0)*fA01+fA11;
if ( fNumer >= fDenom )
{
fT = (double)1.0;
fS = (double)0.0;
fSqrDist = fA11+((double)2.0)*fB1+fC;
}
else
{
fT = fNumer/fDenom;
fS = (double)1.0 - fT;
fSqrDist = fS*(fA00*fS+fA01*fT+((double)2.0)*fB0) +
fT*(fA01*fS+fA11*fT+((double)2.0)*fB1)+fC;
}
}
else
{
fT = (double)0.0;
if ( fTmp1 <= (double)0.0 )
{
fS = (double)1.0;
fSqrDist = fA00+((double)2.0)*fB0+fC;
}
else if ( fB0 >= (double)0.0 )
{
fS = (double)0.0;
fSqrDist = fC;
}
else
{
fS = -fB0/fA00;
fSqrDist = fB0*fS+fC;
}
}
}
else // region 1
{
fNumer = fA11 + fB1 - fA01 - fB0;
if ( fNumer <= (double)0.0 )
{
fS = (double)0.0;
fT = (double)1.0;
fSqrDist = fA11+((double)2.0)*fB1+fC;
}
else
{
fDenom = fA00-2.0f*fA01+fA11;
if ( fNumer >= fDenom )
{
fS = (double)1.0;
fT = (double)0.0;
fSqrDist = fA00+((double)2.0)*fB0+fC;
}
else
{
fS = fNumer/fDenom;
fT = (double)1.0 - fS;
fSqrDist = fS*(fA00*fS+fA01*fT+((double)2.0)*fB0) +
fT*(fA01*fS+fA11*fT+((double)2.0)*fB1)+fC;
}
}
}
}
return fabs(fSqrDist);
}
}
Generated by GNU Enscript 1.6.6.