2004-06-24 10:03:59 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* VCGLib o o *
|
|
|
|
* Visual and Computer Graphics Library o o *
|
|
|
|
* _ O _ *
|
|
|
|
* Copyright(C) 2004 \/)\/ *
|
|
|
|
* Visual Computing Lab /\/| *
|
|
|
|
* ISTI - Italian National Research Council | *
|
|
|
|
* \ *
|
|
|
|
* All rights reserved. *
|
|
|
|
* *
|
|
|
|
* 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 2 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 (http://www.gnu.org/licenses/gpl.txt) *
|
|
|
|
* for more details. *
|
|
|
|
* *
|
|
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
2005-10-11 18:03:40 +02:00
|
|
|
History
|
2004-06-24 10:03:59 +02:00
|
|
|
|
|
|
|
$Log: not supported by cvs2svn $
|
2005-12-21 14:09:03 +01:00
|
|
|
Revision 1.24 2005/12/19 15:13:06 corsini
|
|
|
|
Fix IsOrientedMesh
|
|
|
|
|
2005-12-19 16:13:06 +01:00
|
|
|
Revision 1.23 2005/12/16 13:13:44 cignoni
|
|
|
|
Reimplemented SelfIntersection
|
|
|
|
|
2005-12-16 14:13:44 +01:00
|
|
|
Revision 1.22 2005/12/16 10:54:59 corsini
|
|
|
|
Reimplement isOrientedMesh
|
|
|
|
|
2005-12-16 11:54:59 +01:00
|
|
|
Revision 1.21 2005/12/16 10:53:39 corsini
|
|
|
|
Take account for deletion in isComplexManifold
|
|
|
|
|
2005-12-16 11:53:39 +01:00
|
|
|
Revision 1.20 2005/12/16 10:51:43 corsini
|
|
|
|
Take account for deletion in isRegularMesh
|
|
|
|
|
2005-12-16 11:51:43 +01:00
|
|
|
Revision 1.19 2005/12/15 13:53:13 corsini
|
|
|
|
Reimplement isComplexManifold
|
|
|
|
Reimplement isRegular
|
|
|
|
|
2005-12-15 14:53:13 +01:00
|
|
|
Revision 1.18 2005/12/14 14:04:35 corsini
|
|
|
|
Fix genus computation
|
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
Revision 1.17 2005/12/12 12:11:40 cignoni
|
|
|
|
Removed unuseful detectunreferenced
|
|
|
|
|
2005-12-12 13:11:40 +01:00
|
|
|
Revision 1.16 2005/12/04 00:25:00 cignoni
|
|
|
|
Changed DegeneratedFaces -> RemoveZeroAreaFaces
|
|
|
|
|
2005-12-04 01:25:00 +01:00
|
|
|
Revision 1.15 2005/12/03 22:34:25 cignoni
|
|
|
|
Added missing include and sdt:: (tnx to Mario Latronico)
|
|
|
|
|
2005-12-03 23:34:25 +01:00
|
|
|
Revision 1.14 2005/12/02 00:14:43 cignoni
|
|
|
|
Removed some pointer vs iterator issues that prevented gcc compilation
|
|
|
|
|
2005-12-02 01:14:43 +01:00
|
|
|
Revision 1.13 2005/11/22 14:04:10 rita_borgo
|
|
|
|
Completed and tested self-intersection routine
|
|
|
|
|
2005-11-22 15:04:10 +01:00
|
|
|
Revision 1.12 2005/11/17 00:41:07 cignoni
|
|
|
|
Removed Initialize use updateflags::Clear() instead.
|
|
|
|
|
2005-11-17 01:41:07 +01:00
|
|
|
Revision 1.11 2005/11/16 16:33:23 rita_borgo
|
|
|
|
Changed ComputeSelfintersection
|
|
|
|
|
2005-11-16 17:33:23 +01:00
|
|
|
Revision 1.10 2005/11/15 12:16:34 rita_borgo
|
|
|
|
Changed DegeneratedFaces, sets the D flags for each faces
|
|
|
|
that is found to be degenerated.
|
|
|
|
CounEdges and ConnectedComponents check now if a face IsD()
|
|
|
|
else for degenerated faces many asserts fail.
|
|
|
|
|
2005-11-15 13:16:34 +01:00
|
|
|
Revision 1.9 2005/11/14 09:28:18 cignoni
|
|
|
|
changed access to face functions (border, area)
|
|
|
|
removed some typecast warnings
|
|
|
|
|
2005-11-14 10:28:18 +01:00
|
|
|
Revision 1.8 2005/10/11 16:03:40 rita_borgo
|
|
|
|
Added new functions belonging to triMeshInfo
|
|
|
|
Started the Self-Intersection routine
|
|
|
|
|
2005-10-11 18:03:40 +02:00
|
|
|
Revision 1.7 2005/10/03 15:57:53 rita_borgo
|
|
|
|
Alligned with TriMeshInfo Code
|
|
|
|
|
2005-10-03 17:57:53 +02:00
|
|
|
Revision 1.6 2005/01/28 11:59:35 cignoni
|
|
|
|
Add std:: to stl containers
|
|
|
|
|
2005-01-28 12:59:35 +01:00
|
|
|
Revision 1.5 2004/09/20 08:37:57 cignoni
|
|
|
|
Better Doxygen docs
|
|
|
|
|
2004-09-20 10:37:57 +02:00
|
|
|
Revision 1.4 2004/08/25 15:15:26 ganovelli
|
|
|
|
minor changes to comply gcc compiler (typename's and stuff)
|
|
|
|
|
2004-08-25 17:15:27 +02:00
|
|
|
Revision 1.3 2004/07/18 06:55:37 cignoni
|
|
|
|
NewUserBit -> NewBitFlag
|
|
|
|
|
2004-07-18 08:55:37 +02:00
|
|
|
Revision 1.2 2004/07/09 15:48:37 tarini
|
|
|
|
Added an include (<algorithm>)
|
|
|
|
|
2004-07-09 17:48:37 +02:00
|
|
|
Revision 1.1 2004/06/24 08:03:59 cignoni
|
|
|
|
Initial Release
|
|
|
|
|
2004-06-24 10:03:59 +02:00
|
|
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#ifndef __VCGLIB_CLEAN
|
|
|
|
#define __VCGLIB_CLEAN
|
|
|
|
|
2005-12-15 14:53:13 +01:00
|
|
|
// Standard headers
|
2004-06-24 10:03:59 +02:00
|
|
|
#include <map>
|
2004-07-09 17:48:37 +02:00
|
|
|
#include <algorithm>
|
2005-12-03 23:34:25 +01:00
|
|
|
#include <stack>
|
2004-07-09 17:48:37 +02:00
|
|
|
|
2005-12-15 14:53:13 +01:00
|
|
|
// VCG headers
|
2005-10-11 18:03:40 +02:00
|
|
|
#include <vcg/simplex/face/face.h>
|
2005-12-15 14:53:13 +01:00
|
|
|
#include <vcg/simplex/face/pos.h>
|
|
|
|
#include <vcg/simplex/face/topology.h>
|
2005-10-11 18:03:40 +02:00
|
|
|
#include <vcg/complex/trimesh/base.h>
|
|
|
|
#include <vcg/complex/trimesh/closest.h>
|
|
|
|
#include <vcg/space/index/grid_static_ptr.h>
|
2004-06-24 10:03:59 +02:00
|
|
|
#include<vcg/complex/trimesh/allocate.h>
|
2005-10-11 18:03:40 +02:00
|
|
|
|
|
|
|
|
2004-06-24 10:03:59 +02:00
|
|
|
namespace vcg {
|
2005-10-11 18:03:40 +02:00
|
|
|
namespace tri{
|
|
|
|
///
|
|
|
|
/** \addtogroup trimesh */
|
|
|
|
/*@{*/
|
|
|
|
/// Class of static functions to clean/correct/restore meshs.
|
|
|
|
template <class CleanMeshType>
|
|
|
|
class Clean
|
2004-06-24 10:03:59 +02:00
|
|
|
{
|
2005-10-11 18:03:40 +02:00
|
|
|
|
|
|
|
public:
|
|
|
|
typedef CleanMeshType MeshType;
|
|
|
|
typedef typename MeshType::VertexType VertexType;
|
|
|
|
typedef typename MeshType::VertexPointer VertexPointer;
|
|
|
|
typedef typename MeshType::VertexIterator VertexIterator;
|
|
|
|
typedef typename MeshType::ScalarType ScalarType;
|
|
|
|
typedef typename MeshType::FaceType FaceType;
|
|
|
|
typedef typename MeshType::FacePointer FacePointer;
|
|
|
|
typedef typename MeshType::FaceIterator FaceIterator;
|
|
|
|
typedef typename MeshType::FaceContainer FaceContainer;
|
|
|
|
|
2005-12-02 01:14:43 +01:00
|
|
|
typedef GridStaticPtr<FaceType, ScalarType > TriMeshGrid;
|
|
|
|
typedef Point3<ScalarType> Point3x;
|
2005-10-11 18:03:40 +02:00
|
|
|
|
|
|
|
TriMeshGrid gM;
|
|
|
|
FaceIterator fi;
|
|
|
|
FaceIterator gi;
|
|
|
|
vcg::face::Pos<FaceType> he;
|
|
|
|
vcg::face::Pos<FaceType> hei;
|
|
|
|
|
|
|
|
/* classe di confronto per l'algoritmo di eliminazione vertici duplicati*/
|
|
|
|
class RemoveDuplicateVert_Compare{
|
|
|
|
public:
|
2005-12-02 01:14:43 +01:00
|
|
|
inline bool operator()(VertexPointer const &a, VertexPointer const &b)
|
2004-06-24 10:03:59 +02:00
|
|
|
{
|
2005-11-14 10:28:18 +01:00
|
|
|
return (*a).cP() < (*b).cP();
|
2004-06-24 10:03:59 +02:00
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/** This function removes all duplicate vertices of the mesh by looking only at their spatial positions.
|
|
|
|
Note that it does not update any topology relation that could be affected by this like the VT or TT relation.
|
|
|
|
the reason this function is usually performed BEFORE building any topology information.
|
|
|
|
*/
|
|
|
|
static int RemoveDuplicateVertex( MeshType & m ) // V1.0
|
|
|
|
{
|
|
|
|
if(m.vert.size()==0 || m.vn==0) return 0;
|
|
|
|
|
|
|
|
std::map<VertexPointer, VertexPointer> mp;
|
|
|
|
int i,j;
|
|
|
|
VertexIterator vi;
|
|
|
|
int deleted=0;
|
|
|
|
int k=0;
|
2005-11-14 10:28:18 +01:00
|
|
|
size_t num_vert = m.vert.size();
|
2005-10-11 18:03:40 +02:00
|
|
|
std::vector<VertexPointer> perm(num_vert);
|
|
|
|
for(vi=m.vert.begin(); vi!=m.vert.end(); ++vi, ++k)
|
|
|
|
perm[k] = &(*vi);
|
|
|
|
|
2005-11-14 10:28:18 +01:00
|
|
|
RemoveDuplicateVert_Compare c_obj;
|
2005-10-11 18:03:40 +02:00
|
|
|
|
|
|
|
std::sort(perm.begin(),perm.end(),c_obj);
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
i = j;
|
|
|
|
mp[perm[i]] = perm[j];
|
|
|
|
++i;
|
|
|
|
for(;i!=num_vert;)
|
|
|
|
{
|
|
|
|
if( (! (*perm[i]).IsD()) &&
|
|
|
|
(! (*perm[j]).IsD()) &&
|
|
|
|
(*perm[i]).P() == (*perm[j]).cP() )
|
|
|
|
{
|
|
|
|
VertexPointer t = perm[i];
|
|
|
|
mp[perm[i]] = perm[j];
|
|
|
|
++i;
|
|
|
|
(*t).SetD();
|
|
|
|
deleted++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
j = i;
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
FaceIterator fi;
|
|
|
|
for(fi = m.face.begin(); fi!=m.face.end(); ++fi)
|
|
|
|
for(k = 0; k < 3; ++k)
|
|
|
|
if( !(*fi).IsD() )
|
|
|
|
if( mp.find( (typename MeshType::VertexPointer)(*fi).V(k) ) != mp.end() )
|
|
|
|
{
|
|
|
|
(*fi).V(k) = &*mp[ (*fi).V(k) ];
|
|
|
|
}
|
|
|
|
m.vn -= deleted;
|
|
|
|
return deleted;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** This function removes that are not referenced by any face. The function updates the vn counter.
|
|
|
|
@param m The mesh
|
|
|
|
@return The number of removed vertices
|
|
|
|
*/
|
2005-12-12 13:11:40 +01:00
|
|
|
static int RemoveUnreferencedVertex( MeshType& m, bool DeleteVertexFlag=true) // V1.0
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
|
|
|
FaceIterator fi;
|
|
|
|
VertexIterator vi;
|
|
|
|
int referredBit = VertexType::NewBitFlag();
|
|
|
|
|
|
|
|
int j;
|
|
|
|
int deleted = 0;
|
|
|
|
|
|
|
|
for(vi=m.vert.begin();vi!=m.vert.end();++vi)
|
|
|
|
(*vi).ClearUserBit(referredBit);
|
|
|
|
|
|
|
|
for(fi=m.face.begin();fi!=m.face.end();++fi)
|
|
|
|
if( !(*fi).IsD() )
|
|
|
|
for(j=0;j<3;++j)
|
|
|
|
(*fi).V(j)->SetUserBit(referredBit);
|
|
|
|
|
|
|
|
for(vi=m.vert.begin();vi!=m.vert.end();++vi)
|
|
|
|
if( (!(*vi).IsD()) && (!(*vi).IsUserBit(referredBit)))
|
|
|
|
{
|
2005-12-12 13:11:40 +01:00
|
|
|
if(DeleteVertexFlag)
|
|
|
|
{
|
|
|
|
(*vi).SetD();
|
|
|
|
--m.vn;
|
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
++deleted;
|
|
|
|
}
|
|
|
|
VertexType::DeleteBitFlag(referredBit);
|
|
|
|
return deleted;
|
|
|
|
}
|
|
|
|
|
2005-12-04 01:25:00 +01:00
|
|
|
static int RemoveZeroAreaFace(MeshType& m, ScalarType epsilon=0)
|
|
|
|
{
|
|
|
|
FaceIterator fi;
|
|
|
|
int count_fd = 0;
|
|
|
|
|
|
|
|
for(fi=m.face.begin(); fi!=m.face.end();++fi)
|
|
|
|
if(Area<FaceType>(*fi) <= epsilon)
|
|
|
|
{
|
|
|
|
count_fd++;
|
|
|
|
fi->SetD();
|
|
|
|
m.fn--;
|
|
|
|
}
|
|
|
|
return count_fd;
|
|
|
|
}
|
2005-12-15 14:53:13 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the mesh is a manifold.
|
|
|
|
*
|
|
|
|
* First of all, for each face the FF condition is checked.
|
|
|
|
* Then, a second test is performed: for each vertex the
|
|
|
|
* number of face found have to be the same of the number of
|
|
|
|
* face found with the VF walk trough.
|
|
|
|
*/
|
2005-10-11 18:03:40 +02:00
|
|
|
static bool IsComplexManifold( MeshType & m )
|
|
|
|
{
|
2005-12-15 14:53:13 +01:00
|
|
|
bool flagManifold = true;
|
|
|
|
|
|
|
|
VertexIterator vi;
|
2005-10-11 18:03:40 +02:00
|
|
|
FaceIterator fi;
|
2005-12-15 14:53:13 +01:00
|
|
|
|
|
|
|
// First Test
|
|
|
|
assert(m.HasFFTopology());
|
|
|
|
for (fi = m.face.begin(); fi != m.face.end(); ++fi)
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-16 11:53:39 +01:00
|
|
|
if (!fi->IsD())
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-16 11:53:39 +01:00
|
|
|
if ((!IsManifold(*fi,0))||
|
|
|
|
(!IsManifold(*fi,1))||
|
|
|
|
(!IsManifold(*fi,2)))
|
|
|
|
{
|
|
|
|
flagManifold = false;
|
|
|
|
break;
|
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
2005-12-15 14:53:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Second Test
|
|
|
|
if (flagManifold)
|
|
|
|
{
|
|
|
|
assert(m.HasVFTopology());
|
|
|
|
|
|
|
|
face::VFIterator<FaceType> vfi;
|
|
|
|
int starSizeFF;
|
|
|
|
int starSizeVF;
|
|
|
|
for (vi = m.vert.begin(); vi != m.vert.end(); ++vi)
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-16 11:53:39 +01:00
|
|
|
if (!vi->IsD())
|
|
|
|
{
|
|
|
|
face::VFIterator<FaceType> vfi(&*vi);
|
|
|
|
face::Pos<FaceType> pos((*vi).VFp(), &*vi);
|
2005-12-15 14:53:13 +01:00
|
|
|
|
2005-12-16 11:53:39 +01:00
|
|
|
starSizeFF = pos.StarSize();
|
2005-12-15 14:53:13 +01:00
|
|
|
|
2005-12-16 11:53:39 +01:00
|
|
|
starSizeVF = 0;
|
|
|
|
while(!vfi.End())
|
|
|
|
{
|
|
|
|
++vfi;
|
|
|
|
starSizeVF++;
|
|
|
|
}
|
2005-12-15 14:53:13 +01:00
|
|
|
|
2005-12-16 11:53:39 +01:00
|
|
|
if (starSizeFF != starSizeVF)
|
|
|
|
{
|
|
|
|
flagManifold = false;
|
|
|
|
break;
|
|
|
|
}
|
2005-12-15 14:53:13 +01:00
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
|
|
|
}
|
2005-12-15 14:53:13 +01:00
|
|
|
|
|
|
|
return flagManifold;
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void CountEdges( MeshType & m, int &count_e, int &boundary_e )
|
|
|
|
{
|
|
|
|
FaceIterator fi;
|
|
|
|
vcg::face::Pos<FaceType> he;
|
|
|
|
vcg::face::Pos<FaceType> hei;
|
|
|
|
bool counted =false;
|
|
|
|
for(fi=m.face.begin();fi!=m.face.end();fi++)
|
|
|
|
{
|
2005-11-15 13:16:34 +01:00
|
|
|
if(!((*fi).IsD()))
|
|
|
|
{
|
2005-10-11 18:03:40 +02:00
|
|
|
(*fi).SetS();
|
|
|
|
count_e +=3; //assume that we have to increase the number of edges with three
|
|
|
|
for(int j=0; j<3; j++)
|
|
|
|
{
|
2005-11-14 10:28:18 +01:00
|
|
|
if (face::IsBorder(*fi,j)) //If this edge is a border edge
|
2005-10-11 18:03:40 +02:00
|
|
|
boundary_e++; // then increase the number of boundary edges
|
|
|
|
else if (IsManifold(*fi,j))//If this edge is manifold
|
|
|
|
{
|
|
|
|
if((*fi).FFp(j)->IsS()) //If the face on the other side of the edge is already selected
|
|
|
|
count_e--; // we counted one edge twice
|
|
|
|
}
|
|
|
|
else//We have a non-manifold edge
|
|
|
|
{
|
|
|
|
hei.Set(&(*fi), j , fi->V(j));
|
|
|
|
he=hei;
|
|
|
|
he.NextF();
|
|
|
|
while (he.f!=hei.f)// so we have to iterate all faces that are connected to this edge
|
|
|
|
{
|
|
|
|
if (he.f->IsS())// if one of the other faces was already visited than this edge was counted already.
|
|
|
|
{
|
|
|
|
counted=true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
he.NextF();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (counted)
|
|
|
|
{
|
|
|
|
count_e--;
|
|
|
|
counted=false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-11-15 13:16:34 +01:00
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int CountHoles( MeshType & m)
|
|
|
|
{
|
|
|
|
int numholes=0;
|
|
|
|
int numholev=0;
|
|
|
|
int BEdges=0;
|
|
|
|
FaceIterator fi;
|
|
|
|
FaceIterator gi;
|
|
|
|
vcg::face::Pos<FaceType> he;
|
|
|
|
vcg::face::Pos<FaceType> hei;
|
|
|
|
|
|
|
|
vector<vector<Point3x> > holes; //indices of vertices
|
|
|
|
|
|
|
|
for(fi=m.face.begin();fi!=m.face.end();++fi)
|
|
|
|
(*fi).ClearS();
|
|
|
|
gi=m.face.begin(); fi=gi;
|
|
|
|
|
|
|
|
for(fi=m.face.begin();fi!=m.face.end();fi++)//for all faces do
|
|
|
|
{
|
|
|
|
for(int j=0;j<3;j++)//for all edges
|
|
|
|
{
|
|
|
|
if(fi->V(j)->IsS()) continue;
|
|
|
|
|
2005-11-14 10:28:18 +01:00
|
|
|
if(face::IsBorder(*fi,j))//found an unvisited border edge
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
|
|
|
he.Set(&(*fi),j,fi->V(j)); //set the face-face iterator to the current face, edge and vertex
|
|
|
|
vector<Point3x> hole; //start of a new hole
|
|
|
|
hole.push_back(fi->P(j)); // including the first vertex
|
|
|
|
numholev++;
|
|
|
|
he.v->SetS(); //set the current vertex as selected
|
|
|
|
he.NextB(); //go to the next boundary edge
|
|
|
|
|
|
|
|
|
|
|
|
while(fi->V(j) != he.v)//will we do not encounter the first boundary edge.
|
|
|
|
{
|
|
|
|
Point3x newpoint = he.v->P(); //select its vertex.
|
|
|
|
if(he.v->IsS())//check if this vertex was selected already, because then we have an additional hole.
|
|
|
|
{
|
|
|
|
//cut and paste the additional hole.
|
|
|
|
vector<Point3x> hole2;
|
|
|
|
int index = find(hole.begin(),hole.end(),newpoint) - hole.begin();
|
2005-11-14 10:28:18 +01:00
|
|
|
for(unsigned int i=index; i<hole.size(); i++)
|
2005-10-11 18:03:40 +02:00
|
|
|
hole2.push_back(hole[i]);
|
|
|
|
|
|
|
|
hole.resize(index);
|
|
|
|
if(hole2.size()!=0) //annoying in degenerate cases
|
|
|
|
holes.push_back(hole2);
|
|
|
|
}
|
|
|
|
hole.push_back(newpoint);
|
|
|
|
numholev++;
|
|
|
|
he.v->SetS(); //set the current vertex as selected
|
|
|
|
he.NextB(); //go to the next boundary edge
|
|
|
|
}
|
|
|
|
holes.push_back(hole);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-12-15 14:53:13 +01:00
|
|
|
return static_cast<int>(holes.size());
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int BorderEdges( MeshType & m, int numholes)
|
|
|
|
{
|
|
|
|
int BEdges = 0;
|
|
|
|
for(int i=0; i<numholes; i++)
|
|
|
|
{
|
|
|
|
if(i==numholes-1){ printf("%i)\n",numholes); BEdges++;}
|
|
|
|
else{ printf("%i, ",numholes); BEdges++;}
|
|
|
|
}
|
|
|
|
return BEdges;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ConnectedComponents(MeshType &m)
|
|
|
|
{
|
|
|
|
FaceIterator fi;
|
2005-12-02 01:14:43 +01:00
|
|
|
//FaceIterator gi;
|
2005-10-11 18:03:40 +02:00
|
|
|
vcg::face::Pos<FaceType> he;
|
|
|
|
vcg::face::Pos<FaceType> hei;
|
|
|
|
|
|
|
|
vector<int> nrfaces;
|
|
|
|
nrfaces.reserve(1);
|
2004-06-24 10:03:59 +02:00
|
|
|
|
2005-10-11 18:03:40 +02:00
|
|
|
for(fi=m.face.begin();fi!=m.face.end();++fi)
|
|
|
|
(*fi).ClearS();
|
2005-12-02 01:14:43 +01:00
|
|
|
//gi=m.face.begin(); fi=gi;
|
2005-10-11 18:03:40 +02:00
|
|
|
int Compindex=0;
|
2005-12-03 23:34:25 +01:00
|
|
|
std::stack<FacePointer> sf;
|
2005-12-02 01:14:43 +01:00
|
|
|
FacePointer gi=&*(m.face.begin());
|
|
|
|
FaceType *l;
|
2005-10-11 18:03:40 +02:00
|
|
|
for(fi=m.face.begin();fi!=m.face.end();++fi)
|
|
|
|
{
|
2005-11-15 13:16:34 +01:00
|
|
|
if(!((*fi).IsD()))
|
|
|
|
{
|
2005-10-11 18:03:40 +02:00
|
|
|
if (!(*fi).IsS())
|
|
|
|
{
|
|
|
|
(*fi).SetS();
|
|
|
|
//(*fi).Q()=Compindex;
|
|
|
|
nrfaces.push_back(1);
|
2005-12-02 01:14:43 +01:00
|
|
|
sf.push(&*fi);
|
2005-10-11 18:03:40 +02:00
|
|
|
while (!sf.empty())
|
|
|
|
{
|
|
|
|
gi=sf.top();
|
|
|
|
he.Set(&(*gi),0,gi->V(0));
|
|
|
|
sf.pop();
|
|
|
|
for(int j=0;j<3;++j)
|
|
|
|
{
|
2005-11-14 10:28:18 +01:00
|
|
|
if( !face::IsBorder(*gi,j) )
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
|
|
|
l=he.f->FFp(j);
|
|
|
|
if( !(*l).IsS() )
|
|
|
|
{
|
|
|
|
(*l).SetS();
|
|
|
|
sf.push(l);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Compindex++;
|
|
|
|
}
|
2005-11-15 13:16:34 +01:00
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
|
|
|
return Compindex;
|
|
|
|
}
|
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
GENUS.
|
|
|
|
|
|
|
|
A topologically invariant property of a surface defined as
|
|
|
|
the largest number of non-intersecting simple closed curves that can be
|
|
|
|
drawn on the surface without separating it.
|
|
|
|
|
2005-11-14 10:28:18 +01:00
|
|
|
Roughly speaking, it is the number of holes in a surface.
|
2005-12-14 15:04:35 +01:00
|
|
|
The genus g of a closed surface, also called the geometric genus, is related to the
|
|
|
|
Euler characteristic by the relation $chi$ by $chi==2-2g$.
|
2005-11-14 10:28:18 +01:00
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
The genus of a connected, orientable surface is an integer representing the maximum
|
|
|
|
number of cuttings along closed simple curves without rendering the resultant
|
|
|
|
manifold disconnected. It is equal to the number of handles on it.
|
2005-11-14 10:28:18 +01:00
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
For general polyhedra the <em>Euler Formula</em> is:
|
2005-11-14 10:28:18 +01:00
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
V + F - E = 2 - 2G - B
|
2005-11-14 10:28:18 +01:00
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
where V is the number of vertices, F is the number of faces, E is the
|
|
|
|
number of edges, G is the genus and B is the number of <em>boundary polygons</em>.
|
2005-11-14 10:28:18 +01:00
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
The above formula is valid for a mesh with one single connected component.
|
|
|
|
By considering multiple connected components the formula becomes:
|
2005-11-14 10:28:18 +01:00
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
V + F - E = 2C - 2Gs - B
|
2005-11-14 10:28:18 +01:00
|
|
|
|
2005-12-14 15:04:35 +01:00
|
|
|
where C is the number of connected components and Gs is the sum of
|
|
|
|
the genus of all connected components.
|
|
|
|
|
|
|
|
*/
|
2005-12-21 14:09:03 +01:00
|
|
|
static int MeshGenus(MeshType &m, int numholes, int numcomponents, int count_e)
|
2005-12-14 15:04:35 +01:00
|
|
|
{
|
2005-12-21 14:09:03 +01:00
|
|
|
int V = m.vn;
|
2005-12-14 15:04:35 +01:00
|
|
|
int F = m.fn;
|
|
|
|
int E = count_e;
|
|
|
|
return -((V + F - E + numholes - 2 * numcomponents) / 2);
|
|
|
|
}
|
2004-06-24 10:03:59 +02:00
|
|
|
|
2005-12-15 14:53:13 +01:00
|
|
|
/**
|
|
|
|
* Check if the given mesh is regular, semi-regular or irregular.
|
|
|
|
*
|
|
|
|
* Each vertex of a \em regular mesh has valence 6 except for border vertices
|
|
|
|
* which have valence 4.
|
|
|
|
*
|
|
|
|
* A \em semi-regular mesh is derived from an irregular one applying
|
|
|
|
* 1-to-4 subdivision recursively. (not checked for now)
|
|
|
|
*
|
|
|
|
* All other meshes are \em irregular.
|
|
|
|
*/
|
|
|
|
static void IsRegularMesh(MeshType &m, bool &Regular, bool &Semiregular)
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-15 14:53:13 +01:00
|
|
|
// This algorithm requires Vertex-Face topology
|
|
|
|
assert(m.HasVFTopology());
|
|
|
|
|
|
|
|
Regular = true;
|
|
|
|
|
|
|
|
VertexIterator vi;
|
|
|
|
|
|
|
|
// for each vertex the number of edges are count
|
|
|
|
for (vi = m.vert.begin(); vi != m.vert.end(); ++vi)
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-16 11:51:43 +01:00
|
|
|
if (!vi->IsD())
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-16 11:51:43 +01:00
|
|
|
face::Pos<FaceType> he((*vi).VFp(), &*vi);
|
|
|
|
face::Pos<FaceType> ht = he;
|
2005-10-11 18:03:40 +02:00
|
|
|
|
2005-12-16 11:51:43 +01:00
|
|
|
int n=0;
|
|
|
|
bool border=false;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
++n;
|
|
|
|
ht.NextE();
|
|
|
|
if (ht.IsBorder())
|
|
|
|
border=true;
|
|
|
|
}
|
|
|
|
while (ht != he);
|
2005-10-11 18:03:40 +02:00
|
|
|
|
2005-12-16 11:51:43 +01:00
|
|
|
if (border)
|
|
|
|
n = n/2;
|
|
|
|
|
|
|
|
if ((n != 6)&&(!border && n != 4))
|
|
|
|
{
|
|
|
|
Regular = false;
|
|
|
|
break;
|
|
|
|
}
|
2005-12-15 14:53:13 +01:00
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
2005-10-03 17:57:53 +02:00
|
|
|
|
2005-12-15 14:53:13 +01:00
|
|
|
if (!Regular)
|
|
|
|
Semiregular = false;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// For now we do not account for semi-regularity
|
|
|
|
Semiregular = false;
|
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
2005-10-03 17:57:53 +02:00
|
|
|
|
2005-12-16 11:54:59 +01:00
|
|
|
static void IsOrientedMesh(MeshType &m, bool &Oriented, bool &Orientable)
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-19 16:13:06 +01:00
|
|
|
// This algorithms requires FF topology
|
|
|
|
assert(m.HasFFTopology());
|
2005-12-16 11:54:59 +01:00
|
|
|
|
|
|
|
Orientable = true;
|
|
|
|
Oriented = true;
|
|
|
|
|
2005-12-19 16:13:06 +01:00
|
|
|
// Ensure that each face is deselected
|
|
|
|
FaceIterator fi;
|
|
|
|
for (fi = m.face.begin(); fi != m.face.end(); ++fi)
|
|
|
|
fi->ClearS();
|
|
|
|
|
|
|
|
// initialize stack
|
|
|
|
stack<FacePointer> faces;
|
|
|
|
|
|
|
|
// for each face of the mesh
|
|
|
|
FacePointer fp,fpaux;
|
|
|
|
int iaux;
|
2005-12-16 11:54:59 +01:00
|
|
|
for (fi = m.face.begin(); fi != m.face.end(); ++fi)
|
|
|
|
{
|
2005-12-19 16:13:06 +01:00
|
|
|
if (!fi->IsD() && !fi->IsS())
|
2005-12-16 11:54:59 +01:00
|
|
|
{
|
2005-12-19 16:13:06 +01:00
|
|
|
// each face put in the stack is selected (and oriented)
|
|
|
|
fi->SetS();
|
|
|
|
faces.push(&(*fi));
|
|
|
|
|
|
|
|
// empty the stack
|
|
|
|
while (!faces.empty())
|
|
|
|
{
|
|
|
|
fp = faces.top();
|
|
|
|
faces.pop();
|
|
|
|
|
|
|
|
// make consistently oriented the adjacent faces
|
|
|
|
for (int j = 0; j < 3; j++)
|
2005-12-16 11:54:59 +01:00
|
|
|
{
|
2005-12-19 16:13:06 +01:00
|
|
|
// get one of the adjacent face
|
|
|
|
fpaux = fp->FFp(j);
|
|
|
|
iaux = fp->FFi(j);
|
2005-12-16 11:54:59 +01:00
|
|
|
|
2005-12-19 16:13:06 +01:00
|
|
|
if (!fpaux->IsD() && fpaux != fp)
|
2005-12-16 11:54:59 +01:00
|
|
|
{
|
2005-12-19 16:13:06 +01:00
|
|
|
if (!CheckOrientation(*fpaux, iaux))
|
|
|
|
{
|
|
|
|
Oriented = false;
|
|
|
|
|
|
|
|
if (!fpaux->IsS())
|
|
|
|
{
|
|
|
|
SwapEdge(*fpaux, iaux);
|
|
|
|
assert(CheckOrientation(*fpaux, iaux));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
Orientable = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// put the oriented face into the stack
|
2005-12-16 11:54:59 +01:00
|
|
|
|
2005-12-19 16:13:06 +01:00
|
|
|
if (!fpaux->IsS())
|
|
|
|
{
|
|
|
|
fpaux->SetS();
|
|
|
|
faces.push(fpaux);
|
|
|
|
}
|
|
|
|
}
|
2005-12-16 11:54:59 +01:00
|
|
|
}
|
2005-12-19 16:13:06 +01:00
|
|
|
}
|
2005-12-16 11:54:59 +01:00
|
|
|
}
|
2005-12-19 16:13:06 +01:00
|
|
|
|
|
|
|
if (!Orientable)
|
|
|
|
break;
|
2005-12-16 11:54:59 +01:00
|
|
|
}
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
|
|
|
|
2005-12-16 14:13:44 +01:00
|
|
|
static bool SelfIntersections(MeshType &m, std::vector<FaceType*> &ret)
|
2005-10-11 18:03:40 +02:00
|
|
|
{
|
2005-12-16 14:13:44 +01:00
|
|
|
assert(FaceType::HasMark()); // Needed by the UG
|
|
|
|
|
2005-10-11 18:03:40 +02:00
|
|
|
Box3< ScalarType> bbox;
|
|
|
|
TriMeshGrid gM;
|
2005-12-16 14:13:44 +01:00
|
|
|
ret.clear();
|
2005-10-11 18:03:40 +02:00
|
|
|
FaceIterator fi;
|
2005-12-16 14:13:44 +01:00
|
|
|
int referredBit = FaceType::NewBitFlag();
|
2005-10-11 18:03:40 +02:00
|
|
|
|
2005-12-16 14:13:44 +01:00
|
|
|
int deleted = 0;
|
2005-10-11 18:03:40 +02:00
|
|
|
|
2005-12-16 14:13:44 +01:00
|
|
|
for(fi=m.face.begin();fi!=m.face.end();++fi)
|
|
|
|
(*fi).ClearUserBit(referredBit);
|
|
|
|
|
2005-11-15 13:16:34 +01:00
|
|
|
std::vector<FaceType*> inBox;
|
2005-12-02 01:14:43 +01:00
|
|
|
gM.Set(m.face.begin(),m.face.end());
|
2005-11-22 15:04:10 +01:00
|
|
|
|
2005-10-11 18:03:40 +02:00
|
|
|
for(fi=m.face.begin();fi!=m.face.end();++fi)
|
|
|
|
{
|
2005-12-16 14:13:44 +01:00
|
|
|
(*fi).SetUserBit(referredBit);
|
2005-11-22 15:04:10 +01:00
|
|
|
(*fi).GetBBox(bbox);
|
|
|
|
vcg::trimesh::GetInBoxFace(m, gM, bbox,inBox);
|
2005-12-16 14:13:44 +01:00
|
|
|
bool Intersected=false;
|
|
|
|
std::vector<FaceType*>::iterator fib;
|
|
|
|
for(fib=inBox.begin();fib!=inBox.end();++fib)
|
|
|
|
{
|
|
|
|
if(!(*fib)->IsUserBit(referredBit) && (*fib != &*fi) )
|
|
|
|
if(TestIntersection(&*fi,*fib)){
|
|
|
|
ret.push_back(*fib);
|
|
|
|
if(!Intersected) {
|
|
|
|
ret.push_back(&*fi);
|
|
|
|
Intersected=true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-11-22 15:04:10 +01:00
|
|
|
inBox.clear();
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
2005-12-16 14:13:44 +01:00
|
|
|
|
|
|
|
FaceType::DeleteBitFlag(referredBit);
|
2005-11-22 15:04:10 +01:00
|
|
|
return (ret.size()>0);
|
2005-10-11 18:03:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//test real intersection between faces
|
|
|
|
static bool TestIntersection(FaceType *f0,FaceType *f1)
|
|
|
|
{
|
|
|
|
assert((!f0->IsD())&&(!f1->IsD()));
|
|
|
|
//no adiacent faces
|
2005-11-22 15:04:10 +01:00
|
|
|
if ( (f0!=f1) && (!ShareEdge(f0,f1))
|
|
|
|
&& (!ShareVertex(f0,f1)) )
|
2005-10-11 18:03:40 +02:00
|
|
|
return (vcg::Intersection<FaceType>((*f0),(*f1)));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//control if two faces share an edge
|
|
|
|
static bool ShareEdge(FaceType *f0,FaceType *f1)
|
|
|
|
{
|
|
|
|
assert((!f0->IsD())&&(!f1->IsD()));
|
|
|
|
for (int i=0;i<3;i++)
|
|
|
|
if (f0->FFp(i)==f1)
|
|
|
|
return (true);
|
|
|
|
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
//control if two faces share a vertex
|
|
|
|
static bool ShareVertex(FaceType *f0,FaceType *f1)
|
|
|
|
{
|
|
|
|
assert((!f0->IsD())&&(!f1->IsD()));
|
|
|
|
for (int i=0;i<3;i++)
|
|
|
|
for (int j=0;j<3;j++)
|
|
|
|
if (f0->V(i)==f1->V(j))
|
|
|
|
return (true);
|
|
|
|
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}; // end class
|
|
|
|
/*@}*/
|
|
|
|
|
|
|
|
} //End Namespace Tri
|
2004-06-24 10:03:59 +02:00
|
|
|
} // End Namespace vcg
|
|
|
|
#endif
|