Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

PMP: Non-rigid mesh registration #8329

Open
wants to merge 45 commits into
base: master
Choose a base branch
from
Open
Changes from 36 commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
4c41bda
Non-rigid mesh registration
soesau Jul 2, 2024
97ce59f
added a preliminary license header
soesau Jul 2, 2024
d1de633
fixes for CI
soesau Jul 2, 2024
2f2551b
revert Point_set_3 dependency
soesau Jul 2, 2024
b73ce3e
working on registration and doc
soesau Aug 5, 2024
db8ffbe
doc fix
soesau Aug 5, 2024
d981129
pass on doc
soesau Aug 6, 2024
ba01e61
update on doc
soesau Aug 8, 2024
2fdfb80
update on doc
soesau Aug 8, 2024
06a3187
updated changes.md
soesau Aug 8, 2024
29d1fd1
typo; whitespace
afabri Aug 13, 2024
eb79dfe
proper license; internal::registration; move in CMakeLists
afabri Aug 20, 2024
eebeff2
No 'should' after discussion with @robertod
afabri Aug 20, 2024
c013d13
calculate -> compute https://langeek.co/en/grammar/course/1822
afabri Aug 21, 2024
8667d66
pass on doc
soesau Aug 21, 2024
fb1bb51
Merge branch 'master' into PMP-Non_rigid_registration-GF
soesau Aug 21, 2024
3a274b3
doc of CorrespondenceRange
soesau Aug 21, 2024
3e08965
moving correspondences into named parameters
soesau Aug 21, 2024
5a4fae7
added std::cref to doc of correspondences
soesau Aug 21, 2024
b0e5301
added apply_non_rigid_transformation to CHANGES.md
soesau Oct 2, 2024
4b14289
Merge branch 'master' into PMP-Non_rigid_registration-GF
soesau Oct 2, 2024
357d345
bug fixes
soesau Oct 2, 2024
ab51f73
removed unused variable
soesau Oct 2, 2024
7e0022c
fix warning
soesau Oct 4, 2024
36f59c3
compatibility with Eigen3.3
soesau Oct 8, 2024
d328aac
fix warnings
soesau Oct 9, 2024
09ca6d5
making it easy to switch between both arap terms
soesau Oct 9, 2024
a093496
changes following review
soesau Oct 16, 2024
a948e53
Merge remote-tracking branch 'cgal/master' into PMP-Non_rigid_registr…
soesau Oct 16, 2024
6b363ec
adapt CHANGES.md
soesau Oct 16, 2024
7a10001
another pass for review
soesau Oct 17, 2024
b8937ab
adding proper edge weights for new arap weight as suggested by Roberto
soesau Oct 30, 2024
47b7f67
switching licenses from LGPL to GPL
soesau Nov 18, 2024
688f4a1
moving ARAP visitor into Weights package
soesau Nov 18, 2024
2159bee
vertex normals are estimated if not provided
soesau Nov 18, 2024
37d66b7
documentation of Deformation_algorithm_tag
soesau Nov 20, 2024
d1059ac
doc fix
soesau Nov 21, 2024
608de95
fixing vertex normal rotation after registration
soesau Nov 21, 2024
ff97fb9
doc updates
soesau Nov 21, 2024
09bbae9
doc of geom_traits
soesau Nov 21, 2024
1f897b8
fixed warning
soesau Nov 22, 2024
0a03248
Update Polygon_mesh_processing/doc/Polygon_mesh_processing/Polygon_me…
soesau Dec 3, 2024
fe9e8d9
renaming maximum_matching_distance to maximal_matching_distance
soesau Dec 10, 2024
b2399b9
Merge branch 'master' into PMP-Non_rigid_registration-GF
soesau Dec 10, 2024
c8bb801
CI fix
soesau Dec 10, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions Installation/CHANGES.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
# Release History

## [Release 6.1](https://github.com/CGAL/cgal/releases/tag/v6.1)

### [Polygon Mesh Processing](https://doc.cgal.org/6.1/Manual/packages.html#PkgPolygonMeshProcessing)

- Added two functions for non-rigid registration of mesh to mesh and mesh to point cloud [`CGAL::Polygon_mesh_processing::non_rigid_mesh_to_mesh_registration()`](https://doc.cgal.org/6.1/Polygon_mesh_processing/group__PMP__registration__grp.html#ga04e4101a21663bebb689de30bb7d2f4e) and [`CGAL::Polygon_mesh_processing::non_rigid_mesh_to_points_registration()`](https://doc.cgal.org/6.1/Polygon_mesh_processing/group__PMP__registration__grp.html#ga466fd5b15e5de0b65faf48947a7f2bef) as well as a function to apply a transformation [`CGAL::Polygon_mesh_processing::apply_non_rigid_transformation()`](https://doc.cgal.org/6.1/Polygon_mesh_processing/group__PMP__registration__grp.html#gac022fccce4796cba9ff710865154b127)

## [Release 6.0.1](https://github.com/CGAL/cgal/releases/tag/v6.0.1)

### [Poisson Surface Reconstruction](https://doc.cgal.org/6.0.1/Manual/packages.html#PkgPoissonSurfaceReconstruction3)
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
// Copyright (c) 2016 GeometryFactory SARL (France).
// All rights reserved.
//
// This file is part of CGAL (www.cgal.org)
//
// $URL$
// $Id$
// SPDX-License-Identifier: LGPL-3.0-or-later OR LicenseRef-Commercial
//
// Author(s) : Andreas Fabri
//
// Warning: this file is generated, see include/CGAL/license/README.md

#ifndef CGAL_LICENSE_POLYGON_MESH_PROCESSING_REGISTRATION_H
#define CGAL_LICENSE_POLYGON_MESH_PROCESSING_REGISTRATION_H

#include <CGAL/config.h>
#include <CGAL/license.h>

#ifdef CGAL_POLYGON_MESH_PROCESSING_REGISTRATION_COMMERCIAL_LICENSE

# if CGAL_POLYGON_MESH_PROCESSING_REGISTRATION_COMMERCIAL_LICENSE < CGAL_RELEASE_DATE

# if defined(CGAL_LICENSE_WARNING)

CGAL_pragma_warning("Your commercial license for CGAL does not cover "
"this release of the Polygon Mesh Processing - Registration package.")
# endif

# ifdef CGAL_LICENSE_ERROR
# error "Your commercial license for CGAL does not cover this release \
of the Polygon Mesh Processing - Registration package. \
You get this error, as you defined CGAL_LICENSE_ERROR."
# endif // CGAL_LICENSE_ERROR

# endif // CGAL_POLYGON_MESH_PROCESSING_REGISTRATION_COMMERCIAL_LICENSE < CGAL_RELEASE_DATE

#else // no CGAL_POLYGON_MESH_PROCESSING_REGISTRATION_COMMERCIAL_LICENSE

# if defined(CGAL_LICENSE_WARNING)
CGAL_pragma_warning("\nThe macro CGAL_POLYGON_MESH_PROCESSING_REGISTRATION_COMMERCIAL_LICENSE is not defined."
"\nYou use the CGAL Polygon Mesh Processing - Registration package under "
"the terms of the GPLv3+.")
# endif // CGAL_LICENSE_WARNING

# ifdef CGAL_LICENSE_ERROR
# error "The macro CGAL_POLYGON_MESH_PROCESSING_REGISTRATION_COMMERCIAL_LICENSE is not defined.\
You use the CGAL Polygon Mesh Processing - Registration package under the terms of \
the GPLv3+. You get this error, as you defined CGAL_LICENSE_ERROR."
# endif // CGAL_LICENSE_ERROR

#endif // no CGAL_POLYGON_MESH_PROCESSING_REGISTRATION_COMMERCIAL_LICENSE

#endif // CGAL_LICENSE_POLYGON_MESH_PROCESSING_REGISTRATION_H
Original file line number Diff line number Diff line change
@@ -48,6 +48,10 @@
/// Functions to compute the distance between meshes, between a mesh and a point set and between a point set and a mesh.
/// \ingroup PkgPolygonMeshProcessingRef

/// \defgroup PMP_registration_grp Registration Functions
/// Functions to compute the transformation for mapping one mesh onto another or onto a set of points.
/// \ingroup PkgPolygonMeshProcessingRef

soesau marked this conversation as resolved.
Show resolved Hide resolved
/// \defgroup PMP_corefinement_grp Corefinement and Boolean Operations
/// Functions to corefine triangulated surface meshes and compute triangulated
/// surface meshes of the union, difference and intersection
@@ -257,6 +261,11 @@ The page \ref bgl_namedparameters "Named Parameters" describes their usage.
- `CGAL::Polygon_mesh_processing::region_growing_of_planes_on_faces()`
- `CGAL::Polygon_mesh_processing::detect_corners_of_regions()`

\cgalCRPSection{Registration Functions}
- `CGAL::Polygon_mesh_processing::non_rigid_mesh_to_points_registration()`
- `CGAL::Polygon_mesh_processing::non_rigid_mesh_to_mesh_registration()`
- `CGAL::Polygon_mesh_processing::apply_non_rigid_transformation()`


\cgalCRPSection{Miscellaneous}
- `CGAL::Polygon_mesh_slicer`
Original file line number Diff line number Diff line change
@@ -261,7 +261,7 @@ mesh relaxation function can be used.
- Shape smoothing: `CGAL::Polygon_mesh_processing::smooth_shape()`
moves vertices towards a weighted barycenter of their neighbors along the mean curvature flow.
The curvature flow algorithm for shape smoothing is based on Desbrun et al. \cgalCite{cgal:dmsb-ifamdcf-99} and on Kazhdan et al. \cgalCite{kazhdan2012can}.
The algorithm uses the mean curvature flow to calculate the translation of vertices along the surface normal with a speed equal
The algorithm uses the mean curvature flow to compute the translation of vertices along the surface normal with a speed equal
to the mean curvature of the area that is being smoothed. This means that vertices on sharp corners slide faster.
If the region around a vertex is flat, this vertex does not move (zero curvature). To avoid the formation
of undesirable neck pinches (cylindrical surface areas that form singularities) the algorithm slows
@@ -1416,6 +1416,84 @@ and the number of surface patches that are separated by these edges.

\cgalExample{Polygon_mesh_processing/detect_features_example.cpp}

\section PMPNonRigidRegistration Non-Rigid Mesh Registration

Registration of one mesh onto another is often required to map between multiple acquisitions of an object or between an acquisition and a CAD model, e.g., in quality assurance. A simple linear transformation is only sufficient in the case of rigid objects.

The functions `CGAL::Polygon_mesh_processing::non_rigid_mesh_to_mesh_registration()` and `CGAL::Polygon_mesh_processing::non_rigid_mesh_to_points_registration()` compute a non-rigid registration from a source mesh onto a target mesh or a target oriented point cloud. For each vertex in the source mesh a translation vector and a rotation matrix is returned.
The registration is formulated as an energy to minimize the distance between source and target. The energy is iteratively minimized by solving sparse linear systems and finding closest rotation matrices.
The core algorithm is ICP, iterative closest point. In each iteration, the method identifies vertex or point pairs between the intermediate, initially a copy of the source, and the target and subsequently computes the transformations which minimize the distance between them. The intermediate is updated after each iteration and converges towards the target.

\f{equation}{
\mathbf{E_{reg}} = w_1\mathbf{E_{match}} + w_2\mathbf{E_{point\_to\_plane}} + w_3\mathbf{E_{arap}},
\label{eq:e_reg}
\f}

with the single energy terms:

\f{equation}{
\mathbf{E_{match}} = \sum_{\mathbf{v}_i, \mathbf{\tilde{v}}_i \in P} \left\|\mathbf{\tilde{v}}_i - (\mathbf{R}_i\mathbf{v}_i + \mathbf{t_i})\right\|^2_2,
\label{eq:e_match}
\f}

\f{equation}{
\mathbf{E_{point\_to\_plane}} = \sum_{\mathbf{v}_i, \mathbf{\tilde{v}}_i \in P} \left\|(\mathbf{\tilde{v}}_i - (\mathbf{R}_i\mathbf{v}_i + \mathbf{t_i}))\mathbf{\tilde{n}}_i\right\|^2_2,
\label{eq:e_plane}
\f}

\f{equation}{
\mathbf{E_{arap}} =
\sum_{\mathbf{v}_i \in M}
\sum_{\mathbf{v}_j \in N(\mathbf{v}_i)}
\left\| (\mathbf{R}_i\mathbf{v}_i + \mathbf{t}_i - \mathbf{R}_j\mathbf{v}_j + \mathbf{t}_j) - \mathbf{R}_i(\mathbf{v}_i - \mathbf{v}_j) \right\|^2_2,
\label{eq:e_arap}
\f}

where:
- \f$w_1, w_2\f$ and \f$w_3\f$ denote the weights for the energy terms.
- \f$\mathbf{P}\f$ is the set of vertex/point pairs with \f$\mathbf{v}_i \in\f$ intermediate and \f$\mathbf{\tilde{v}}_i \in\f$ target. It is recreated in each iteration.
- \f$\mathbf{R}_i\f$ is a \f$ 3 \times 3 \f$ rotation matrix for source vertex \f$\mathbf{v}_i\f$.
- \f$\mathbf{t}_i\f$ is the translation vector for source vertex \f$\mathbf{v}_i\f$.
- \f$\mathbf{M}\f$ is the intermediate mesh (initially equal to the source mesh).
- \f$N(\mathbf{v}_i)\f$ denotes the set of vertices adjacent to \f$\mathbf{v}_i\f$.

The \f$\mathbf{E_{match}}\f$ energy penalizes the distance between point pairs \f$\mathbf{v}_i\f$ and \f$\mathbf{\tilde{v}}_i\f$. The point pairs are restablished after each iteration. In addition, the method can take fixed point pairs of correspondences between source and target. This greatly improves the quality of the registration and is generally required when the initial alignment of the source and target is poor.

The \f$\mathbf{E_{point\_to\_plane}}\f$ energy is similar to \f$\mathbf{E_{match}}\f$, but instead penalizes the distance to the plane given by the target vertex and its normal. This energy compensates for different discretizations, i.e., for source and target mesh that were not created by deforming one mesh into the other.

The \f$\mathbf{E_{arap}}\f$ energy, as-rigid-as-possible, controls the rigidity of the transformation. It measures the deformation of the edges connected to each vertex that results from applying an individual transformation to each vertex of the source mesh.

\subsection PMPNonRigidRegistrationParameters Parameters

The algorithm has six parameters:
- `point_to_point_energy`:
Sets the weight `w1` of the \f$\mathbf{E_{match}}\f$. Penalizes the distance between vertices of intermediate and target. A higher value promotes a tighter fit.
- `point_to_plane_energy`:
Sets the weight `w2` of the \f$\mathbf{E_{point\_to\_plane}}\f$. Also penalizes the distances between the meshes, but is more forgiving towards different discretization.
- `as_rigid_as_possible_energy`:
Controls the rigidity weight `w3` of the \f$\mathbf{E_{arap}}\f$. Penalizes the deformation of the source and may thus prefers rigidity over a tight fit.
soesau marked this conversation as resolved.
Show resolved Hide resolved
- maximum_matching_distance:
soesau marked this conversation as resolved.
Show resolved Hide resolved
The maximal distance for finding vertex/point pairs between the intermediate and target. A pair is ignored during an iteration if the specified distance is exceeded.
- number_of_iterations:
soesau marked this conversation as resolved.
Show resolved Hide resolved
The number of ICP iterations. The default value is 50.
- correspondences:
Optional fixed vertex/point pairs can be provided to guide the registration. These are especially helpful when source and target are not roughly aligned initially.

\cgalFigureAnchor{registration_results}
<center>
<img src="non-rigid_registration.png" style="max-width:80%;"/>
</center>
\cgalFigureCaptionBegin{registration_results}
sloriot marked this conversation as resolved.
Show resolved Hide resolved
Non-rigid registration using five correspondences on top of the head and at the tips of hand and feet. A low rigidity leads to a distortion and a self-intersection of the transformed mesh. The topology of the mesh as well as the number of vertices and faces remain unchanged. The lack of correspondences also causes self-intersection and leads to a flattening of the left arm.
\cgalFigureCaptionEnd

\subsection PMPNonRigidRegistrationExample Non-rigid registration Example
soesau marked this conversation as resolved.
Show resolved Hide resolved

In the following example the bear_simple.off model is registered onto the bear_simple_bis.off model. The resulting transformed mesh is then saved as off file with a name depending on the chosen parameters.

\cgalExample{Polygon_mesh_processing/non_rigid_mesh_registration_example.cpp}


\section PMPHistory Implementation History

A first version of this package was started by Ilker %O. Yaz and Sébastien Loriot.
@@ -1441,5 +1519,7 @@ used as a reference during the project.

The curvature-based sizing field version of isotropic remeshing was added by Ivan Pađen during GSoC 2023, under the supervision of Sébastien Loriot and Jane Tournois.

The non-rigid mesh registration was developed by Roberto M. Dyke. Sven Oesau integrated the method into \cgal.

*/
} /* namespace CGAL */
Original file line number Diff line number Diff line change
@@ -48,5 +48,6 @@
\example Polygon_mesh_processing/remesh_almost_planar_patches.cpp
\example Polygon_mesh_processing/sample_example.cpp
\example Polygon_mesh_processing/soup_autorefinement.cpp
\example Polygon_mesh_processing/non_rigid_mesh_registration_example.cpp
\example Polygon_mesh_processing/isotropic_remeshing_with_allow_move.cpp
*/
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Original file line number Diff line number Diff line change
@@ -90,6 +90,8 @@ if(TARGET CGAL::Eigen3_support)
target_link_libraries(interpolated_corrected_curvatures_PH PUBLIC CGAL::Eigen3_support)
create_single_source_cgal_program("interpolated_corrected_curvatures_vertex.cpp")
target_link_libraries(interpolated_corrected_curvatures_vertex PUBLIC CGAL::Eigen3_support)
create_single_source_cgal_program("non_rigid_mesh_registration_example.cpp")
target_link_libraries(non_rigid_mesh_registration_example PUBLIC CGAL::Eigen3_support)
else()
message(STATUS "NOTICE: Examples that use Eigen will not be compiled.")
endif()
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
1136 824
1229 1747
1089 1426
1914 129
1957 144
1714 1293
Loading