515 lines
14 KiB
515 lines
14 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 "plPhysXCooking.h"
|
||
|
#include "hsGeometry3.h"
|
||
|
#include "../plPhysX/plSimulationMgr.h"
|
||
|
#include "../plPhysX/plPXStream.h"
|
||
|
#include "../plPhysX/plPXConvert.h"
|
||
|
#include "hsSTLStream.h"
|
||
|
|
||
|
#include "Nx.h"
|
||
|
#include "NxStream.h"
|
||
|
#include "NxPhysics.h"
|
||
|
#include "NxCooking.h"
|
||
|
#include "NxPlane.h"
|
||
|
#include "NxUtilLib.h"
|
||
|
#include "NxMat33.h"
|
||
|
bool plPhysXCooking::fSkipErrors = false;
|
||
|
|
||
|
NxUtilLib* plPhysXCooking::fUtilLib =nil;
|
||
|
//assumes that the Vectors are normalized
|
||
|
hsBool ThreePlaneIntersect(const NxVec3& norm0, const NxVec3& point0,
|
||
|
const NxVec3& norm1, const NxVec3& point1,
|
||
|
const NxVec3& norm2, const NxVec3& point2, NxVec3& loc)
|
||
|
{
|
||
|
//need to make sure these planes aren't parallel
|
||
|
hsBool suc=0;
|
||
|
NxVec3 cross=norm1.cross( norm2);
|
||
|
hsScalar denom=norm0.dot(cross);
|
||
|
if(abs(denom)<0.0001) return 0;//basically paralell
|
||
|
// if we are here there must be a point in 3 space
|
||
|
try{
|
||
|
hsScalar d1,d2,d3;
|
||
|
d1=norm0.dot(point0);
|
||
|
d2=norm1.dot(point1);
|
||
|
d3=norm2.dot(point2);
|
||
|
NxVec3 n1Xn2=norm1.cross(norm2);
|
||
|
NxVec3 n2Xn0=norm2.cross(norm0);
|
||
|
NxVec3 n0Xn1=norm0.cross(norm1);
|
||
|
NxVec3 pos=(d1*n1Xn2+ d2*n2Xn0 + d3*n0Xn1)/(denom);
|
||
|
loc.x=pos.x;
|
||
|
loc.y=pos.y;
|
||
|
loc.z=pos.z;
|
||
|
suc= 1;
|
||
|
}
|
||
|
catch(...)
|
||
|
{
|
||
|
suc=0;
|
||
|
}
|
||
|
|
||
|
return suc;
|
||
|
|
||
|
}
|
||
|
|
||
|
void plPhysXCooking::Init()
|
||
|
{
|
||
|
NxInitCooking();
|
||
|
NxUtilLib* fUtilLib=NxGetUtilLib();
|
||
|
NxCookingParams parms=NxGetCookingParams();
|
||
|
parms.skinWidth=.05;
|
||
|
NxSetCookingParams(parms);
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
void plPhysXCooking::Shutdown()
|
||
|
{
|
||
|
NxCloseCooking();
|
||
|
fUtilLib=nil;
|
||
|
fSkipErrors = false;
|
||
|
}
|
||
|
|
||
|
hsVectorStream* plPhysXCooking::CookTrimesh(int nVerts, hsPoint3* verts, int nFaces, UInt16* faces)
|
||
|
{
|
||
|
NxTriangleMeshDesc triDesc;
|
||
|
triDesc.numVertices = nVerts;
|
||
|
triDesc.pointStrideBytes = sizeof(hsPoint3);
|
||
|
triDesc.points = verts;
|
||
|
triDesc.numTriangles = nFaces;
|
||
|
triDesc.triangleStrideBytes = sizeof(UInt16) * 3;
|
||
|
triDesc.triangles = faces;
|
||
|
triDesc.flags = NX_MF_16_BIT_INDICES;
|
||
|
|
||
|
hsVectorStream* ram = TRACKED_NEW hsVectorStream;
|
||
|
plPXStream buf(ram);
|
||
|
bool status = NxCookTriangleMesh(triDesc, buf);
|
||
|
hsAssert(status, "Trimesh failed to cook");
|
||
|
|
||
|
if (status)
|
||
|
{
|
||
|
ram->Rewind();
|
||
|
return ram;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
delete ram;
|
||
|
return nil;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool plPhysXCooking::IsPointInsideHull(hsPlane3* hull, int nPlanes, const hsPoint3& pos)
|
||
|
{
|
||
|
int i;
|
||
|
for( i = 0; i < nPlanes; i++ )
|
||
|
{
|
||
|
// add a fudge to the point so not to trip on the ever so slightly concave
|
||
|
// ... so pull the point out in the direction of the normal of the plane we are testing.
|
||
|
hsPoint3 fudgepos = pos + (hull[i].GetNormal()*0.005f);
|
||
|
if (!ITestPlane(fudgepos, hull[i]))
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool plPhysXCooking::TestIfConvex(NxConvexMesh* convexMesh, int nVerts, hsPoint3* verts)
|
||
|
{
|
||
|
bool retVal = true;
|
||
|
|
||
|
// build planes from the convex mesh
|
||
|
NxConvexMeshDesc desc;
|
||
|
convexMesh->saveToDesc(desc);
|
||
|
|
||
|
hsPlane3* planes = TRACKED_NEW hsPlane3[desc.numTriangles];
|
||
|
|
||
|
int i;
|
||
|
for ( i = 0; i < desc.numTriangles; i++)
|
||
|
{
|
||
|
UInt32* triangle = (UInt32*)(((char*)desc.triangles) + desc.triangleStrideBytes*i);
|
||
|
float* vertex1 = (float*)(((char*)desc.points) + desc.pointStrideBytes*triangle[0]);
|
||
|
float* vertex2 = (float*)(((char*)desc.points) + desc.pointStrideBytes*triangle[1]);
|
||
|
float* vertex3 = (float*)(((char*)desc.points) + desc.pointStrideBytes*triangle[2]);
|
||
|
hsPoint3 pt1(vertex1[0],vertex1[1],vertex1[2]);
|
||
|
hsPoint3 pt2(vertex2[0],vertex2[1],vertex2[2]);
|
||
|
hsPoint3 pt3(vertex3[0],vertex3[1],vertex3[2]);
|
||
|
|
||
|
planes[i] = hsPlane3(&pt1,&pt2,&pt3);
|
||
|
}
|
||
|
// now see if any of the points from the mesh are inside the hull
|
||
|
for (int j=0; j<nVerts && retVal; j++)
|
||
|
{
|
||
|
if ( IsPointInsideHull(planes,desc.numTriangles,verts[j]) )
|
||
|
retVal = false;
|
||
|
}
|
||
|
|
||
|
delete [] planes;
|
||
|
|
||
|
return retVal;
|
||
|
}
|
||
|
|
||
|
|
||
|
hsVectorStream* plPhysXCooking::CookHull(int nVerts, hsPoint3* verts, bool inflate)
|
||
|
{
|
||
|
NxConvexMeshDesc convexDesc;
|
||
|
convexDesc.numVertices = nVerts;
|
||
|
convexDesc.pointStrideBytes = sizeof(hsPoint3);
|
||
|
convexDesc.points = verts;
|
||
|
convexDesc.flags = NX_CF_COMPUTE_CONVEX;
|
||
|
|
||
|
if(inflate)
|
||
|
{
|
||
|
|
||
|
convexDesc.flags|= NX_CF_INFLATE_CONVEX ;
|
||
|
}
|
||
|
hsVectorStream* ram = TRACKED_NEW hsVectorStream;
|
||
|
plPXStream buf(ram);
|
||
|
bool status = NxCookConvexMesh(convexDesc, buf);
|
||
|
hsAssert(status, "Convex mesh failed to cook");
|
||
|
|
||
|
if (status)
|
||
|
{
|
||
|
ram->Rewind();
|
||
|
return ram;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
delete ram;
|
||
|
return nil;
|
||
|
}
|
||
|
}
|
||
|
/*
|
||
|
NxTriangleMesh* ReadExplicit(hsStream* stream)
|
||
|
{
|
||
|
const int nVertices = stream->ReadSwap32();
|
||
|
hsPoint3* pVertices = TRACKED_NEW hsPoint3[nVertices];
|
||
|
stream->ReadSwapScalar(nVertices*3, (float*)pVertices);
|
||
|
|
||
|
const int nFaces = stream->ReadSwap32();
|
||
|
unsigned short* pTriangles = TRACKED_NEW unsigned short[nFaces * 3];
|
||
|
stream->ReadSwap16(nFaces * 3, pTriangles);
|
||
|
|
||
|
NxTriangleMeshDesc triDesc;
|
||
|
triDesc.numVertices = nVertices;
|
||
|
triDesc.pointStrideBytes = sizeof(hsPoint3);
|
||
|
triDesc.points = pVertices;
|
||
|
triDesc.numTriangles = nFaces;
|
||
|
triDesc.triangleStrideBytes = sizeof(UInt16) * 3;
|
||
|
triDesc.triangles = pTriangles;
|
||
|
triDesc.flags = NX_MF_16_BIT_INDICES;// | NX_MF_FLIPNORMALS;
|
||
|
|
||
|
hsRAMStream ram;
|
||
|
plNxStream buf(&ram);
|
||
|
NxInitCooking();
|
||
|
bool status = NxCookTriangleMesh(triDesc, buf);
|
||
|
hsAssert(status, "Trimesh failed to cook");
|
||
|
NxCloseCooking();
|
||
|
|
||
|
delete[] pVertices;
|
||
|
delete[] pTriangles;
|
||
|
|
||
|
if (status)
|
||
|
{
|
||
|
ram.Rewind();
|
||
|
return plSimulationMgr::GetInstance()->GetSDK()->createTriangleMesh(buf);
|
||
|
}
|
||
|
|
||
|
return nil;
|
||
|
}
|
||
|
|
||
|
NxConvexMesh* ReadConvexHull(hsStream* stream)
|
||
|
{
|
||
|
const int nVertices = stream->ReadSwap32();
|
||
|
hsPoint3* pVertices = TRACKED_NEW hsPoint3[nVertices];
|
||
|
stream->ReadSwapScalar(nVertices*3, (float*)pVertices);
|
||
|
|
||
|
NxConvexMeshDesc convexDesc;
|
||
|
convexDesc.numVertices = nVertices;
|
||
|
convexDesc.pointStrideBytes = sizeof(hsPoint3);
|
||
|
convexDesc.points = pVertices;
|
||
|
convexDesc.flags = NX_CF_COMPUTE_CONVEX;
|
||
|
|
||
|
hsRAMStream ram;
|
||
|
plNxStream buf(&ram);
|
||
|
NxInitCooking();
|
||
|
bool status = NxCookConvexMesh(convexDesc, buf);
|
||
|
hsAssert(status, "Convex mesh failed to cook");
|
||
|
NxCloseCooking();
|
||
|
|
||
|
delete[] pVertices;
|
||
|
|
||
|
if (status)
|
||
|
{
|
||
|
ram.Rewind();
|
||
|
return plSimulationMgr::GetInstance()->GetSDK()->createConvexMesh(buf);
|
||
|
}
|
||
|
|
||
|
return nil;
|
||
|
}
|
||
|
|
||
|
void ReadBoxFromHull(hsStream* stream, NxBoxShapeDesc& box)
|
||
|
{
|
||
|
const int nVertices = stream->ReadSwap32();
|
||
|
hsPoint3* pVertices = TRACKED_NEW hsPoint3[nVertices];
|
||
|
stream->ReadSwapScalar(nVertices*3, (float*)pVertices);
|
||
|
|
||
|
hsScalar minX, minY, minZ, maxX, maxY, maxZ;
|
||
|
minX = minY = minZ = FLT_MAX;
|
||
|
maxX = maxY = maxZ = -FLT_MAX;
|
||
|
for (int i = 0; i < nVertices; i++)
|
||
|
{
|
||
|
hsPoint3& vec = pVertices[i];
|
||
|
minX = hsMinimum(minX, vec.fX);
|
||
|
minY = hsMinimum(minY, vec.fY);
|
||
|
minZ = hsMinimum(minZ, vec.fZ);
|
||
|
maxX = hsMaximum(maxX, vec.fX);
|
||
|
maxY = hsMaximum(maxY, vec.fY);
|
||
|
maxZ = hsMaximum(maxZ, vec.fZ);
|
||
|
}
|
||
|
|
||
|
delete[] pVertices;
|
||
|
|
||
|
float xWidth = maxX - minX;
|
||
|
float yWidth = maxY - minY;
|
||
|
float zWidth = maxZ - minZ;
|
||
|
box.dimensions.x = xWidth / 2;
|
||
|
box.dimensions.y = yWidth / 2;
|
||
|
box.dimensions.z = zWidth / 2;
|
||
|
|
||
|
// hsMatrix44 mat;
|
||
|
// box.localPose.getRowMajor44(&mat.fMap[0][0]);
|
||
|
// hsPoint3 trans(minX + (xWidth / 2), minY + (yWidth / 2), minY + (yWidth / 2));
|
||
|
// mat.SetTranslate(&trans);
|
||
|
// box.localPose.setRowMajor44(&mat.fMap[0][0]);
|
||
|
}
|
||
|
*/
|
||
|
hsBool ProjectPointOnToPlane(const hsVector3& planeNormal,hsScalar& d0,
|
||
|
const hsVector3 pointToProject, hsPoint3& res)
|
||
|
{
|
||
|
|
||
|
NxVec3 vec=plPXConvert::Vector(planeNormal);
|
||
|
NxVec3 orig,projected;
|
||
|
orig=plPXConvert::Vector(pointToProject);
|
||
|
NxPlane* pl=new NxPlane(vec,d0);
|
||
|
projected=pl->project(orig);
|
||
|
res.fX=projected.x;
|
||
|
res.fY=projected.y;
|
||
|
res.fZ=projected.z;
|
||
|
return 1;
|
||
|
}
|
||
|
void plPhysXCooking::PCA(const NxVec3* points,int numPoints, NxMat33& out)
|
||
|
{
|
||
|
NxVec3 mean(0.f,0.f,0.f);
|
||
|
hsScalar Cov[3][3];
|
||
|
memset(Cov,0,9* sizeof hsScalar);
|
||
|
for(int i=0; i<numPoints;i++)
|
||
|
{
|
||
|
mean+=points[i];
|
||
|
}
|
||
|
mean=mean/(hsScalar)numPoints;
|
||
|
for(int i=0;i<numPoints;i++)
|
||
|
{
|
||
|
Cov[0][0]+=pow(points[i].x-mean.x ,2.0f)/(hsScalar)(numPoints);
|
||
|
Cov[1][1]+=pow(points[i].y-mean.y ,2.0f)/(hsScalar)(numPoints);
|
||
|
Cov[2][2]+=pow(points[i].z-mean.z ,2.0f)/(hsScalar)(numPoints);
|
||
|
Cov[0][1]+=(points[i].x-mean.x)*(points[i].y-mean.y)/(hsScalar)(numPoints);
|
||
|
Cov[0][2]+=(points[i].x-mean.x)*(points[i].z-mean.z)/(hsScalar)(numPoints);
|
||
|
Cov[1][2]+=(points[i].y-mean.y)*(points[i].z-mean.z)/(hsScalar)(numPoints);
|
||
|
}
|
||
|
Cov[2][0]=Cov[0][2];
|
||
|
Cov[1][0]=Cov[0][1];
|
||
|
Cov[2][1]=Cov[1][2];
|
||
|
NxF32 Covun[9];
|
||
|
for(int i=0;i<3;i++)
|
||
|
{
|
||
|
for(int j=0; j<3;j++)
|
||
|
{
|
||
|
Covun[3*i +j]=Cov[i][j];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
NxVec3 eigenVals;
|
||
|
NxMat33 CovNx,Rot;
|
||
|
CovNx.setRowMajor(Covun);
|
||
|
if(fUtilLib==nil)Init();
|
||
|
NxDiagonalizeInertiaTensor(CovNx,eigenVals,out);
|
||
|
|
||
|
|
||
|
|
||
|
}
|
||
|
hsVectorStream* plPhysXCooking::IMakePolytope(const plMaxMeshExtractor::NeutralMesh& inMesh)
|
||
|
{
|
||
|
hsBool success=0;
|
||
|
std::vector<hsPoint3> outCloud;
|
||
|
hsPoint3 offset;
|
||
|
int numPlanes=26;
|
||
|
float planeMax[26];
|
||
|
int indexMax[26];
|
||
|
hsPoint3 AABBMin(FLT_MAX,FLT_MAX,FLT_MAX);
|
||
|
hsPoint3 AABBMax(-FLT_MAX,-FLT_MAX,-FLT_MAX);
|
||
|
//prep
|
||
|
NxVec3* vectors = TRACKED_NEW NxVec3[26];
|
||
|
|
||
|
int curvec=0;
|
||
|
for(int xcomp= -1;xcomp<2;xcomp++)
|
||
|
{
|
||
|
for(int ycomp= -1;ycomp<2;ycomp++)
|
||
|
{
|
||
|
for(int zcomp= -1;zcomp<2;zcomp++)
|
||
|
{
|
||
|
if(!((xcomp==0)&&(ycomp==0)&&(zcomp==0)))
|
||
|
{
|
||
|
vectors[curvec].set((hsScalar)(xcomp),(hsScalar)(ycomp),(hsScalar)(zcomp));
|
||
|
vectors[curvec].normalize();
|
||
|
planeMax[curvec]=(-FLT_MAX);
|
||
|
//indexMax[curvec]=0;
|
||
|
curvec++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
/*
|
||
|
for(int i=0;i<26;i++)
|
||
|
{//make your max and mins
|
||
|
planeMax[i]=(-FLT_MAX);
|
||
|
}
|
||
|
*/
|
||
|
hsPoint3 centroid(0.0f,0.0f,0.0f);
|
||
|
for(int i=0;i<inMesh.fNumVerts;i++) centroid+=inMesh.fVerts[i];
|
||
|
centroid=centroid/(float)inMesh.fNumVerts;
|
||
|
//temp
|
||
|
NxVec3* nxLocs=new NxVec3[inMesh.fNumVerts];
|
||
|
NxVec3* nxLocs2=new NxVec3[inMesh.fNumVerts];
|
||
|
for(int i=0;i<inMesh.fNumVerts;i++)
|
||
|
{
|
||
|
hsPoint3 temppt=inMesh.fVerts[i] - centroid;
|
||
|
nxLocs[i]=plPXConvert::Point(temppt);
|
||
|
}
|
||
|
NxMat33 rot;
|
||
|
NxVec3 eigen;
|
||
|
PCA(nxLocs,inMesh.fNumVerts,rot);
|
||
|
NxMat33 invrot;
|
||
|
rot.getInverse(invrot);
|
||
|
for(int i=0; i<inMesh.fNumVerts;i++)
|
||
|
{
|
||
|
nxLocs2[i]=invrot*nxLocs[i];
|
||
|
}
|
||
|
for(int i=0;i<inMesh.fNumVerts;i++)
|
||
|
{
|
||
|
for(int plane=0;plane<26;plane++)
|
||
|
{
|
||
|
float dist=nxLocs2[i].dot(vectors[plane]);
|
||
|
if(dist>=planeMax[plane])
|
||
|
{
|
||
|
planeMax[plane]=dist;
|
||
|
indexMax[plane]=i;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
for(int i=0;i<inMesh.fNumVerts;i++)
|
||
|
{
|
||
|
AABBMin.fX = hsMinimum(nxLocs2[i].x, AABBMin.fX);
|
||
|
AABBMin.fY = hsMinimum(nxLocs2[i].y, AABBMin.fY);
|
||
|
AABBMin.fZ = hsMinimum(nxLocs2[i].z, AABBMin.fZ);
|
||
|
AABBMax.fX = hsMaximum(nxLocs2[i].x, AABBMax.fX);
|
||
|
AABBMax.fY = hsMaximum(nxLocs2[i].y, AABBMax.fY);
|
||
|
AABBMax.fZ = hsMaximum(nxLocs2[i].z, AABBMax.fZ);
|
||
|
}
|
||
|
|
||
|
int resultingPoints=0;
|
||
|
for(int i=0;i<26;i++)
|
||
|
{
|
||
|
for(int j=0;j<26;j++)
|
||
|
{
|
||
|
for(int k=0;k<26;k++)
|
||
|
{
|
||
|
NxVec3 res;
|
||
|
if(ThreePlaneIntersect(vectors[i],nxLocs2[indexMax[i]],vectors[j],nxLocs2[indexMax[j]], vectors[k],nxLocs2[indexMax[k]],res))
|
||
|
{
|
||
|
//check it is within all slabs
|
||
|
bool within=true;
|
||
|
int curplane=0;
|
||
|
do
|
||
|
{
|
||
|
float intersecdist=res.dot(vectors[curplane]);
|
||
|
if((intersecdist-planeMax[curplane])>0.0001)
|
||
|
|
||
|
{
|
||
|
within=false;
|
||
|
|
||
|
}
|
||
|
curplane++;
|
||
|
}
|
||
|
while((curplane<26)&&within);
|
||
|
if(within)
|
||
|
// if((res.x>=AABBMin.fX)&&(res.x<=AABBMax.fX)&&
|
||
|
// (res.y>=AABBMin.fY)&&(res.y<=AABBMax.fY)&&
|
||
|
// (res.z>=AABBMin.fZ)&&(res.z<=AABBMax.fZ))
|
||
|
{
|
||
|
NxVec3 reverted;
|
||
|
reverted=rot*res;
|
||
|
reverted.x=reverted.x +centroid.fX;
|
||
|
reverted.y=reverted.y +centroid.fY;
|
||
|
reverted.z=reverted.z +centroid.fZ;
|
||
|
hsPoint3 out;
|
||
|
out=plPXConvert::Point(reverted);
|
||
|
outCloud.push_back(out);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//planes discovered
|
||
|
//this is'nt right
|
||
|
//cleanup
|
||
|
offset=centroid;
|
||
|
|
||
|
delete[] vectors;
|
||
|
hsPoint3* pointages=TRACKED_NEW hsPoint3[outCloud.size()];
|
||
|
for(int x=0;x<outCloud.size();x++)pointages[x]=outCloud[x];
|
||
|
hsVectorStream* vectorstrm;
|
||
|
vectorstrm= CookHull(outCloud.size(),pointages,true);
|
||
|
delete[] pointages;
|
||
|
delete[] nxLocs;
|
||
|
delete[] nxLocs2;
|
||
|
return vectorstrm;
|
||
|
}
|