28 const std::string &root_path,
29 const bool non_conforming)
34 if (j_mesh[
"extract"].get<std::string>() !=
"volume")
42 if (j_mesh[
"advanced"][
"normalize_mesh"])
48 mesh->bounding_box(bbox[0], bbox[1]);
50 const std::string unit = j_mesh[
"unit"];
51 double unit_scale = 1;
60 j_mesh[
"transformation"],
61 (bbox[1] - bbox[0]).cwiseAbs().transpose(),
63 mesh->apply_affine_transformation(A, b);
66 mesh->bounding_box(bbox[0], bbox[1]);
70 const int n_refs = j_mesh[
"n_refs"];
71 const double refinement_location = j_mesh[
"advanced"][
"refinement_location"];
83 assert(mesh->n_elements() > 0);
84 const int uniform_value = mesh->get_body_id(0);
85 for (
int i = 1; i < mesh->n_elements(); ++i)
86 if (mesh->get_body_id(i) != uniform_value)
87 log_and_throw_error(
"Unable to apply stored nonuniform volume_selection because n_refs={} > 0!", n_refs);
89 logger().info(
"Performing global h-refinement with {} refinements", n_refs);
90 mesh->refine(n_refs, refinement_location);
91 mesh->set_body_ids(std::vector<int>(mesh->n_elements(), uniform_value));
96 if (j_mesh[
"advanced"][
"min_component"].get<int>() != -1)
128 if (j_mesh[
"advanced"][
"force_linear_geometry"].get<bool>())
142 const std::vector<std::shared_ptr<Selection>> node_selections =
145 if (!node_selections.empty())
147 bool boundary_only =
true;
148 for (
const auto &selection : node_selections)
150 if (!selection->boundary_only())
152 boundary_only =
false;
157 mesh->compute_node_ids([&](
const size_t n_id,
const RowVectorNd &p,
bool is_boundary) {
158 if (boundary_only && !is_boundary)
161 const std::vector<int> tmp = {int(n_id)};
162 for (
const auto &selection : node_selections)
164 if (selection->boundary_only() && !is_boundary)
167 if (selection->inside(n_id, tmp, p))
168 return selection->id(n_id, tmp, p);
170 return std::numeric_limits<int>::max();
174 if (!j_mesh[
"curve_selection"].is_null())
179 std::vector<std::shared_ptr<Selection>> surface_selections =
182 if (!surface_selections.empty())
184 bool boundary_only =
true;
185 for (
const auto &selection : surface_selections)
187 if (!selection->boundary_only())
189 boundary_only =
false;
194 mesh->compute_boundary_ids([&](
const size_t p_id,
const std::vector<int> &vs,
const RowVectorNd &p,
bool is_boundary) {
195 if (boundary_only && !is_boundary)
198 for (
const auto &selection : surface_selections)
200 if (selection->boundary_only() && !is_boundary)
203 if (selection->inside(p_id, vs, p))
204 return selection->id(p_id, vs, p);
206 return std::numeric_limits<int>::max();
213 const json volume_selection = j_mesh[
"volume_selection"];
214 if (volume_selection.is_object()
215 && volume_selection.size() == 1
216 && volume_selection.contains(
"id_offset"))
218 const int id_offset = volume_selection[
"id_offset"].get<
int>();
221 const int n_body_ids = mesh->n_elements();
222 std::vector<int> body_ids(n_body_ids);
223 for (
int i = 0; i < n_body_ids; ++i)
224 body_ids[i] = mesh->get_body_id(i) + id_offset;
225 mesh->set_body_ids(body_ids);
231 std::vector<std::shared_ptr<Selection>> volume_selections =
235 if (mesh->has_body_ids())
236 volume_selections.push_back(std::make_shared<SpecifiedSelection>(mesh->get_body_ids()));
238 mesh->compute_body_ids([&](
const size_t cell_id,
const std::vector<int> &vs,
const RowVectorNd &p) ->
int {
239 for (
const auto &selection : volume_selections)
242 if (selection->inside(cell_id, vs, p))
243 return selection->id(cell_id, vs, p);
258 const json &geometry,
259 const std::string &root_path,
260 const std::vector<std::string> &_names,
261 const std::vector<Eigen::MatrixXd> &_vertices,
262 const std::vector<Eigen::MatrixXi> &_cells,
263 const bool non_conforming)
283 assert(_names.empty());
284 assert(_vertices.empty());
285 assert(_cells.empty());
289 if (geometry.empty())
296 std::unique_ptr<Mesh> mesh =
nullptr;
298 for (
const json &geometry : geometries)
300 if (!geometry[
"enabled"].get<bool>() || geometry[
"is_obstacle"].get<
bool>())
303 if (geometry[
"type"] !=
"mesh" && geometry[
"type"] !=
"mesh_array")
306 const std::unique_ptr<Mesh> tmp_mesh =
read_fem_mesh(units, geometry, root_path, non_conforming);
309 mesh = tmp_mesh->copy();
311 mesh->append(tmp_mesh);
313 if (geometry[
"type"] ==
"mesh_array")
316 tmp_mesh->bounding_box(bbox[0], bbox[1]);
318 const long dim = tmp_mesh->dimension();
319 const bool is_offset_relative = geometry[
"array"][
"relative"];
320 const double offset = geometry[
"array"][
"offset"];
321 const VectorNd dimensions = (bbox[1] - bbox[0]);
322 const VectorNi size = geometry[
"array"][
"size"];
324 for (
int i = 0; i < size[0]; ++i)
326 for (
int j = 0; j < size[1]; ++j)
328 for (
int k = 0; k < (size.size() > 2 ? size[2] : 1); ++k)
330 if (i == 0 && j == 0 && k == 0)
333 RowVectorNd translation = offset * Eigen::RowVector3d(i, j, k).head(dim);
334 if (is_offset_relative)
335 translation.array() *= dimensions.array();
337 const std::unique_ptr<Mesh> copy_mesh = tmp_mesh->copy();
338 copy_mesh->apply_affine_transformation(MatrixNd::Identity(dim, dim), translation);
339 mesh->append(copy_mesh);
356 const std::string &root_path,
358 Eigen::MatrixXd &vertices,
359 Eigen::VectorXi &codim_vertices,
360 Eigen::MatrixXi &codim_edges,
361 Eigen::MatrixXi &
faces)
366 const std::string mesh_path =
resolve_path(j_mesh[
"mesh"], root_path);
369 mesh_path, vertices, codim_vertices, codim_edges,
faces);
373 throw std::runtime_error(fmt::format(
"Unable to read mesh: {}", mesh_path));
375 const int prev_dim = vertices.cols();
376 vertices.conservativeResize(vertices.rows(), dim);
378 vertices.rightCols(dim - prev_dim).setZero();
383 const std::string unit = j_mesh[
"unit"];
384 double unit_scale = 1;
388 const VectorNd mesh_dimensions = (vertices.colwise().maxCoeff() - vertices.colwise().minCoeff()).cwiseAbs();
392 vertices = vertices * A.transpose();
393 vertices.rowwise() += b.transpose();
396 std::string extract = j_mesh[
"extract"];
398 if (extract ==
"volume")
401 if (extract ==
"points")
404 codim_edges.resize(0, 0);
406 codim_vertices.resize(vertices.rows());
407 for (
int i = 0; i < codim_vertices.size(); ++i)
408 codim_vertices[i] = i;
410 else if (extract ==
"edges" &&
faces.size() != 0)
413 Eigen::MatrixXi edges;
414 igl::edges(
faces, edges);
416 codim_edges.conservativeResize(codim_edges.rows() + edges.rows(), 2);
417 codim_edges.bottomRows(edges.rows()) = edges;
419 else if (extract ==
"surface" && dim == 2 &&
faces.size() != 0)
422 Eigen::MatrixXi boundary_edges;
423 igl::boundary_facets(
faces, boundary_edges);
424 codim_edges.conservativeResize(codim_edges.rows() + boundary_edges.rows(), 2);
425 codim_edges.bottomRows(boundary_edges.rows()) = boundary_edges;
430 else if (extract ==
"volume")
436 if (j_mesh[
"n_refs"].get<int>() != 0)
438 if (
faces.size() != 0)
441 const int n_refs = j_mesh[
"n_refs"];
442 const double refinement_location = j_mesh[
"advanced"][
"refinement_location"];
443 for (
int i = 0; i < n_refs; i++)
445 const size_t n_vertices = vertices.rows();
446 const size_t n_edges = codim_edges.rows();
447 vertices.conservativeResize(n_vertices + n_edges, vertices.cols());
448 codim_edges.conservativeResize(2 * n_edges, codim_edges.cols());
449 for (
size_t ei = 0; ei < n_edges; ei++)
451 const int v0i = codim_edges(ei, 0);
452 const int v1i = codim_edges(ei, 1);
453 const int v2i = n_vertices + ei;
454 vertices.row(v2i) = (vertices.row(v1i) - vertices.row(v0i)) * refinement_location + vertices.row(v0i);
455 codim_edges.row(ei) << v0i, v2i;
456 codim_edges.row(n_edges + ei) << v2i, v1i;
466 const json &geometry,
467 const std::vector<json> &displacements,
468 const std::vector<json> &dirichlets,
469 const std::string &root_path,
471 const std::vector<std::string> &_names,
472 const std::vector<Eigen::MatrixXd> &_vertices,
473 const std::vector<Eigen::MatrixXi> &_cells,
474 const bool non_conforming)
494 assert(_names.empty());
495 assert(_vertices.empty());
496 assert(_cells.empty());
500 if (geometry.empty())
505 for (
const json &geometry : geometries)
508 if (!geometry[
"is_obstacle"].get<bool>())
511 if (!geometry[
"enabled"].get<bool>())
514 if (geometry[
"type"] ==
"mesh" || geometry[
"type"] ==
"mesh_array")
516 Eigen::MatrixXd vertices;
517 Eigen::VectorXi codim_vertices;
518 Eigen::MatrixXi codim_edges;
519 Eigen::MatrixXi
faces;
521 geometry, root_path, dim, vertices, codim_vertices,
524 if (geometry[
"type"] ==
"mesh_array")
526 const Selection::BBox bbox{{vertices.colwise().minCoeff(), vertices.colwise().maxCoeff()}};
528 const bool is_offset_relative = geometry[
"array"][
"relative"];
529 const double offset = geometry[
"array"][
"offset"];
530 const VectorNd dimensions = (bbox[1] - bbox[0]);
531 const VectorNi size = geometry[
"array"][
"size"];
533 const int N = size.head(dim).prod();
534 const int nV = vertices.rows(), nCV = codim_vertices.rows(), nCE = codim_edges.rows(), nF =
faces.rows();
536 vertices.conservativeResize(N * nV, Eigen::NoChange);
537 codim_vertices.conservativeResize(N * nCV, Eigen::NoChange);
538 codim_edges.conservativeResize(N * nCE, Eigen::NoChange);
539 faces.conservativeResize(N * nF, Eigen::NoChange);
541 for (
int i = 0; i < size[0]; ++i)
543 for (
int j = 0; j < size[1]; ++j)
545 for (
int k = 0; k < (size.size() > 2 ? size[2] : 1); ++k)
547 RowVectorNd translation = offset * Eigen::RowVector3d(i, j, k).head(vertices.cols());
548 if (is_offset_relative)
549 translation.array() *= dimensions.array();
551 int n = i * size[1] + j;
557 vertices.middleRows(n * nV, nV) = vertices.topRows(nV).rowwise() + translation;
559 codim_vertices.segment(n * nV, nV) = codim_vertices.head(nV).array() + n * nV;
561 codim_edges.middleRows(n * nCE, nCE) = codim_edges.topRows(nCE).array() + n * nV;
563 faces.middleRows(n * nF, nF) =
faces.topRows(nF).array() + n * nV;
569 json displacement =
"{\"value\":[0, 0, 0]}"_json;
572 if (!geometry[
"surface_selection"].is_number())
575 const int id = geometry[
"surface_selection"];
576 for (
const json &disp : dirichlets)
578 if ((disp[
"id"].is_string() && disp[
"id"].get<std::string>() ==
"all")
579 || (disp[
"id"].is_number_integer() && disp[
"id"].get<
int>() ==
id))
584 else if (disp[
"id"].is_array())
586 for (
const json &disp_id : disp[
"id"])
588 assert(disp_id.is_number_integer());
589 if (disp_id.get<
int>() ==
id)
597 for (
const json &disp : displacements)
599 if ((disp[
"id"].is_string() && disp[
"id"].get<std::string>() ==
"all")
600 || (disp[
"id"].is_number_integer() && disp[
"id"].get<
int>() ==
id))
605 else if (disp[
"id"].is_array())
607 for (
const json &disp_id : disp[
"id"])
609 assert(disp_id.is_number_integer());
610 if (disp_id.get<
int>() ==
id)
621 vertices, codim_vertices, codim_edges,
faces, displacement, root_path);
623 else if (geometry[
"type"] ==
"plane")
625 obstacle.
append_plane(geometry[
"point"], geometry[
"normal"]);
627 else if (geometry[
"type"] ==
"ground")
629 VectorNd gravity = VectorNd::Zero(dim);
631 const double height = geometry[
"height"];
632 assert(gravity.norm() != 0);
633 const VectorNd normal = -gravity.normalized();
634 const VectorNd point = height * normal;
637 else if (geometry[
"type"] ==
"mesh_sequence")
639 namespace fs = std::filesystem;
640 std::vector<fs::path> mesh_files;
641 if (geometry[
"mesh_sequence"].is_array())
643 mesh_files = geometry[
"mesh_sequence"].get<std::vector<fs::path>>();
647 assert(geometry[
"mesh_sequence"].is_string());
648 const fs::path meshes(
resolve_path(geometry[
"mesh_sequence"], root_path));
650 if (fs::is_directory(meshes))
652 for (
const auto &entry : std::filesystem::directory_iterator(meshes))
654 if (entry.is_regular_file())
655 mesh_files.push_back(entry.path());
660 mesh_files = glob::rglob(meshes.string());
663 std::sort(mesh_files.begin(), mesh_files.end(), [](
const fs::path &p1,
const fs::path &p2) {
664 return strnatcmp(p1.string().c_str(), p2.string().c_str()) < 0;
668 std::vector<Eigen::MatrixXd> vertices(mesh_files.size());
669 Eigen::VectorXi codim_vertices;
670 Eigen::MatrixXi codim_edges;
671 Eigen::MatrixXi
faces;
673 for (
int i = 0; i < mesh_files.size(); ++i)
675 json jmesh = geometry;
676 jmesh[
"mesh"] = mesh_files[i];
679 Eigen::VectorXi tmp_codim_vertices;
680 Eigen::MatrixXi tmp_codim_edges;
681 Eigen::MatrixXi tmp_faces;
683 jmesh, root_path, dim, vertices[i],
684 tmp_codim_vertices, tmp_codim_edges, tmp_faces);
687 codim_vertices = tmp_codim_vertices;
688 codim_edges = tmp_codim_edges;
693 assert((codim_vertices.array() == tmp_codim_vertices.array()).all());
694 assert((codim_edges.array() == tmp_codim_edges.array()).all());
695 assert((
faces.array() == tmp_faces.array()).all());
700 vertices, codim_vertices, codim_edges,
faces, geometry[
"fps"]);
715 const double unit_scale,
716 const json &transform,
721 const int dim = mesh_dimensions.size();
728 if (transform[
"dimensions"].is_array())
731 (mesh_dimensions.array() == 0).select(1, mesh_dimensions);
733 scale = transform[
"dimensions"];
734 const int scale_size = scale.size();
735 scale.conservativeResize(dim);
736 if (scale_size < dim)
737 scale.tail(dim - scale_size).setZero();
739 scale.array() /= modified_dimensions.array();
741 else if (transform[
"scale"].is_number())
743 scale.setConstant(dim, transform[
"scale"].get<double>());
747 assert(transform[
"scale"].is_array());
748 scale = transform[
"scale"];
749 const int scale_size = scale.size();
750 scale.conservativeResize(dim);
751 if (scale_size < dim)
752 scale.tail(dim - scale_size).setZero();
758 A = (unit_scale * scale).asDiagonal();
766 MatrixNd R = MatrixNd::Identity(dim, dim);
767 if (!transform[
"rotation"].is_null())
771 if (transform[
"rotation"].is_number())
772 R = Eigen::Rotation2Dd(
deg2rad(transform[
"rotation"].get<double>()))
774 else if (!transform[
"rotation"].is_array() || !transform[
"rotation"].empty())
789 b = transform[
"translation"];
790 const int translation_size = b.size();
791 b.conservativeResize(dim);
792 if (translation_size < dim)
793 b.tail(dim - translation_size).setZero();