mirror of https://github.com/CGAL/cgal
1571 lines
64 KiB
C++
1571 lines
64 KiB
C++
// Copyright (c) 2009 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) : Stéphane Tayeb, Aymeric PELLE
|
|
|
|
|
|
#ifndef CGAL_LABELED_MESH_DOMAIN_3_H
|
|
#define CGAL_LABELED_MESH_DOMAIN_3_H
|
|
|
|
#include <CGAL/license/Mesh_3.h>
|
|
|
|
#include <CGAL/disable_warnings.h>
|
|
|
|
#include <CGAL/Mesh_3/config.h>
|
|
|
|
#include <CGAL/Bbox_3.h>
|
|
#include <CGAL/point_generators_3.h>
|
|
#include <memory>
|
|
#include <CGAL/tuple.h>
|
|
#include <CGAL/Origin.h>
|
|
|
|
#include <functional>
|
|
#include <type_traits>
|
|
|
|
#include <CGAL/SMDS_3/internal/Handle_IO_for_pair_of_int.h>
|
|
#include <CGAL/SMDS_3/internal/indices_management.h>
|
|
|
|
// support for `CGAL::Image_3`
|
|
#include <CGAL/Image_3.h>
|
|
#include <CGAL/Mesh_3/Image_to_labeled_function_wrapper.h>
|
|
#include <CGAL/Mesh_3/Image_plus_weights_to_labeled_function_wrapper.h>
|
|
|
|
// support for implicit functions
|
|
#include <CGAL/Implicit_to_labeling_function_wrapper.h>
|
|
#include <CGAL/Named_function_parameters.h>
|
|
#ifdef CGAL_MESH_3_VERBOSE
|
|
# include <boost/format.hpp>
|
|
#endif
|
|
#include <optional>
|
|
|
|
#include <CGAL/Mesh_3/Null_subdomain_index.h>
|
|
#include <CGAL/Mesh_domain_with_polyline_features_3.h>
|
|
|
|
#include <CGAL/Mesh_3/polylines_to_protect.h>
|
|
|
|
namespace CGAL {
|
|
namespace Mesh_3 {
|
|
namespace internal {
|
|
|
|
struct Identity {
|
|
template <typename T>
|
|
const T& operator()(const T& x) { return x; }
|
|
};
|
|
|
|
template<typename T>
|
|
struct Greater_than {
|
|
typedef T argument_type;
|
|
typedef bool result_type;
|
|
Greater_than(const T& second) : second(second) {}
|
|
bool operator()(const T& first) const {
|
|
return std::greater<T>()(first, second);
|
|
}
|
|
T second;
|
|
};
|
|
|
|
struct Do_not_delete {
|
|
template <typename T> void operator()(T*) const { }
|
|
};
|
|
|
|
// Returns a box enclosing image `im`
|
|
inline Bbox_3 compute_bounding_box(const Image_3& im)
|
|
{
|
|
return Bbox_3(-1+im.tx(),-1+im.ty(),-1+im.tz(),
|
|
double(im.xdim())*im.vx()+im.tx()+1,
|
|
double(im.ydim())*im.vy()+im.ty()+1,
|
|
double(im.zdim())*im.vz()+im.tz()+1);
|
|
}
|
|
|
|
template <typename Image_values_to_subdom_indices>
|
|
struct Create_gray_image_values_to_subdomain_indices {
|
|
typedef Image_values_to_subdom_indices type;
|
|
template <typename FT>
|
|
type operator()(Image_values_to_subdom_indices functor, const FT&) const {
|
|
return functor;
|
|
}
|
|
};
|
|
|
|
// specialization for `Null_functor`: create the default functor
|
|
template <>
|
|
struct Create_gray_image_values_to_subdomain_indices<Null_functor> {
|
|
typedef Mesh_3::internal::Greater_than<double> type;
|
|
template <typename FT>
|
|
type operator()(Null_functor, const FT& iso_value) const {
|
|
return type(iso_value);
|
|
}
|
|
};
|
|
|
|
template <typename Image_values_to_subdom_indices>
|
|
struct Create_labeled_image_values_to_subdomain_indices {
|
|
typedef Image_values_to_subdom_indices type;
|
|
type operator()(Image_values_to_subdom_indices functor) const {
|
|
return functor;
|
|
}
|
|
};
|
|
|
|
// specialization for `Null_functor`: create the default functor
|
|
template <>
|
|
struct Create_labeled_image_values_to_subdomain_indices<Null_functor> {
|
|
typedef Identity type;
|
|
type operator()(Null_functor) const {
|
|
return type();
|
|
}
|
|
};
|
|
|
|
// Detect_features_in_domain
|
|
template<typename Point, typename DetectFunctor>
|
|
struct Detect_features_in_domain {
|
|
|
|
std::vector<std::vector<Point>>
|
|
operator()(const CGAL::Image_3& image, CGAL::Image_3& weights, DetectFunctor functor) const {
|
|
#if defined(BOOST_MSVC) && (BOOST_MSVC < 1910) //before msvc2017
|
|
if(weights.is_valid())
|
|
return functor.operator()<Point>(image, weights);
|
|
else
|
|
return functor.operator()<Point>(image);
|
|
#else
|
|
if(weights.is_valid())
|
|
return functor.template operator()<Point>(image, weights);
|
|
else
|
|
return functor.template operator()<Point>(image);
|
|
#endif
|
|
}
|
|
};
|
|
// specialization for `Null_functor`: create the default functor
|
|
template<typename Point>
|
|
struct Detect_features_in_domain<Point, Null_functor> {
|
|
std::vector<std::vector<Point>>
|
|
operator()(const CGAL::Image_3&, CGAL::Image_3&, Null_functor) const {
|
|
return std::vector<std::vector<Point>>();
|
|
}
|
|
};
|
|
|
|
template<typename Point, typename DetectFunctor>
|
|
std::vector<std::vector<Point>>
|
|
detect_features(const CGAL::Image_3& image, CGAL::Image_3& weights, DetectFunctor functor)
|
|
{
|
|
Detect_features_in_domain<Point, DetectFunctor> detector;
|
|
return detector(image, weights, functor);
|
|
}
|
|
|
|
template<bool WithFeatures>
|
|
struct Add_features_in_domain {
|
|
template<typename MeshDomain, typename InputFeatureRange, typename DetectFunctor>
|
|
void operator()(const CGAL::Image_3&, CGAL::Image_3&, MeshDomain&, const InputFeatureRange&, DetectFunctor)
|
|
{}
|
|
};
|
|
|
|
template<>
|
|
struct Add_features_in_domain<true>
|
|
{
|
|
template<typename MeshDomain, typename InputFeatureRange, typename DetectFunctor>
|
|
void operator()(const CGAL::Image_3& image,
|
|
CGAL::Image_3& weights,
|
|
MeshDomain& domain,
|
|
const InputFeatureRange& input_features,
|
|
DetectFunctor functor)
|
|
{
|
|
using P = typename MeshDomain::Point_3;
|
|
auto detected_feature_range
|
|
= CGAL::Mesh_3::internal::detect_features<P>(image, weights, functor);
|
|
|
|
CGAL::merge_and_snap_polylines(image, detected_feature_range, input_features);
|
|
|
|
if (!input_features.empty())
|
|
domain.add_features(input_features.begin(), input_features.end());
|
|
domain.add_features(detected_feature_range.begin(), detected_feature_range.end());
|
|
}
|
|
};
|
|
|
|
} // end namespace CGAL::Mesh_3::internal
|
|
} // end namespace CGAL::Mesh_3
|
|
|
|
#ifndef DOXYGEN_RUNNING
|
|
template <typename Subdomain_index>
|
|
struct Construct_pair_from_subdomain_indices {
|
|
typedef std::pair<Subdomain_index, Subdomain_index> result_type;
|
|
|
|
result_type operator()(Subdomain_index a, Subdomain_index b) const {
|
|
return result_type(a, b);
|
|
}
|
|
}; // end class template Construct_pair_from_subdomain_indices
|
|
#endif
|
|
|
|
namespace details
|
|
{
|
|
|
|
template <typename Geom_traits,
|
|
typename Subdomain_index,
|
|
typename Surface_patch_index_>
|
|
class Labeled_mesh_domain_3_impl
|
|
{
|
|
protected:
|
|
typedef Surface_patch_index_ Surface_patch_index;
|
|
typedef typename Geom_traits::Point_3 Point_3;
|
|
typedef typename Geom_traits::Sphere_3 Sphere_3;
|
|
typedef typename Geom_traits::Iso_cuboid_3 Iso_cuboid_3;
|
|
typedef typename Geom_traits::FT FT;
|
|
typedef std::shared_ptr<CGAL::Random> CGAL_Random_share_ptr_t;
|
|
// Returns squared error bound from `bbox` and `error`
|
|
FT squared_error_bound(const Iso_cuboid_3& bbox, const FT& error) const
|
|
{
|
|
typename Geom_traits::Compute_squared_distance_3 squared_distance =
|
|
Geom_traits().compute_squared_distance_3_object();
|
|
return squared_distance((bbox.min)(), (bbox.max)())*error*error/4;
|
|
}
|
|
|
|
static Iso_cuboid_3 iso_cuboid(const Bbox_3& bbox)
|
|
{
|
|
const Point_3 p_min(bbox.xmin(), bbox.ymin(), bbox.zmin());
|
|
const Point_3 p_max(bbox.xmax(), bbox.ymax(), bbox.zmax());
|
|
|
|
return Iso_cuboid_3(p_min,p_max);
|
|
}
|
|
|
|
static Iso_cuboid_3 iso_cuboid(const typename Geom_traits::Sphere_3& sphere)
|
|
{
|
|
return iso_cuboid(sphere.bbox());
|
|
}
|
|
|
|
static Iso_cuboid_3 iso_cuboid(const typename Geom_traits::Iso_cuboid_3& c)
|
|
{
|
|
return c;
|
|
}
|
|
|
|
static Construct_pair_from_subdomain_indices<Subdomain_index>
|
|
construct_pair_functor() {
|
|
return Construct_pair_from_subdomain_indices<Subdomain_index>();
|
|
}
|
|
|
|
template <typename Function,
|
|
typename Bounding_object,
|
|
typename Null,
|
|
typename Construct_surface_patch_index>
|
|
Labeled_mesh_domain_3_impl(const Function& f,
|
|
const Bounding_object& bounding,
|
|
const FT& error_bound,
|
|
Construct_surface_patch_index cstr_s_p_i,
|
|
Null null,
|
|
CGAL::Random* p_rng)
|
|
: function_(f)
|
|
, bbox_(iso_cuboid(bounding))
|
|
, cstr_s_p_index(cstr_s_p_i)
|
|
, null(null)
|
|
, p_rng_(p_rng == 0 ?
|
|
CGAL_Random_share_ptr_t(new CGAL::Random(0)) :
|
|
CGAL_Random_share_ptr_t(p_rng, Mesh_3::internal::Do_not_delete()))
|
|
, squared_error_bound_(squared_error_bound(bbox_,error_bound))
|
|
{}
|
|
|
|
// The function which answers subdomain queries
|
|
typedef std::function<Subdomain_index(const Point_3&)> Function;
|
|
Function function_;
|
|
// The bounding box
|
|
const Iso_cuboid_3 bbox_;
|
|
|
|
typedef std::function<
|
|
Surface_patch_index(Subdomain_index,
|
|
Subdomain_index)> Construct_surface_patch_index;
|
|
Construct_surface_patch_index cstr_s_p_index;
|
|
// The functor that decides which sub-domain indices correspond to the
|
|
// outside of the domain.
|
|
typedef std::function<bool(Subdomain_index)> Null;
|
|
Null null;
|
|
// The random number generator used by Construct_initial_points
|
|
CGAL_Random_share_ptr_t p_rng_;
|
|
// Error bound relative to sphere radius
|
|
FT squared_error_bound_;
|
|
}; // Labeled_mesh_domain_3_impl
|
|
|
|
} // namespace details
|
|
|
|
/*!
|
|
\ingroup PkgMesh3Domains
|
|
|
|
\brief The class `Labeled_mesh_domain_3` implements indexed domains.
|
|
|
|
This class is a model of concept `MeshDomain_3`.
|
|
|
|
Any boundary facet is labeled <a,b>, with a<b, where a and b are the
|
|
tags of its incident subdomains.
|
|
Thus, a boundary facet of the domain is labeled <0,b>, where b!=0.
|
|
|
|
This class includes a <em>labeling function</em> that provides the index of the subdomain in which any
|
|
query point lies. An intersection between a segment and bounding
|
|
surfaces is detected when both segment endpoints are associated with different
|
|
values of subdomain indices. The intersection is then constructed by bisection.
|
|
The bisection stops when the query segment is shorter than an error bound
|
|
`e` given by the product of the
|
|
length of the diagonal of the bounding box (in world coordinates), or the radius of the bounding sphere, and
|
|
a relative error bound passed as argument to the constructor of `Labeled_mesh_domain_3`.
|
|
|
|
This class has a constructor taking a labeling function. It has also three
|
|
static template member functions that act as named constructors:
|
|
<ul>
|
|
<li>`create_gray_image_mesh_domain()`, to create a domain from a 3D gray image,
|
|
<li>`create_labeled_image_mesh_domain()`, to create a domain from a 3D labeled image, and
|
|
<li>`create_implicit_mesh_domain()`, to create a domain from an implicit function.
|
|
</ul>
|
|
|
|
\tparam BGT is a geometric traits class that provides
|
|
the basic operations to implement
|
|
intersection tests and intersection computations
|
|
through a bisection method. This parameter must be instantiated
|
|
with a model of the concept `BisectionGeometricTraits_3`.
|
|
|
|
\cgalHeading{Labeling function}
|
|
|
|
A labeling function `f` must return `0` if the point isn't located in any subdomain. The return type of labeling functions is an integer.
|
|
|
|
Let `p` be a Point.
|
|
<ul>
|
|
<li>`f(p)=0` means that `p` is outside domain.</li>
|
|
<li>`f(p)=a`, `a!=0` means that `p` is inside subdomain `a`.</li>
|
|
</ul>
|
|
`CGAL::Implicit_multi_domain_to_labeling_function_wrapper` is a good candidate for this template parameter
|
|
if there are several components to mesh.
|
|
|
|
The function type can be any model of the concept `Callable` compatible with the signature
|
|
`Subdomain_index(const %Point_3&)`: it can be a function, a function object, a lambda expression...
|
|
that takes a `%Point_3` as argument, and returns a type convertible to `Subdomain_index`.
|
|
|
|
\cgalModels{MeshDomain_3}
|
|
|
|
\sa `CGAL::Implicit_multi_domain_to_labeling_function_wrapper`
|
|
\sa `CGAL::make_mesh_3()`
|
|
|
|
*/
|
|
template<class BGT,
|
|
class Subdomain_index_ = int,
|
|
class Surface_patch_index_ = std::pair<Subdomain_index_,
|
|
Subdomain_index_> >
|
|
class Labeled_mesh_domain_3
|
|
#ifndef DOXYGEN_RUNNING
|
|
: protected details::Labeled_mesh_domain_3_impl<BGT,
|
|
Subdomain_index_,
|
|
Surface_patch_index_>
|
|
#endif
|
|
{
|
|
public:
|
|
//-------------------------------------------------------
|
|
// Index Types
|
|
//-------------------------------------------------------
|
|
// Type of indexes for cells of the input complex
|
|
/// \name Types
|
|
///@{
|
|
/// The subdomain index of this model of `MeshDomain_3`
|
|
typedef Subdomain_index_ Subdomain_index;
|
|
//
|
|
#ifdef DOXYGEN_RUNNING
|
|
/// The type of object that stores the function using type-erasure.
|
|
typedef std::function< Subdomain_index(const Point_3 &)> Labeling_function;
|
|
///@}
|
|
|
|
/// \name Types imported from the geometric traits class
|
|
///@{
|
|
/// The point type of the geometric traits class
|
|
typedef typename Geom_traits::Point_3 Point_3;
|
|
/// The sphere type of the geometric traits class
|
|
typedef typename Geom_traits::Sphere_3 Sphere_3;
|
|
/// The iso-cuboid type of the geometric traits class
|
|
typedef typename Geom_traits::Iso_cuboid_3 Iso_cuboid_3;
|
|
/// The number type (a field type) of the geometric traits class
|
|
typedef typename Geom_traits::FT FT;
|
|
///@}
|
|
#else // DOXYGEN_RUNNING
|
|
typedef std::optional<Subdomain_index> Subdomain;
|
|
|
|
// Type of indexes for cells of the input complex
|
|
typedef Surface_patch_index_ Surface_patch_index;
|
|
typedef std::optional<Surface_patch_index> Surface_patch;
|
|
|
|
// Type of indexes to characterize the lowest dimensional face of the input
|
|
// complex on which a vertex lie
|
|
typedef typename CGAL::Mesh_3::internal::
|
|
Index_generator<Subdomain_index, Surface_patch_index>::Index Index;
|
|
|
|
private:
|
|
typedef details::Labeled_mesh_domain_3_impl<BGT,
|
|
Subdomain_index,
|
|
Surface_patch_index
|
|
> Impl_details;
|
|
typedef typename Impl_details::Null Null;
|
|
typedef typename Impl_details::Construct_surface_patch_index
|
|
Construct_surface_patch_index;
|
|
typedef typename Impl_details::Function Function;
|
|
|
|
public:
|
|
// Geometric object types
|
|
typedef typename BGT::Point_3 Point_3;
|
|
typedef typename BGT::Segment_3 Segment_3;
|
|
typedef typename BGT::Ray_3 Ray_3;
|
|
typedef typename BGT::Line_3 Line_3;
|
|
typedef typename BGT::Vector_3 Vector_3;
|
|
typedef typename BGT::Sphere_3 Sphere_3;
|
|
typedef CGAL::Bbox_3 Bbox_3;
|
|
|
|
protected:
|
|
typedef typename BGT::Iso_cuboid_3 Iso_cuboid_3;
|
|
|
|
public:
|
|
// Kernel_traits compatibility
|
|
typedef BGT R;
|
|
// access Function type from inherited class
|
|
typedef Function Fct;
|
|
|
|
typedef std::tuple<Point_3,Index,int> Intersection;
|
|
|
|
|
|
typedef typename BGT::FT FT;
|
|
typedef BGT Geom_traits;
|
|
using Impl_details::construct_pair_functor;
|
|
#endif // DOXYGEN_RUNNING
|
|
|
|
/// \name Creation
|
|
/// @{
|
|
/*! \brief Construction from a function, a bounding object and a relative error bound.
|
|
*
|
|
* \tparam Function a type compatible with `Labeling_function`
|
|
* \tparam NamedParameters a sequence of \ref bgl_namedparameters "Named Parameters"
|
|
* \tparam Bounding_object either a bounding sphere (of type `Sphere_3`), a bounding box (type `Bbox_3`),
|
|
* or a bounding `Iso_cuboid_3`
|
|
*
|
|
* \param function the labeling function
|
|
* \param bounding_object the bounding object bounding the meshable space.
|
|
* \param np an optional sequence of \ref bgl_namedparameters "Named Parameters" among the ones listed below:
|
|
*
|
|
* \cgalNamedParamsBegin
|
|
* \cgalParamNBegin{relative_error_bound}
|
|
* \cgalParamDescription{the relative error bound used to compute intersection points between the implicit surface and query segments.
|
|
* The bisection is stopped when the length of the intersected segment is less than the product
|
|
* of `relative_error_bound` by the diameter of the bounding object.}
|
|
* \cgalParamDefault{FT(1e-3)}
|
|
* \cgalParamNEnd
|
|
* \cgalNamedParamsEnd
|
|
*
|
|
* \cgalHeading{Example}
|
|
* From the example (\ref Mesh_3/mesh_implicit_domains_2.cpp):
|
|
* \snippet Mesh_3/mesh_implicit_domains_2.cpp Domain creation
|
|
*/
|
|
template<typename Function, typename Bounding_object, typename CGAL_NP_TEMPLATE_PARAMETERS>
|
|
Labeled_mesh_domain_3(const Function& function,
|
|
const Bounding_object& bounding_object,
|
|
const CGAL_NP_CLASS& np = parameters::default_values()
|
|
#ifndef DOXYGEN_RUNNING
|
|
, typename std::enable_if<!is_named_function_parameter<Function>>::type* = nullptr
|
|
#endif // DOXYGEN_RUNNING
|
|
)
|
|
:Impl_details(function,
|
|
bounding_object,
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::error_bound), FT(1e-3)),
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::surface_patch_index), construct_pair_functor()),
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::null_subdomain_index_param), Null_subdomain_index()),
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::rng), nullptr))
|
|
{}
|
|
///@}
|
|
|
|
#ifndef DOXYGEN_RUNNING
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT>
|
|
Labeled_mesh_domain_3(const CGAL_NP_CLASS& np)
|
|
:Impl_details(parameters::get_parameter(np, internal_np::function_param),
|
|
parameters::get_parameter(np, internal_np::bounding_object_param),
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::error_bound), FT(1e-3)),
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::surface_patch_index), construct_pair_functor()),
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::null_subdomain_index_param), Null_subdomain_index()),
|
|
parameters::choose_parameter(parameters::get_parameter(np, internal_np::rng), nullptr))
|
|
{}
|
|
|
|
// Overload handling parameters passed with operator=
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_1,
|
|
typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_2,
|
|
typename ... NP>
|
|
Labeled_mesh_domain_3(const CGAL_NP_CLASS_1& np1,
|
|
const CGAL_NP_CLASS_2& np2,
|
|
const NP& ... nps)
|
|
: Labeled_mesh_domain_3(internal_np::combine_named_parameters(np1, np2, nps...))
|
|
{}
|
|
|
|
|
|
#ifndef CGAL_NO_DEPRECATED_CODE
|
|
template<typename Function, typename Bounding_object>
|
|
#if !defined(BOOST_MSVC)
|
|
CGAL_DEPRECATED
|
|
#endif // BOOST_MSVC
|
|
Labeled_mesh_domain_3(const Function& function,
|
|
const Bounding_object& bounding_object,
|
|
double error_bound,
|
|
typename std::enable_if<!is_named_function_parameter<Function>>::type* = nullptr)
|
|
: Labeled_mesh_domain_3(function,
|
|
bounding_object,
|
|
parameters::relative_error_bound(error_bound))
|
|
{}
|
|
#endif // CGAL_NO_DEPRECATED_CODE
|
|
#endif // DOXYGEN_RUNNING
|
|
|
|
/// \name Creation of domains from 3D images
|
|
/// @{
|
|
/*!
|
|
* \brief Construction from a 3D gray image
|
|
*
|
|
* This static method is a <em>named constructor</em>. It constructs a domain
|
|
* described by a 3D gray image. A 3D gray image is a grid of voxels,
|
|
* where each voxel is associated with a gray level value. Unless otherwise specified by the parameter `image_values_to_subdom_indices`, the domain to
|
|
* be discretized is the union of voxels that lie inside a surface
|
|
* described by an isolevel value, called \a isovalue. The voxels lying
|
|
* inside the domain have gray level values that are larger than the
|
|
* isovalue.
|
|
*
|
|
* The value of voxels is interpolated to a gray level value at any query point.
|
|
*
|
|
* \tparam NamedParameters a sequence of \ref bgl_namedparameters "Named Parameters"
|
|
*
|
|
* \param image_ the input 3D image.
|
|
* \param np an optional sequence of \ref bgl_namedparameters "Named Parameters" among the ones listed below:
|
|
*
|
|
* \cgalNamedParamsBegin
|
|
* \cgalParamNBegin{iso_value}
|
|
* \cgalParamDescription{the isovalue, inside
|
|
* `image`, of the surface describing the boundary of the object to be
|
|
* meshed.}
|
|
* \cgalParamDefault{0}
|
|
* \cgalParamNEnd
|
|
* \cgalParamNBegin{image_values_to_subdomain_indices}
|
|
* \cgalParamDescription{a function or a function object, compatible with the signature
|
|
* `Subdomain_index(double)`. This function returns the subdomain index
|
|
* corresponding to a pixel value. If this parameter is used, then the
|
|
* parameter `iso_value` is ignored.}
|
|
* \cgalParamDefault{Null_functor()}
|
|
* \cgalParamNEnd
|
|
* \cgalParamNBegin{value_outside}
|
|
* \cgalParamDescription{the value attached to voxels
|
|
* outside of the domain to be meshed. It should be lower than
|
|
* `iso_value`.}
|
|
* \cgalParamDefault{0}
|
|
* \cgalParamNEnd
|
|
* \cgalParamNBegin{relative_error_bound}
|
|
* \cgalParamDescription{ is the relative error
|
|
* bound, relative to the diameter of the box of the image.}
|
|
* \cgalParamDefault{FT(1e-3)}
|
|
* \cgalParamNEnd
|
|
* \cgalNamedParamsEnd
|
|
*
|
|
* \cgalHeading{Examples}
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_3D_gray_image.cpp):
|
|
*
|
|
* \snippet Mesh_3/mesh_3D_gray_image.cpp Domain creation
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_3D_gray_vtk_image.cpp):
|
|
*
|
|
* \snippet Mesh_3/mesh_3D_gray_vtk_image.cpp Domain creation
|
|
*
|
|
*/
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS>
|
|
static Labeled_mesh_domain_3 create_gray_image_mesh_domain(const CGAL::Image_3& image_, const CGAL_NP_CLASS& np = parameters::default_values())
|
|
{
|
|
using parameters::get_parameter;
|
|
using parameters::choose_parameter;
|
|
auto iso_value_ = choose_parameter(get_parameter(np, internal_np::iso_value_param), 0);
|
|
auto value_outside_ = choose_parameter(get_parameter(np, internal_np::voxel_value), 0);
|
|
FT relative_error_bound_ = choose_parameter(get_parameter(np, internal_np::error_bound), FT(1e-3));
|
|
auto image_values_to_subdomain_indices_ = choose_parameter(get_parameter(np, internal_np::image_subdomain_index), Null_functor());
|
|
CGAL::Random* p_rng_ = choose_parameter(get_parameter(np, internal_np::rng), nullptr);
|
|
auto null_subdomain_index_ = choose_parameter(get_parameter(np, internal_np::null_subdomain_index_param), Null_functor());
|
|
auto construct_surface_patch_index_ = choose_parameter(get_parameter(np, internal_np::surface_patch_index), Null_functor());
|
|
namespace p = CGAL::parameters;
|
|
return Labeled_mesh_domain_3
|
|
(p::function = create_gray_image_wrapper
|
|
(image_,
|
|
iso_value_,
|
|
image_values_to_subdomain_indices_,
|
|
value_outside_),
|
|
p::bounding_object = Mesh_3::internal::compute_bounding_box(image_),
|
|
p::relative_error_bound = relative_error_bound_,
|
|
p::p_rng = p_rng_,
|
|
p::null_subdomain_index =
|
|
create_null_subdomain_index(null_subdomain_index_),
|
|
p::construct_surface_patch_index =
|
|
create_construct_surface_patch_index(construct_surface_patch_index_));
|
|
|
|
}
|
|
|
|
/*!
|
|
* \brief Construction from a 3D labeled image
|
|
*
|
|
* This static method is a <em>named constructor</em>. It constructs a
|
|
* domain described by a 3D labeled image. A 3D labeled image is a grid
|
|
* of voxels, where each voxel is associated with an index (a subdomain
|
|
* index) characterizing the subdomain in which the voxel lies. The
|
|
* domain to be discretized is the union of voxels that have non-zero
|
|
* values.
|
|
*
|
|
* \returns either a `Labeled_mesh_domain_3`,
|
|
* or a `Mesh_domain_with_polyline_features_3<Labeled_mesh_domain_3>`
|
|
* depending on whether one or more of the named parameters
|
|
* `features_detector` and `input_features` are provided.
|
|
*
|
|
* \tparam NamedParameters a sequence of \ref bgl_namedparameters "Named Parameters"
|
|
* \param image_ the input 3D image.
|
|
* \param np an optional sequence of \ref bgl_namedparameters "Named Parameters" among the ones listed below:
|
|
*
|
|
* \cgalNamedParamsBegin
|
|
* \cgalParamNBegin{weights}
|
|
* \cgalParamDescription{a reference to an input 3D image that provides
|
|
* weights associated to each voxel (the word type is `unsigned char`,
|
|
* and the voxels values are integers between 0 and 255).
|
|
* The weights image can be generated with `CGAL::Mesh_3::generate_label_weights()`.
|
|
* Its dimensions must be the same as the dimensions of `parameters::image`.}
|
|
* \cgalParamDefault{CGAL::Image_3()}
|
|
* \cgalParamType{CGAL::Image_3&}
|
|
* \cgalParamExtra{if `features_detector` is provided, `weights` should be modified accordingly.
|
|
* The available functors described in See \ref PkgMesh3FeatureDetection
|
|
* implement the necessary modifications.}
|
|
* \cgalParamExtra{if `input_features` is provided, `weights` should be modified accordingly
|
|
* to keep consistency of the output `MeshDomainWithFeatures_3`}
|
|
* \cgalParamNEnd
|
|
* \cgalParamNBegin{value_outside}
|
|
* \cgalParamDescription{the value attached to voxels
|
|
* outside of the domain to be meshed. It should be lower than
|
|
* `iso_value`.}
|
|
* \cgalParamDefault{0}
|
|
* \cgalParamNEnd
|
|
*
|
|
* \cgalParamNBegin{relative_error_bound}
|
|
* \cgalParamDescription{ is the relative error
|
|
* bound, relative to the diameter of the box of the image.}
|
|
* \cgalParamDefault{FT(1e-3)}
|
|
* \cgalParamNEnd
|
|
*
|
|
* \cgalParamNBegin{features_detector}
|
|
* \cgalParamDescription{ a functor that implements
|
|
* `std::vector<std::vector<Point>> operator()(const Image_3& img) const`,
|
|
* and `std::vector<std::vector<Point>> operator()(const Image_3& img, Image_3& weights) const`,
|
|
* where `%Point` matches the mesh domain point type,
|
|
* that both return a range of detected polyline features for feature protection.
|
|
* Only one implementation is used, depending on whether the named parameter `weights`
|
|
* is provided or not.
|
|
* Polyline features are added to the domain for further feature protection.
|
|
* See \ref PkgMesh3FeatureDetection for available functors.}
|
|
* \cgalParamDefault{CGAL::Null_functor()}
|
|
* \cgalParamExtra{The return type of the function depends on whether this parameter
|
|
* or `input_features` are provided or not.}
|
|
* \cgalParamExtra{If `weights` is provided, it must either be adapted to the detected features,
|
|
* or postprocessed during feature detection to keep consistency
|
|
* of the output `MeshDomainWithFeatures_3`.
|
|
* Available functors implement the necessary modifications.}
|
|
* \cgalParamNEnd
|
|
*
|
|
* \cgalParamNBegin{input_features}
|
|
* \cgalParamDescription{ a `Range` of polyline features, represented as `Range`s of `Point_3`.
|
|
* Polyline features are added to the domain for further feature protection.
|
|
* Input polyline features must be different from the detected features
|
|
* and can intersect only at vertices, if they do. Otherwise,
|
|
* the meshing process may not terminate.}
|
|
* \cgalParamDefault{`std::vector<std::vector<Point_3>>()`}
|
|
* \cgalParamExtra{The return type of the function depends on whether this parameter
|
|
or `input_features` are provided or not.}
|
|
* \cgalParamExtra{It is recommended to pass a const-reference for this parameter,
|
|
* possibly using `std::cref(polylines_range)` to avoid useless copies.}
|
|
* \cgalParamExtra{If `weights` is provided, it must be adapted to the input features,
|
|
* to keep consistency of the output `MeshDomainWithFeatures_3`}
|
|
* \cgalParamNEnd
|
|
*
|
|
* \cgalNamedParamsEnd
|
|
*
|
|
* \cgalHeading{Example}
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_3D_image.cpp):
|
|
*
|
|
* \snippet Mesh_3/mesh_3D_image.cpp Domain creation
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_3D_weighted_image.cpp),
|
|
* where the labeled image is used with a precomputed 3D image of weights :
|
|
*
|
|
* \snippet Mesh_3/mesh_3D_weighted_image.cpp Domain creation
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_3D_image_with_detection_of_features.cpp)
|
|
* where the features are detected in `image`:
|
|
*
|
|
* \snippet Mesh_3/mesh_3D_image_with_detection_of_features.cpp Domain creation
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_3D_image_with_input_features.cpp)
|
|
* where the features are provided by the user:
|
|
*
|
|
* \snippet Mesh_3/mesh_3D_image_with_input_features.cpp Domain creation
|
|
*
|
|
*/
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS>
|
|
static auto
|
|
create_labeled_image_mesh_domain(const CGAL::Image_3& image_, const CGAL_NP_CLASS& np = parameters::default_values())
|
|
{
|
|
using parameters::get_parameter;
|
|
using parameters::get_parameter_reference;
|
|
using parameters::choose_parameter;
|
|
|
|
auto iso_value_ = choose_parameter(get_parameter(np, internal_np::iso_value_param), 0);
|
|
auto value_outside_ = choose_parameter(get_parameter(np, internal_np::voxel_value), 0);
|
|
FT relative_error_bound_ = choose_parameter(get_parameter(np, internal_np::error_bound), FT(1e-3));
|
|
auto image_values_to_subdomain_indices_ = choose_parameter(get_parameter(np, internal_np::image_subdomain_index), Null_functor());
|
|
CGAL::Random* p_rng_ = choose_parameter(get_parameter(np, internal_np::rng), nullptr);
|
|
auto null_subdomain_index_ = choose_parameter(get_parameter(np, internal_np::null_subdomain_index_param), Null_functor());
|
|
auto construct_surface_patch_index_ = choose_parameter(get_parameter(np, internal_np::surface_patch_index), Null_functor());
|
|
|
|
using Image_ref_type = typename internal_np::Lookup_named_param_def<internal_np::weights_param_t,
|
|
CGAL_NP_CLASS,
|
|
CGAL::Image_3>::reference;
|
|
CGAL::Image_3 no_weights_;
|
|
Image_ref_type weights_ = choose_parameter(get_parameter_reference(np, internal_np::weights_param), no_weights_);
|
|
auto features_detector_ = choose_parameter(get_parameter(np, internal_np::features_detector_param), Null_functor());
|
|
|
|
using Default_input_features = std::vector<std::vector<typename Labeled_mesh_domain_3::Point_3>>;
|
|
using Input_features_ref_type = typename internal_np::Lookup_named_param_def<internal_np::input_features_param_t,
|
|
CGAL_NP_CLASS,
|
|
Default_input_features>::reference;
|
|
Default_input_features empty_vec;
|
|
Input_features_ref_type input_features_
|
|
= choose_parameter(get_parameter_reference(np, internal_np::input_features_param), empty_vec);
|
|
|
|
CGAL_USE(iso_value_);
|
|
namespace p = CGAL::parameters;
|
|
|
|
const bool use_weights = weights_.is_valid();
|
|
auto image_wrapper = use_weights
|
|
? create_weighted_labeled_image_wrapper(image_,
|
|
weights_,
|
|
image_values_to_subdomain_indices_,
|
|
value_outside_)
|
|
: create_labeled_image_wrapper(image_,
|
|
image_values_to_subdomain_indices_,
|
|
value_outside_);
|
|
|
|
// warning : keep Return_type consistent with actual return type
|
|
const bool no_features
|
|
= CGAL::parameters::is_default_parameter<CGAL_NP_CLASS, internal_np::features_detector_param_t>::value
|
|
&& CGAL::parameters::is_default_parameter<CGAL_NP_CLASS, internal_np::input_features_param_t>::value;
|
|
using Return_type = std::conditional_t <
|
|
no_features,
|
|
Labeled_mesh_domain_3,
|
|
Mesh_domain_with_polyline_features_3<Labeled_mesh_domain_3>
|
|
>;
|
|
|
|
Return_type domain
|
|
(p::function = image_wrapper,
|
|
p::bounding_object = Mesh_3::internal::compute_bounding_box(image_),
|
|
p::relative_error_bound = relative_error_bound_,
|
|
p::p_rng = p_rng_,
|
|
p::null_subdomain_index =
|
|
create_null_subdomain_index(null_subdomain_index_),
|
|
p::construct_surface_patch_index =
|
|
create_construct_surface_patch_index(construct_surface_patch_index_));
|
|
|
|
// features
|
|
Mesh_3::internal::Add_features_in_domain<!no_features>()
|
|
(image_, weights_, domain, input_features_, features_detector_);
|
|
|
|
return domain;
|
|
}
|
|
/// @}
|
|
|
|
#ifndef DOXYGEN_RUNNING
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT>
|
|
static Labeled_mesh_domain_3 create_gray_image_mesh_domain(const CGAL_NP_CLASS& np)
|
|
{
|
|
static_assert(!parameters::is_default_parameter<CGAL_NP_CLASS, internal_np::image_3_param_t>::value, "Value for required parameter not found");
|
|
using parameters::get_parameter;
|
|
using parameters::get_parameter_reference;
|
|
using parameters::choose_parameter;
|
|
const CGAL::Image_3& image_ = get_parameter_reference(np,internal_np::image_3_param);
|
|
auto iso_value_ = choose_parameter(get_parameter(np, internal_np::iso_value_param), 0);
|
|
auto value_outside_ = choose_parameter(get_parameter(np, internal_np::voxel_value), 0);
|
|
FT relative_error_bound_ = choose_parameter(get_parameter(np, internal_np::error_bound), FT(1e-3));
|
|
auto image_values_to_subdomain_indices_ = choose_parameter(get_parameter(np, internal_np::image_subdomain_index), Null_functor());
|
|
CGAL::Random* p_rng_ = choose_parameter(get_parameter(np, internal_np::rng), nullptr);
|
|
auto null_subdomain_index_ = choose_parameter(get_parameter(np, internal_np::null_subdomain_index_param), Null_functor());
|
|
auto construct_surface_patch_index_ = choose_parameter(get_parameter(np, internal_np::surface_patch_index), Null_functor());
|
|
namespace p = CGAL::parameters;
|
|
return Labeled_mesh_domain_3
|
|
(p::function = create_gray_image_wrapper
|
|
(image_,
|
|
iso_value_,
|
|
image_values_to_subdomain_indices_,
|
|
value_outside_),
|
|
p::bounding_object = Mesh_3::internal::compute_bounding_box(image_),
|
|
p::relative_error_bound = relative_error_bound_,
|
|
p::p_rng = p_rng_,
|
|
p::null_subdomain_index =
|
|
create_null_subdomain_index(null_subdomain_index_),
|
|
p::construct_surface_patch_index =
|
|
create_construct_surface_patch_index(construct_surface_patch_index_));
|
|
|
|
}
|
|
|
|
// Overload handling parameters passed with operator=
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_1,
|
|
typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_2,
|
|
typename ... NP>
|
|
static Labeled_mesh_domain_3 create_gray_image_mesh_domain(const CGAL::Image_3& image_,
|
|
const CGAL_NP_CLASS_1& np1,
|
|
const CGAL_NP_CLASS_2& np2,
|
|
const NP& ... nps)
|
|
{
|
|
return create_gray_image_mesh_domain(image_, internal_np::combine_named_parameters(np1, np2, nps...));
|
|
}
|
|
// Overload handling parameters passed with operator=
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_1,
|
|
typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_2,
|
|
typename ... NP>
|
|
static Labeled_mesh_domain_3 create_gray_image_mesh_domain(const CGAL_NP_CLASS_1& np1,
|
|
const CGAL_NP_CLASS_2& np2,
|
|
const NP& ... nps)
|
|
{
|
|
return create_gray_image_mesh_domain(internal_np::combine_named_parameters(np1, np2, nps...));
|
|
}
|
|
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT>
|
|
static auto create_labeled_image_mesh_domain(const CGAL_NP_CLASS& np)
|
|
{
|
|
static_assert(!parameters::is_default_parameter<CGAL_NP_CLASS, internal_np::image_3_param_t>::value, "Value for required parameter not found");
|
|
using parameters::get_parameter_reference;
|
|
const CGAL::Image_3& image_ = get_parameter_reference(np,internal_np::image_3_param);
|
|
return create_labeled_image_mesh_domain(image_, np);
|
|
}
|
|
|
|
// Overload handling parameters passed with operator=
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_1,
|
|
typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_2,
|
|
typename ... NP>
|
|
static auto create_labeled_image_mesh_domain(const CGAL::Image_3& image_,
|
|
const CGAL_NP_CLASS_1& np1,
|
|
const CGAL_NP_CLASS_2& np2,
|
|
const NP& ... nps)
|
|
{
|
|
return create_labeled_image_mesh_domain(image_, internal_np::combine_named_parameters(np1, np2, nps...));
|
|
}
|
|
// Overload handling parameters passed with operator=
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_1,
|
|
typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_2,
|
|
typename ... NP>
|
|
static auto create_labeled_image_mesh_domain(const CGAL_NP_CLASS_1& np1,
|
|
const CGAL_NP_CLASS_2& np2,
|
|
const NP& ... nps)
|
|
{
|
|
return create_labeled_image_mesh_domain(internal_np::combine_named_parameters(np1, np2, nps...));
|
|
}
|
|
#endif
|
|
|
|
/// \name Creation of domains from implicit functions
|
|
/// @{
|
|
|
|
/*!
|
|
* \brief Construction from an implicit function
|
|
*
|
|
* This static method is a <em>named constructor</em>. It constructs a domain
|
|
* whose bounding surface is described implicitly as the zero level set of a
|
|
* function. The domain to be discretized is assumed to be the domain where
|
|
* the function has negative values.
|
|
*
|
|
* \tparam Function a type compatible with the signature `FT(Point_3)`: it takes a point as argument,
|
|
* and returns a scalar value. That object must be model of `CopyConstructible`
|
|
* \tparam NamedParameters a sequence of \ref bgl_namedparameters "Named Parameters"
|
|
* \tparam Bounding_object either a bounding sphere (of type `Sphere_3`), a bounding box (type `Bbox_3`),
|
|
* or a bounding `Iso_cuboid_3` which is required to circumscribe
|
|
* the surface and to have its center inside the domain.
|
|
*
|
|
* \param function the implicit function
|
|
* \param bounding_object object bounding the meshable domain and its center is inside the domain.
|
|
* \param np an optional sequence of \ref bgl_namedparameters "Named Parameters" among the ones listed below:
|
|
*
|
|
* \cgalNamedParamsBegin
|
|
* \cgalParamNBegin{relative_error_bound}
|
|
* \cgalParamDescription{ is the relative error
|
|
* bound, relative to the diameter of the box of the image.}
|
|
* \cgalParamDefault{FT(1e-3)}
|
|
* \cgalParamNEnd
|
|
* \cgalNamedParamsEnd
|
|
*
|
|
* \cgalHeading{Examples}
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_implicit_sphere.cpp):
|
|
*
|
|
* \snippet Mesh_3/mesh_implicit_sphere.cpp Domain creation
|
|
*
|
|
* From the example (\ref Mesh_3/mesh_implicit_sphere_variable_size.cpp):
|
|
*
|
|
* \snippet Mesh_3/mesh_implicit_sphere_variable_size.cpp Domain creation
|
|
*
|
|
*/
|
|
template<typename Function, typename Bounding_object, typename CGAL_NP_TEMPLATE_PARAMETERS>
|
|
static Labeled_mesh_domain_3 create_implicit_mesh_domain(const Function& function,
|
|
const Bounding_object& bounding_object,
|
|
const CGAL_NP_CLASS& np = parameters::default_values()
|
|
#ifndef DOXYGEN_RUNNING
|
|
, typename std::enable_if<!is_named_function_parameter<Function>>::type* = nullptr
|
|
#endif
|
|
)
|
|
{
|
|
using parameters::get_parameter;
|
|
using parameters::choose_parameter;
|
|
FT relative_error_bound_ = choose_parameter(get_parameter(np, internal_np::error_bound), FT(1e-3));
|
|
CGAL::Random* p_rng_ = choose_parameter(get_parameter(np, internal_np::rng), nullptr);
|
|
auto null_subdomain_index_ = choose_parameter(get_parameter(np, internal_np::null_subdomain_index_param), Null_functor());
|
|
auto construct_surface_patch_index_ = choose_parameter(get_parameter(np, internal_np::surface_patch_index), Null_functor());
|
|
namespace p = CGAL::parameters;
|
|
return Labeled_mesh_domain_3
|
|
(p::function = make_implicit_to_labeling_function_wrapper<BGT>(function),
|
|
p::bounding_object = bounding_object,
|
|
p::relative_error_bound = relative_error_bound_,
|
|
p::p_rng = p_rng_,
|
|
p::null_subdomain_index =
|
|
create_null_subdomain_index(null_subdomain_index_),
|
|
p::construct_surface_patch_index =
|
|
create_construct_surface_patch_index(construct_surface_patch_index_));
|
|
}
|
|
/// @}
|
|
#ifndef DOXYGEN_RUNNING
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS>
|
|
static Labeled_mesh_domain_3 create_implicit_mesh_domain(const CGAL_NP_CLASS &np)
|
|
{
|
|
static_assert(!parameters::is_default_parameter<CGAL_NP_CLASS, internal_np::function_param_t>::value, "Value for required parameter not found");
|
|
static_assert(!parameters::is_default_parameter<CGAL_NP_CLASS, internal_np::bounding_object_param_t>::value, "Value for required parameter not found");
|
|
|
|
return create_implicit_mesh_domain(parameters::get_parameter(np, internal_np::function_param),
|
|
parameters::get_parameter(np, internal_np::bounding_object_param),
|
|
np);
|
|
}
|
|
|
|
// Overload handling parameters passed with operator=
|
|
template<typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_1,
|
|
typename CGAL_NP_TEMPLATE_PARAMETERS_NO_DEFAULT_2,
|
|
typename ... NP>
|
|
static Labeled_mesh_domain_3 create_implicit_mesh_domain(const CGAL_NP_CLASS_1& np1,
|
|
const CGAL_NP_CLASS_2& np2,
|
|
const NP& ... nps)
|
|
{
|
|
return create_implicit_mesh_domain(internal_np::combine_named_parameters(np1, np2, nps...));
|
|
}
|
|
|
|
|
|
#ifndef CGAL_NO_DEPRECATED_CODE
|
|
template<typename SubdomainIndex = Null_functor,
|
|
typename NullSubdomainIndex = Null_functor,
|
|
typename ConstructSurfacePatchIndex = Null_functor>
|
|
CGAL_DEPRECATED
|
|
static Labeled_mesh_domain_3
|
|
create_gray_image_mesh_domain(const CGAL::Image_3& image_,
|
|
double iso_value,
|
|
double value_outside=0,
|
|
double relative_error_bound = 1e-3,
|
|
CGAL::Random* rng = nullptr,
|
|
SubdomainIndex image_values_to_subdom_indices = SubdomainIndex(),
|
|
NullSubdomainIndex null_subdomain_index_ = NullSubdomainIndex(),
|
|
ConstructSurfacePatchIndex construct_surface_patch_index_ = ConstructSurfacePatchIndex())
|
|
{
|
|
return create_gray_image_mesh_domain(image_, parameters::iso_value(iso_value)
|
|
.image_values_to_subdomain_indices(image_values_to_subdom_indices)
|
|
.value_outside(value_outside)
|
|
.relative_error_bound(relative_error_bound)
|
|
.p_rng(rng).null_subdomain_index(null_subdomain_index_)
|
|
.construct_surface_patch_index(construct_surface_patch_index_));
|
|
}
|
|
|
|
template<typename SubdomainIndex = Null_functor,
|
|
typename NullSubdomainIndex = Null_functor,
|
|
typename ConstructSurfacePatchIndex = Null_functor>
|
|
CGAL_DEPRECATED
|
|
static Labeled_mesh_domain_3
|
|
create_labeled_image_mesh_domain(const CGAL::Image_3& image_,
|
|
double relative_error_bound,
|
|
const CGAL::Image_3& weights_ = CGAL::Image_3(),
|
|
int value_outside=0,
|
|
CGAL::Random* rng = nullptr,
|
|
SubdomainIndex image_values_to_subdom_indices = SubdomainIndex(),
|
|
NullSubdomainIndex null_subdomain_index_ = NullSubdomainIndex(),
|
|
ConstructSurfacePatchIndex construct_surface_patch_index_ = ConstructSurfacePatchIndex())
|
|
{
|
|
return create_labeled_image_mesh_domain(image_, parameters::weights(weights_)
|
|
.image_values_to_subdomain_indices(image_values_to_subdom_indices)
|
|
.value_outside(value_outside)
|
|
.relative_error_bound(relative_error_bound)
|
|
.p_rng(rng)
|
|
.null_subdomain_index(null_subdomain_index_)
|
|
.construct_surface_patch_index(construct_surface_patch_index_));
|
|
}
|
|
#endif
|
|
|
|
|
|
/*
|
|
* Constructs a set of `n` points on the surface, and output them to
|
|
* the output iterator `pts` whose value type is required to be
|
|
* `std::pair<Points_3, Index>`.
|
|
*/
|
|
struct Construct_initial_points
|
|
{
|
|
Construct_initial_points(const Labeled_mesh_domain_3& domain)
|
|
: r_domain_(domain) {}
|
|
|
|
template<class OutputIterator>
|
|
OutputIterator operator()(OutputIterator pts, const int n = 12) const;
|
|
|
|
private:
|
|
const Labeled_mesh_domain_3& r_domain_;
|
|
};
|
|
|
|
// Returns Construct_initial_points object
|
|
Construct_initial_points construct_initial_points_object() const
|
|
{
|
|
return Construct_initial_points(*this);
|
|
}
|
|
|
|
/*
|
|
* Returns a bounding box of the domain
|
|
*/
|
|
Bbox_3 bbox() const {
|
|
return this->bbox_.bbox();
|
|
}
|
|
|
|
/*
|
|
* Returns true if point `p` is in the domain. If `p` is in the
|
|
* domain, the parameter index is set to the index of the subdomain
|
|
* including $p$. It is set to the default value otherwise.
|
|
*/
|
|
struct Is_in_domain
|
|
{
|
|
Is_in_domain(const Labeled_mesh_domain_3& domain) : r_domain_(domain) {}
|
|
|
|
Subdomain operator()(const Point_3& p) const
|
|
{
|
|
// null(f(p)) means p is outside the domain
|
|
Subdomain_index index = (r_domain_.function_)(p);
|
|
if ( r_domain_.null(index) )
|
|
return Subdomain{};
|
|
else
|
|
return Subdomain{ index };
|
|
}
|
|
private:
|
|
const Labeled_mesh_domain_3& r_domain_;
|
|
};
|
|
|
|
// Returns Is_in_domain object
|
|
Is_in_domain is_in_domain_object() const { return Is_in_domain(*this); }
|
|
|
|
/*
|
|
* Returns `true` if the element `type` intersect properly any of the
|
|
* surface patches describing the either the domain boundary or some
|
|
* subdomain boundary.
|
|
* `Type` is either `Segment_3`, `Ray_3` or `Line_3`.
|
|
* Parameter index is set to the index of the intersected surface patch
|
|
* if `true` is returned and to the default `Surface_patch_index`
|
|
* value otherwise.
|
|
*/
|
|
struct Do_intersect_surface
|
|
{
|
|
Do_intersect_surface(const Labeled_mesh_domain_3& domain)
|
|
: r_domain_(domain) {}
|
|
|
|
Surface_patch operator()(const Segment_3& s) const
|
|
{
|
|
return this->operator()(s.source(), s.target());
|
|
}
|
|
|
|
Surface_patch operator()(const Ray_3& r) const
|
|
{
|
|
return clip_to_segment(r);
|
|
}
|
|
|
|
Surface_patch operator()(const Line_3& l) const
|
|
{
|
|
return clip_to_segment(l);
|
|
}
|
|
|
|
private:
|
|
// Returns true if points `a` and `b` do not belong to the same subdomain
|
|
// `index` is set to the surface index of subdomains f(a), f(b)
|
|
Surface_patch operator()(const Point_3& a, const Point_3& b) const
|
|
{
|
|
// If f(a) != f(b), then [a,b] intersects some surface. Here we consider
|
|
// [a,b] intersects surface_patch labeled <f(a),f(b)> (or <f(b),f(a)>).
|
|
// It may be false, further rafinement will improve precision
|
|
const Subdomain_index value_a = r_domain_.function_(a);
|
|
const Subdomain_index value_b = r_domain_.function_(b);
|
|
|
|
if ( value_a != value_b ) {
|
|
if( r_domain_.null(value_a) && r_domain_.null(value_b) )
|
|
return Surface_patch();
|
|
else
|
|
return Surface_patch(r_domain_.make_surface_index(value_a, value_b));
|
|
}
|
|
else
|
|
return Surface_patch();
|
|
}
|
|
|
|
/*
|
|
* Clips `query` to a segment `s`, and call `operator()(s)`
|
|
*/
|
|
template<typename Query>
|
|
Surface_patch clip_to_segment(const Query& query) const
|
|
{
|
|
const auto clipped = CGAL::intersection(query, r_domain_.bbox_);
|
|
if(clipped)
|
|
if(const Segment_3* s = std::get_if<Segment_3>(&*clipped))
|
|
return this->operator()(*s);
|
|
|
|
return Surface_patch();
|
|
}
|
|
|
|
private:
|
|
const Labeled_mesh_domain_3& r_domain_;
|
|
};
|
|
|
|
// Returns Do_intersect_surface object
|
|
Do_intersect_surface do_intersect_surface_object() const
|
|
{
|
|
return Do_intersect_surface(*this);
|
|
}
|
|
|
|
/*
|
|
* Returns a point in the intersection of the primitive `type`
|
|
* with some boundary surface.
|
|
* `Type` is either `Segment_3`, `Ray_3` or `Line_3`.
|
|
*/
|
|
struct Construct_intersection
|
|
{
|
|
Construct_intersection(const Labeled_mesh_domain_3& domain)
|
|
: r_domain_(domain) {}
|
|
|
|
Intersection operator()(const Segment_3& s) const
|
|
{
|
|
#ifndef CGAL_MESH_3_NO_LONGER_CALLS_DO_INTERSECT_3
|
|
CGAL_precondition(r_domain_.do_intersect_surface_object()(s) != std::nullopt);
|
|
#endif // NOT CGAL_MESH_3_NO_LONGER_CALLS_DO_INTERSECT_3
|
|
return this->operator()(s.source(),s.target());
|
|
}
|
|
|
|
Intersection operator()(const Ray_3& r) const
|
|
{
|
|
return clip_to_segment(r);
|
|
}
|
|
|
|
Intersection operator()(const Line_3& l) const
|
|
{
|
|
return clip_to_segment(l);
|
|
}
|
|
|
|
private:
|
|
/*
|
|
* Returns a point in the intersection of `[a,b]` with the surface
|
|
* `a` must be the source point, and `b` the out point. It is important
|
|
* because it drives bisection cuts.
|
|
* Indeed, the returned point is the first intersection of `[a,b]`
|
|
* with a subdomain surface.
|
|
*/
|
|
Intersection operator()(const Point_3& a, const Point_3& b) const
|
|
{
|
|
// Functors
|
|
typename BGT::Compute_squared_distance_3 squared_distance =
|
|
BGT().compute_squared_distance_3_object();
|
|
typename BGT::Construct_midpoint_3 midpoint =
|
|
BGT().construct_midpoint_3_object();
|
|
|
|
// Non const points
|
|
Point_3 p1 = a;
|
|
Point_3 p2 = b;
|
|
Point_3 mid = midpoint(p1, p2);
|
|
|
|
// Cannot be const: those values are modified below.
|
|
Subdomain_index value_at_p1 = r_domain_.function_(p1);
|
|
Subdomain_index value_at_p2 = r_domain_.function_(p2);
|
|
Subdomain_index value_at_mid = r_domain_.function_(mid);
|
|
|
|
// If both extremities are in the same subdomain,
|
|
// there is no intersection.
|
|
// Should only be able to happen during initial point generation.
|
|
if( value_at_p1 == value_at_p2 )
|
|
{
|
|
return Intersection();
|
|
}
|
|
if( r_domain_.null(value_at_p1) && r_domain_.null(value_at_p2) ) {
|
|
return Intersection();
|
|
}
|
|
|
|
// Else lets find a point (by bisection)
|
|
// Bisection ends when the point is near than error bound from surface
|
|
while(true)
|
|
{
|
|
// If the two points are enough close, then we return midpoint
|
|
if ( squared_distance(p1, p2) < r_domain_.squared_error_bound_ )
|
|
{
|
|
CGAL_assertion(value_at_p1 != value_at_p2 &&
|
|
! ( r_domain_.null(value_at_p1) && r_domain_.null(value_at_p2) ) );
|
|
const Surface_patch_index sp_index =
|
|
r_domain_.make_surface_index(value_at_p1, value_at_p2);
|
|
const Index index = r_domain_.index_from_surface_patch_index(sp_index);
|
|
return Intersection(mid, index, 2);
|
|
}
|
|
|
|
// Else we must go on
|
|
// Here we consider that p1(a) is the source point. Thus, we keep p1 and
|
|
// change p2 if f(p1)!=f(p2).
|
|
// That allows us to find the first intersection from a of [a,b] with
|
|
// a surface.
|
|
if ( value_at_p1 != value_at_mid &&
|
|
! ( r_domain_.null(value_at_p1) && r_domain_.null(value_at_mid) ) )
|
|
{
|
|
p2 = mid;
|
|
value_at_p2 = value_at_mid;
|
|
}
|
|
else
|
|
{
|
|
p1 = mid;
|
|
value_at_p1 = value_at_mid;
|
|
}
|
|
|
|
mid = midpoint(p1, p2);
|
|
value_at_mid = r_domain_.function_(mid);
|
|
}
|
|
}
|
|
|
|
// Clips `query` to a segment `s`, and call `operator()(s)`
|
|
template<typename Query>
|
|
Intersection clip_to_segment(const Query& query) const
|
|
{
|
|
const auto clipped = CGAL::intersection(query, r_domain_.bbox_);
|
|
if(clipped)
|
|
if(const Segment_3* s = std::get_if<Segment_3>(&*clipped))
|
|
return this->operator()(*s);
|
|
|
|
return Intersection();
|
|
}
|
|
|
|
private:
|
|
const Labeled_mesh_domain_3& r_domain_;
|
|
};
|
|
|
|
// Returns Construct_intersection object
|
|
Construct_intersection construct_intersection_object() const
|
|
{
|
|
return Construct_intersection(*this);
|
|
}
|
|
|
|
/*
|
|
* Returns the index to be stored in a vertex lying on the surface identified
|
|
* by `index`.
|
|
*/
|
|
Index index_from_surface_patch_index(const Surface_patch_index& index) const
|
|
{ return Index(index); }
|
|
|
|
/*
|
|
* Returns the index to be stored in a vertex lying in the subdomain
|
|
* identified by `index`.
|
|
*/
|
|
Index index_from_subdomain_index(const Subdomain_index& index) const
|
|
{ return Index(index); }
|
|
|
|
/*
|
|
* Returns the `Surface_patch_index` of the surface patch
|
|
* where lies a vertex with dimension 2 and index `index`.
|
|
*/
|
|
Surface_patch_index surface_patch_index(const Index& index) const
|
|
{ return Mesh_3::internal::get_index<Surface_patch_index>(index); }
|
|
|
|
/*
|
|
* Returns the index of the subdomain containing a vertex
|
|
* with dimension 3 and index `index`.
|
|
*/
|
|
Subdomain_index subdomain_index(const Index& index) const
|
|
{ return Mesh_3::internal::get_index<Subdomain_index>(index); }
|
|
|
|
// -----------------------------------
|
|
// Backward Compatibility
|
|
// -----------------------------------
|
|
#ifndef CGAL_MESH_3_NO_DEPRECATED_SURFACE_INDEX
|
|
typedef Surface_patch_index Surface_index;
|
|
|
|
Index index_from_surface_index(const Surface_index& index) const
|
|
{ return index_from_surface_patch_index(index); }
|
|
|
|
Surface_index surface_index(const Index& index) const
|
|
{ return surface_patch_index(index); }
|
|
#endif // CGAL_MESH_3_NO_DEPRECATED_SURFACE_INDEX
|
|
// -----------------------------------
|
|
// End backward Compatibility
|
|
// -----------------------------------
|
|
|
|
protected:
|
|
// Returns Surface_patch_index from `i` and `j`
|
|
Surface_patch_index make_surface_index(const Subdomain_index i,
|
|
const Subdomain_index j) const
|
|
{
|
|
if(i < j)
|
|
return this->cstr_s_p_index(i, j);
|
|
else
|
|
return this->cstr_s_p_index(j, i);
|
|
}
|
|
|
|
// Returns the bounding sphere of an Iso_cuboid_3
|
|
Sphere_3 bounding_sphere(const Iso_cuboid_3& bbox) const
|
|
{
|
|
typename BGT::Construct_sphere_3 sphere = BGT().construct_sphere_3_object();
|
|
return sphere((bbox.min)(), (bbox.max)());
|
|
}
|
|
|
|
template <typename Image_word_type,
|
|
typename FT, typename FT2, typename Functor>
|
|
static
|
|
Function
|
|
create_gray_image_wrapper_with_known_word_type
|
|
(const CGAL::Image_3& image,
|
|
const FT& iso_value,
|
|
const Functor& image_values_to_subdomain_indices,
|
|
const FT2& value_outside)
|
|
{
|
|
using Mesh_3::internal::Create_gray_image_values_to_subdomain_indices;
|
|
typedef Create_gray_image_values_to_subdomain_indices<Functor> C_i_v_t_s_i;
|
|
typedef typename C_i_v_t_s_i::type Image_values_to_subdomain_indices;
|
|
Image_values_to_subdomain_indices transform_fct =
|
|
C_i_v_t_s_i()(image_values_to_subdomain_indices, iso_value);
|
|
|
|
typedef Mesh_3::Image_to_labeled_function_wrapper<Image_word_type,
|
|
double,
|
|
Subdomain_index,
|
|
false> Wrapper;
|
|
return Wrapper(image,
|
|
transform_fct,
|
|
value_outside) ;
|
|
}
|
|
|
|
template <typename FT, typename FT2, typename Functor>
|
|
static
|
|
Function
|
|
create_gray_image_wrapper(const CGAL::Image_3& image,
|
|
const FT& iso_value,
|
|
const Functor& image_values_to_subdomain_indices,
|
|
const FT2& value_outside)
|
|
{
|
|
CGAL_IMAGE_IO_CASE(image.image(),
|
|
return create_gray_image_wrapper_with_known_word_type<Word>
|
|
(image,
|
|
iso_value,
|
|
image_values_to_subdomain_indices,
|
|
value_outside);
|
|
);
|
|
CGAL_error_msg("This place should never be reached, because it would mean "
|
|
"the image word type is a type that is not handled by "
|
|
"CGAL_ImageIO.");
|
|
return Function();
|
|
}
|
|
|
|
template <typename Image_word_type,
|
|
typename FT, typename Functor>
|
|
static
|
|
Function
|
|
create_labeled_image_wrapper_with_known_word_type
|
|
(const CGAL::Image_3& image,
|
|
const Functor& image_values_to_subdomain_indices,
|
|
const FT& value_outside)
|
|
{
|
|
using Mesh_3::internal::Create_labeled_image_values_to_subdomain_indices;
|
|
typedef Create_labeled_image_values_to_subdomain_indices<Functor> C_i_v_t_s_i;
|
|
typedef typename C_i_v_t_s_i::type Image_values_to_subdomain_indices;
|
|
Image_values_to_subdomain_indices transform_fct =
|
|
C_i_v_t_s_i()(image_values_to_subdomain_indices);
|
|
|
|
typedef Mesh_3::Image_to_labeled_function_wrapper<Image_word_type,
|
|
int,
|
|
Subdomain_index> Wrapper;
|
|
return Wrapper(image,
|
|
transform_fct,
|
|
transform_fct(value_outside));
|
|
}
|
|
|
|
template <typename Image_word_type,
|
|
typename FT, typename Functor>
|
|
static
|
|
Function
|
|
create_weighted_labeled_image_wrapper_with_know_word_type
|
|
(const CGAL::Image_3& image,
|
|
const CGAL::Image_3& weights,
|
|
const Functor& image_values_to_subdomain_indices,
|
|
const FT& value_outside)
|
|
{
|
|
using Mesh_3::internal::Create_labeled_image_values_to_subdomain_indices;
|
|
typedef Create_labeled_image_values_to_subdomain_indices<Functor> C_i_v_t_s_i;
|
|
typedef typename C_i_v_t_s_i::type Image_values_to_subdomain_indices;
|
|
Image_values_to_subdomain_indices transform_fct =
|
|
C_i_v_t_s_i()(image_values_to_subdomain_indices);
|
|
|
|
typedef Mesh_3::Image_plus_weights_to_labeled_function_wrapper<
|
|
Image_word_type,
|
|
int, //interpolation type
|
|
unsigned char, // Weights_type,
|
|
Subdomain_index> Wrapper;
|
|
return Wrapper(image,
|
|
weights,
|
|
transform_fct,
|
|
transform_fct(value_outside));
|
|
}
|
|
|
|
template <typename FT, typename Functor>
|
|
static
|
|
Function
|
|
create_labeled_image_wrapper(const CGAL::Image_3& image,
|
|
const Functor& image_values_to_subdomain_indices,
|
|
const FT& value_outside)
|
|
{
|
|
CGAL_IMAGE_IO_CASE(image.image(),
|
|
return create_labeled_image_wrapper_with_known_word_type<Word>
|
|
(image,
|
|
image_values_to_subdomain_indices,
|
|
value_outside);
|
|
);
|
|
CGAL_error_msg("This place should never be reached, because it would mean "
|
|
"the image word type is a type that is not handled by "
|
|
"CGAL_ImageIO.");
|
|
return Function();
|
|
}
|
|
|
|
template <typename FT, typename Functor>
|
|
static
|
|
Function
|
|
create_weighted_labeled_image_wrapper(const CGAL::Image_3& image,
|
|
const CGAL::Image_3& weights,
|
|
const Functor& image_values_to_subdomain_indices,
|
|
const FT& value_outside)
|
|
{
|
|
CGAL_IMAGE_IO_CASE(image.image(),
|
|
return create_weighted_labeled_image_wrapper_with_know_word_type<Word>
|
|
(image,
|
|
weights,
|
|
image_values_to_subdomain_indices,
|
|
value_outside);
|
|
);
|
|
CGAL_error_msg("This place should never be reached, because it would mean "
|
|
"the image word type is a type that is not handled by "
|
|
"CGAL_ImageIO.");
|
|
return Function();
|
|
}
|
|
|
|
static
|
|
Construct_surface_patch_index
|
|
create_construct_surface_patch_index(const Null_functor&) {
|
|
return Impl_details::construct_pair_functor();
|
|
}
|
|
|
|
template <typename Functor>
|
|
static
|
|
Construct_surface_patch_index
|
|
create_construct_surface_patch_index(const Functor& functor) {
|
|
return functor;
|
|
}
|
|
|
|
static Null create_null_subdomain_index(const Null_functor&) {
|
|
return Null_subdomain_index();
|
|
}
|
|
|
|
template <typename Functor>
|
|
static Null create_null_subdomain_index(const Functor& functor) {
|
|
return functor;
|
|
}
|
|
|
|
public:
|
|
// Returns bounding box
|
|
const Iso_cuboid_3& bounding_box() const { return this->bbox_; }
|
|
#endif //DOXYGEN_RUNNING
|
|
|
|
}; // end class Labeled_mesh_domain_3
|
|
|
|
#ifndef DOXYGEN_RUNNING
|
|
//-------------------------------------------------------
|
|
// Method implementation
|
|
//-------------------------------------------------------
|
|
template<class BGT, class Subdomain_index, class Surface_patch_index>
|
|
template<class OutputIterator>
|
|
OutputIterator
|
|
Labeled_mesh_domain_3<BGT, Subdomain_index, Surface_patch_index>::
|
|
Construct_initial_points::operator()(OutputIterator pts,
|
|
const int nb_points) const
|
|
{
|
|
// Create point_iterator on and in bounding_sphere
|
|
typedef Random_points_on_sphere_3<Point_3> Random_points_on_sphere_3;
|
|
typedef Random_points_in_sphere_3<Point_3> Random_points_in_sphere_3;
|
|
|
|
const FT squared_radius = BGT().compute_squared_radius_3_object()(
|
|
r_domain_.bounding_sphere(r_domain_.bbox_));
|
|
|
|
const double radius = std::sqrt(CGAL::to_double(squared_radius));
|
|
|
|
CGAL::Random& rng = *(r_domain_.p_rng_);
|
|
Random_points_on_sphere_3 random_point_on_sphere(radius, rng);
|
|
Random_points_in_sphere_3 random_point_in_sphere(radius, rng);
|
|
|
|
// Get some functors
|
|
typename BGT::Construct_segment_3 segment_3 =
|
|
BGT().construct_segment_3_object();
|
|
typename BGT::Construct_vector_3 vector_3 =
|
|
BGT().construct_vector_3_object();
|
|
typename BGT::Construct_translated_point_3 translate =
|
|
BGT().construct_translated_point_3_object();
|
|
typename BGT::Construct_center_3 center = BGT().construct_center_3_object();
|
|
|
|
// Get translation from origin to sphere center
|
|
Point_3 center_pt = center(r_domain_.bounding_sphere(r_domain_.bbox_));
|
|
const Vector_3 sphere_translation = vector_3(CGAL::ORIGIN, center_pt);
|
|
|
|
// Create nb_point points
|
|
int n = nb_points;
|
|
#ifdef CGAL_MESH_3_VERBOSE
|
|
std::cerr << "construct initial points (nb_points: " << nb_points << ")\n";
|
|
#endif
|
|
while ( 0 != n )
|
|
{
|
|
// Get a random segment
|
|
const Point_3 random_point = translate(*random_point_on_sphere,
|
|
sphere_translation);
|
|
const Segment_3 random_seg = segment_3(center_pt, random_point);
|
|
|
|
// Add the intersection to the output if it exists
|
|
Surface_patch surface = r_domain_.do_intersect_surface_object()(random_seg);
|
|
if ( surface )
|
|
{
|
|
const Point_3 intersect_pt = std::get<0>(
|
|
r_domain_.construct_intersection_object()(random_seg));
|
|
*pts++ = std::make_pair(intersect_pt,
|
|
r_domain_.index_from_surface_patch_index(*surface));
|
|
--n;
|
|
|
|
#ifdef CGAL_MESH_3_VERBOSE
|
|
std::cerr << boost::format("\r \r"
|
|
"%1%/%2% initial point(s) found...")
|
|
% (nb_points - n)
|
|
% nb_points;
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
// Get a new random point into sphere as center of object
|
|
// It may be necessary if the center of the domain is empty, e.g. torus
|
|
// In general case, it is good for input point dispersion
|
|
++random_point_in_sphere;
|
|
center_pt = translate(*random_point_in_sphere, sphere_translation);
|
|
}
|
|
++random_point_on_sphere;
|
|
}
|
|
|
|
#ifdef CGAL_MESH_3_VERBOSE
|
|
std::cerr << "\n";
|
|
#endif
|
|
return pts;
|
|
}
|
|
#endif
|
|
|
|
} // end namespace CGAL
|
|
|
|
#include <CGAL/enable_warnings.h>
|
|
|
|
#endif // CGAL_LABELED_MESH_DOMAIN_3_H
|