Skip to content
Snippets Groups Projects
Commit f32ab4d3 authored by Christophe Geuzaine's avatar Christophe Geuzaine
Browse files

Merge branch 'enhance-crack-plugin' into 'master'

enhance crack plugin

See merge request !468
parents da80b2e0 73d691ae
No related branches found
No related tags found
No related merge requests found
4.10.2 (Work-in-progress): fixed regression introduced in 4.9 for recombined 4.10.2 (Work-in-progress): fixed regression introduced in 4.9 for recombined
meshes with boundary layers; new HealShapes command in .geo files; simplified meshes with boundary layers; new HealShapes command in .geo files; simplified
calculation of OCC STL bounding boxes; small bug fixes. calculation of OCC STL bounding boxes; generalized Crack plugin; small bug fixes.
4.10.1 (May 1, 2022): small bug fixes. 4.10.1 (May 1, 2022): small bug fixes.
......
...@@ -31,8 +31,14 @@ gmsh.model.addPhysicalGroup(1, new_lines, 101) ...@@ -31,8 +31,14 @@ gmsh.model.addPhysicalGroup(1, new_lines, 101)
gmsh.model.mesh.generate(2) gmsh.model.mesh.generate(2)
gmsh.plugin.setNumber("Crack", "PhysicalGroup", 101) gmsh.plugin.setNumber("Crack", "PhysicalGroup", 101)
# gmsh.plugin.setNumber("Crack", "DebugView", 1)
gmsh.plugin.run("Crack") gmsh.plugin.run("Crack")
# save all the elements in the mesh (even those that do not belong to any
# physical group):
gmsh.option.setNumber("Mesh.SaveAll", 1)
gmsh.write("crack.msh")
if '-nopopup' not in sys.argv: if '-nopopup' not in sys.argv:
gmsh.fltk.run() gmsh.fltk.run()
......
...@@ -28,6 +28,7 @@ gmsh.model.mesh.generate(3) ...@@ -28,6 +28,7 @@ gmsh.model.mesh.generate(3)
# "crack" the mesh by duplicating the elements and nodes on the small surface # "crack" the mesh by duplicating the elements and nodes on the small surface
gmsh.plugin.setNumber("Crack", "Dimension", 2) gmsh.plugin.setNumber("Crack", "Dimension", 2)
gmsh.plugin.setNumber("Crack", "PhysicalGroup", phys) gmsh.plugin.setNumber("Crack", "PhysicalGroup", phys)
#gmsh.plugin.setNumber("Crack", "DebugView", 1)
gmsh.plugin.run("Crack") gmsh.plugin.run("Crack")
# save all the elements in the mesh (even those that do not belong to any # save all the elements in the mesh (even those that do not belong to any
......
...@@ -21,7 +21,8 @@ StringXNumber CrackOptions_Number[] = { ...@@ -21,7 +21,8 @@ StringXNumber CrackOptions_Number[] = {
{GMSH_FULLRC, "NormalX", nullptr, 0.}, {GMSH_FULLRC, "NormalX", nullptr, 0.},
{GMSH_FULLRC, "NormalY", nullptr, 0.}, {GMSH_FULLRC, "NormalY", nullptr, 0.},
{GMSH_FULLRC, "NormalZ", nullptr, 1.}, {GMSH_FULLRC, "NormalZ", nullptr, 1.},
{GMSH_FULLRC, "NewPhysicalGroup", nullptr, 0} {GMSH_FULLRC, "NewPhysicalGroup", nullptr, 0},
{GMSH_FULLRC, "DebugView", nullptr, 0}
}; };
extern "C" { extern "C" {
...@@ -86,6 +87,7 @@ PView *GMSH_CrackPlugin::execute(PView *view) ...@@ -86,6 +87,7 @@ PView *GMSH_CrackPlugin::execute(PView *view)
SVector3 normal1d(CrackOptions_Number[3].def, CrackOptions_Number[4].def, SVector3 normal1d(CrackOptions_Number[3].def, CrackOptions_Number[4].def,
CrackOptions_Number[5].def); CrackOptions_Number[5].def);
int newPhysical = (int)CrackOptions_Number[6].def; int newPhysical = (int)CrackOptions_Number[6].def;
int debug = (int)CrackOptions_Number[7].def;
if(dim != 1 && dim != 2) { if(dim != 1 && dim != 2) {
Msg::Error("Crack dimension should be 1 or 2"); Msg::Error("Crack dimension should be 1 or 2");
...@@ -122,13 +124,19 @@ PView *GMSH_CrackPlugin::execute(PView *view) ...@@ -122,13 +124,19 @@ PView *GMSH_CrackPlugin::execute(PView *view)
for(std::size_t j = 0; j < entities[i]->getNumMeshElements(); j++) for(std::size_t j = 0; j < entities[i]->getNumMeshElements(); j++)
crackElements.push_back(entities[i]->getMeshElement(j)); crackElements.push_back(entities[i]->getMeshElement(j));
// get internal crack nodes and boundary nodes // get internal crack nodes (and list of connected crack elements), as well as
std::set<MVertex *> crackVertices, bndVertices; // and boundary nodes
std::map<MVertex *, std::vector<MElement *> > crackVertices;
std::set<MVertex *> bndVertices;
if(dim == 1) { if(dim == 1) {
for(std::size_t i = 0; i < crackElements.size(); i++) { for(std::size_t i = 0; i < crackElements.size(); i++) {
for(std::size_t j = 0; j < crackElements[i]->getNumVertices(); j++) { for(std::size_t j = 0; j < crackElements[i]->getNumVertices(); j++) {
MVertex *v = crackElements[i]->getVertex(j); MVertex *v = crackElements[i]->getVertex(j);
crackVertices.insert(v); auto it = crackVertices.find(v);
if(it == crackVertices.end())
crackVertices[v] = {crackElements[i]};
else
it->second.push_back(crackElements[i]);
} }
for(std::size_t j = 0; j < crackElements[i]->getNumPrimaryVertices(); for(std::size_t j = 0; j < crackElements[i]->getNumPrimaryVertices();
j++) { j++) {
...@@ -145,7 +153,11 @@ PView *GMSH_CrackPlugin::execute(PView *view) ...@@ -145,7 +153,11 @@ PView *GMSH_CrackPlugin::execute(PView *view)
for(std::size_t i = 0; i < crackElements.size(); i++) { for(std::size_t i = 0; i < crackElements.size(); i++) {
for(std::size_t j = 0; j < crackElements[i]->getNumVertices(); j++) { for(std::size_t j = 0; j < crackElements[i]->getNumVertices(); j++) {
MVertex *v = crackElements[i]->getVertex(j); MVertex *v = crackElements[i]->getVertex(j);
crackVertices.insert(v); auto it = crackVertices.find(v);
if(it == crackVertices.end())
crackVertices[v] = {crackElements[i]};
else
it->second.push_back(crackElements[i]);
} }
for(int j = 0; j < crackElements[i]->getNumEdges(); j++) { for(int j = 0; j < crackElements[i]->getNumEdges(); j++) {
EdgeData ed(crackElements[i]->getEdge(j)); EdgeData ed(crackElements[i]->getEdge(j));
...@@ -199,8 +211,11 @@ PView *GMSH_CrackPlugin::execute(PView *view) ...@@ -199,8 +211,11 @@ PView *GMSH_CrackPlugin::execute(PView *view)
for(auto it = bndVertices.begin(); it != bndVertices.end(); it++) for(auto it = bndVertices.begin(); it != bndVertices.end(); it++)
crackVertices.erase(*it); crackVertices.erase(*it);
// compute elements on one side of the crack // compute elements on the positive side of the crack, and keep track of each
std::set<MElement *> oneside; // node in the element that leads to categorizing the element on this side
// (this allows to handle the degenerate case where the same element touches
// the same crack on both sides, with different nodes - cf. #1750)
std::map<MElement *, std::vector<std::size_t> > oneside;
std::vector<GEntity *> allentities; std::vector<GEntity *> allentities;
m->getEntities(allentities); m->getEntities(allentities);
for(std::size_t ent = 0; ent < allentities.size(); ent++) { for(std::size_t ent = 0; ent < allentities.size(); ent++) {
...@@ -208,40 +223,43 @@ PView *GMSH_CrackPlugin::execute(PView *view) ...@@ -208,40 +223,43 @@ PView *GMSH_CrackPlugin::execute(PView *view)
for(std::size_t i = 0; i < allentities[ent]->getNumMeshElements(); i++) { for(std::size_t i = 0; i < allentities[ent]->getNumMeshElements(); i++) {
MElement *e = allentities[ent]->getMeshElement(i); MElement *e = allentities[ent]->getMeshElement(i);
for(std::size_t j = 0; j < e->getNumVertices(); j++) { for(std::size_t j = 0; j < e->getNumVertices(); j++) {
if(crackVertices.find(e->getVertex(j)) != crackVertices.end()) { auto it = crackVertices.find(e->getVertex(j));
// element touches the crack: find the closest crack element if(it == crackVertices.end()) continue;
SPoint3 b = e->barycenter(); // the element touches the crack by at least one node: determine if the
double d = 1e200; // vector joining its barycenter to the barycenter of one of the
MElement *ce = nullptr; // connected crack elements is not in the same direction as the normal
for(std::size_t k = 0; k < crackElements.size(); k++) { // to the crack element; if the condition is fulfilled for one of the
double d2 = b.distance(crackElements[k]->barycenter()); // connected crack elements, we consider the element lies on the
if(d2 < d) { // "positive side" of the crack
d = d2; SPoint3 b = e->barycenter();
ce = crackElements[k]; for(auto ce : it->second) {
} SVector3 dv = SVector3(b, ce->barycenter());
}
SVector3 dv = SVector3(e->barycenter(), ce->barycenter());
SVector3 n; SVector3 n;
if(dim == 1) if(dim == 1)
n = crossprod(normal1d, ce->getEdge(0).tangent()); n = crossprod(normal1d, ce->getEdge(0).tangent());
else else
n = ce->getFace(0).normal(); n = ce->getFace(0).normal();
if(dot(n, dv) < 0) { oneside.insert(e); } if(dot(n, dv) < 0) {
auto it2 = oneside.find(e);
if(it2 == oneside.end())
oneside[e] = {j};
else {
if(std::find(it2->second.begin(), it2->second.end(), j) ==
it2->second.end())
it2->second.push_back(j);
}
}
} }
} }
} }
} }
/* if(debug) {
FILE *fp = fopen("debug.pos", "w"); std::map<int, std::vector<double> > d;
if(fp){ for(auto e : oneside) d[e.first->getNum()] = {(double)e.first->getNum()};
fprintf(fp, "View \"Ele < 0\" {\n"); view = new PView("Elements on positive side of crack", "ElementData",
for(auto it = oneside.begin(); it != oneside.end(); it++) GModel::current(), d);
(*it)->writePOS(fp, false, true, false, false, false, false);
fprintf(fp, "};\n");
fclose(fp);
} }
*/
// create new crack entity // create new crack entity
...@@ -278,7 +296,7 @@ PView *GMSH_CrackPlugin::execute(PView *view) ...@@ -278,7 +296,7 @@ PView *GMSH_CrackPlugin::execute(PView *view)
// duplicate internal crack nodes // duplicate internal crack nodes
std::map<MVertex *, MVertex *> vxv; std::map<MVertex *, MVertex *> vxv;
for(auto it = crackVertices.begin(); it != crackVertices.end(); it++) { for(auto it = crackVertices.begin(); it != crackVertices.end(); it++) {
MVertex *v = *it; MVertex *v = it->first;
MVertex *newv = new MVertex(v->x(), v->y(), v->z(), crackEntity); MVertex *newv = new MVertex(v->x(), v->y(), v->z(), crackEntity);
crackEntity->mesh_vertices.push_back(newv); crackEntity->mesh_vertices.push_back(newv);
vxv[v] = newv; vxv[v] = newv;
...@@ -304,14 +322,16 @@ PView *GMSH_CrackPlugin::execute(PView *view) ...@@ -304,14 +322,16 @@ PView *GMSH_CrackPlugin::execute(PView *view)
// replace vertices in elements on one side of the crack // replace vertices in elements on one side of the crack
for(auto it = oneside.begin(); it != oneside.end(); it++) { for(auto it = oneside.begin(); it != oneside.end(); it++) {
MElement *e = *it; MElement *e = it->first;
for(std::size_t i = 0; i < e->getNumVertices(); i++) { for(auto i : it->second) {
if(vxv.count(e->getVertex(i))) e->setVertex(i, vxv[e->getVertex(i)]); MVertex *v = e->getVertex(i);
if(vxv.count(v))
e->setVertex(i, vxv[v]);
else
Msg::Warning("Mesh node %lu not found in cracked nodes", v->getNum());
} }
} }
// m->pruneMeshVertexAssociations();
CTX::instance()->mesh.changed = ENT_ALL; CTX::instance()->mesh.changed = ENT_ALL;
return view; return view;
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment