mirror of
https://github.com/NGSolve/netgen.git
synced 2025-01-11 21:50:34 +05:00
automatic 2d mesh-size control
This commit is contained in:
parent
4dcae7ec4d
commit
2637b547b5
@ -9,51 +9,116 @@ namespace netgen
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
void CalcPartition (const SplineSegExt & spline, double l,
|
|
||||||
MeshingParameters & mp, double h, double h1, double h2,
|
void CalcPartition (const SplineSegExt & spline,
|
||||||
double hcurve, double elto0,
|
// double l,
|
||||||
Array<double> & points);
|
MeshingParameters & mp, Mesh & mesh,
|
||||||
|
// double h, double h1, double h2, double hcurve,
|
||||||
|
double elto0, Array<double> & points)
|
||||||
|
{
|
||||||
|
double fperel, oldf, f;
|
||||||
|
|
||||||
|
int n = 10000;
|
||||||
|
|
||||||
|
Array<Point<2> > xi(n);
|
||||||
|
Array<double> hi(n);
|
||||||
|
|
||||||
|
for (int i = 0; i < n; i++)
|
||||||
|
{
|
||||||
|
xi[i] = spline.GetPoint ( (i+0.5) / n );
|
||||||
|
hi[i] = mesh.GetH (Point<3> (xi[i](0), xi[i](1), 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
// limit slope
|
||||||
|
double gradh = 1/elto0;
|
||||||
|
for (int i = 0; i < n-1; i++)
|
||||||
|
{
|
||||||
|
double hnext = hi[i] + gradh * (xi[i+1]-xi[i]).Length();
|
||||||
|
hi[i+1] = min(hi[i+1], hnext);
|
||||||
|
}
|
||||||
|
for (int i = n-1; i > 1; i--)
|
||||||
|
{
|
||||||
|
double hnext = hi[i] + gradh * (xi[i-1]-xi[i]).Length();
|
||||||
|
hi[i-1] = min(hi[i-1], hnext);
|
||||||
|
}
|
||||||
|
|
||||||
|
points.SetSize (0);
|
||||||
|
|
||||||
|
double len = spline.Length();
|
||||||
|
double dt = len / n;
|
||||||
|
|
||||||
|
double sum = 0;
|
||||||
|
for (int i = 1; i <= n; i++)
|
||||||
|
{
|
||||||
|
double t = (i-0.5)*dt;
|
||||||
|
double fun = hi[i-1];
|
||||||
|
sum += dt / fun;
|
||||||
|
}
|
||||||
|
|
||||||
|
int nel = int (sum+1);
|
||||||
|
fperel = sum / nel;
|
||||||
|
|
||||||
|
points.Append (0);
|
||||||
|
|
||||||
|
int i = 1;
|
||||||
|
oldf = 0;
|
||||||
|
|
||||||
|
for (int j = 1; j <= n && i < nel; j++)
|
||||||
|
{
|
||||||
|
double t = (j-0.5)*dt;
|
||||||
|
double fun = hi[j-1];
|
||||||
|
|
||||||
|
f = oldf + dt / fun;
|
||||||
|
|
||||||
|
while (i * fperel < f && i < nel)
|
||||||
|
{
|
||||||
|
points.Append ( dt * (j-1) + (i * fperel - oldf) * fun);
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
oldf = f;
|
||||||
|
t += dt;
|
||||||
|
}
|
||||||
|
points.Append (len);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// partitionizes spline curve
|
// partitionizes spline curve
|
||||||
void Partition (const SplineSegExt & spline,
|
void Partition (const SplineSegExt & spline,
|
||||||
MeshingParameters & mp, double h, double elto0,
|
MeshingParameters & mp, double h, double elto0,
|
||||||
Mesh & mesh, Point3dTree & searchtree, int segnr)
|
Mesh & mesh, Point3dTree & searchtree, int segnr)
|
||||||
{
|
{
|
||||||
enum { D = 2 };
|
|
||||||
int i, j;
|
|
||||||
double l; // , r1, r2, ra;
|
|
||||||
double lold, dt, frac;
|
|
||||||
int n = 100;
|
int n = 100;
|
||||||
Point<D> p, pold, mark, oldmark;
|
|
||||||
|
Point<2> mark, oldmark;
|
||||||
Array<double> curvepoints;
|
Array<double> curvepoints;
|
||||||
double edgelength, edgelengthold;
|
double edgelength, edgelengthold;
|
||||||
l = spline.Length();
|
|
||||||
|
|
||||||
double h1 = min (spline.StartPI().hmax, h/spline.StartPI().refatpoint);
|
CalcPartition (spline, mp, mesh, elto0, curvepoints);
|
||||||
double h2 = min (spline.EndPI().hmax, h/spline.EndPI().refatpoint);
|
|
||||||
double hcurve = min (spline.hmax, h/spline.reffak);
|
|
||||||
|
|
||||||
CalcPartition (spline, l, mp, h, h1, h2, hcurve, elto0, curvepoints);
|
double dt = 1.0 / n;
|
||||||
// cout << "curvepoints = " << curvepoints << endl;
|
|
||||||
|
|
||||||
dt = 1.0 / n;
|
int j = 1;
|
||||||
|
|
||||||
l = 0;
|
Point<2> pold = spline.GetPoint (0);
|
||||||
j = 1;
|
double lold = 0;
|
||||||
|
|
||||||
pold = spline.GetPoint (0);
|
|
||||||
lold = 0;
|
|
||||||
oldmark = pold;
|
oldmark = pold;
|
||||||
edgelengthold = 0;
|
edgelengthold = 0;
|
||||||
Array<int> locsearch;
|
Array<int> locsearch;
|
||||||
|
|
||||||
for (i = 1; i <= n; i++)
|
for (int i = 1; i <= n; i++)
|
||||||
{
|
{
|
||||||
p = spline.GetPoint (i*dt);
|
Point<2> p = spline.GetPoint (i*dt);
|
||||||
l = lold + Dist (p, pold);
|
double l = lold + Dist (p, pold);
|
||||||
while (j < curvepoints.Size() && (l >= curvepoints[j] || i == n))
|
while (j < curvepoints.Size() && (l >= curvepoints[j] || i == n))
|
||||||
{
|
{
|
||||||
frac = (curvepoints[j]-lold) / (l-lold);
|
double frac = (curvepoints[j]-lold) / (l-lold);
|
||||||
edgelength = i*dt + (frac-1)*dt;
|
edgelength = i*dt + (frac-1)*dt;
|
||||||
// mark = pold + frac * (p-pold);
|
// mark = pold + frac * (p-pold);
|
||||||
mark = spline.GetPoint (edgelength);
|
mark = spline.GetPoint (edgelength);
|
||||||
@ -72,26 +137,11 @@ namespace netgen
|
|||||||
for (int k = 0; k < locsearch.Size(); k++)
|
for (int k = 0; k < locsearch.Size(); k++)
|
||||||
if ( mesh[PointIndex(locsearch[k])].GetLayer() == spline.layer)
|
if ( mesh[PointIndex(locsearch[k])].GetLayer() == spline.layer)
|
||||||
pi1 = locsearch[k];
|
pi1 = locsearch[k];
|
||||||
// if (locsearch.Size()) pi1 = locsearch[0];
|
|
||||||
|
|
||||||
searchtree.GetIntersecting (mark3 - v, mark3 + v, locsearch);
|
searchtree.GetIntersecting (mark3 - v, mark3 + v, locsearch);
|
||||||
for (int k = 0; k < locsearch.Size(); k++)
|
for (int k = 0; k < locsearch.Size(); k++)
|
||||||
if ( mesh[PointIndex(locsearch[k])].GetLayer() == spline.layer)
|
if ( mesh[PointIndex(locsearch[k])].GetLayer() == spline.layer)
|
||||||
pi2 = locsearch[k];
|
pi2 = locsearch[k];
|
||||||
// if (locsearch.Size()) pi2 = locsearch[0];
|
|
||||||
|
|
||||||
/*
|
|
||||||
for (PointIndex pk = PointIndex::BASE;
|
|
||||||
pk < mesh.GetNP()+PointIndex::BASE; pk++)
|
|
||||||
{
|
|
||||||
if (Dist (mesh[pk], oldmark3) < 1e-4 * h) pi1 = pk;
|
|
||||||
if (Dist (mesh[pk], mark3) < 1e-4 * h) pi2 = pk;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
// cout << "pi1 = " << pi1 << endl;
|
|
||||||
// cout << "pi2 = " << pi2 << endl;
|
|
||||||
|
|
||||||
if (pi1 == -1)
|
if (pi1 == -1)
|
||||||
{
|
{
|
||||||
@ -157,24 +207,52 @@ namespace netgen
|
|||||||
if (dom != 0) GetSpline(i).layer = GetDomainLayer (dom);
|
if (dom != 0) GetSpline(i).layer = GetDomainLayer (dom);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// mesh size restrictions ...
|
||||||
|
|
||||||
|
for (int i = 0; i < splines.Size(); i++)
|
||||||
|
{
|
||||||
|
const SplineSegExt & spline = GetSpline(i);
|
||||||
|
const GeomPoint<2> & p1 = spline.StartPI();
|
||||||
|
const GeomPoint<2> & p2 = spline.EndPI();
|
||||||
|
|
||||||
|
double h1 = min (p1.hmax, h/p1.refatpoint);
|
||||||
|
mesh2d.RestrictLocalH (Point<3>(p1(0),p1(1),0), h1);
|
||||||
|
double h2 = min (p2.hmax, h/p2.refatpoint);
|
||||||
|
mesh2d.RestrictLocalH (Point<3>(p2(0),p2(1),0), h2);
|
||||||
|
|
||||||
|
double len = spline.Length();
|
||||||
|
mesh2d.RestrictLocalHLine (Point<3>(p1(0),p1(1),0),
|
||||||
|
Point<3>(p2(0),p2(1),0), len/mp.segmentsperedge);
|
||||||
|
|
||||||
|
double hcurve = min (spline.hmax, h/spline.reffak);
|
||||||
|
double hl = GetDomainMaxh (spline.leftdom);
|
||||||
|
if (hl > 0) hcurve = min2 (hcurve, hl);
|
||||||
|
double hr = GetDomainMaxh (spline.rightdom);
|
||||||
|
if (hr > 0) hcurve = min2 (hcurve, hr);
|
||||||
|
|
||||||
|
int np = 1000;
|
||||||
|
for (double t = 0.5/np; t < 1; t += 1.0/np)
|
||||||
|
{
|
||||||
|
Point<2> x = spline.GetPoint(t);
|
||||||
|
double hc = 1.0/mp.curvaturesafety / (1e-99+spline.CalcCurvature (t));
|
||||||
|
mesh2d.RestrictLocalH (Point<3> (x(0), x(1), 0), min2(hc, spline.hmax));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (int i = 0; i < splines.Size(); i++)
|
for (int i = 0; i < splines.Size(); i++)
|
||||||
if (GetSpline(i).copyfrom == -1)
|
if (GetSpline(i).copyfrom == -1)
|
||||||
{
|
{
|
||||||
// astrid - set boundary meshsize to domain meshsize h
|
// astrid - set boundary meshsize to domain meshsize h
|
||||||
// if no domain mesh size is given, the max h value from the bounding box is used
|
// if no domain mesh size is given, the max h value from the bounding box is used
|
||||||
double minimum = min2 ( GetDomainMaxh ( GetSpline(i).leftdom ), GetDomainMaxh ( GetSpline(i).rightdom ) );
|
double hl = GetDomainMaxh ( GetSpline(i).leftdom );
|
||||||
double maximum = max2 ( GetDomainMaxh ( GetSpline(i).leftdom ), GetDomainMaxh ( GetSpline(i).rightdom ) );
|
double hr = GetDomainMaxh ( GetSpline(i).rightdom );
|
||||||
minimum = min2 ( minimum, h );
|
|
||||||
maximum = min2 ( maximum, h);
|
double useh = h;
|
||||||
if ( minimum > 0 )
|
if (hl > 0) useh = min2 (h, hl);
|
||||||
// GetSpline(i).Partition(minimum, elto0, mesh2d, searchtree, i+1);
|
if (hr > 0) useh = min2 (h, hr);
|
||||||
Partition(GetSpline(i), mp, minimum, elto0, mesh2d, searchtree, i+1);
|
|
||||||
else if ( maximum > 0 )
|
Partition(GetSpline(i), mp, useh, elto0, mesh2d, searchtree, i+1);
|
||||||
// GetSpline(i).Partition(maximum, elto0, mesh2d, searchtree, i+1);
|
|
||||||
Partition(GetSpline(i), mp, maximum, elto0, mesh2d, searchtree, i+1);
|
|
||||||
else
|
|
||||||
// GetSpline(i).Partition(h, elto0, mesh2d, searchtree, i+1);
|
|
||||||
Partition(GetSpline(i), mp, h, elto0, mesh2d, searchtree, i+1);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -298,6 +376,14 @@ namespace netgen
|
|||||||
mesh = new Mesh;
|
mesh = new Mesh;
|
||||||
mesh->SetDimension (2);
|
mesh->SetDimension (2);
|
||||||
|
|
||||||
|
Point3d pmin(bbox.PMin()(0), bbox.PMin()(1), -bbox.Diam());
|
||||||
|
Point3d pmax(bbox.PMax()(0), bbox.PMax()(1), bbox.Diam());
|
||||||
|
|
||||||
|
mesh->SetLocalH (pmin, pmax, mparam.grading);
|
||||||
|
mesh->SetGlobalH (h);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
geometry.PartitionBoundary (mp, h, *mesh);
|
geometry.PartitionBoundary (mp, h, *mesh);
|
||||||
|
|
||||||
|
|
||||||
@ -347,11 +433,6 @@ namespace netgen
|
|||||||
for (SegmentIndex si = 0; si < mesh->GetNSeg(); si++)
|
for (SegmentIndex si = 0; si < mesh->GetNSeg(); si++)
|
||||||
(*mesh)[si].SetBCName ( (*mesh).GetBCNamePtr( (*mesh)[si].si-1 ) );
|
(*mesh)[si].SetBCName ( (*mesh).GetBCNamePtr( (*mesh)[si].si-1 ) );
|
||||||
|
|
||||||
Point3d pmin(bbox.PMin()(0), bbox.PMin()(1), -bbox.Diam());
|
|
||||||
Point3d pmax(bbox.PMax()(0), bbox.PMax()(1), bbox.Diam());
|
|
||||||
|
|
||||||
mesh->SetLocalH (pmin, pmax, mparam.grading);
|
|
||||||
mesh->SetGlobalH (h);
|
|
||||||
|
|
||||||
mesh->CalcLocalH(mparam.grading);
|
mesh->CalcLocalH(mparam.grading);
|
||||||
|
|
||||||
@ -489,11 +570,6 @@ namespace netgen
|
|||||||
|
|
||||||
mparam.checkoverlap = 0;
|
mparam.checkoverlap = 0;
|
||||||
|
|
||||||
/*
|
|
||||||
if (!mparam.quad)
|
|
||||||
meshing.Delaunay (*mesh, domnr, mparam);
|
|
||||||
else
|
|
||||||
*/
|
|
||||||
meshing.GenerateMesh (*mesh, mparam, h, domnr);
|
meshing.GenerateMesh (*mesh, mparam, h, domnr);
|
||||||
|
|
||||||
for (SurfaceElementIndex sei = oldnf; sei < mesh->GetNSE(); sei++)
|
for (SurfaceElementIndex sei = oldnf; sei < mesh->GetNSE(); sei++)
|
||||||
|
@ -884,99 +884,6 @@ namespace netgen
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
void CalcPartition (const SplineSegExt & spline, double l,
|
|
||||||
MeshingParameters & mp, double h, double h1, double h2,
|
|
||||||
double hcurve, double elto0, Array<double> & points)
|
|
||||||
{
|
|
||||||
double fperel, oldf, f;
|
|
||||||
|
|
||||||
int n = 10000;
|
|
||||||
|
|
||||||
Array<Point<2> > xi(n);
|
|
||||||
Array<double> hi(n);
|
|
||||||
|
|
||||||
hi = h;
|
|
||||||
for (int i = 0; i < n; i++)
|
|
||||||
{
|
|
||||||
xi[i] = spline.GetPoint ( (i+0.5) / n );
|
|
||||||
double hc = 1.0/mp.curvaturesafety / (1e-20+spline.CalcCurvature ( (i+0.5) / n ));
|
|
||||||
// *testout << "t = " << (i+0.5)/n << ", xi = " << xi[i] << ", hi = " << hc << endl;
|
|
||||||
hi[i] = min2(hi[i], hc);
|
|
||||||
}
|
|
||||||
hi[0] = min (hi[0], h1);
|
|
||||||
hi[n-1] = min (hi[n-1], h2);
|
|
||||||
|
|
||||||
if ( (spline.GetPoint(0)-spline.GetPoint(1)).Length() < 1e-12 )
|
|
||||||
hi[0] = hi[n-1] = min2 (hi[0], hi[n-1]);
|
|
||||||
|
|
||||||
// limit slope
|
|
||||||
double gradh = 1/elto0;
|
|
||||||
for (int i = 0; i < n-1; i++)
|
|
||||||
{
|
|
||||||
double hnext = hi[i] + gradh * (xi[i+1]-xi[i]).Length();
|
|
||||||
hi[i+1] = min(hi[i+1], hnext);
|
|
||||||
}
|
|
||||||
for (int i = n-1; i > 1; i--)
|
|
||||||
{
|
|
||||||
double hnext = hi[i] + gradh * (xi[i-1]-xi[i]).Length();
|
|
||||||
hi[i-1] = min(hi[i-1], hnext);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
points.SetSize (0);
|
|
||||||
|
|
||||||
double dt = l / n;
|
|
||||||
|
|
||||||
double sum = 0;
|
|
||||||
for (int i = 1; i <= n; i++)
|
|
||||||
{
|
|
||||||
double t = (i-0.5)*dt;
|
|
||||||
/*
|
|
||||||
double fun = min3 (hcurve, t/elto0 + h1, (l-t)/elto0 + h2);
|
|
||||||
double curv = spline.CalcCurvature (t/l);
|
|
||||||
cout << "curv = " << curv << endl;
|
|
||||||
if (curv < 1e-10) curv = 1e-10;
|
|
||||||
fun = min2 (fun, 0.1/curv);
|
|
||||||
*/
|
|
||||||
double fun = hi[i-1];
|
|
||||||
sum += dt / fun;
|
|
||||||
}
|
|
||||||
|
|
||||||
int nel = int (sum+1);
|
|
||||||
fperel = sum / nel;
|
|
||||||
|
|
||||||
points.Append (0);
|
|
||||||
|
|
||||||
int i = 1;
|
|
||||||
oldf = 0;
|
|
||||||
// t = 0.5 * dt;
|
|
||||||
for (int j = 1; j <= n && i < nel; j++)
|
|
||||||
{
|
|
||||||
double t = (j-0.5)*dt;
|
|
||||||
double fun = hi[j-1];
|
|
||||||
/*
|
|
||||||
double fun = min3 (hcurve, t/elto0 + h1, (l-t)/elto0 + h2);
|
|
||||||
double curv = spline.CalcCurvature (t/l);
|
|
||||||
if (curv < 1e-10) curv = 1e-10;
|
|
||||||
fun = min2 (fun, 0.1/curv);
|
|
||||||
*/
|
|
||||||
f = oldf + dt / fun;
|
|
||||||
|
|
||||||
while (i * fperel < f && i < nel)
|
|
||||||
{
|
|
||||||
points.Append ( dt * (j-1) + (i * fperel - oldf) * fun);
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
oldf = f;
|
|
||||||
t += dt;
|
|
||||||
}
|
|
||||||
points.Append (l);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user