Wildmeshing Toolkit
Loading...
Searching...
No Matches
extract_child_mesh_from_tag.cpp
Go to the documentation of this file.
2#include <map>
3#include <numeric>
4#include <wmtk/EdgeMesh.hpp>
5#include <wmtk/Mesh.hpp>
7#include <wmtk/PointMesh.hpp>
8#include <wmtk/Primitive.hpp>
9#include <wmtk/TetMesh.hpp>
10#include <wmtk/TriMesh.hpp>
13#include <wmtk/utils/Logger.hpp>
14#include "internal/TupleTag.hpp"
15
16namespace wmtk::multimesh::utils {
17
18
19template <typename T>
20
22 Mesh& m,
24 const T& tag_value,
25 bool child_is_free)
26{
27 auto tags = m.create_const_accessor(tag_handle);
28 const PrimitiveType pt = tag_handle.primitive_type();
29
30 std::vector<Tuple> tagged_tuples;
31 for (const Tuple& t : m.get_all(pt)) {
32 if (tags.const_scalar_attribute(t) == tag_value) {
33 tagged_tuples.emplace_back(t);
34 }
35 }
36
37 auto run_edge = [&]() {
38 std::map<int64_t, int64_t> parent_to_child_vertex_map;
39 // int64_t child_vertex_count = 0;
40
41 RowVectors2l edge_mesh_matrix;
42 edge_mesh_matrix.resize(tagged_tuples.size(), 2);
43
44 for (size_t i = 0; i < tagged_tuples.size(); ++i) {
45 const std::array<int64_t, 2> vs = {
46 {m.id(tagged_tuples[i], PrimitiveType::Vertex),
47 m.id(
48 m.switch_tuple(tagged_tuples[i], PrimitiveType::Vertex),
50
51
52 // check and add v0, v1 to the vertex map
53 for (int k = 0; k < 2; k++) {
54 size_t size = parent_to_child_vertex_map.size();
55 parent_to_child_vertex_map.try_emplace(vs[k], size);
56 edge_mesh_matrix(i, k) = parent_to_child_vertex_map[vs[k]];
57 }
58 }
59
60 std::shared_ptr<EdgeMesh> child_ptr = std::make_shared<EdgeMesh>();
61 auto& child = *child_ptr;
62 child.initialize(edge_mesh_matrix);
63 return child_ptr;
64 };
65
66 auto run_face = [&]() {
67 std::map<int64_t, int64_t> parent_to_child_vertex_map;
68 int64_t child_vertex_count = 0;
69
70 RowVectors3l tri_mesh_matrix;
71 tri_mesh_matrix.resize(tagged_tuples.size(), 3);
72 for (int64_t i = 0; i < tagged_tuples.size(); ++i) {
73 // TODO: check if this break the orientation of the map
74 const std::array<int64_t, 3> vs = {
75 {m.id(tagged_tuples[i], PrimitiveType::Vertex),
76 m.id(
77 m.switch_tuple(tagged_tuples[i], PrimitiveType::Vertex),
79 m.id(
81 tagged_tuples[i],
82 {PrimitiveType::Edge, PrimitiveType::Vertex}),
84
85 // check and add v0, v1, v2 to the vertex map
86 for (int k = 0; k < 3; k++) {
87 size_t size = parent_to_child_vertex_map.size();
88 parent_to_child_vertex_map.try_emplace(vs[k], size);
89 tri_mesh_matrix(i, k) = parent_to_child_vertex_map.at(vs[k]);
90 }
91 }
92 std::shared_ptr<TriMesh> child_ptr = std::make_shared<TriMesh>();
93 auto& child = *child_ptr;
94 child.initialize(tri_mesh_matrix);
95
96 return child_ptr;
97 };
98
99 auto run_tet = [&]() {
100 std::map<int64_t, int64_t> parent_to_child_vertex_map;
101 int64_t child_vertex_count = 0;
102
103 RowVectors4l tet_mesh_matrix;
104
105 tet_mesh_matrix.resize(tagged_tuples.size(), 4);
106 for (int64_t i = 0; i < tagged_tuples.size(); ++i) {
107 const std::array<int64_t, 4> vs = {
108 {m.id(tagged_tuples[i], PrimitiveType::Vertex),
109 m.id(
110 m.switch_tuple(tagged_tuples[i], PrimitiveType::Vertex),
112 m.id(
114 tagged_tuples[i],
115 {PrimitiveType::Triangle, PrimitiveType::Edge, PrimitiveType::Vertex}),
117 m.id(
119 tagged_tuples[i],
120 {PrimitiveType::Edge, PrimitiveType::Vertex}),
122
123 for (int k = 0; k < 4; ++k) {
124 size_t size = parent_to_child_vertex_map.size();
125 parent_to_child_vertex_map.try_emplace(vs[k], size);
126 tet_mesh_matrix(i, k) = parent_to_child_vertex_map[vs[k]];
127 }
128 }
129
130 // TODO: use TV matrix instead of switches
131
132
133 std::shared_ptr<TetMesh> child_ptr = std::make_shared<TetMesh>();
134 auto& child = *child_ptr;
135 child.initialize(tet_mesh_matrix);
136 return child_ptr;
137 };
138
139
140 std::shared_ptr<Mesh> child_mesh_ptr;
141
142 if (child_is_free) {
143 switch (pt) {
144 case PrimitiveType::Vertex: throw("not implemented");
145 case PrimitiveType::Edge: {
146 std::shared_ptr<EdgeMesh> meshptr = std::make_shared<EdgeMesh>();
147 meshptr->initialize_free(tagged_tuples.size());
148 child_mesh_ptr = meshptr;
149 break;
150 }
152 std::shared_ptr<TriMesh> meshptr = std::make_shared<TriMesh>();
153 meshptr->initialize_free(tagged_tuples.size());
154 child_mesh_ptr = meshptr;
155 break;
156 }
158 std::shared_ptr<TetMesh> meshptr = std::make_shared<TetMesh>();
159 meshptr->initialize_free(tagged_tuples.size());
160 child_mesh_ptr = meshptr;
161 break;
162 }
163 default: throw("invalid child mesh type");
164 }
165
166 } else {
167 switch (pt) {
168 case PrimitiveType::Vertex: throw("not implemented");
169 case PrimitiveType::Edge: {
170 child_mesh_ptr = run_edge();
171 break;
172 }
174 child_mesh_ptr = run_face();
175 break;
176 }
178 child_mesh_ptr = run_tet();
179 break;
180 }
181 default: throw("invalid child mesh type");
182 }
183 }
184
185 assert(bool(child_mesh_ptr));
186 auto& child = *child_mesh_ptr;
187
188 std::vector<std::array<Tuple, 2>> child_to_parent_map(tagged_tuples.size());
189 assert(tagged_tuples.size() == child.capacity(pt));
190
191 const auto child_top_dimension_tuples = child.get_all(pt);
192
193 for (size_t i = 0; i < tagged_tuples.size(); ++i) {
194 child_to_parent_map[i] = {{child_top_dimension_tuples[i], tagged_tuples[i]}};
195 }
196
197 m.register_child_mesh(child_mesh_ptr, child_to_parent_map);
198 return child_mesh_ptr;
199}
200
202 Mesh& m,
203 const std::string& tag,
204 const int64_t tag_value,
205 const PrimitiveType pt,
206 bool child_is_free)
207{
208 assert(m.top_simplex_type() >= pt);
209 auto tag_handle = m.get_attribute_handle<int64_t>(tag, pt).as<int64_t>();
210 return extract_and_register_child_mesh_from_tag_handle(m, tag_handle, tag_value, child_is_free);
211}
212
214 Mesh& m,
216 const int64_t tag_value,
217 bool child_is_free)
218{
220 m,
221 tag_handle,
222 tag_value,
223 child_is_free);
224}
225
229 bool child_is_free)
230{
231 return std::visit(
232 [&](auto&& handle) noexcept -> std::shared_ptr<Mesh> {
233 using HandleType = typename std::decay_t<decltype(handle)>;
234 using T = typename HandleType::Type;
235 if constexpr (wmtk::attribute::MeshAttributeHandle::template handle_type_is_basic<
236 HandleType>()) {
237 return std::visit(
238 [&](auto&& value) noexcept -> std::shared_ptr<Mesh> {
239 if constexpr (std::is_convertible_v<std::decay_t<decltype(value)>, T>) {
242 tag_handle.mesh(),
243 handle,
244 T(value),
245 child_is_free);
246 } else {
248 "Tried to use a tag value that was not convertible to "
249 "the tag attribute type");
250 return {};
251 }
252 },
253 tag_value);
254 } else {
255 log_and_throw_error("Tried to use a non-primitive meshattributehandle when "
256 "extracting a child mesh fro ma tag");
257 }
258 return nullptr;
259 },
260 tag_handle.handle());
261}
262
263} // namespace wmtk::multimesh::utils
attribute::MeshAttributeHandle get_attribute_handle(const std::string &name, const PrimitiveType ptype) const
Definition Mesh.hpp:901
const attribute::Accessor< T, Mesh, attribute::CachingAttribute< T >, D > create_const_accessor(const attribute::MeshAttributeHandle &handle) const
int64_t id(const Tuple &tuple, PrimitiveType type) const
return the global id of the Tuple of the given dimension
Definition Mesh.hpp:1010
std::vector< Tuple > get_all(PrimitiveType type) const
Generate a vector of Tuples from global vertex/edge/triangle/tetrahedron index.
Definition Mesh.cpp:18
Tuple switch_tuples(const Tuple &tuple, const ContainerType &op_sequence) const
Definition Mesh.hpp:951
void register_child_mesh(const std::shared_ptr< Mesh > &child_mesh_ptr, const std::vector< std::array< Tuple, 2 > > &map_tuples)
register a mesh as the child of this mesh
virtual Tuple switch_tuple(const Tuple &tuple, PrimitiveType type) const =0
switch the orientation of the Tuple of the given dimension
PrimitiveType top_simplex_type() const
Definition Mesh.hpp:980
The Tuple is the basic navigation tool in our mesh data structure.
Definition Tuple.hpp:19
std::variant< char, int64_t, double, wmtk::Rational, std::tuple< char, wmtk::Rational, double > > ValueVariant
Handle that represents attributes for some mesh.
static std::shared_ptr< Mesh > extract_and_register_child_mesh_from_tag_handle(Mesh &m, const wmtk::attribute::TypedAttributeHandle< T > &tag_handle, const T &tag_value, bool child_is_free)
std::shared_ptr< Mesh > extract_and_register_child_mesh_from_tag_handle(Mesh &m, const wmtk::attribute::TypedAttributeHandle< int64_t > &tag_handle, const int64_t tag_value, bool child_is_free)
extract a child mesh based on the tag handle and the tag value, and register it to the input (parent)...
std::shared_ptr< Mesh > extract_and_register_child_mesh_from_tag(Mesh &m, const std::string &tag, const int64_t tag_value, const PrimitiveType pt, bool child_is_free)
extract a child mesh based on the given tag and tag value, and register it to the input (parent) mesh
RowVectors< int64_t, 3 > RowVectors3l
Definition Types.hpp:47
void log_and_throw_error(const std::string &msg)
Definition Logger.cpp:101
RowVectors< int64_t, 4 > RowVectors4l
Definition Types.hpp:48
RowVectors< int64_t, 2 > RowVectors2l
Definition Types.hpp:46