mirror of https://github.com/CGAL/cgal
171 lines
5.5 KiB
C++
171 lines
5.5 KiB
C++
#ifndef CGAL_PSURF_RINGS_H_
|
|
#define CGAL_PSURF_RINGS_H_
|
|
|
|
#include <cassert>
|
|
#include <vector>
|
|
#include <map>
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
//T_PolyhedralSurf_rings
|
|
//---------------------------------------------------------------------------
|
|
template < class TPoly >
|
|
class T_PolyhedralSurf_rings
|
|
{
|
|
|
|
const TPoly& P;
|
|
|
|
protected:
|
|
//Polyhedron
|
|
typedef typename boost::graph_traits<TPoly>::vertex_descriptor Vertex_const_handle;
|
|
typedef typename boost::graph_traits<TPoly>::halfedge_descriptor Halfedge_const_handle;
|
|
typedef typename boost::graph_traits<TPoly>::vertex_iterator Vertex_const_iterator;
|
|
typedef CGAL::Halfedge_around_target_circulator<TPoly> Halfedge_around_vertex_const_circulator;
|
|
|
|
typedef std::map<Vertex_const_handle, int> Vertex2int_map;
|
|
Vertex2int_map ring_index_map;
|
|
|
|
//vertex indices are initialised to -1
|
|
void reset_ring_indices(std::vector <Vertex_const_handle> &vces);
|
|
|
|
//i >= 1; from a start vertex on the current i-1 ring, push non-visited neighbors
|
|
//of start in the nextRing and set indices to i. Also add these vertices in all.
|
|
void push_neighbors_of(const Vertex_const_handle start, const int ith,
|
|
std::vector < Vertex_const_handle > &nextRing,
|
|
std::vector < Vertex_const_handle > &all);
|
|
|
|
//i >= 1, from a currentRing i-1, collect all neighbors, set indices
|
|
//to i and store them in nextRing and all.
|
|
void collect_ith_ring(const int ith,
|
|
std::vector < Vertex_const_handle > ¤tRing,
|
|
std::vector < Vertex_const_handle > &nextRing,
|
|
std::vector < Vertex_const_handle > &all);
|
|
|
|
public:
|
|
T_PolyhedralSurf_rings(const TPoly& P);
|
|
|
|
//collect i>=1 rings : all neighbors up to the ith ring,
|
|
void collect_i_rings(const Vertex_const_handle v,
|
|
const int ring_i,
|
|
std::vector < Vertex_const_handle >& all);
|
|
|
|
//collect enough rings (at least 1), to get at least min_nb of neighbors
|
|
void collect_enough_rings(const Vertex_const_handle v,
|
|
const unsigned int min_nb,
|
|
std::vector < Vertex_const_handle >& all);
|
|
};
|
|
|
|
////IMPLEMENTATION/////////////////////////////////////////////////////////////////////
|
|
|
|
template < class TPoly >
|
|
T_PolyhedralSurf_rings <TPoly>::
|
|
T_PolyhedralSurf_rings(const TPoly& P)
|
|
: P(P)
|
|
{
|
|
//init the ring_index_map
|
|
Vertex_const_iterator itb, ite;
|
|
boost::tie(itb,ite) = vertices(P);
|
|
for(;itb!=ite;itb++) ring_index_map[*itb] = -1;
|
|
}
|
|
|
|
template < class TPoly >
|
|
void T_PolyhedralSurf_rings <TPoly>::
|
|
push_neighbors_of(const Vertex_const_handle start, const int ith,
|
|
std::vector < Vertex_const_handle > &nextRing,
|
|
std::vector < Vertex_const_handle > &all)
|
|
{
|
|
Vertex_const_handle v;
|
|
Halfedge_around_vertex_const_circulator
|
|
hedgeb(halfedge(start,P),P), hedgee = hedgeb;
|
|
|
|
CGAL_For_all(hedgeb, hedgee)
|
|
{
|
|
v = target(opposite(*hedgeb,P),P);
|
|
if (ring_index_map[v] != -1) continue;//if visited: next
|
|
|
|
ring_index_map[v] = ith;
|
|
nextRing.push_back(v);
|
|
all.push_back(v);
|
|
}
|
|
}
|
|
|
|
template <class TPoly>
|
|
void T_PolyhedralSurf_rings <TPoly>::
|
|
collect_ith_ring(const int ith, std::vector < Vertex_const_handle > ¤tRing,
|
|
std::vector < Vertex_const_handle > &nextRing,
|
|
std::vector < Vertex_const_handle > &all)
|
|
{
|
|
typename std::vector < Vertex_const_handle >::const_iterator
|
|
itb = currentRing.begin(), ite = currentRing.end();
|
|
|
|
CGAL_For_all(itb, ite) push_neighbors_of(*itb, ith, nextRing, all);
|
|
}
|
|
|
|
template <class TPoly>
|
|
void T_PolyhedralSurf_rings <TPoly>::
|
|
reset_ring_indices(std::vector < Vertex_const_handle >&vces)
|
|
{
|
|
typename std::vector < Vertex_const_handle >::const_iterator
|
|
itb = vces.begin(), ite = vces.end();
|
|
CGAL_For_all(itb, ite) ring_index_map[*itb] = -1;
|
|
}
|
|
|
|
template <class TPoly>
|
|
void T_PolyhedralSurf_rings <TPoly>::
|
|
collect_i_rings(const Vertex_const_handle v,
|
|
const int ring_i,
|
|
std::vector < Vertex_const_handle >& all)
|
|
{
|
|
std::vector<Vertex_const_handle> current_ring, next_ring;
|
|
std::vector<Vertex_const_handle> *p_current_ring, *p_next_ring;
|
|
assert(ring_i >= 1);
|
|
//initialize
|
|
p_current_ring = ¤t_ring;
|
|
p_next_ring = &next_ring;
|
|
ring_index_map[v] = 0;
|
|
current_ring.push_back(v);
|
|
all.push_back(v);
|
|
|
|
for (int i=1; i<=ring_i; i++)
|
|
{
|
|
collect_ith_ring(i, *p_current_ring, *p_next_ring, all);
|
|
//next round must be launched from p_nextRing...
|
|
p_current_ring->clear();
|
|
std::swap(p_current_ring, p_next_ring);
|
|
}
|
|
//clean up
|
|
reset_ring_indices(all);
|
|
}
|
|
|
|
template <class TPoly>
|
|
void T_PolyhedralSurf_rings <TPoly>::
|
|
collect_enough_rings(const Vertex_const_handle v,
|
|
const unsigned int min_nb,
|
|
std::vector < Vertex_const_handle >& all)
|
|
{
|
|
std::vector<Vertex_const_handle> current_ring, next_ring;
|
|
std::vector<Vertex_const_handle> *p_current_ring, *p_next_ring;
|
|
|
|
//initialize
|
|
p_current_ring = ¤t_ring;
|
|
p_next_ring = &next_ring;
|
|
ring_index_map[v] = 0;
|
|
current_ring.push_back(v);
|
|
all.push_back(v);
|
|
|
|
int i = 1;
|
|
|
|
while ( (all.size() < min_nb) && (p_current_ring->size() != 0) )
|
|
{
|
|
collect_ith_ring(i, *p_current_ring, *p_next_ring, all);
|
|
//next round must be launched from p_nextRing...
|
|
p_current_ring->clear();
|
|
std::swap(p_current_ring, p_next_ring);
|
|
i++;
|
|
}
|
|
//clean up
|
|
reset_ring_indices(all);
|
|
}
|
|
|
|
#endif
|