Recompiled from previous out of date version. Still to revise but working

This commit is contained in:
ganovelli 2008-01-12 19:07:05 +00:00
parent 491317ecd5
commit c919dca603
1 changed files with 225 additions and 172 deletions

View File

@ -19,6 +19,9 @@
/*#************************************************************************** /*#**************************************************************************
History History
$Log: not supported by cvs2svn $ $Log: not supported by cvs2svn $
Revision 1.5 2005/12/13 17:17:19 ganovelli
first importing from old version. NOT optimized! It works with VertexFace Adjacency even over non manifolds
*#**************************************************************************/ *#**************************************************************************/
@ -40,92 +43,106 @@ class Geo{
public: public:
typedef typename MeshType::VertexPointer VertexPointer; typedef typename MeshType::VertexPointer VertexPointer;
typedef typename MeshType::ScalarType ScalarType;
/* Auxiliary class for keeping the heap of vertices to visit and their estimated distance
*/
struct VertDist{
VertDist(){}
VertDist(VertexPointer _v, ScalarType _d):v(_v),d(_d){}
VertexPointer v;
ScalarType d;
};
/* Temporary data to associate to all the vertices: estimated distance and boolean flag
*/
template <class MeshType> template <class MeshType>
struct TempData{ struct TempData{
TempData(){} TempData(){}
TempData(const double & d_){d=d_;visited=false;} TempData(const ScalarType & d_){d=d_;visited=false;}
double d; ScalarType d;
bool visited; bool visited;
}; };
typedef SimpleTempData<std::vector<typename MeshType::VertexType>, TempData<MeshType> > TempDataType; typedef SimpleTempData<std::vector<typename MeshType::VertexType>, TempData<MeshType> > TempDataType;
static TempDataType & TD(){ static TempDataType td; return td;} TempDataType * TD;
struct pred: public std::binary_function<VertexPointer,VertexPointer,bool>{ struct pred: public std::binary_function<VertDist,VertDist,bool>{
bool operator()(const VertexPointer& v0, const VertexPointer& v1) const pred(){};
{return (Geo<MeshType>::TD()[v0].d > Geo<MeshType>::TD()[v1].d);} bool operator()(const VertDist& v0, const VertDist& v1) const
{return (v0.d > v1.d);}
}; };
/*
static typename MeshType::VertexPointer BuildSP( starting from the seeds, it assign a distance value to each vertex. The distance of a vertex is its
approximated geodesic distance to the closest seeds.
This is function is not meant to be called (although is not prevented). Instead, it is invoked by
wrapping function.
*/
typename MeshType::VertexPointer Visit(
MeshType & m, MeshType & m,
std::vector<typename MeshType::VertexPointer> & _frontier, std::vector<VertDist> & _frontier,
double & max_distance, ScalarType & max_distance,
bool fartestOnBorder = false bool fartestOnBorder = false
) )
{ {
TD().c = &m.vert; bool isLeaf,toQueue;
std::vector<VertDist> frontier;
bool isLeaf;
std::vector<typename MeshType::VertexPointer> frontier;
std::vector<typename MeshType::VertexPointer> :: iterator tmp;
frontier.clear();
//Requirements
assert(m.HasVFTopology());
if(m.vn==0) return NULL;
MeshType::VertexIterator ii; MeshType::VertexIterator ii;
std::list<typename MeshType::VertexPointer> children; std::list<typename MeshType::VertexPointer> children;
typename MeshType::VertexPointer curr,fartest,pw1; typename MeshType::VertexPointer curr,fartest,pw1;
bool toQueue;
TD().Start(TempData<MeshType>(-1.0));
std::list<typename MeshType::VertexPointer>::iterator is; std::list<typename MeshType::VertexPointer>::iterator is;
std::deque<typename MeshType::VertexPointer> leaves; std::deque<typename MeshType::VertexPointer> leaves;
std::vector <std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType> > expansion; std::vector <std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType> > expansion;
std::vector <VertDist >::iterator ifr;
std::vector <typename MeshType::VertexPointer >::const_iterator ifr;
face::VFIterator<typename MeshType::FaceType> x;int k; face::VFIterator<typename MeshType::FaceType> x;int k;
typename MeshType::VertexPointer pw; typename MeshType::VertexPointer pw;
//Requirements
assert(m.HasVFTopology());
assert(!_frontier.empty());
TD = new TempDataType(m.vert);
TD->Start(TempData<MeshType>(-1.0));
for(ifr = _frontier.begin(); ifr != _frontier.end(); ++ifr){ for(ifr = _frontier.begin(); ifr != _frontier.end(); ++ifr){
TD()[*ifr].visited= true; (*TD)[(*ifr).v].visited= true;
TD()[*ifr].d = 0.0; (*TD)[(*ifr).v].d = 0.0;
(*ifr).d = 0.0;
} }
for(ifr = _frontier.begin(); ifr != _frontier.end(); ++ifr) for(ifr = _frontier.begin(); ifr != _frontier.end(); ++ifr)
{ {
// determina la distanza dei vertici della fan // determina la distanza dei vertici della fan
for( x.f = (*ifr)->VFp(), x.z = (*ifr)->VFi(); x.f!=0; ++x ) for( x.f = (*ifr).v->VFp(), x.z = (*ifr).v->VFi(); x.f!=0; ++x )
for(k=0;k<2;++k) for(k=0;k<2;++k)
{ {
if(k==0) pw = x.f->V1(x.z); if(k==0) pw = x.f->V1(x.z);
else pw = x.f->V2(x.z); else pw = x.f->V2(x.z);
if(TD()[pw].d ==-1){ if((*TD)[pw].d ==-1){
TD()[pw].d = Distance(pw->cP(),(*ifr)->cP()); (*TD)[pw].d = Distance(pw->cP(),(*ifr).v->cP());
frontier.push_back(pw); frontier.push_back(VertDist(pw,(*TD)[pw].d));
} }
} }
} }
// initialize Heap // initialize Heap
make_heap(frontier.begin(),frontier.end(),pred()); make_heap(frontier.begin(),frontier.end(),pred());
double curr_d,d_curr = 0.0; ScalarType curr_d,d_curr = 0.0;
max_distance=0.0; max_distance=0.0;
std::vector<typename MeshType::VertexPointer >:: iterator iv; std::vector<VertDist >:: iterator iv;
while(!frontier.empty()) while(!frontier.empty())
{ //printf("size: %d\n", frontier.size()); { //printf("size: %d\n", frontier.size());
expansion.clear(); expansion.clear();
pop_heap(frontier.begin(),frontier.end(),pred()); pop_heap(frontier.begin(),frontier.end(),pred());
curr = frontier.back(); curr = (frontier.back()).v;
frontier.pop_back(); frontier.pop_back();
d_curr = TD()[curr].d; d_curr = (*TD)[curr].d;
TD()[curr].visited = true; (*TD)[curr].visited = true;
isLeaf = (!fartestOnBorder || curr->IsB()); isLeaf = (!fartestOnBorder || curr->IsB());
@ -144,18 +161,18 @@ static typename MeshType::VertexPointer BuildSP(
pw1=x.f->V1(x.z); pw1=x.f->V1(x.z);
} }
const double & d_pw1 = TD()[pw1].d; const ScalarType & d_pw1 = (*TD)[pw1].d;
if((!TD()[pw1].visited ) || d_curr == 0.0) if((! (*TD)[pw1].visited ) || d_curr == 0.0)
{ {
if(TD()[pw].d == -1){ if( (*TD)[pw].d == -1){
curr_d = TD()[curr].d + (pw->P()-curr->P()).Norm(); curr_d = (*TD)[curr].d + (pw->P()-curr->P()).Norm();
expansion.push_back(std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType>(pw,curr_d)); expansion.push_back(std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType>(pw,curr_d));
} }
continue; continue;
} }
assert( TD()[pw1].d != -1); assert( (*TD)[pw1].d != -1);
assert( (curr!=pw) && (pw!=pw1) && (pw1 != curr)); assert( (curr!=pw) && (pw!=pw1) && (pw1 != curr));
assert(d_pw1!=-1.0); assert(d_pw1!=-1.0);
@ -166,43 +183,43 @@ static typename MeshType::VertexPointer BuildSP(
Point3<MeshType::ScalarType> w_w1 = pw->cP()- pw1->cP(); Point3<MeshType::ScalarType> w_w1 = pw->cP()- pw1->cP();
Point3<MeshType::ScalarType> w1_c = pw1->cP()- curr->cP(); Point3<MeshType::ScalarType> w1_c = pw1->cP()- curr->cP();
double ew_c = (w_c).Norm(); ScalarType ew_c = (w_c).Norm();
double ew_w1 = (w_w1).Norm(); ScalarType ew_w1 = (w_w1).Norm();
double ec_w1 = (w1_c).Norm(); ScalarType ec_w1 = (w1_c).Norm();
double alpha,alpha_, beta,beta_,theta_c,theta,h,delta,s,a,b; ScalarType alpha,alpha_, beta,beta_,theta,h,delta,s,a,b;
alpha = acos((w_c*w1_c)/(ew_c*ec_w1)); alpha = acos((w_c*w1_c)/(ew_c*ec_w1));
s = (d_curr + d_pw1+ec_w1)/2; s = (d_curr + d_pw1+ec_w1)/2;
a = s/ec_w1; a = s/ec_w1;
b = a*s; b = a*s;
alpha_ = 2*acos ( math::Min(1.0,sqrt( (b- a* d_pw1)/d_curr))); alpha_ = 2*acos ( math::Min<ScalarType>(1.0,sqrt( (b- a* d_pw1)/d_curr)));
if ( alpha+alpha_ > M_PI){ if ( alpha+alpha_ > M_PI){
curr_d = d_curr + ew_c; curr_d = d_curr + ew_c;
}else }else
{ {
beta_ = 2*acos ( math::Min(1.0,sqrt( (b- a* d_curr)/d_pw1))); beta_ = 2*acos ( math::Min<ScalarType>(1.0,sqrt( (b- a* d_curr)/d_pw1)));
beta = acos((w_w1)*(-w1_c)/(ew_w1*ec_w1)); beta = acos((w_w1)*(-w1_c)/(ew_w1*ec_w1));
if ( beta+beta_ > M_PI) if ( beta+beta_ > M_PI)
curr_d = d_pw1 + ew_w1; curr_d = d_pw1 + ew_w1;
else else
{ {
theta = M_PI-alpha-alpha_; theta = ScalarType(M_PI)-alpha-alpha_;
delta = cos(theta)* ew_c; delta = cos(theta)* ew_c;
h = sin(theta)* ew_c; h = sin(theta)* ew_c;
curr_d = sqrt( pow(h,2)+ pow(d_curr + delta,2)); curr_d = sqrt( pow(h,2)+ pow(d_curr + delta,2));
} }
} }
//************************************************************************************** //**************************************************************************************
toQueue = (TD()[(pw)].d==-1); toQueue = ( (*TD)[(pw)].d==-1);
if(toQueue){// se non e'gia' in coda ce lo mette if(toQueue){// se non e'gia' in coda ce lo mette
expansion.push_back(std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType>(pw,curr_d)); expansion.push_back(std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType>(pw,curr_d));
}else }else
{ {
if( TD()[(pw)].d > curr_d ) if( (*TD)[(pw)].d > curr_d )
TD()[(pw)].d = curr_d; (*TD)[(pw)].d = curr_d;
} }
if(isLeaf){ if(isLeaf){
@ -217,19 +234,21 @@ static typename MeshType::VertexPointer BuildSP(
std::vector <std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType> > ::iterator i; std::vector <std::pair<typename MeshType::VertexPointer,typename MeshType::ScalarType> > ::iterator i;
for(i = expansion.begin(); i!= expansion.end(); ++i) for(i = expansion.begin(); i!= expansion.end(); ++i)
{ {
TD()[(*i).first].d = (*i).second; (*TD)[(*i).first].d = (*i).second;
frontier.push_back((*i).first); frontier.push_back(VertDist((*i).first,(*TD)[(*i).first].d));
push_heap(frontier.begin(),frontier.end(),pred()); push_heap(frontier.begin(),frontier.end(),pred());
} // end for } // end for
}// end while }// end while
// scrivi le distanze sul campo qualita' (nn: farlo parametrico) // scrivi le distanze sul campo qualita' (nn: farlo parametrico)
MeshType::VertexIterator vi; MeshType::VertexIterator vi;
for(vi = m.vert.begin(); vi != m.vert.end(); ++vi) for(vi = m.vert.begin(); vi != m.vert.end(); ++vi)
(*vi).Q() = TD()[&(*vi)].d; (*vi).Q() = (*TD)[&(*vi)].d;
TD().Stop(); (*TD).Stop();
delete TD;
return fartest; return fartest;
@ -237,26 +256,78 @@ static typename MeshType::VertexPointer BuildSP(
public: public:
static void FartestPoint( MeshType & m, /*
std::vector<typename MeshType::VertexPointer> & fro,//insieme di vertici da cui trovare le distanze Given a mesh and a vector of pointers to vertices (sources), assigns the approximated geodesic
typename MeshType::VertexPointer & fartest, //punto piu'lontano distance from the cloasest source to all the mesh vertices and returns the pointer to the fartest.
double & distance){ //distaza geodesica Note: update the field Q() of the vertices
fartest = BuildSP(m,fro,distance,false); */
void FartestPoint( MeshType & m,
std::vector<typename MeshType::VertexPointer> & fro,
typename MeshType::VertexPointer & fartest,
ScalarType & distance){
std::vector<typename MeshType::VertexPointer>::iterator fi;
std::vector<VertDist>fr;
for( fi = fro.begin(); fi != fro.end() ; ++fi)
fr.push_back(VertDist(*fi,-1));
fartest = Visit(m,fr,distance,false);
} }
static void FartestBPoint( /*
MeshType & m, Given a mesh and a pointers to a vertex-source (source), assigns the approximated geodesic
std::vector<typename MeshType::VertexPointer> & fro, //insieme di vertici da cui trovare le distanze distance from the vertex-source to all the mesh vertices and returns the pointer to the fartest
typename MeshType::VertexPointer & fartest, //punto piu'lontano Note: update the field Q() of the vertices
double & distance){ */
fartest = BuildSP(m,fro,distance,true); void FartestPoint( MeshType & m,
typename MeshType::VertexPointer seed,
typename MeshType::VertexPointer & fartest,
ScalarType & distance){
std::vector<typename MeshType::VertexPointer> fro;
fro.push_back( seed );
typename MeshType::VertexPointer v0;
FartestPoint(m,fro,v0,distance);
fartest = v0;
}
/*
Same as FartestPoint but the returned pointer is to a border vertex
Note: update the field Q() of the vertices
*/
void FartestBPoint(MeshType & m,
std::vector<typename MeshType::VertexPointer> & fro,
typename MeshType::VertexPointer & fartest,
ScalarType & distance){
std::vector<typename MeshType::VertexPointer>::iterator fi;
std::vector<VertDist>fr;
for( fi = fro.begin(); fi != fro.end() ; ++fi)
fr.push_back(VertDist(*fi,-1));
fartest = Visit(m,fr,distance,true);
}
/*
Same as FartestPoint but the returned pointer is to a border vertex
Note: update the field Q() of the vertices
*/
void FartestBPoint( MeshType & m,
typename MeshType::VertexPointer seed,
typename MeshType::VertexPointer & fartest,
ScalarType & distance){
std::vector<typename MeshType::VertexPointer> fro;
fro.push_back( seed );
typename MeshType::VertexPointer v0;
FartestBPoint(m,fro,v0,distance);
fartest = v0;
} }
static void DistanceFromBorder( MeshType & m, /*
typename MeshType::VertexPointer & v0, //ritorna il vertice piu'lontano da ogni punto sul bordo Assigns to each vertex of the mesh its distance to the closest vertex on the border
typename MeshType::VertexPointer & v1, // ritorna il vertice di bordo piu'vicino a v0 Note: update the field Q() of the vertices
double & distance // distanza geodesica tra v0 e v1 */
) void DistanceFromBorder( MeshType & m,
{ typename MeshType::VertexPointer & v0,
ScalarType & distance
){
std::vector<typename MeshType::VertexPointer> fro; std::vector<typename MeshType::VertexPointer> fro;
MeshType::VertexIterator vi; MeshType::VertexIterator vi;
MeshType::VertexPointer fartest; MeshType::VertexPointer fartest;
@ -264,25 +335,7 @@ static void DistanceFromBorder( MeshType & m,
if( (*vi).IsB()) if( (*vi).IsB())
fro.push_back(&(*vi)); fro.push_back(&(*vi));
FartestPoint(m,fro,fartest,distance); FartestPoint(m,fro,fartest,distance);
} }
static void MostInternal( MeshType & m, };
typename MeshType::VertexPointer & v0, //ritorna il vertice piu'lontano da ogni punto sul bordo
typename MeshType::VertexPointer & v1, // ritorna il vertice di bordo piu'vicino a v0
double & distance // distanza geodesica tra v0 e v1
)
{
std::vector<typename MeshType::VertexPointer> fro;
MeshType::VertexIterator vi;
MeshType::VertexPointer fartest;
for(vi = m.vert.begin(); vi != m.vert.end(); ++vi)
if( (*vi).IsB())
fro.push_back(&(*vi));
FartestPoint(m,fro,fartest,distance);
fro.clear();
fro.push_back(fartest);
FartestBPoint(m,fro,fartest,distance);
}
};
};// end namespace };// end namespace