diff --git a/Kernel_23/include/CGAL/Kernel/function_objects.h b/Kernel_23/include/CGAL/Kernel/function_objects.h index 5a7b0774e6df..55431d98d48d 100644 --- a/Kernel_23/include/CGAL/Kernel/function_objects.h +++ b/Kernel_23/include/CGAL/Kernel/function_objects.h @@ -284,6 +284,40 @@ namespace CommonKernelFunctors { * sq_length(ba)*sq_length(bc)); } } + + result_type + operator()(const Point_3& a1, const Point_3& b1, const Point_3& c1, + const Point_3& a2, const Point_3& b2, const Point_3& c2, + const FT& cosine) const + { + typename K::Compute_scalar_product_3 scalar_product = K().compute_scalar_product_3_object(); + typename K::Compute_squared_length_3 sq_length = K().compute_squared_length_3_object(); + typename K::Construct_normal_3 normal = K().construct_normal_3_object(); + + Vector_3 n1 = normal(a1,b1,c1); + Vector_3 n2 = normal(a2,b2,c2); + + typename K::FT sc_prod = scalar_product(n1, n2); + + if (sc_prod >= 0) + { + if (cosine >= 0) + return CGAL::compare(CGAL::square(cosine) + * sq_length(n1)*sq_length(n2), + CGAL::square(sc_prod)); + else + return SMALLER; + } + else + { + if (cosine >= 0) + return LARGER; + else + return CGAL::compare(CGAL::square(sc_prod), + CGAL::square(cosine) + * sq_length(n1)*sq_length(n2)); + } + } }; template @@ -930,6 +964,9 @@ namespace CommonKernelFunctors { class Compare_squared_distance_3 { typedef typename K::FT FT; + typedef typename K::Point_3 Point_3; + typedef typename K::Plane_3 Plane_3; + typename K::Construct_plane_3 construct_plane; public: typedef typename K::Comparison_result result_type; @@ -947,6 +984,14 @@ namespace CommonKernelFunctors { return CGAL::compare(internal::squared_distance(p, q, K()), internal::squared_distance(r, s, K())); } + + + Needs_FT + operator()(const Point_3& p, const Point_3& q, const Point_3& r, const Point_3& query, const FT& sqd) const + { + Plane_3 plane = construct_plane(p,q,r); + return this->operator()(plane, query, sqd); + } }; template diff --git a/Shape_detection/include/CGAL/Polygon_mesh_processing/region_growing.h b/Shape_detection/include/CGAL/Polygon_mesh_processing/region_growing.h index d6f78085c448..4923b5cb6349 100644 --- a/Shape_detection/include/CGAL/Polygon_mesh_processing/region_growing.h +++ b/Shape_detection/include/CGAL/Polygon_mesh_processing/region_growing.h @@ -192,13 +192,13 @@ region_growing_of_planes_on_faces(const PolygonMesh& mesh, Static_boolean_property_map()); using Neighbor_query = internal::One_ring_neighbor_query_with_constraints; - using Region_type = RG_PM::Least_squares_plane_fit_region; - using Sorting = RG_PM::Least_squares_plane_fit_sorting; + using Region_type = RG_PM::Plane_face_region; + using Sorting = RG_PM::Face_area_sorting; using Region_growing = CGAL::Shape_detection::Region_growing; Neighbor_query neighbor_query(mesh, ecm); Region_type region_type(mesh, np); - Sorting sorting(mesh, neighbor_query, np); + Sorting sorting(mesh, np); sorting.sort(); std::vector regions; diff --git a/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh.h b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh.h index 73443a13b8f5..6b571861b82c 100644 --- a/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh.h +++ b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh.h @@ -29,5 +29,7 @@ #include #include +#include +#include #endif // CGAL_SHAPE_DETECTION_REGION_GROWING_POLYGON_MESH_H diff --git a/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Face_area_sorting.h b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Face_area_sorting.h new file mode 100644 index 000000000000..203f71143e4f --- /dev/null +++ b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Face_area_sorting.h @@ -0,0 +1,193 @@ +// Copyright (c) 2018 INRIA Sophia-Antipolis (France). +// All rights reserved. +// +// This file is part of CGAL (www.cgal.org). +// +// $URL$ +// $Id$ +// SPDX-License-Identifier: GPL-3.0-or-later OR LicenseRef-Commercial +// +// +// Author(s) : Florent Lafarge, Simon Giraudot, Thien Hoang, Dmitry Anisimov +// + +#ifndef CGAL_SHAPE_DETECTION_REGION_GROWING_POLYGON_MESH_FACE_AREA_SORTING_H +#define CGAL_SHAPE_DETECTION_REGION_GROWING_POLYGON_MESH_FACE_AREA_SORTING_H + +#include +#include + +namespace CGAL { +namespace Shape_detection { +namespace Polygon_mesh { + + /*! + \ingroup PkgShapeDetectionRGOnMesh + + \brief Sorting of polygon mesh faces with respect to the local plane fit quality. + + `Items` of faces in a polygon mesh are sorted with respect to the quality of the + least squares plane fit applied to the vertices of incident faces of each face. + + \tparam GeomTraits + a model of `Kernel` + + \tparam PolygonMesh + a model of `FaceListGraph` + + \tparam VertexToPointMap + a model of `ReadablePropertyMap` whose key type is the vertex type of a polygon mesh and + value type is `Kernel::Point_3` + */ + template< + typename GeomTraits, + typename PolygonMesh, + typename VertexToPointMap = typename boost::property_map::const_type> + class Face_area_sorting + { + using vertex_descriptor = typename boost::graph_traits::vertex_descriptor; + using face_descriptor = typename boost::graph_traits::face_descriptor; + using halfedge_descriptor = typename boost::graph_traits::halfedge_descriptor; + using FT = typename GeomTraits::FT; + + public: + /// \name Types + /// @{ + + /// Item type. + using Item = typename boost::graph_traits::face_descriptor; + + /// Seed range. + using Seed_range = std::vector; + + /// @} + + /// \name Initialization + /// @{ + + /*! + \brief initializes all internal data structures. + + \tparam NamedParameters + a sequence of \ref bgl_namedparameters "Named Parameters" + + \param pmesh + an instance of `PolygonMesh` that represents a polygon mesh + + \param np + a sequence of \ref bgl_namedparameters "Named Parameters" + among the ones listed below + + \cgalNamedParamsBegin + \cgalParamNBegin{vertex_point_map} + \cgalParamDescription{an instance of `VertexToPointMap` that maps a polygon mesh + vertex to `Kernel::Point_3`} + \cgalParamDefault{`boost::get(CGAL::vertex_point, pmesh)`} + \cgalParamNEnd + \cgalParamNBegin{geom_traits} + \cgalParamDescription{an instance of `GeomTraits`} + \cgalParamDefault{`GeomTraits()`} + \cgalParamNEnd + \cgalNamedParamsEnd + + \pre `faces(pmesh).size() > 0` + */ + template + Face_area_sorting( + const PolygonMesh& pmesh, + const CGAL_NP_CLASS& np = parameters::default_values()) + : m_face_graph(pmesh) + , m_vpm(parameters::choose_parameter(parameters::get_parameter(np, internal_np::vertex_point), + get_const_property_map(CGAL::vertex_point, pmesh))) + , m_traits(parameters::choose_parameter(parameters::get_parameter(np, internal_np::geom_traits))) + { + CGAL_precondition(faces(pmesh).size() > 0); + + m_ordered.resize(faces(pmesh).size()); + + std::size_t index = 0; + for (Item item : faces(pmesh)) + m_ordered[index++] = item; + m_scores.resize(m_ordered.size(), 0.); + } + + /// @} + + /// \name Sorting + /// @{ + + /*! + \brief sorts `Items` of input faces. + */ + void sort() { + compute_scores(); + CGAL_precondition(m_scores.size() > 0); + + auto cmp = [this](const std::size_t i, const std::size_t j) + { + CGAL_precondition(i < m_scores.size()); + CGAL_precondition(j < m_scores.size()); + return m_scores[i] > m_scores[j]; + }; + std::vector order(m_ordered.size()); + std::iota(order.begin(), order.end(), 0); + std::sort(order.begin(), order.end(), cmp); + + std::vector tmp(m_ordered.size()); + for (std::size_t i = 0; i < m_ordered.size(); i++) + tmp[i] = m_ordered[order[i]]; + + m_ordered.swap(tmp); + } + /// @} + + /// \name Access + /// @{ + + /*! + \brief returns an instance of `Seed_range` to access the ordered `Items` + of input faces. + */ + const Seed_range &ordered() { + return m_ordered; + } + /// @} + + private: + const PolygonMesh& m_face_graph; + VertexToPointMap m_vpm; + const GeomTraits m_traits; + Seed_range m_ordered; + std::vector m_scores; + + void compute_scores() + { + auto squared_area = m_traits.compute_squared_area_3_object(); + std::size_t idx = 0; + for (Item item : m_ordered) + { + halfedge_descriptor hd = halfedge(item, m_face_graph); + std::vector pts; + + for (vertex_descriptor v : vertices_around_face(hd,m_face_graph)) + pts.push_back( get(m_vpm, v) ); + + if (pts.size()==3) + m_scores[idx++] = approximate_sqrt(squared_area(pts[0], pts[1], pts[2])); + else + { + std::vector triangles; + internal::triangulate_face(pts, triangles); + for (const typename GeomTraits::Triangle_3& tr : triangles) + m_scores[idx++] += approximate_sqrt(squared_area(tr[0], tr[1], tr[2])); + } + } + } + }; + +} // namespace Polygon_mesh +} // namespace Shape_detection +} // namespace CGAL + +#endif // CGAL_SHAPE_DETECTION_REGION_GROWING_POLYGON_MESH_FACE_AREA_SORTING_H + diff --git a/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Least_squares_plane_fit_region.h b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Least_squares_plane_fit_region.h index fdc710665819..9f7a9f74f770 100644 --- a/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Least_squares_plane_fit_region.h +++ b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Least_squares_plane_fit_region.h @@ -137,6 +137,13 @@ namespace Polygon_mesh { vertex to `Kernel::Point_3`} \cgalParamDefault{`boost::get(CGAL::vertex_point, pmesh)`} \cgalParamNEnd + \cgalParamNBegin{face_normal_map} + \cgalParamDescription{a property map associating normal vectors to the faces of `pmesh`.} + \cgalParamType{a class model of `ReadablePropertyMap` with `boost::graph_traits::%face_descriptor` + as key type and `GT::Vector_3` as value type, `GT` being the type of the parameter `geom_traits`.} + \cgalParamDefault{If this parameter is omitted, face normals will be estimated using crossproducts of vectors created + from consecutive vertices of the face.} + \cgalParamNEnd \cgalParamNBegin{geom_traits} \cgalParamDescription{an instance of `GeomTraits`} \cgalParamDefault{`GeomTraits()`} @@ -161,10 +168,13 @@ namespace Polygon_mesh { m_squared_distance_3(m_traits.compute_squared_distance_3_object()), m_scalar_product_3(m_traits.compute_scalar_product_3_object()), m_cross_product_3(m_traits.construct_cross_product_vector_3_object()), - m_face_normals( get(CGAL::dynamic_face_property_t(), pmesh) ), + m_face_normals(get(CGAL::dynamic_face_property_t(), pmesh)), m_face_triangulations( get(CGAL::dynamic_face_property_t>(), pmesh) ) { + static constexpr bool use_input_face_normal = + !parameters::is_default_parameter::value; + #ifdef CGAL_SD_RG_USE_PMP auto get_face_normal = [this](Item face, const PolygonMesh& pmesh) { @@ -193,8 +203,19 @@ namespace Polygon_mesh { }; #endif + if constexpr (!use_input_face_normal) + { + for (const Item &i : faces(pmesh)) + put(m_face_normals, i, get_face_normal(i, pmesh)); + } + else + { + auto fnm = parameters::get_parameter(np, internal_np::face_normal); + for (const Item &i : faces(pmesh)) + put(m_face_normals, i, get(fnm, i)); + } + for (const Item &i : faces(pmesh)) { - put(m_face_normals, i, get_face_normal(i, pmesh)); std::vector pts; auto h = halfedge(i, pmesh); auto s = h; diff --git a/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Plane_face_region.h b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Plane_face_region.h new file mode 100644 index 000000000000..a36bfef6e20d --- /dev/null +++ b/Shape_detection/include/CGAL/Shape_detection/Region_growing/Polygon_mesh/Plane_face_region.h @@ -0,0 +1,523 @@ +// Copyright (c) 2018 INRIA Sophia-Antipolis (France). +// All rights reserved. +// +// This file is part of CGAL (www.cgal.org). +// +// $URL$ +// $Id$ +// SPDX-License-Identifier: GPL-3.0-or-later OR LicenseRef-Commercial +// +// +// Author(s) : Florent Lafarge, Simon Giraudot, Thien Hoang, Dmitry Anisimov +// + +#ifndef CGAL_SHAPE_DETECTION_REGION_GROWING_POLYGON_MESH_PLANE_FACE_REGION_H +#define CGAL_SHAPE_DETECTION_REGION_GROWING_POLYGON_MESH_PLANE_FACE_REGION_H + +#include + +// Internal includes. +#include +#include +#ifdef CGAL_SD_RG_USE_PMP +#include +#endif + +namespace CGAL { +namespace Shape_detection { +namespace Polygon_mesh { + + /*! + \ingroup PkgShapeDetectionRGOnMesh + + \brief Region type based on the plane of the first face selected. + + This class uses the supporting plane of the first face picked for the region + and expand it for all faces with a normal close that that of the first face + (close being defined by the input angle threshold) and such that vertices are + not far from that supporting plane (far being defined by the input distance threshold). + + \tparam GeomTraits + a model of `Kernel` + + \tparam PolygonMesh + a model of `FaceListGraph` + + \tparam VertexToPointMap + a model of `ReadablePropertyMap` whose key type is the vertex type of a polygon mesh and + value type is `Kernel::Point_3` + + \cgalModels{RegionType} + */ + template< + typename GeomTraits, + typename PolygonMesh, + typename VertexToPointMap = typename boost::property_map::const_type> + class Plane_face_region { + + public: + /// \name Types + /// @{ + + /// \cond SKIP_IN_MANUAL + using Face_graph = PolygonMesh; + using Vertex_to_point_map = VertexToPointMap; + + using face_descriptor = typename boost::graph_traits::face_descriptor; + using halfedge_descriptor = typename boost::graph_traits::halfedge_descriptor; + using vertex_descriptor = typename boost::graph_traits::vertex_descriptor; + /// \endcond + + /// Number type. + typedef typename GeomTraits::FT FT; + + /// Item type. + using Item = face_descriptor; + using Region = std::vector; + + /// Primitive + using Primitive = typename GeomTraits::Plane_3; + + /// Region map + using Region_index_map = typename boost::property_map >::const_type; + + /// @} + + private: + using Point_3 = typename GeomTraits::Point_3; + using Vector_3 = typename GeomTraits::Vector_3; + using Plane_3 = typename GeomTraits::Plane_3; + using Triangle_3 = typename GeomTraits::Triangle_3; + + using Squared_length_3 = typename GeomTraits::Compute_squared_length_3; + using Squared_distance_3 = typename GeomTraits::Compute_squared_distance_3; + using Scalar_product_3 = typename GeomTraits::Compute_scalar_product_3; + using Cross_product_3 = typename GeomTraits::Construct_cross_product_vector_3; + + public: + /// \name Initialization + /// @{ + + /*! + \brief initializes all internal data structures. + + \tparam NamedParameters + a sequence of \ref bgl_namedparameters "Named Parameters" + + \param pmesh + an instance of `PolygonMesh` that represents a polygon mesh + + \param np + a sequence of \ref bgl_namedparameters "Named Parameters" + among the ones listed below + + \cgalNamedParamsBegin + \cgalParamNBegin{maximum_distance} + \cgalParamDescription{the maximum distance from the furthest vertex of a face to a plane} + \cgalParamType{`GeomTraits::FT`} + \cgalParamDefault{1} + \cgalParamNEnd + \cgalParamNBegin{maximum_angle} + \cgalParamDescription{the maximum angle in degrees between + the normal of a face and the normal of a plane} + \cgalParamType{`GeomTraits::FT`} + \cgalParamDefault{25 degrees} + \cgalParamNEnd + \cgalParamNBegin{cosine_of_maximum_angle} + \cgalParamDescription{the cosine value `cos(maximum_angle * PI / 180)` to be used instead of the parameter `maximum_angle()`} + \cgalParamType{`GeomTraits::FT`} + \cgalParamDefault{`cos(25 * PI / 180)`} + \cgalParamNEnd + \cgalParamNBegin{minimum_region_size} + \cgalParamDescription{the minimum number of faces a region must have} + \cgalParamType{`std::size_t`} + \cgalParamDefault{1} + \cgalParamNEnd + \cgalParamNBegin{vertex_point_map} + \cgalParamDescription{an instance of `VertexToPointMap` that maps a polygon mesh + vertex to `Kernel::Point_3`} + \cgalParamDefault{`boost::get(CGAL::vertex_point, pmesh)`} + \cgalParamNEnd + \cgalParamNBegin{face_normal_map} + \cgalParamDescription{a property map associating normal vectors to the faces of `pmesh`.} + \cgalParamType{a class model of `ReadablePropertyMap` with `boost::graph_traits::%face_descriptor` + as key type and `GT::Vector_3` as value type, `GT` being the type of the parameter `geom_traits`.} + \cgalParamDefault{If this parameter is omitted, face normals will be estimated using crossproducts of vectors created + from consecutive vertices of the face.} + \cgalParamNEnd + \cgalParamNBegin{geom_traits} + \cgalParamDescription{an instance of `GeomTraits`} + \cgalParamDefault{`GeomTraits()`} + \cgalParamNEnd + \cgalNamedParamsEnd + + \pre `faces(tmesh).size() > 0` + \pre `maximum_distance >= 0` + \pre `maximum_angle >= 0 && maximum_angle <= 90` + \pre `cosine_of_maximum_angle >= 0 && cosine_of_maximum_angle <= 1` + \pre `minimum_region_size > 0` + */ + template + Plane_face_region( + const PolygonMesh& pmesh, + const CGAL_NP_CLASS& np = parameters::default_values()) : + m_face_graph(pmesh), + m_vpm(parameters::choose_parameter(parameters::get_parameter( + np, internal_np::vertex_point), get_const_property_map(CGAL::vertex_point, pmesh))), + m_traits(parameters::choose_parameter(parameters::get_parameter(np, internal_np::geom_traits))), + m_squared_length_3(m_traits.compute_squared_length_3_object()), + m_squared_distance_3(m_traits.compute_squared_distance_3_object()), + m_scalar_product_3(m_traits.compute_scalar_product_3_object()), + m_cross_product_3(m_traits.construct_cross_product_vector_3_object()), + m_face_normals(get(CGAL::dynamic_face_property_t(), pmesh)), + m_face_triangulations( get(CGAL::dynamic_face_property_t>(), pmesh) ) + { + static constexpr bool use_input_face_normal = + !parameters::is_default_parameter::value; + +#ifdef CGAL_SD_RG_USE_PMP + auto get_face_normal = [this](Item face, const PolygonMesh& pmesh) + { + return Polygon_mesh_processing::compute_face_normal(face, pmesh, parameters::vertex_point_map(m_vpm)); + }; +#else + auto get_face_normal = [this](Item face, const PolygonMesh& pmesh) -> Vector_3 + { + const auto hedge = halfedge(face, pmesh); + const auto vertices = vertices_around_face(hedge, pmesh); + CGAL_precondition(vertices.size() >= 3); + + auto vertex = vertices.begin(); + const Point_3& p1 = get(m_vpm, *vertex); ++vertex; + const Point_3& p2 = get(m_vpm, *vertex); ++vertex; + Point_3 p3 = get(m_vpm, *vertex); + while(collinear(p1, p2, p3)) + { + if (++vertex == vertices.end()) return NULL_VECTOR; + p3 = get(m_vpm, *vertex); + } + + const Vector_3 u = p2 - p1; + const Vector_3 v = p3 - p1; + return m_cross_product_3(u, v); + }; +#endif + + if constexpr (!use_input_face_normal) + { + for (const Item &i : faces(pmesh)) + put(m_face_normals, i, get_face_normal(i, pmesh)); + } + else + { + auto fnm = parameters::get_parameter(np, internal_np::face_normal); + for (const Item &i : faces(pmesh)) + put(m_face_normals, i, get(fnm, i)); + } + + for (const Item &i : faces(pmesh)) { + std::vector pts; + auto h = halfedge(i, pmesh); + auto s = h; + + do { + pts.push_back(get(m_vpm, target(h, pmesh))); + h = next(h, pmesh); + } while (h != s); + + std::vector face_triangulation; + internal::triangulate_face(pts, face_triangulation); + put(m_face_triangulations, i, face_triangulation); + } + + CGAL_precondition(faces(m_face_graph).size() > 0); + const FT max_distance = parameters::choose_parameter( + parameters::get_parameter(np, internal_np::maximum_distance), FT(1)); + CGAL_precondition(max_distance >= FT(0)); + m_distance_threshold = max_distance; + + const FT max_angle = parameters::choose_parameter( + parameters::get_parameter(np, internal_np::maximum_angle), FT(25)); + CGAL_precondition(max_angle >= FT(0) && max_angle <= FT(90)); + + m_min_region_size = parameters::choose_parameter( + parameters::get_parameter(np, internal_np::minimum_region_size), 1); + CGAL_precondition(m_min_region_size > 0); + + const FT default_cos_value = static_cast(std::cos(CGAL::to_double( + (max_angle * static_cast(CGAL_PI)) / FT(180)))); + const FT cos_value = parameters::choose_parameter( + parameters::get_parameter(np, internal_np::cosine_of_maximum_angle), default_cos_value); + CGAL_precondition(cos_value >= FT(0) && cos_value <= FT(1)); + m_cos_value_threshold = cos_value; + } + + /// @} + + /// \name Access + /// @{ + + /*! + \brief implements `RegionType::region_index_map()`. + + This function creates an empty property map that maps each face to a std::size_t + */ + Region_index_map region_index_map() { + return get(CGAL::dynamic_face_property_t(), m_face_graph); + } + + /*! + \brief implements `RegionType::primitive()`. + + This function provides the last primitive that has been fitted with the region. + + \return Primitive parameters that fits the region. + + \pre `successful fitted primitive via successful call of update(region) with a sufficient large region` + */ + + // TODO: we probably want to return the m_seed_face instead + Primitive primitive() const { + return m_plane; + } + + /*! + \brief implements `RegionType::is_part_of_region()`. + + This function controls if the face `query` is within + the `maximum_distance` from the corresponding plane and if the angle between + its normal and the plane's normal is within the `maximum_angle`. If both conditions + are satisfied, it returns `true`, otherwise `false`. + + \param query + `Item` of the query face + + The last parameter is not used in this implementation. + + \return Boolean `true` or `false` + + \pre `query` is a valid const_iterator of `input_range` + */ +#ifdef FAST // TODO: bench me + bool is_part_of_region( + const Item query, + const Region&) const + { + //TODO: that's a bit dummy that we retest points that are already in the region... + + halfedge_descriptor h = halfedge(m_seed_face, m_face_graph); + + + const typename GeomTraits::Point_3& p=get(m_vpm,source(h, m_face_graph)); + const typename GeomTraits::Point_3& q=get(m_vpm,target(h, m_face_graph)); + typename GeomTraits::Point_3 r; + //TODO: add safety checks for degenerate faces + do{ + h=next(h, m_face_graph); + r=get(m_vpm,target(h, m_face_graph)); + } + while(!collinear(p,q,r)); + + + if (m_cos_value_threshold==1 || m_distance_threshold == 0) + { + h = halfedge(query, m_face_graph); + for (vertex_descriptor v : vertices_around_face(h, m_face_graph)) + { + if (!coplanar(p,q,r, get(m_vpm, v))) + return false; + } + } + else + { + // test on distance of points to the plane of the seed face + const FT squared_distance_threshold = m_distance_threshold * m_distance_threshold; + h = halfedge(query, m_face_graph); + typename GeomTraits::Plane_3 plane(p,q,r); + for (vertex_descriptor v : vertices_around_face(h, m_face_graph)) + { + FT sd = m_squared_distance_3(plane, get(m_vpm, v)); + if (sd>squared_distance_threshold) return false; + } + + + // test on the normal of the query face to the normal of the seed face + const Vector_3 face_normal = get(m_face_normals, query); + const FT cos_value = m_scalar_product_3(face_normal, m_normal_of_best_fit); + const FT squared_cos_value = cos_value * cos_value; + + FT squared_cos_value_threshold = + m_cos_value_threshold * m_cos_value_threshold; + squared_cos_value_threshold *= m_squared_length_3(face_normal); + squared_cos_value_threshold *= m_squared_length_3(m_normal_of_best_fit); + + if (squared_cos_value < squared_cos_value_threshold) + return false; + } + + return true; + } +#else + bool is_part_of_region( + const Item query, + const Region&) const + { + halfedge_descriptor h = halfedge(m_seed_face, m_face_graph); + + //TODO: store me! + const typename GeomTraits::Point_3& p=get(m_vpm,source(h, m_face_graph)); + const typename GeomTraits::Point_3& q=get(m_vpm,target(h, m_face_graph)); + typename GeomTraits::Point_3 r; + //TODO: add safety checks for degenerate faces + halfedge_descriptor guard = prev(h, m_face_graph); + + do{ + h=next(h, m_face_graph); + if (h == guard) return false; + r=get(m_vpm,target(h, m_face_graph)); + } + while(collinear(p,q,r)); + + + if (m_cos_value_threshold==1 || m_distance_threshold == 0) + { + h = halfedge(query, m_face_graph); + for (vertex_descriptor v : vertices_around_face(h, m_face_graph)) + { + if (!coplanar(p,q,r, get(m_vpm, v))) + return false; + } + return true; + } + else + { + // test on distance of points to the plane of the seed face + const FT squared_distance_threshold = m_distance_threshold * m_distance_threshold; + h = halfedge(query, m_face_graph); + for (vertex_descriptor v : vertices_around_face(h, m_face_graph)) + { + //TODO: that's a bit dummy that we retest points that are already in the region... + // not sure caching in a vpm does worth it (need reset for each region) + if (typename GeomTraits::Compare_squared_distance_3()(p,q,r,get(m_vpm, v), squared_distance_threshold) != SMALLER) + return false; + } + + const typename GeomTraits::Point_3& p2=get(m_vpm,source(h, m_face_graph)); + const typename GeomTraits::Point_3& q2=get(m_vpm,target(h, m_face_graph)); + typename GeomTraits::Point_3 r2; + + halfedge_descriptor guard = prev(h, m_face_graph); + do{ + h=next(h, m_face_graph); + if (h == guard) return true; + r2=get(m_vpm,target(h, m_face_graph)); + } + while(collinear(p2,q2,r2)); + + // test on the normal of the query face to the normal of the seed face + return typename GeomTraits::Compare_angle_3()(p,q,r, + p2,q2,r2, + m_cos_value_threshold) == SMALLER; + } + } +#endif + + /*! + \brief implements `RegionType::is_valid_region()`. + + This function controls if the `region` contains at least `minimum_region_size` faces. + + \param region + Faces of the region represented as `Items`. + + \return Boolean `true` or `false` + */ + inline bool is_valid_region(const Region& region) const { + return (region.size() >= m_min_region_size); + } + + /*! + \brief implements `RegionType::update()`. + + This function fits the least squares plane to all vertices of the faces + from the `region`. + + \param region + Faces of the region represented as `Items`. + + \return Boolean `true` if the plane fitting succeeded and `false` otherwise + + \pre `region.size() > 0` + */ + bool update(const Region& region) { + + CGAL_precondition(region.size() > 0); + if (region.size() == 1) { // init reference plane and normal + m_seed_face = region[0]; + + // The best fit plane will be a plane through this face centroid with + // its normal being the face's normal. + const Point_3 face_centroid = get_face_centroid(m_seed_face); + const Vector_3 face_normal = get(m_face_normals, m_seed_face); + if (face_normal == CGAL::NULL_VECTOR) return false; + + CGAL_precondition(face_normal != CGAL::NULL_VECTOR); + m_plane = Plane_3(face_centroid, face_normal); + m_normal = face_normal; + } + //TODO: shall we try to find a better seed face in the region? + return true; + } + + /// @} + + private: + const Face_graph& m_face_graph; + const Vertex_to_point_map m_vpm; + const GeomTraits m_traits; + + FT m_distance_threshold; + FT m_cos_value_threshold; + std::size_t m_min_region_size; + + const Squared_length_3 m_squared_length_3; + const Squared_distance_3 m_squared_distance_3; + const Scalar_product_3 m_scalar_product_3; + const Cross_product_3 m_cross_product_3; + + typename boost::property_map >::const_type m_face_normals; + typename boost::property_map> >::const_type m_face_triangulations; + + Plane_3 m_plane; + Vector_3 m_normal; + face_descriptor m_seed_face; + + // Compute centroid of the face. + template + Point_3 get_face_centroid(const Face& face) const { + + const auto hedge = halfedge(face, m_face_graph); + const auto vertices = vertices_around_face(hedge, m_face_graph); + CGAL_precondition(vertices.size() > 0); + + FT sum = FT(0), x = FT(0), y = FT(0), z = FT(0); + for (const auto vertex : vertices) { + const Point_3& point = get(m_vpm, vertex); + x += point.x(); + y += point.y(); + z += point.z(); + sum += FT(1); + } + CGAL_precondition(sum > FT(0)); + x /= sum; + y /= sum; + z /= sum; + return Point_3(x, y, z); + } + }; + +} // namespace Polygon_mesh +} // namespace Shape_detection +} // namespace CGAL + +#endif // CGAL_SHAPE_DETECTION_REGION_GROWING_POLYGON_MESH_PLANE_FACE_REGION_H