16 #include "libmesh/unstructured_mesh.h" 19 StitchedMeshGenerator,
28 params.
addRequiredParam<std::vector<MeshGeneratorName>>(
"inputs",
"The input MeshGenerators.");
29 params.
addParam<
bool>(
"prevent_boundary_ids_overlap",
31 "Whether to re-number boundaries in stitched meshes to prevent merging of " 32 "unrelated boundaries");
34 "merge_boundaries_with_same_name",
36 "If the input meshes have boundaries with the same name (but different IDs), merge them");
38 "subdomain_remapping",
40 "Treat input subdomain names as primary, preserving them and remapping IDs as needed");
43 "Allows multiple mesh files to be stitched together to form a single mesh.");
52 _mesh_ptrs(getMeshes(
"inputs")),
53 _input_names(getParam<
std::vector<MeshGeneratorName>>(
"inputs")),
54 _prevent_boundary_ids_overlap(getParam<bool>(
"prevent_boundary_ids_overlap")),
55 _merge_boundaries_with_same_name(getParam<bool>(
"merge_boundaries_with_same_name"))
60 "Can only stitch one pair of boundary per pair of mesh. We have '" +
62 "' meshes specified (=" + std::to_string(
_input_names.size() - 1) +
64 " pairs of boundaries specified");
69 "' is not of length 2, but of length: " + std::to_string(pair.size()));
72 std::unique_ptr<MeshBase>
78 mooseError(
"StitchMeshGenerator is only implemented for unstructured meshes");
81 std::vector<std::unique_ptr<UnstructuredMesh>> meshes(
_mesh_ptrs.size() - 1);
85 meshes[i - 1] = dynamic_pointer_cast<UnstructuredMesh>(*
_mesh_ptrs[i]);
88 for (MooseIndex(meshes) i = 0; i < meshes.size(); i++)
96 ((i == 0) ?
"" : (
" (stitched with " + std::to_string(i) +
" previous meshes)")),
100 const bool use_binary_search = (
_algorithm ==
"BINARY");
103 if (!
mesh->is_prepared())
104 mesh->prepare_for_use();
105 if (!meshes[i]->is_prepared())
106 meshes[i]->prepare_for_use();
113 const auto & base_mesh_bids =
mesh->get_boundary_info().get_global_boundary_ids();
114 const auto stitched_mesh_bids = meshes[i]->get_boundary_info().get_global_boundary_ids();
117 bool overlap_found =
false;
118 for (
const auto & bid : stitched_mesh_bids)
119 if (base_mesh_bids.count(bid))
120 overlap_found =
true;
124 const auto max_boundary_id =
125 std::max(*base_mesh_bids.rbegin(), *stitched_mesh_bids.rbegin());
127 for (
const auto bid : stitched_mesh_bids)
129 const auto new_bid = max_boundary_id + (new_index++);
130 meshes[i]->get_boundary_info().renumber_id(bid, new_bid);
142 const auto & base_mesh_bids =
mesh->get_boundary_info().get_global_boundary_ids();
143 const auto & other_mesh_bids = meshes[i]->get_boundary_info().get_global_boundary_ids();
146 std::set<boundary_id_type> bd_id_intersection;
147 std::set_intersection(base_mesh_bids.begin(),
148 base_mesh_bids.end(),
149 other_mesh_bids.begin(),
150 other_mesh_bids.end(),
151 std::inserter(bd_id_intersection, bd_id_intersection.begin()));
153 for (
const auto bid : bd_id_intersection)
155 const auto & sideset_name_on_first_mesh =
mesh->get_boundary_info().get_sideset_name(bid);
156 const auto & sideset_name_on_second_mesh =
157 meshes[i]->get_boundary_info().get_sideset_name(bid);
159 if (sideset_name_on_first_mesh != sideset_name_on_second_mesh)
163 " corresponds to different boundary names on the input meshes! On the first " 164 "mesh it corresponds to `",
165 sideset_name_on_first_mesh,
166 "` while on the second mesh it corresponds to `",
167 sideset_name_on_second_mesh,
168 "`. The final mesh will replace boundary `",
169 sideset_name_on_second_mesh,
171 sideset_name_on_first_mesh,
172 "`. To avoid this situation, use the `prevent_boundary_ids_overlap` parameter!");
176 mesh->stitch_meshes(*meshes[i],
179 getParam<Real>(
"stitching_hmin_tolerance_factor"),
181 getParam<bool>(
"verbose_stitching"),
185 getParam<bool>(
"subdomain_remapping"));
191 mesh->set_isnt_prepared();
registerMooseObjectRenamed("MooseApp", StitchedMeshGenerator, "06/30/2026 24:00", StitchMeshGenerator)
registerMooseObject("MooseApp", StitchMeshGenerator)
boundary_id_type getBoundaryIdToStitch(const MeshBase &mesh, const std::string &input_mg_name, const BoundaryName &bname) const
Get the boundary id from the name of a boundary to stitch.
std::unique_ptr< T_DEST, T_DELETER > dynamic_pointer_cast(std::unique_ptr< T_SRC, T_DELETER > &src)
These are reworked from https://stackoverflow.com/a/11003103.
virtual const std::string & name() const
Get the name of the class.
static InputParameters validParams()
void mooseWarning(Args &&... args) const
Emits a warning prefixed with object name and type.
const bool _clear_stitched_boundary_ids
Whether or not to clear (remove) the stitched boundary IDs.
static InputParameters validParams()
auto max(const L &left, const R &right)
const bool _merge_boundaries_with_same_name
Whether to merge boundaries if they have the same name but different boundary IDs.
const bool _prevent_boundary_ids_overlap
Whether to renumber all boundaries in stitched meshes to prevent accidental merging of sidesets with ...
boundary_id_type BoundaryID
A base class for mesh generators that stitch boundaries together.
Allows multiple mesh files to be "stitched" together to form a single mesh.
void paramError(const std::string ¶m, Args... args) const
Emits an error prefixed with the file and line number of the given param (from the input file) along ...
std::string stringify(const T &t)
conversion to string
const MooseEnum _algorithm
Type of algorithm used to find matching nodes (binary or exhaustive)
const std::vector< MeshGeneratorName > & _input_names
The mesh generator inputs to read.
StitchMeshGenerator(const InputParameters ¶meters)
std::vector< std::unique_ptr< MeshBase > * > _mesh_ptrs
std::unique_ptr< MeshBase > generate() override
Generate / modify the mesh.
void mooseError(Args &&... args) const
Emits an error prefixed with object name and type.
void mergeBoundaryIDsWithSameName(MeshBase &mesh)
Merges the boundary IDs of boundaries that have the same names but different IDs. ...
const std::vector< std::vector< std::string > > _stitch_boundaries_pairs
A transformed version of _stitch_boundaries into a more logical "pairwise" structure.