14#include <lagrange/Mesh.h>
15#include <lagrange/MeshTrait.h>
16#include <lagrange/attributes/map_attributes.h>
17#include <lagrange/common.h>
18#include <lagrange/create_mesh.h>
19#include <lagrange/internal/sortrows.h>
20#include <lagrange/legacy/inline.h>
21#include <lagrange/mesh_cleanup/remove_topologically_degenerate_triangles.h>
22#include <lagrange/reorder_mesh_vertices.h>
23#include <lagrange/utils/range.h>
31namespace legacy_internal {
33template <
typename DerivedA,
typename Index>
35 const Eigen::DenseBase<DerivedA>& A,
36 Index& num_unique_vertices,
37 std::vector<Index>& forward_mapping)
43 constexpr bool is_ascending =
true;
44 lagrange::internal::sortrows(A, is_ascending, __, IM);
48 const Index num_rows = safe_cast<Index>(A.rows());
49 forward_mapping.resize(num_rows);
50 num_unique_vertices = 0;
51 forward_mapping[IM(0)] = num_unique_vertices;
52 for (
auto i : range<Index>(1, num_rows)) {
53 if ((A.row(IM(i)) != A.row(IM(i - 1)))) {
54 ++num_unique_vertices;
56 forward_mapping[IM(i)] =
static_cast<Index
>(num_unique_vertices);
58 ++num_unique_vertices;
75template <
typename MeshType>
78 const std::vector<std::string>& vertex_attribute_names,
79 const std::vector<std::string>& indexed_attribute_names = {})
81 static_assert(MeshTrait<MeshType>::is_mesh(),
"Input type is not Mesh");
83 mesh.get_vertex_per_facet() == 3,
84 std::string(
"vertex per facet is ") + std::to_string(mesh.get_vertex_per_facet()));
86 using Scalar =
typename MeshType::Scalar;
87 using Index =
typename MeshType::Index;
88 using Entries = Eigen::Matrix<Scalar, Eigen::Dynamic, Eigen::Dynamic>;
90 const auto& vertices = mesh.get_vertices();
91 const auto& facets = mesh.get_facets();
92 const Index num_vertices = mesh.get_num_vertices();
93 const Index num_facets = mesh.get_num_facets();
95 const Index dim = mesh.get_dim();
97 for (
const auto& attr_name : vertex_attribute_names) {
99 num_cols +=
static_cast<Index
>(mesh.get_vertex_attribute(attr_name).cols());
101 for (
const auto& attr_name : indexed_attribute_names) {
103 auto attr = mesh.get_indexed_attribute(attr_name);
104 num_cols +=
static_cast<Index
>(std::get<0>(attr).cols());
107 Eigen::Matrix<Scalar, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor> vertices_and_keys;
108 vertices_and_keys.resize(num_vertices, num_cols);
109 vertices_and_keys.leftCols(dim) = vertices;
110 Index col_count = dim;
112 for (
const auto& attr_name : vertex_attribute_names) {
113 const auto& attr = mesh.get_vertex_attribute(attr_name);
114 vertices_and_keys.block(0, col_count, num_vertices, attr.cols()) = attr;
115 col_count +=
static_cast<Index
>(attr.cols());
117 for (
const auto& attr_name : indexed_attribute_names) {
118 const auto attr = mesh.get_indexed_attribute(attr_name);
119 const auto& attr_values = std::get<0>(attr);
120 const auto& attr_indices = std::get<1>(attr);
121 for (
auto i :
range(num_facets)) {
122 vertices_and_keys.row(facets(i, 0)).segment(col_count, attr_values.cols()) =
123 attr_values.row(attr_indices(i, 0));
124 vertices_and_keys.row(facets(i, 1)).segment(col_count, attr_values.cols()) =
125 attr_values.row(attr_indices(i, 1));
126 vertices_and_keys.row(facets(i, 2)).segment(col_count, attr_values.cols()) =
127 attr_values.row(attr_indices(i, 2));
129 col_count +=
static_cast<Index
>(attr_values.cols());
132 Entries unique_vertices_and_keys;
133 std::vector<Index> forward_mapping;
134 Index num_unique_vertices;
135 legacy_internal::unique_rows(vertices_and_keys, num_unique_vertices, forward_mapping);
137 la_runtime_assert(safe_cast<Eigen::Index>(forward_mapping.size()) == vertices.rows());
139 if (num_unique_vertices < mesh.get_num_vertices()) {
140 auto mesh2 = reorder_mesh_vertices(mesh, forward_mapping);
141 mesh2 = remove_topologically_degenerate_triangles(*mesh2);
145 auto mesh2 =
create_mesh(mesh.get_vertices(), mesh.get_facets());
168template <
typename MeshType>
171 const std::string& key_name =
"",
174 std::vector<std::string> vertex_attributes, indexed_attributes;
175 if (key_name !=
"") {
176 vertex_attributes.push_back(key_name);
178 if (with_uv && mesh.is_uv_initialized()) {
179 indexed_attributes.push_back(
"uv");
#define la_runtime_assert(...)
Runtime assertion check.
Definition: assert.h:169
internal::Range< Index > range(Index end)
Returns an iterable object representing the range [0, end).
Definition: range.h:176
void map_attributes(const SurfaceMesh< Scalar, Index > &source_mesh, SurfaceMesh< Scalar, Index > &target_mesh, span< const Index > mapping_data, span< const Index > mapping_offsets={}, const MapAttributesOptions &options={})
Map attributes from the source mesh to the target mesh.
Definition: map_attributes.cpp:26
Main namespace for Lagrange.
Definition: AABBIGL.h:30
auto create_mesh(const Eigen::MatrixBase< DerivedV > &vertices, const Eigen::MatrixBase< DerivedF > &facets)
This function create a new mesh given the vertex and facet arrays by copying data into the Mesh objec...
Definition: create_mesh.h:39
void remove_duplicate_vertices(SurfaceMesh< Scalar, Index > &mesh, const RemoveDuplicateVerticesOptions &options={})
Removes duplicate vertices from a mesh.
Definition: remove_duplicate_vertices.cpp:33