dust3d/thirdparty/cgal/CGAL-5.1/include/CGAL/Umbilics.h

307 lines
11 KiB
C
Raw Normal View History

// Copyright (c) 2007 INRIA Sophia-Antipolis (France), INRIA Lorraine LORIA.
// All rights reserved.
//
// This file is part of CGAL (www.cgal.org).
//
2020-10-13 12:44:25 +00:00
// $URL: https://github.com/CGAL/cgal/blob/v5.1/Ridges_3/include/CGAL/Umbilics.h $
// $Id: Umbilics.h 8bb22d5 2020-03-26T14:23:37+01:00 Sébastien Loriot
// SPDX-License-Identifier: GPL-3.0-or-later OR LicenseRef-Commercial
//
// Author(s) : Marc Pouget and Frédéric Cazals
#ifndef CGAL_UMBILIC_H_
#define CGAL_UMBILIC_H_
#include <CGAL/license/Ridges_3.h>
#include <CGAL/basic.h>
#include <CGAL/boost/graph/helpers.h>
2020-10-13 12:44:25 +00:00
#include <CGAL/number_type_config.h>
#include <CGAL/PolyhedralSurf_neighbors.h>
#include <CGAL/Kernel/global_functions_3.h>
2020-10-13 12:44:25 +00:00
#include <boost/shared_ptr.hpp>
2020-10-13 12:44:25 +00:00
#include <list>
#include <vector>
#include <math.h>
namespace CGAL {
enum Umbilic_type { NON_GENERIC_UMBILIC = 0, ELLIPTIC_UMBILIC, HYPERBOLIC_UMBILIC};
//-------------------------------------------------------------------
//Umbilic : stores umbilic data, its location given by a vertex, its
//type and a circle of edges bording a disk containing the vertex
//------------------------------------------------------------------
template < class TriangleMesh >
class Umbilic
{
public:
typedef typename boost::graph_traits<TriangleMesh>::vertex_descriptor vertex_descriptor;
typedef typename boost::graph_traits<TriangleMesh>::halfedge_descriptor halfedge_descriptor;
2020-10-13 12:44:25 +00:00
//constructor
Umbilic(const TriangleMesh& P,
const vertex_descriptor v_init,
2020-10-13 12:44:25 +00:00
const std::list<halfedge_descriptor> contour_init);
//access fct
vertex_descriptor vertex() const { return v;}
Umbilic_type umbilic_type() const { return umb_type;}
Umbilic_type& umbilic_type() { return umb_type;}
const TriangleMesh& mesh()const { return P;}
const std::list<halfedge_descriptor>& contour_list() const { return contour;}
protected:
const TriangleMesh& P;
const vertex_descriptor v;
Umbilic_type umb_type;
const std::list<halfedge_descriptor> contour;
};
//constructor
template <class TriangleMesh>
Umbilic<TriangleMesh>::
Umbilic(const TriangleMesh& P,
const vertex_descriptor v_init,
2020-10-13 12:44:25 +00:00
const std::list<halfedge_descriptor> contour_init)
: P(P), v(v_init), contour(contour_init)
2020-10-13 12:44:25 +00:00
{}
template <class TriangleMesh>
2020-10-13 12:44:25 +00:00
std::ostream&
operator<<(std::ostream& out_stream, const Umbilic<TriangleMesh>& umbilic)
{
typedef typename boost::property_map<TriangleMesh,vertex_point_t>::const_type VPM;
2020-10-13 12:44:25 +00:00
VPM vpm = get(vertex_point, umbilic.mesh());
out_stream << "Umbilic at location (" << get(vpm, umbilic.vertex()) << ") of type ";
switch (umbilic.umbilic_type())
{
case CGAL::NON_GENERIC_UMBILIC: out_stream << "non generic" << std::endl; break;
case CGAL::ELLIPTIC_UMBILIC: out_stream << "elliptic" << std::endl; break;
case CGAL::HYPERBOLIC_UMBILIC: out_stream << "hyperbolic" << std::endl; break;
2020-10-13 12:44:25 +00:00
default : out_stream << "Something wrong occurred for sure..." << std::endl; break;
}
return out_stream;
}
//---------------------------------------------------------------------------
//Umbilic_approximation : enable computation of umbilics of a
//TriangleMesh. It uses the class
//T_PolyhedralSurf_neighbors to compute topological disk patches
//around vertices
//--------------------------------------------------------------------------
template < class TriangleMesh, class VertexFTMap, class VertexVectorMap >
class Umbilic_approximation
{
2020-10-13 12:44:25 +00:00
typedef typename boost::property_map<TriangleMesh,vertex_point_t>::const_type VPM;
typedef typename boost::property_traits<VPM>::value_type Point_3;
typedef typename Kernel_traits<Point_3>::Kernel Kernel;
public:
typedef typename Kernel::FT FT;
typedef typename Kernel::Vector_3 Vector_3;
typedef typename boost::graph_traits<TriangleMesh>::vertex_descriptor vertex_descriptor;
typedef typename boost::graph_traits<TriangleMesh>::halfedge_descriptor halfedge_descriptor;
typedef typename boost::graph_traits<TriangleMesh>::face_iterator Facet_const_iterator;
typedef typename boost::graph_traits<TriangleMesh>::vertex_iterator Vertex_const_iterator;
//requirements for the templates TriangleMesh and VertexFTMap or VertexVectorMap
CGAL_static_assertion((boost::is_same<vertex_descriptor, typename VertexFTMap::key_type>::value));
CGAL_static_assertion((boost::is_same<vertex_descriptor, typename VertexVectorMap::key_type>::value));
CGAL_static_assertion((boost::is_same<FT, typename VertexFTMap::value_type>::value));
CGAL_static_assertion((boost::is_same<Vector_3, typename VertexVectorMap::value_type>::value));
typedef CGAL::Umbilic<TriangleMesh> Umbilic;
//constructor : sets propertymaps and the poly_neighbors
2020-10-13 12:44:25 +00:00
Umbilic_approximation(const TriangleMesh& P,
const VertexFTMap& vertex2k1_pm,
const VertexFTMap& vertex2k2_pm,
const VertexVectorMap& vertex2d1_pm,
const VertexVectorMap& vertex2d2_pm);
//identify umbilics as vertices minimizing the function k1-k2 on
//their patch and for which the index is not 0. We avoid
//potential umbilics whose contours touch the border.
template <class OutputIterator>
OutputIterator compute(OutputIterator it, FT size);
protected:
const TriangleMesh& P;
2020-10-13 12:44:25 +00:00
typedef T_PolyhedralSurf_neighbors<TriangleMesh> Poly_neighbors;
boost::shared_ptr<Poly_neighbors> poly_neighbors;
CGAL::Abs<FT> cgal_abs;
CGAL::To_double<FT> To_double;
//Property maps
const VertexFTMap &k1, &k2;
const VertexVectorMap &d1, &d2;
// index: following CW the contour, we choose an orientation for the
// max dir of an arbitrary starting point, the max dir field is
// oriented on the next point so that the scalar product of the
// consecutive vectors is positive. Adding all the angles between
// consecutive vectors around the contour gives ~ -/+180 for a
// wedge/trisector, ~ 0 gives a false umbilic, everything else gives
// a non_generic umbilic.
int compute_type(Umbilic& umb);
};
template < class TriangleMesh, class VertexFTMap, class VertexVectorMap >
Umbilic_approximation< TriangleMesh, VertexFTMap, VertexVectorMap >::
2020-10-13 12:44:25 +00:00
Umbilic_approximation(const TriangleMesh& p,
const VertexFTMap& vertex2k1_pm,
const VertexFTMap& vertex2k2_pm,
const VertexVectorMap& vertex2d1_pm,
const VertexVectorMap& vertex2d2_pm)
: P(p), k1(vertex2k1_pm), k2(vertex2k2_pm),
d1(vertex2d1_pm), d2(vertex2d2_pm)
{
CGAL_precondition(is_triangle_mesh(P));
poly_neighbors = boost::shared_ptr<Poly_neighbors>(new Poly_neighbors(P));
}
template < class TriangleMesh, class VertexFTMap, class VertexVectorMap >
template <class OutputIterator>
OutputIterator Umbilic_approximation< TriangleMesh, VertexFTMap, VertexVectorMap >::
compute(OutputIterator umbilics_it, FT size)
{
CGAL_precondition( size >= 1 );
2020-10-13 12:44:25 +00:00
std::vector<vertex_descriptor> vces;
std::list<halfedge_descriptor> contour;
FT umbilicEstimatorVertex, umbilicEstimatorNeigh;
2020-10-13 12:44:25 +00:00
bool is_umbilic = true;
//MAIN loop on P vertices
Vertex_const_iterator itb, ite;
boost::tie(itb,ite) = vertices(P);
for (;itb != ite; itb++) {
vertex_descriptor vh = *itb;
umbilicEstimatorVertex = cgal_abs(get(k1,vh)-get(k2,vh));
//reset vector, list and bool
vces.clear();
contour.clear();
is_umbilic = true;
//the size of neighbourhood is (size * OneRingSize)
poly_neighbors->compute_neighbors(vh, vces, contour, size);
2020-10-13 12:44:25 +00:00
// avoid umbilics whose contours touch the border (Note may not be
// desirable?)
typename std::list<halfedge_descriptor>::const_iterator itb_cont = contour.begin(),
ite_cont = contour.end();
for (; itb_cont != ite_cont; itb_cont++)
if ( is_border(*itb_cont, P) ) {is_umbilic = false; continue;}
if (is_umbilic == false) continue;
2020-10-13 12:44:25 +00:00
//is v an umbilic?
2020-10-13 12:44:25 +00:00
//a priori is_umbilic = true, and it switches to false as soon as a
// neigh vertex has a lower umbilicEstimator value
typename std::vector<vertex_descriptor>::const_iterator itbv = vces.begin(),
itev = vces.end();
itbv++;
for (; itbv != itev; itbv++)
2020-10-13 12:44:25 +00:00
{ umbilicEstimatorNeigh = cgal_abs( get(k1,*itbv) - get(k2,*itbv) );
if ( umbilicEstimatorNeigh < umbilicEstimatorVertex )
{is_umbilic = false; break;}
}
if (is_umbilic == false) continue;
2020-10-13 12:44:25 +00:00
//v is an umbilic (wrt the min of k1-k2), compute the index. If
//the index is not 0 then we have actually an umbilic which is output
Umbilic* cur_umbilic = new Umbilic(P, vh, contour);
if (compute_type(*cur_umbilic) != 0) *umbilics_it++ = cur_umbilic;
}
return umbilics_it;
}
template < class TriangleMesh, class VertexFTMap, class VertexVectorMap >
int Umbilic_approximation< TriangleMesh, VertexFTMap, VertexVectorMap >::
compute_type(Umbilic& umb)
{
Vector_3 dir, dirnext, normal;
double cosinus, angle=0, angleSum=0;
vertex_descriptor v;
typename std::list<halfedge_descriptor>::const_iterator itb = umb.contour_list().begin(),
itlast = --umb.contour_list().end();
v = target(*itb, P);
dir = get(d1,v);
normal = CGAL::cross_product(get(d1,v), get(d2,v));
//sum angles along the contour
do{
itb++;
v = target(*itb, P);
dirnext = get(d1,v);
cosinus = To_double(dir*dirnext);
if (cosinus < 0) {dirnext = dirnext*(-1); cosinus *= -1;}
if (cosinus>1) cosinus = 1;
//orientation of (dir, dirnext, normal)
if ( (dir * CGAL::cross_product(dirnext, normal)) > 0) angle = acos(cosinus);
else angle = -acos(cosinus);
angleSum += angle;
dir = dirnext;
normal = CGAL::cross_product(get(d1,v), get(d2,v));
}
while (itb != (itlast));
2020-10-13 12:44:25 +00:00
//angle (v_last, v_0)
v = target(*umb.contour_list().begin(), P);
dirnext = get(d1,v);
cosinus = To_double(dir*dirnext);
if (cosinus < 0) {dirnext = dirnext*(-1); cosinus *= -1;}
if (cosinus>1) cosinus = 1;
if ( (dir * CGAL::cross_product(dirnext, normal)) > 0) angle = acos(cosinus);
else angle = -acos(cosinus);
angleSum += angle;
2020-10-13 12:44:25 +00:00
if ((angleSum > (CGAL_PI/2)) && (angleSum < (3*CGAL_PI/2))) umb.umbilic_type() = HYPERBOLIC_UMBILIC ;
else if ((angleSum < (-CGAL_PI/2)) && (angleSum > (-3*CGAL_PI/2))) umb.umbilic_type() = ELLIPTIC_UMBILIC;
else if ((angleSum <= (CGAL_PI/2)) && (angleSum >= (-CGAL_PI/2))) return 0;//is not considered as an umbilic
else umb.umbilic_type() = NON_GENERIC_UMBILIC;
return 1;
}
//Global function
2020-10-13 12:44:25 +00:00
template < class TriangleMesh,
class VertexFTMap,
class VertexVectorMap,
class OutputIterator>
OutputIterator compute_umbilics(const TriangleMesh &P,
2020-10-13 12:44:25 +00:00
const VertexFTMap& vertex2k1_pm,
const VertexFTMap& vertex2k2_pm,
2020-10-13 12:44:25 +00:00
const VertexVectorMap& vertex2d1_pm,
const VertexVectorMap& vertex2d2_pm,
2020-10-13 12:44:25 +00:00
OutputIterator it,
double size)
{
2020-10-13 12:44:25 +00:00
typedef Umbilic_approximation < TriangleMesh,
VertexFTMap, VertexVectorMap > Umbilic_approximation;
2020-10-13 12:44:25 +00:00
Umbilic_approximation umbilic_approximation(P,
vertex2k1_pm, vertex2k2_pm,
vertex2d1_pm, vertex2d2_pm);
return umbilic_approximation.compute(it, size);
}
} //namespace CGAL
#endif