mirror of
https://github.com/NGSolve/netgen.git
synced 2025-01-27 13:20:34 +05:00
SplitImprove for triangles
This commit is contained in:
parent
28d05d9ae7
commit
4987d12556
@ -432,6 +432,7 @@ namespace netgen
|
|||||||
{
|
{
|
||||||
if (!faceindex)
|
if (!faceindex)
|
||||||
{
|
{
|
||||||
|
SplitImprove(mesh);
|
||||||
PrintMessage (3, "Combine improve");
|
PrintMessage (3, "Combine improve");
|
||||||
|
|
||||||
for (faceindex = 1; faceindex <= mesh.GetNFD(); faceindex++)
|
for (faceindex = 1; faceindex <= mesh.GetNFD(); faceindex++)
|
||||||
@ -696,7 +697,11 @@ namespace netgen
|
|||||||
if ( (normals[el[l]] * nv) < 0.5)
|
if ( (normals[el[l]] * nv) < 0.5)
|
||||||
bad2 += 1e10;
|
bad2 += 1e10;
|
||||||
|
|
||||||
illegal2 += 1-mesh.LegalTrig(el);
|
Element2d el1 = el;
|
||||||
|
for (auto i : Range(3))
|
||||||
|
if(el1[i]==pi2)
|
||||||
|
el1[i] = pi1;
|
||||||
|
illegal2 += 1-mesh.LegalTrig(el1);
|
||||||
}
|
}
|
||||||
bad2 /= hasonepi.Size();
|
bad2 /= hasonepi.Size();
|
||||||
|
|
||||||
@ -791,6 +796,142 @@ namespace netgen
|
|||||||
mesh.SetNextTimeStamp();
|
mesh.SetNextTimeStamp();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MeshOptimize2d :: SplitImprove (Mesh & mesh)
|
||||||
|
{
|
||||||
|
if (!faceindex)
|
||||||
|
{
|
||||||
|
PrintMessage (3, "Split improve");
|
||||||
|
|
||||||
|
mesh.CalcSurfacesOfNode(); // TODO: needed?
|
||||||
|
for (faceindex = 1; faceindex <= mesh.GetNFD(); faceindex++)
|
||||||
|
{
|
||||||
|
SplitImprove (mesh);
|
||||||
|
|
||||||
|
if (multithread.terminate)
|
||||||
|
throw NgException ("Meshing stopped");
|
||||||
|
}
|
||||||
|
|
||||||
|
faceindex = 0;
|
||||||
|
mesh.Compress(); // TODO: needed?
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Array<SurfaceElementIndex> elements;
|
||||||
|
mesh.GetSurfaceElementsOfFace (faceindex, elements);
|
||||||
|
|
||||||
|
// return if we have quads in this surface
|
||||||
|
for (auto & ei : elements)
|
||||||
|
if (mesh[ei].GetNP() != 3)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// maps from edges to adjacent trigs
|
||||||
|
INDEX_2_HASHTABLE<tuple<SurfaceElementIndex, SurfaceElementIndex>> els_on_edge(2*elements.Size() + 2);
|
||||||
|
|
||||||
|
// build els_on_edge table
|
||||||
|
for (SurfaceElementIndex sei : elements)
|
||||||
|
{
|
||||||
|
const Element2d & sel = mesh[sei];
|
||||||
|
|
||||||
|
for (int j = 0; j < 3; j++)
|
||||||
|
{
|
||||||
|
PointIndex pi1 = sel.PNumMod(j+2);
|
||||||
|
PointIndex pi2 = sel.PNumMod(j+3);
|
||||||
|
|
||||||
|
if (mesh.IsSegment (pi1, pi2)) continue;
|
||||||
|
|
||||||
|
INDEX_2 ii2 (pi1, pi2);
|
||||||
|
ii2.Sort();
|
||||||
|
if (els_on_edge.Used (ii2))
|
||||||
|
{
|
||||||
|
auto els = els_on_edge.Get(ii2);
|
||||||
|
get<1>(els) = sei;
|
||||||
|
els_on_edge.Set(ii2, els);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
els_on_edge.Set (ii2, make_tuple(sei, sei));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// split edges of illegal trigs
|
||||||
|
for (SurfaceElementIndex sei : elements)
|
||||||
|
{
|
||||||
|
Element2d & sel = mesh[sei];
|
||||||
|
|
||||||
|
if (sel.IsDeleted()) continue;
|
||||||
|
|
||||||
|
// TODO: split also bad trigs, nut just illegal ones
|
||||||
|
if (mesh.LegalTrig(sel)) continue;
|
||||||
|
|
||||||
|
for (int j = 0; j < 3; j++)
|
||||||
|
{
|
||||||
|
PointIndex pi1 = sel.PNumMod(j+2);
|
||||||
|
PointIndex pi2 = sel.PNumMod(j+3);
|
||||||
|
PointIndex pi3 = sel.PNumMod(j+1);
|
||||||
|
PointIndex pi4;
|
||||||
|
PointGeomInfo gi1 = sel.GeomInfoPiMod(j+2);
|
||||||
|
PointGeomInfo gi2 = sel.GeomInfoPiMod(j+3);
|
||||||
|
PointGeomInfo gi3 = sel.GeomInfoPiMod(j+1);
|
||||||
|
PointGeomInfo gi4;
|
||||||
|
|
||||||
|
if (mesh.IsSegment (pi1, pi2)) continue;
|
||||||
|
|
||||||
|
// get neighbor element
|
||||||
|
INDEX_2 ii2 (pi1, pi2);
|
||||||
|
ii2.Sort();
|
||||||
|
auto els = els_on_edge.Get(ii2);
|
||||||
|
SurfaceElementIndex other_i = get<0>(els);
|
||||||
|
if(other_i==sei) other_i = get<1>(els);
|
||||||
|
auto & other = mesh[other_i];
|
||||||
|
|
||||||
|
// find opposite point of neighbor element
|
||||||
|
for (int j = 0; j < 3; j++)
|
||||||
|
if(other[j]!=pi1 && other[j]!=pi2)
|
||||||
|
{
|
||||||
|
pi4 = other[j];
|
||||||
|
gi4 = other.GeomInfoPi(j);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// split edge pi1,pi2
|
||||||
|
Point<3> p5;
|
||||||
|
PointIndex pi5;
|
||||||
|
PointGeomInfo gi5;
|
||||||
|
|
||||||
|
mesh.GetGeometry()->GetRefinement().PointBetween (mesh[pi1], mesh[pi2], 0.5,
|
||||||
|
faceindex,
|
||||||
|
gi1, gi2, p5, gi5);
|
||||||
|
|
||||||
|
pi5 = mesh.AddPoint(p5);
|
||||||
|
|
||||||
|
Element2d e1(3);
|
||||||
|
e1.SetIndex(faceindex);
|
||||||
|
e1={ {pi1,gi1}, {pi5,gi5}, {pi3,gi3} };
|
||||||
|
mesh.AddSurfaceElement( e1 );
|
||||||
|
|
||||||
|
Element2d e2(3);
|
||||||
|
e2.SetIndex(faceindex);
|
||||||
|
e2 ={ {pi5,gi5}, {pi2,gi2}, {pi3,gi3} };
|
||||||
|
mesh.AddSurfaceElement( e2 );
|
||||||
|
|
||||||
|
Element2d e3(3);
|
||||||
|
e3.SetIndex(faceindex);
|
||||||
|
e3 ={ {pi1,gi1}, {pi4,gi4}, {pi5,gi5} };
|
||||||
|
mesh.AddSurfaceElement( e3 );
|
||||||
|
|
||||||
|
Element2d e4(3);
|
||||||
|
e4.SetIndex(faceindex);
|
||||||
|
e4 ={ {pi4,gi4}, {pi2,gi2}, {pi5,gi5} };
|
||||||
|
mesh.AddSurfaceElement( e4 );
|
||||||
|
|
||||||
|
sel.Delete();
|
||||||
|
other.Delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mesh.SetNextTimeStamp();
|
||||||
|
}
|
||||||
|
|
||||||
void MeshOptimize2d :: CheckMeshApproximation (Mesh & mesh)
|
void MeshOptimize2d :: CheckMeshApproximation (Mesh & mesh)
|
||||||
{
|
{
|
||||||
|
@ -24,6 +24,7 @@ public:
|
|||||||
|
|
||||||
void EdgeSwapping (Mesh & mesh, int usemetric);
|
void EdgeSwapping (Mesh & mesh, int usemetric);
|
||||||
void CombineImprove (Mesh & mesh);
|
void CombineImprove (Mesh & mesh);
|
||||||
|
void SplitImprove (Mesh & mesh);
|
||||||
|
|
||||||
void GenericImprove (Mesh & mesh);
|
void GenericImprove (Mesh & mesh);
|
||||||
|
|
||||||
|
@ -3692,6 +3692,30 @@ namespace netgen
|
|||||||
|
|
||||||
bool Mesh :: LegalTrig (const Element2d & el) const
|
bool Mesh :: LegalTrig (const Element2d & el) const
|
||||||
{
|
{
|
||||||
|
// Search for surface trigs with same vertices ( may happen for instance with close surfaces in stl geometies )
|
||||||
|
if(!illegal_trigs)
|
||||||
|
{
|
||||||
|
auto & tab = const_cast<decltype(illegal_trigs)&>(illegal_trigs);
|
||||||
|
tab = make_unique<INDEX_3_CLOSED_HASHTABLE<int>> (3*GetNSE() + 1);
|
||||||
|
for (SurfaceElementIndex sei = 0; sei < GetNSE(); sei++)
|
||||||
|
{
|
||||||
|
const Element2d & sel = surfelements[sei];
|
||||||
|
if (sel.IsDeleted()) continue;
|
||||||
|
|
||||||
|
INDEX_3 i3(sel[0], sel[1], sel[2]);
|
||||||
|
i3.Sort();
|
||||||
|
if(tab->Used(i3) && tab->Get(i3)!=sei)
|
||||||
|
tab -> Set (i3, -1);
|
||||||
|
else
|
||||||
|
tab -> Set (i3, sei);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
INDEX_3 i3 (el[0], el[1], el[2]);
|
||||||
|
i3.Sort();
|
||||||
|
if(illegal_trigs->Used(i3) && illegal_trigs->Get(i3)==-1)
|
||||||
|
return false;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
if ( /* hp */ 1) // needed for old, simple hp-refinement
|
if ( /* hp */ 1) // needed for old, simple hp-refinement
|
||||||
{
|
{
|
||||||
|
@ -55,6 +55,7 @@ namespace netgen
|
|||||||
unique_ptr<INDEX_2_CLOSED_HASHTABLE<int>> segmentht;
|
unique_ptr<INDEX_2_CLOSED_HASHTABLE<int>> segmentht;
|
||||||
///
|
///
|
||||||
unique_ptr<INDEX_3_CLOSED_HASHTABLE<int>> surfelementht;
|
unique_ptr<INDEX_3_CLOSED_HASHTABLE<int>> surfelementht;
|
||||||
|
unique_ptr<INDEX_3_CLOSED_HASHTABLE<int>> illegal_trigs;
|
||||||
|
|
||||||
/// faces of rest-solid
|
/// faces of rest-solid
|
||||||
NgArray<Element2d> openelements;
|
NgArray<Element2d> openelements;
|
||||||
|
Loading…
Reference in New Issue
Block a user