// NeL - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "std3d.h"
#include "nel/3d/ray_mesh.h"
#include "nel/misc/vector_2f.h"
#include "nel/misc/fast_mem.h"
#include "nel/misc/plane.h"
#include "nel/3d/matrix_3x4.h"
using namespace NLMISC;
using namespace std;
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NL3D
{
// ***************************************************************************
// The number of byte to process per block
const uint NL_BlockByteL1= 4096;
// Number of vertices per block to process For vertices mul
uint CRayMesh::NumCacheVertex= NL_BlockByteL1 / sizeof(CVector);
// ***************************************************************************
template
static bool getRayIntersectionT(std::vector &vertices, const std::vector &tris,
float &dist2D, float &distZ, bool computeDist2D)
{
uint numTris= (uint)tris.size()/3;
if(!numTris)
return false;
// test all tris
const TIndex *pTri= &tris[0];
float minSkinDistZ= FLT_MAX;
for(uint i=0;i=0;
// Line p1-p2.
a= -p12.y;
b= p12.x;
c= (p1.x*a + p1.y*b);
allInf= allInf && c<=0;
allSup= allSup && c>=0;
// Line p2-p0.
a= -p20.y;
b= p20.x;
c= (p2.x*a + p2.y*b);
allInf= allInf && c<=0;
allSup= allSup && c>=0;
// all on same side (don't bother front or backfaces)?
if(allInf || allSup)
{
// => ray intersect. compute the intersection now.
// This code is called for a very small subset of faces, hence don't bother optim.
CPlane plane;
plane.make(p0,p1,p2);
// intersect with the ray. Since vertices are in ray basis, the Ray is (Null, K).
CVector hit= plane.intersect(CVector::Null, CVector::K);
// then dist of Null to the tri is just Z.
float distToTri= hit.z;
// avoid problems if the plane is // to the ray. take maximum proj dist with the 3 points
float minVertDist= min(p0.z, p1.z);
minVertDist= min(minVertDist, p2.z);
// distToTri cannot be less than minVertDist
distToTri= max(minVertDist, distToTri);
// NB: it is possible than distToTri<0 (face behind or clipped by camera). clamp then to 0
distToTri= max(distToTri, 0.f);
minSkinDistZ= min(minSkinDistZ, distToTri);
}
}
// don't intersect?
if(minSkinDistZ==FLT_MAX)
{
// get the nearest distance to the ray (do the compute only if no IT found at all => optim)
const TIndex *pTri= &tris[0];
float minSkinSqrDist2D= FLT_MAX;
// only if user want this feature
if(computeDist2D)
{
for(uint i=0;i value between 0 and p01sqdist (if in segment)
fSeg= -p0.x*p01.x -p0.y*p01.y;
if(fSeg>0 && fSeg0 && fSeg0 && fSeg &vertices, const std::vector &tris,
float &dist2D, float &distZ, bool computeDist2D)
{
return getRayIntersectionT(vertices, tris, dist2D, distZ, computeDist2D);
}
// ***************************************************************************
bool CRayMesh::getRayIntersection(std::vector &vertices, const std::vector &tris,
float &dist2D, float &distZ, bool computeDist2D)
{
return getRayIntersectionT(vertices, tris, dist2D, distZ, computeDist2D);
}
// ***************************************************************************
bool CRayMesh::fastIntersect(const NLMISC::CMatrix &worldMatrix, const NLMISC::CVector &p0, const NLMISC::CVector &dir, float &dist2D, float &distZ, bool computeDist2D) const
{
if(empty())
return false;
// *** Compute toRaySpace matrix
// The skinning must be done in final RaySpace.
CMatrix toRaySpace;
// compute the ray matrix
CVector dirn= dir;
if(dirn.isNull())
dirn= CVector::K;
dirn.normalize();
toRaySpace.setArbitraryRotK(dirn);
toRaySpace.setPos(p0);
// The skinning must be done in ray space: (RayMat-1)*worldMatrix;
toRaySpace.invert();
toRaySpace*= worldMatrix;
CMatrix3x4 fastMat;
fastMat.set(toRaySpace);
// *** Make all points in ray space
uint numVerts= (uint)Vertices.size();
const CVector *src= &Vertices[0];
// enlarge temp buffer
static std::vector meshInRaySpace;
if(Vertices.size()>meshInRaySpace.size())
meshInRaySpace.resize(Vertices.size());
CVector *dst= &meshInRaySpace[0];
// Then do the skin
for(;numVerts>0;)
{
// number of vertices to process for this block.
uint nBlockInf= min(NumCacheVertex, numVerts);
// next block.
numVerts-= nBlockInf;
// cache the data in L1 cache.
CFastMem::precache(src, nBlockInf * sizeof(CVector));
// for all InfluencedVertices only.
for(;nBlockInf>0;nBlockInf--, src++, dst++)
{
fastMat.mulSetPoint( *src, *dst );
}
}
// *** and get ray intersection
return getRayIntersection(meshInRaySpace, Triangles, dist2D, distZ, computeDist2D);
}
} // NL3D