2d Delaunay, array-iterators

This commit is contained in:
Joachim Schoeberl 2014-12-02 13:23:36 +00:00
parent 9df9eeca0b
commit 8e8e390f2e
15 changed files with 749 additions and 282 deletions

View File

@ -16,6 +16,54 @@ namespace netgen
template <typename TSIZE>
class ArrayRangeIterator
{
TSIZE ind;
public:
ArrayRangeIterator (TSIZE ai) : ind(ai) { ; }
ArrayRangeIterator operator++ (int) { return ind++; }
ArrayRangeIterator operator++ () { return ++ind; }
TSIZE operator*() const { return ind; }
bool operator != (ArrayRangeIterator d2) { return ind != d2.ind; }
};
/// a range of intergers
template <typename T>
class T_Range
{
T first, next;
public:
T_Range (T f, T n) : first(f), next(n) {;}
T Size() const { return next-first; }
T operator[] (T i) const { return first+i; }
bool Contains (T i) const { return ((i >= first) && (i < next)); }
ArrayRangeIterator<T> begin() const { return first; }
ArrayRangeIterator<T> end() const { return next; }
};
template <typename T, int BASE = 0, typename TIND = int>
class FlatArray;
template <typename T, int BASE, typename TIND>
class ArrayIterator
{
FlatArray<T,BASE,TIND> ar;
TIND ind;
public:
ArrayIterator (FlatArray<T,BASE,TIND> aar, TIND ai) : ar(aar), ind(ai) { ; }
ArrayIterator operator++ (int) { return ArrayIterator(ar, ind++); }
ArrayIterator operator++ () { return ArrayIterator(ar, ++ind); }
T operator*() const { return ar[ind]; }
T & operator*() { return ar[ind]; }
bool operator != (ArrayIterator d2) { return ind != d2.ind; }
};
/** /**
A simple array container. A simple array container.
Array represented by size and data-pointer. Array represented by size and data-pointer.
@ -24,7 +72,7 @@ namespace netgen
Optional range check by macro RANGE_CHECK Optional range check by macro RANGE_CHECK
*/ */
template <typename T, int BASE = 0, typename TIND = int> template <typename T, int BASE, typename TIND>
class FlatArray class FlatArray
{ {
protected: protected:
@ -42,8 +90,14 @@ namespace netgen
/// the size /// the size
int Size() const { return size; } int Size() const { return size; }
ArrayIterator<T,BASE,TIND> begin() const
{ return ArrayIterator<T,BASE,TIND> (*this, BASE); }
ArrayIterator<T,BASE,TIND> end() const
{ return ArrayIterator<T,BASE,TIND> (*this, BASE+size); }
TIND Begin() const { return TIND(BASE); } TIND Begin() const { return TIND(BASE); }
TIND End() const { return TIND(size+BASE); } TIND End() const { return TIND(size+BASE); }
T_Range<TIND> Range() const { return T_Range<TIND>(BASE, size+BASE); }
/// Access array. BASE-based /// Access array. BASE-based
T & operator[] (TIND i) const T & operator[] (TIND i) const

View File

@ -428,6 +428,7 @@ namespace netgen
mesh->CalcLocalH(mp.grading); mesh->CalcLocalH(mp.grading);
int bnp = mesh->GetNP(); // boundary points int bnp = mesh->GetNP(); // boundary points
auto BndPntRange = mesh->Points().Range();
int hquad = mp.quad; int hquad = mp.quad;
@ -532,10 +533,10 @@ namespace netgen
Array<int, PointIndex::BASE> compress(bnp); Array<int, PointIndex::BASE> compress(bnp);
compress = -1; compress = -1;
int cnt = 0; int cnt = 0;
for (PointIndex pi = PointIndex::BASE; pi < bnp+PointIndex::BASE; pi++) for (PointIndex pi : BndPntRange)
if ( (*mesh)[pi].GetLayer() == geometry.GetDomainLayer(domnr)) if ( (*mesh)[pi].GetLayer() == geometry.GetDomainLayer(domnr))
{ {
meshing.AddPoint ( (*mesh)[pi], pi); meshing.AddPoint ((*mesh)[pi], pi);
cnt++; cnt++;
compress[pi] = cnt; compress[pi] = cnt;
} }
@ -546,35 +547,29 @@ namespace netgen
{ {
if ( (*mesh)[si].domin == domnr) if ( (*mesh)[si].domin == domnr)
{ {
meshing.AddBoundaryElement ( compress[(*mesh)[si][0]], meshing.AddBoundaryElement (compress[(*mesh)[si][0]],
compress[(*mesh)[si][1]], gi, gi); compress[(*mesh)[si][1]], gi, gi);
} }
if ( (*mesh)[si].domout == domnr) if ( (*mesh)[si].domout == domnr)
{ {
meshing.AddBoundaryElement ( compress[(*mesh)[si][1]], meshing.AddBoundaryElement (compress[(*mesh)[si][1]],
compress[(*mesh)[si][0]], gi, gi); compress[(*mesh)[si][0]], gi, gi);
}
} }
} // not complete, use at own risk ...
// meshing.Delaunay(*mesh, domnr, mp);
mp.checkoverlap = 0; mp.checkoverlap = 0;
meshing.GenerateMesh (*mesh, mp, h, domnr); meshing.GenerateMesh (*mesh, mp, h, domnr);
for (SurfaceElementIndex sei = oldnf; sei < mesh->GetNSE(); sei++) for (SurfaceElementIndex sei = oldnf; sei < mesh->GetNSE(); sei++)
(*mesh)[sei].SetIndex (domnr); (*mesh)[sei].SetIndex (domnr);
// astrid // astrid
char * material; char * material;
geometry.GetMaterial( domnr, material ); geometry.GetMaterial (domnr, material);
if ( material ) if (material)
{ mesh->SetMaterial (domnr, material);
(*mesh).SetMaterial ( domnr, material );
}
} }
mp.quad = hquad; mp.quad = hquad;

View File

@ -492,4 +492,72 @@ namespace netgen
return ((cnt % 2) != 0); return ((cnt % 2) != 0);
} }
bool AdFront2 :: SameSide (const Point<2> & lp1, const Point<2> & lp2,
const Array<int> * testfaces) const
{
int cnt = 0;
if (testfaces)
{
for (int ii = 0; ii < testfaces->Size(); ii++)
if (lines[(*testfaces)[ii]].Valid())
{
int i = (*testfaces)[ii];
const Point<3> & p13d = points[lines[i].L().I1()].P();
const Point<3> & p23d = points[lines[i].L().I2()].P();
Point<2> p1(p13d(0), p13d(1));
Point<2> p2(p23d(0), p23d(1));
// p1 + alpha v = lp1 + beta vl
Vec<2> v = p2-p1;
Vec<2> vl = lp2 - lp1;
Mat<2,2> mat, inv;
Vec<2> rhs, sol;
mat(0,0) = v(0);
mat(1,0) = v(1);
mat(0,1) = -vl(0);
mat(1,1) = -vl(1);
rhs = lp1-p1;
if (Det(mat) == 0) continue;
CalcInverse (mat, inv);
sol = inv * rhs;
if (sol(0) >= 0 && sol(0) <= 1 & sol(1) >= 0 && sol(1) <= 1)
{ cnt++; }
}
}
else
{
for (int i = 0; i < lines.Size(); i++)
if (lines[i].Valid())
{
const Point<3> & p13d = points[lines[i].L().I1()].P();
const Point<3> & p23d = points[lines[i].L().I2()].P();
Point<2> p1(p13d(0), p13d(1));
Point<2> p2(p23d(0), p23d(1));
// p1 + alpha v = lp1 + beta vl
Vec<2> v = p2-p1;
Vec<2> vl = lp2 - lp1;
Mat<2,2> mat, inv;
Vec<2> rhs, sol;
mat(0,0) = v(0);
mat(1,0) = v(1);
mat(0,1) = -vl(0);
mat(1,1) = -vl(1);
rhs = lp1-p1;
if (Det(mat) == 0) continue;
CalcInverse (mat, inv);
sol = inv * rhs;
if (sol(0) >= 0 && sol(0) <= 1 & sol(1) >= 0 && sol(1) <= 1)
{ cnt++; }
}
}
return ((cnt % 2) == 0);
}
} }

View File

@ -262,11 +262,12 @@ public:
bool Inside (const Point<2> & p) const; bool Inside (const Point<2> & p) const;
bool SameSide (const Point<2> & lp1, const Point<2> & lp2, bool SameSide (const Point<2> & lp1, const Point<2> & lp2,
const Array<int> * /* testfaces */ = NULL) const const Array<int> * /* testfaces */ = NULL) const;
/*
{ {
return Inside (lp1) == Inside (lp2); return Inside (lp1) == Inside (lp2);
} }
*/
/// ///
void SetStartFront (); void SetStartFront ();

View File

@ -116,8 +116,8 @@ namespace netgen
Vec3d Vect_A,Vect_B; Vec3d Vect_A,Vect_B;
Vect_A = mesh.Point(el.PNum(Vertex_A)) - mesh.Point(el.PNum(Vertex)); Vect_A = mesh[el.PNum(Vertex_A)] - mesh[el.PNum(Vertex)];
Vect_B = mesh.Point(el.PNum(Vertex_B)) - mesh.Point(el.PNum(Vertex)); Vect_B = mesh[el.PNum(Vertex_B)] - mesh[el.PNum(Vertex)];
SurfaceNormal = Cross(Vect_A,Vect_B); SurfaceNormal = Cross(Vect_A,Vect_B);
SurfaceNormal.Normalize(); SurfaceNormal.Normalize();

View File

@ -6,8 +6,68 @@
namespace netgen namespace netgen
{ {
class DelaunayTrig
{
PointIndex pnums[3];
Point<3> c;
double r;
double rad2;
public:
DelaunayTrig () { ; }
DelaunayTrig (int p1, int p2, int p3)
{ pnums[0] = p1; pnums[1] = p2; pnums[2] = p3; }
PointIndex & operator[] (int j) { return pnums[j]; }
const PointIndex & operator[] (int j) const { return pnums[j]; }
void CalcCenter (Mesh & mesh)
{
Point<3> p1 = mesh[pnums[0]];
Point<3> p2 = mesh[pnums[1]];
Point<3> p3 = mesh[pnums[2]];
Vec<3> v1 = p2-p1;
Vec<3> v2 = p3-p1;
Mat<2,2> mat, inv;
mat(0,0) = v1*v1;
mat(0,1) = v1*v2;
mat(1,0) = v2*v1;
mat(1,1) = v2*v2;
Vec<2> rhs, sol;
rhs(0) = 0.5 * v1*v1;
rhs(1) = 0.5 * v2*v2;
CalcInverse (mat, inv);
sol = inv * rhs;
c = p1 + sol(0) * v1 + sol(1) * v2;
rad2 = Dist2(c, p1);
r = sqrt(rad2);
}
Point<3> Center() const { return c; }
double Radius2() const { return rad2; }
Box<3> BoundingBox() const { return Box<3> (c-Vec<3>(r,r,0.1), c+Vec<3>(r,r,0.1)); }
};
ostream & operator<< (ostream & ost, DelaunayTrig trig)
{
ost << trig[0] << "-" << trig[1] << "-" << trig[2] << endl;
return ost;
}
void Meshing2 :: BlockFillLocalH (Mesh & mesh, const MeshingParameters & mp) void Meshing2 :: BlockFillLocalH (Mesh & mesh, const MeshingParameters & mp)
{ {
static int timer = NgProfiler::CreateTimer ("Meshing2::BlockFill");
static int timer1 = NgProfiler::CreateTimer ("Meshing2::BlockFill 1");
static int timer2 = NgProfiler::CreateTimer ("Meshing2::BlockFill 2");
static int timer3 = NgProfiler::CreateTimer ("Meshing2::BlockFill 3");
static int timer4 = NgProfiler::CreateTimer ("Meshing2::BlockFill 4");
NgProfiler::RegionTimer reg (timer);
NgProfiler::StartTimer (timer1);
double filldist = mp.filldist; double filldist = mp.filldist;
cout << "blockfill local h" << endl; cout << "blockfill local h" << endl;
@ -28,8 +88,7 @@ namespace netgen
const Point<3> & p1 = adfront->GetPoint(line.L().I1()); const Point<3> & p1 = adfront->GetPoint(line.L().I1());
const Point<3> & p2 = adfront->GetPoint(line.L().I2()); const Point<3> & p2 = adfront->GetPoint(line.L().I2());
double hi = Dist (p1, p2); maxh = max (maxh, Dist (p1, p2));
if (hi > maxh) maxh = hi;
bbox.Add (p1); bbox.Add (p1);
bbox.Add (p2); bbox.Add (p2);
@ -43,7 +102,11 @@ namespace netgen
bbox.Increase (bbox.Diam()/2); bbox.Increase (bbox.Diam()/2);
Box<3> meshbox = bbox; Box<3> meshbox = bbox;
LocalH loch2 (bbox, 1); NgProfiler::StopTimer (timer1);
NgProfiler::StartTimer (timer2);
LocalH loch2 (bbox, 1, 2);
if (mp.maxh < maxh) maxh = mp.maxh; if (mp.maxh < maxh) maxh = mp.maxh;
@ -75,7 +138,7 @@ namespace netgen
changed = false; changed = false;
for (int i = 0; i < npoints.Size(); i++) for (int i = 0; i < npoints.Size(); i++)
{ {
if (mesh.LocalHFunction().GetH(npoints[i]) > 1.5 * maxh) if (mesh.LocalHFunction().GetH(npoints[i]) > 1.2 * maxh)
{ {
mesh.LocalHFunction().SetH (npoints[i], maxh); mesh.LocalHFunction().SetH (npoints[i], maxh);
changed = true; changed = true;
@ -84,6 +147,10 @@ namespace netgen
} }
while (changed); while (changed);
NgProfiler::StopTimer (timer2);
NgProfiler::StartTimer (timer3);
if (debugparam.slowchecks) if (debugparam.slowchecks)
(*testout) << "Blockfill with points: " << endl; (*testout) << "Blockfill with points: " << endl;
*testout << "loch = " << mesh.LocalHFunction() << endl; *testout << "loch = " << mesh.LocalHFunction() << endl;
@ -112,6 +179,8 @@ namespace netgen
} }
} }
NgProfiler::StopTimer (timer3);
NgProfiler::StartTimer (timer4);
// find outer points // find outer points
@ -142,6 +211,7 @@ namespace netgen
loch2.FindInnerBoxes (adfront, NULL); loch2.FindInnerBoxes (adfront, NULL);
// outer points : smooth mesh-grading
npoints.SetSize(0); npoints.SetSize(0);
loch2.GetOuterPoints (npoints); loch2.GetOuterPoints (npoints);
@ -154,21 +224,199 @@ namespace netgen
} }
} }
NgProfiler::StopTimer (timer4);
} }
void Meshing2 :: Delaunay (Mesh & mesh, int domainnr, const MeshingParameters & mp) void Meshing2 :: Delaunay (Mesh & mesh, int domainnr, const MeshingParameters & mp)
{ {
static int timer = NgProfiler::CreateTimer ("Meshing2::Delaunay - total");
static int timerstart = NgProfiler::CreateTimer ("Meshing2::Delaunay - start");
static int timerfinish = NgProfiler::CreateTimer ("Meshing2::Delaunay - finish");
static int timer1 = NgProfiler::CreateTimer ("Meshing2::Delaunay - incremental");
static int timer1a = NgProfiler::CreateTimer ("Meshing2::Delaunay - incremental a");
static int timer1b = NgProfiler::CreateTimer ("Meshing2::Delaunay - incremental b");
static int timer1c = NgProfiler::CreateTimer ("Meshing2::Delaunay - incremental c");
static int timer1d = NgProfiler::CreateTimer ("Meshing2::Delaunay - incremental d");
NgProfiler::RegionTimer reg (timer);
cout << "2D Delaunay meshing (in progress)" << endl; cout << "2D Delaunay meshing (in progress)" << endl;
// int oldnp = mesh.GetNP();
cout << "np, old = " << mesh.GetNP() << endl;
BlockFillLocalH (mesh, mp); BlockFillLocalH (mesh, mp);
NgProfiler::StartTimer (timerstart);
cout << "np, now = " << mesh.GetNP() << endl; // do the delaunay
// face bounding box:
Box<3> bbox (Box<3>::EMPTY_BOX);
for (int i = 0; i < adfront->GetNFL(); i++)
{
const FrontLine & line = adfront->GetLine(i);
bbox.Add (Point<3> (adfront->GetPoint (line.L()[0])));
bbox.Add (Point<3> (adfront->GetPoint (line.L()[1])));
}
for (int i = 0; i < mesh.LockedPoints().Size(); i++)
bbox.Add (mesh.Point (mesh.LockedPoints()[i]));
cout << "bbox = " << bbox << endl;
// external point
Vec<3> vdiag = bbox.PMax()-bbox.PMin();
auto old_points = mesh.Points().Range();
DelaunayTrig startel;
startel[0] = mesh.AddPoint (bbox.PMin() + Vec<3> (-8*vdiag(0), -8*vdiag(1), 0));
startel[1] = mesh.AddPoint (bbox.PMin() + Vec<3> (+8*vdiag(0), -8*vdiag(1), 0));
startel[2] = mesh.AddPoint (bbox.PMin() + Vec<3> (0, 8*vdiag(1), 0));
Box<3> hbox;
hbox.Set (mesh[startel[0]]);
hbox.Add (mesh[startel[1]]);
hbox.Add (mesh[startel[2]]);
Point<3> hp = mesh[startel[0]];
hp(2) = 1; hbox.Add (hp);
hp(2) = -1; hbox.Add (hp);
Box3dTree searchtree(hbox);
Array<DelaunayTrig> tempels;
startel.CalcCenter (mesh);
tempels.Append (startel);
searchtree.Insert(startel.BoundingBox(), 0);
Array<int> closeels;
Array<int> intersecting;
Array<INDEX_2> edges;
// reorder points
Array<PointIndex, PointIndex::BASE, PointIndex> mixed(old_points.Size());
int prims[] = { 11, 13, 17, 19, 23, 29, 31, 37 };
int prim;
{
int i = 0;
while (old_points.Size() % prims[i] == 0) i++;
prim = prims[i];
}
for (PointIndex pi : old_points)
mixed[pi] = PointIndex ( (prim * pi) % old_points.Size() + PointIndex::BASE );
NgProfiler::StopTimer (timerstart);
NgProfiler::StartTimer (timer1);
for (PointIndex i1 : old_points)
{
PointIndex i = mixed[i1];
NgProfiler::StartTimer (timer1a);
Point<3> newp = mesh[i];
intersecting.SetSize(0);
edges.SetSize(0);
searchtree.GetIntersecting (newp, newp, closeels);
// for (int jj = 0; jj < closeels.Size(); jj++)
// for (int j = 0; j < tempels.Size(); j++)
for (int j : closeels)
{
if (tempels[j][0] < 0) continue;
Point<3> c = tempels[j].Center();
double r2 = tempels[j].Radius2();
bool inside = Dist2(mesh[i], c) < r2;
if (inside) intersecting.Append (j);
}
NgProfiler::StopTimer (timer1a);
NgProfiler::StartTimer (timer1b);
// find outer edges
for (auto j : intersecting)
{
const DelaunayTrig & trig = tempels[j];
for (int k = 0; k < 3; k++)
{
int p1 = trig[k];
int p2 = trig[(k+1)%3];
INDEX_2 edge(p1,p2);
edge.Sort();
bool found = false;
for (int l = 0; l < edges.Size(); l++)
if (edges[l] == edge)
{
edges.Delete(l);
found = true;
break;
}
if (!found) edges.Append (edge);
}
}
NgProfiler::StopTimer (timer1b);
NgProfiler::StartTimer (timer1c);
/*
for (int j = intersecting.Size()-1; j >= 0; j--)
tempels.Delete (intersecting[j]);
*/
for (int j : intersecting)
{
searchtree.DeleteElement (j);
tempels[j][0] = -1;
tempels[j][1] = -1;
tempels[j][2] = -1;
}
NgProfiler::StopTimer (timer1c);
NgProfiler::StartTimer (timer1d);
for (auto edge : edges)
{
DelaunayTrig trig (edge[0], edge[1], i);
trig.CalcCenter (mesh);
tempels.Append (trig);
searchtree.Insert(trig.BoundingBox(), tempels.Size()-1);
}
NgProfiler::StopTimer (timer1d);
}
NgProfiler::StopTimer (timer1);
NgProfiler::StartTimer (timerfinish);
for (DelaunayTrig & trig : tempels)
{
if (trig[0] < 0) continue;
Point<3> c = Center (mesh[trig[0]], mesh[trig[1]], mesh[trig[2]]);
if (!adfront->Inside (Point<2> (c(0),c(1)))) continue;
Vec<3> n = Cross (mesh[trig[1]]-mesh[trig[0]],
mesh[trig[2]]-mesh[trig[0]]);
if (n(2) < 0) Swap (trig[1], trig[2]);
Element2d el(trig[0], trig[1], trig[2]);
el.SetIndex (domainnr);
mesh.AddSurfaceElement (el);
}
for (PointIndex pi : mesh.Points().Range())
*testout << pi << ": " << mesh[pi].Type() << endl;
NgProfiler::StopTimer (timerfinish);
} }
} }

View File

@ -722,9 +722,12 @@ namespace netgen
if (should) if (should)
{ {
// (*testout) << "combine !" << endl; /*
// (*testout) << "bad1 = " << bad1 << ", bad2 = " << bad2 << endl; (*testout) << "combine !" << endl;
(*testout) << "bad1 = " << bad1 << ", bad2 = " << bad2 << endl;
(*testout) << "illegal1 = " << illegal1 << ", illegal2 = " << illegal2 << endl;
(*testout) << "loch = " << loch << endl;
*/
mesh[pi1] = pnew; mesh[pi1] = pnew;
PointGeomInfo gi; PointGeomInfo gi;

View File

@ -52,69 +52,36 @@ namespace netgen
} }
LocalH :: LocalH (const Point3d & pmin, const Point3d & pmax, double agrading) LocalH :: LocalH (Point<3> pmin, Point<3> pmax, double agrading, int adimension)
: dimension(adimension)
{ {
double x1[3], x2[3]; double x1[3], x2[3];
double hmax; double hmax;
boundingbox = Box3d (pmin, pmax); boundingbox = Box<3> (pmin, pmax);
grading = agrading; grading = agrading;
// a small enlargement, non-regular points // a small enlargement, non-regular points
double val = 0.0879; double val = 0.0879;
for (int i = 1; i <= 3; i++) for (int i = 0; i < dimension; i++)
{ {
x1[i-1] = (1 + val * i) * pmin.X(i) - val * i * pmax.X(i); x1[i] = (1 + val * (i+1)) * pmin(i) - val * (i+1) * pmax(i);
x2[i-1] = 1.1 * pmax.X(i) - 0.1 * pmin.X(i); x2[i] = 1.1 * pmax(i) - 0.1 * pmin(i);
} }
for (int i = dimension; i < 3; i++)
x1[i] = x2[i] = 0;
hmax = x2[0] - x1[0]; hmax = x2[0] - x1[0];
for (int i = 1; i <= 2; i++) for (int i = 1; i < dimension; i++)
if (x2[i] - x1[i] > hmax) hmax = max2(x2[i]-x1[i], hmax);
hmax = x2[i] - x1[i];
for (int i = 0; i <= 2; i++) for (int i = 0; i < dimension; i++)
x2[i] = x1[i] + hmax; x2[i] = x1[i] + hmax;
root = new GradingBox (x1, x2); root = new GradingBox (x1, x2);
boxes.Append (root); boxes.Append (root);
} }
LocalH :: LocalH (const Box<3> & box, double agrading)
{
Point3d pmin = box.PMin();
Point3d pmax = box.PMax();
double x1[3], x2[3];
double hmax;
boundingbox = Box3d (pmin, pmax);
grading = agrading;
// a small enlargement, non-regular points
double val = 0.0879;
for (int i = 1; i <= 3; i++)
{
x1[i-1] = (1 + val * i) * pmin.X(i) - val * i * pmax.X(i);
x2[i-1] = 1.1 * pmax.X(i) - 0.1 * pmin.X(i);
}
hmax = x2[0] - x1[0];
for (int i = 1; i <= 2; i++)
if (x2[i] - x1[i] > hmax)
hmax = x2[i] - x1[i];
for (int i = 0; i <= 2; i++)
x2[i] = x1[i] + hmax;
root = new GradingBox (x1, x2);
boxes.Append (root);
}
LocalH :: ~LocalH () LocalH :: ~LocalH ()
{ {
root->DeleteChilds(); root->DeleteChilds();
@ -126,33 +93,16 @@ namespace netgen
root->DeleteChilds(); root->DeleteChilds();
} }
void LocalH :: SetH (const Point3d & p, double h) void LocalH :: SetH (Point<3> p, double h)
{ {
/* if (dimension == 2)
(*testout) << "Set h at " << p << " to " << h << endl;
if (h < 1e-8)
{ {
cout << "do not set h to " << h << endl; if (fabs (p(0) - root->xmid[0]) > root->h2 ||
fabs (p(1) - root->xmid[1]) > root->h2)
return; return;
}
*/
if (fabs (p.X() - root->xmid[0]) > root->h2 ||
fabs (p.Y() - root->xmid[1]) > root->h2 ||
fabs (p.Z() - root->xmid[2]) > root->h2)
return;
/*
if (p.X() < root->x1[0] || p.X() > root->x2[0] ||
p.Y() < root->x1[1] || p.Y() > root->x2[1] ||
p.Z() < root->x1[2] || p.Z() > root->x2[2])
return;
*/
if (GetH(p) <= 1.2 * h) return; if (GetH(p) <= 1.2 * h) return;
GradingBox * box = root; GradingBox * box = root;
GradingBox * nbox = root; GradingBox * nbox = root;
GradingBox * ngb; GradingBox * ngb;
@ -163,9 +113,89 @@ namespace netgen
{ {
box = nbox; box = nbox;
childnr = 0; childnr = 0;
if (p.X() > box->xmid[0]) childnr += 1; if (p(0) > box->xmid[0]) childnr += 1;
if (p.Y() > box->xmid[1]) childnr += 2; if (p(1) > box->xmid[1]) childnr += 2;
if (p.Z() > box->xmid[2]) childnr += 4; nbox = box->childs[childnr];
};
while (2 * box->h2 > h)
{
childnr = 0;
if (p(0) > box->xmid[0]) childnr += 1;
if (p(1) > box->xmid[1]) childnr += 2;
double h2 = box->h2;
if (childnr & 1)
{
x1[0] = box->xmid[0];
x2[0] = x1[0]+h2; // box->x2[0];
}
else
{
x2[0] = box->xmid[0];
x1[0] = x2[0]-h2; // box->x1[0];
}
if (childnr & 2)
{
x1[1] = box->xmid[1];
x2[1] = x1[1]+h2; // box->x2[1];
}
else
{
x2[1] = box->xmid[1];
x1[1] = x2[1]-h2; // box->x1[1];
}
x1[2] = x2[2] = 0;
ngb = new GradingBox (x1, x2);
box->childs[childnr] = ngb;
ngb->father = box;
boxes.Append (ngb);
box = box->childs[childnr];
}
box->hopt = h;
double hbox = 2 * box->h2; // box->x2[0] - box->x1[0];
double hnp = h + grading * hbox;
Point<3> np;
for (int i = 0; i < 2; i++)
{
np = p;
np(i) = p(i) + hbox;
SetH (np, hnp);
np(i) = p(i) - hbox;
SetH (np, hnp);
}
}
else
{
if (fabs (p(0) - root->xmid[0]) > root->h2 ||
fabs (p(1) - root->xmid[1]) > root->h2 ||
fabs (p(2) - root->xmid[2]) > root->h2)
return;
if (GetH(p) <= 1.2 * h) return;
GradingBox * box = root;
GradingBox * nbox = root;
GradingBox * ngb;
int childnr;
double x1[3], x2[3];
while (nbox)
{
box = nbox;
childnr = 0;
if (p(0) > box->xmid[0]) childnr += 1;
if (p(1) > box->xmid[1]) childnr += 2;
if (p(2) > box->xmid[2]) childnr += 4;
nbox = box->childs[childnr]; nbox = box->childs[childnr];
}; };
@ -173,9 +203,9 @@ namespace netgen
while (2 * box->h2 > h) while (2 * box->h2 > h)
{ {
childnr = 0; childnr = 0;
if (p.X() > box->xmid[0]) childnr += 1; if (p(0) > box->xmid[0]) childnr += 1;
if (p.Y() > box->xmid[1]) childnr += 2; if (p(1) > box->xmid[1]) childnr += 2;
if (p.Z() > box->xmid[2]) childnr += 4; if (p(2) > box->xmid[2]) childnr += 4;
double h2 = box->h2; double h2 = box->h2;
if (childnr & 1) if (childnr & 1)
@ -221,34 +251,49 @@ namespace netgen
box->hopt = h; box->hopt = h;
double hbox = 2 * box->h2; // box->x2[0] - box->x1[0]; double hbox = 2 * box->h2; // box->x2[0] - box->x1[0];
double hnp = h + grading * hbox; double hnp = h + grading * hbox;
Point3d np; Point<3> np;
for (int i = 1; i <= 3; i++) for (int i = 0; i < 3; i++)
{ {
np = p; np = p;
np.X(i) = p.X(i) + hbox; np(i) = p(i) + hbox;
SetH (np, hnp); SetH (np, hnp);
np.X(i) = p.X(i) - hbox; np(i) = p(i) - hbox;
SetH (np, hnp); SetH (np, hnp);
} }
} }
}
double LocalH :: GetH (const Point3d & x) const double LocalH :: GetH (Point<3> x) const
{ {
const GradingBox * box = root; const GradingBox * box = root;
if (dimension == 2)
{
while (1) while (1)
{ {
int childnr = 0; int childnr = 0;
if (x.X() > box->xmid[0]) childnr += 1; if (x(0) > box->xmid[0]) childnr += 1;
if (x.Y() > box->xmid[1]) childnr += 2; if (x(1) > box->xmid[1]) childnr += 2;
if (x.Z() > box->xmid[2]) childnr += 4;
if (box->childs[childnr])
box = box->childs[childnr];
else
return box->hopt;
}
}
else
{
while (1)
{
int childnr = 0;
if (x(0) > box->xmid[0]) childnr += 1;
if (x(1) > box->xmid[1]) childnr += 2;
if (x(2) > box->xmid[2]) childnr += 4;
if (box->childs[childnr]) if (box->childs[childnr])
box = box->childs[childnr]; box = box->childs[childnr];
@ -257,16 +302,18 @@ namespace netgen
} }
} }
}
/// minimal h in box (pmin, pmax) /// minimal h in box (pmin, pmax)
double LocalH :: GetMinH (const Point3d & pmin, const Point3d & pmax) const double LocalH :: GetMinH (Point<3> pmin, Point<3> pmax) const
{ {
Point3d pmin2, pmax2; Point<3> pmin2, pmax2;
for (int j = 1; j <= 3; j++) for (int j = 0; j < 3; j++)
if (pmin.X(j) < pmax.X(j)) if (pmin(j) < pmax(j))
{ pmin2.X(j) = pmin.X(j); pmax2.X(j) = pmax.X(j); } { pmin2(j) = pmin(j); pmax2(j) = pmax(j); }
else else
{ pmin2.X(j) = pmax.X(j); pmax2.X(j) = pmin.X(j); } { pmin2(j) = pmax(j); pmax2(j) = pmin(j); }
return GetMinHRec (pmin2, pmax2, root); return GetMinHRec (pmin2, pmax2, root);
} }
@ -274,6 +321,23 @@ namespace netgen
double LocalH :: GetMinHRec (const Point3d & pmin, const Point3d & pmax, double LocalH :: GetMinHRec (const Point3d & pmin, const Point3d & pmax,
const GradingBox * box) const const GradingBox * box) const
{
if (dimension == 2)
{
double h2 = box->h2;
if (pmax.X() < box->xmid[0]-h2 || pmin.X() > box->xmid[0]+h2 ||
pmax.Y() < box->xmid[1]-h2 || pmin.Y() > box->xmid[1]+h2)
return 1e8;
double hmin = 2 * box->h2; // box->x2[0] - box->x1[0];
for (int i = 0; i < 8; i++)
if (box->childs[i])
hmin = min2 (hmin, GetMinHRec (pmin, pmax, box->childs[i]));
return hmin;
}
else
{ {
double h2 = box->h2; double h2 = box->h2;
if (pmax.X() < box->xmid[0]-h2 || pmin.X() > box->xmid[0]+h2 || if (pmax.X() < box->xmid[0]-h2 || pmin.X() > box->xmid[0]+h2 ||
@ -289,17 +353,33 @@ namespace netgen
return hmin; return hmin;
} }
}
void LocalH :: CutBoundaryRec (const Point3d & pmin, const Point3d & pmax, void LocalH :: CutBoundaryRec (const Point3d & pmin, const Point3d & pmax,
GradingBox * box) GradingBox * box)
{ {
double h2 = box->h2; double h2 = box->h2;
if (dimension == 2)
{
if (pmax.X() < box->xmid[0]-h2 || pmin.X() > box->xmid[0]+h2 ||
pmax.Y() < box->xmid[1]-h2 || pmin.Y() > box->xmid[1]+h2)
return;
}
else
{
if (pmax.X() < box->xmid[0]-h2 || pmin.X() > box->xmid[0]+h2 || if (pmax.X() < box->xmid[0]-h2 || pmin.X() > box->xmid[0]+h2 ||
pmax.Y() < box->xmid[1]-h2 || pmin.Y() > box->xmid[1]+h2 || pmax.Y() < box->xmid[1]-h2 || pmin.Y() > box->xmid[1]+h2 ||
pmax.Z() < box->xmid[2]-h2 || pmin.Z() > box->xmid[2]+h2) pmax.Z() < box->xmid[2]-h2 || pmin.Z() > box->xmid[2]+h2)
return; return;
}
box->flags.cutboundary = 1; box->flags.cutboundary = 1;
for (int i = 0; i < 8; i++) for (int i = 0; i < 8; i++)
@ -309,10 +389,13 @@ namespace netgen
void LocalH :: FindInnerBoxes (AdFront3 * adfront, void LocalH :: FindInnerBoxes (AdFront3 * adfront,
int (*testinner)(const Point3d & p1)) int (*testinner)(const Point3d & p1))
{ {
static int timer = NgProfiler::CreateTimer ("LocalH::FindInnerBoxes");
NgProfiler::RegionTimer reg (timer);
int nf = adfront->GetNF(); int nf = adfront->GetNF();
for (int i = 0; i < boxes.Size(); i++) for (int i = 0; i < boxes.Size(); i++)
@ -461,13 +544,11 @@ namespace netgen
void LocalH :: FindInnerBoxes (AdFront2 * adfront, void LocalH :: FindInnerBoxes (AdFront2 * adfront,
int (*testinner)(const Point<2> & p1)) int (*testinner)(const Point<2> & p1))
{ {
int nf = adfront->GetNFL(); static int timer = NgProfiler::CreateTimer ("LocalH::FindInnerBoxes 2d");
NgProfiler::RegionTimer reg (timer);
for (int i = 0; i < boxes.Size(); i++) for (int i = 0; i < boxes.Size(); i++)
boxes[i] -> flags.isinner = 0; boxes[i] -> flags.isinner = 0;
@ -486,6 +567,8 @@ namespace netgen
(*testout) << "inner = " << root->flags.pinner << " =?= " (*testout) << "inner = " << root->flags.pinner << " =?= "
<< testinner(rpmid) << endl; << testinner(rpmid) << endl;
int nf = adfront->GetNFL();
Array<int> faceinds(nf); Array<int> faceinds(nf);
Array<Box<3> > faceboxes(nf); Array<Box<3> > faceboxes(nf);
@ -497,7 +580,6 @@ namespace netgen
const FrontLine & line = adfront->GetLine(i); const FrontLine & line = adfront->GetLine(i);
faceboxes[i].Set (adfront->GetPoint (line.L().I1())); faceboxes[i].Set (adfront->GetPoint (line.L().I1()));
faceboxes[i].Add (adfront->GetPoint (line.L().I2())); faceboxes[i].Add (adfront->GetPoint (line.L().I2()));
} }
for (int i = 0; i < 8; i++) for (int i = 0; i < 8; i++)
@ -515,40 +597,37 @@ namespace netgen
GradingBox * father = box -> father; GradingBox * father = box -> father;
Point3d c(box->xmid[0], box->xmid[1], box->xmid[2]); Point3d c(box->xmid[0], box->xmid[1], 0); // box->xmid[2]);
Vec3d v(box->h2, box->h2, box->h2); Vec3d v(box->h2, box->h2, box->h2);
Box3d boxc(c-v, c+v); Box3d boxc(c-v, c+v);
Point3d fc(father->xmid[0], father->xmid[1], father->xmid[2]); Point3d fc(father->xmid[0], father->xmid[1], 0); // father->xmid[2]);
Vec3d fv(father->h2, father->h2, father->h2); Vec3d fv(father->h2, father->h2, father->h2);
Box3d fboxc(fc-fv, fc+fv); Box3d fboxc(fc-fv, fc+fv);
Box3d boxcfc(c,fc); Box3d boxcfc(c,fc);
ArrayMem<int, 100> faceused; ArrayMem<int, 100> faceused;
ArrayMem<int, 100> faceused2; ArrayMem<int, 100> faceused2;
ArrayMem<int, 100> facenotused; ArrayMem<int, 100> facenotused;
for (int j = 0; j < nfinbox; j++)
for (int j = 1; j <= nfinbox; j++)
{ {
// adfront->GetFaceBoundingBox (faceinds.Get(j), facebox); // adfront->GetFaceBoundingBox (faceinds.Get(j), facebox);
const Box3d & facebox = faceboxes.Get(faceinds.Get(j)); const Box3d & facebox = faceboxes[faceinds[j]];
if (boxc.Intersect (facebox)) if (boxc.Intersect (facebox))
faceused.Append(faceinds.Get(j)); faceused.Append(faceinds[j]);
else else
facenotused.Append(faceinds.Get(j)); facenotused.Append(faceinds[j]);
if (boxcfc.Intersect (facebox)) if (boxcfc.Intersect (facebox))
faceused2.Append (faceinds.Get(j)); faceused2.Append (faceinds[j]);
} }
for (int j = 1; j <= faceused.Size(); j++) for (int j = 0; j < faceused.Size(); j++)
faceinds.Elem(j) = faceused.Get(j); faceinds[j] = faceused[j];
for (int j = 1; j <= facenotused.Size(); j++) for (int j = 0; j < facenotused.Size(); j++)
faceinds.Elem(j+faceused.Size()) = facenotused.Get(j); faceinds[j+faceused.Size()] = facenotused[j];
if (!father->flags.cutboundary) if (!father->flags.cutboundary)
{ {
@ -560,12 +639,15 @@ namespace netgen
Point3d cf(father->xmid[0], father->xmid[1], father->xmid[2]); Point3d cf(father->xmid[0], father->xmid[1], father->xmid[2]);
if (father->flags.isinner) if (father->flags.isinner)
{
box->flags.pinner = 1; box->flags.pinner = 1;
}
else else
{ {
Point<2> c2d (c.X(), c.Y()); Point<2> c2d (c.X(), c.Y());
Point<2> cf2d (cf.X(), cf.Y()); Point<2> cf2d (cf.X(), cf.Y());
if (adfront->SameSide (c2d, cf2d, &faceused2)) bool sameside = adfront->SameSide (c2d, cf2d, &faceused2);
if (sameside)
box->flags.pinner = father->flags.pinner; box->flags.pinner = father->flags.pinner;
else else
box->flags.pinner = 1 - father->flags.pinner; box->flags.pinner = 1 - father->flags.pinner;
@ -654,12 +736,22 @@ namespace netgen
} }
void LocalH :: GetInnerPoints (Array<Point<3> > & points) void LocalH :: GetInnerPoints (Array<Point<3> > & points)
{
if (dimension == 2)
{
for (int i = 0; i < boxes.Size(); i++)
if (boxes[i] -> flags.isinner && boxes[i] -> HasChilds())
points.Append ( boxes[i] -> PMid() );
}
else
{ {
for (int i = 0; i < boxes.Size(); i++) for (int i = 0; i < boxes.Size(); i++)
if (boxes[i] -> flags.isinner) if (boxes[i] -> flags.isinner)
points.Append ( boxes[i] -> PMid() ); points.Append ( boxes[i] -> PMid() );
} }
}
void LocalH :: GetOuterPoints (Array<Point<3> > & points) void LocalH :: GetOuterPoints (Array<Point<3> > & points)
{ {

View File

@ -45,6 +45,13 @@ namespace netgen
Point<3> PMid() const { return Point<3> (xmid[0], xmid[1], xmid[2]); } Point<3> PMid() const { return Point<3> (xmid[0], xmid[1], xmid[2]); }
double H2() const { return h2; } double H2() const { return h2; }
bool HasChilds() const
{
for (int i = 0; i < 8; i++)
if (childs[i]) return true;
return false;
}
friend class LocalH; friend class LocalH;
static BlockAllocator ball; static BlockAllocator ball;
@ -67,12 +74,15 @@ namespace netgen
/// ///
Array<GradingBox*> boxes; Array<GradingBox*> boxes;
/// ///
Box3d boundingbox; Box<3> boundingbox;
/// octree or quadtree
int dimension;
public: public:
/// ///
LocalH (const Point3d & pmin, const Point3d & pmax, double grading); LocalH (Point<3> pmin, Point<3> pmax, double grading, int adimension = 3);
/// ///
LocalH (const Box<3> & box, double grading); LocalH (const Box<3> & box, double grading, int adimension = 3)
: LocalH (box.PMin(), box.PMax(), grading, adimension) { ; }
/// ///
~LocalH(); ~LocalH();
/// ///
@ -80,15 +90,16 @@ namespace netgen
/// ///
void SetGrading (double agrading) { grading = agrading; } void SetGrading (double agrading) { grading = agrading; }
/// ///
void SetH (const Point3d & x, double h); void SetH (Point<3> x, double h);
/// ///
double GetH (const Point3d & x) const; double GetH (Point<3> x) const;
/// minimal h in box (pmin, pmax) /// minimal h in box (pmin, pmax)
double GetMinH (const Point3d & pmin, const Point3d & pmax) const; double GetMinH (Point<3> pmin, Point<3> pmax) const;
/// mark boxes intersecting with boundary-box /// mark boxes intersecting with boundary-box
// void CutBoundary (const Point3d & pmin, const Point3d & pmax) // void CutBoundary (const Point3d & pmin, const Point3d & pmax)
// { CutBoundaryRec (pmin, pmax, root); } // { CutBoundaryRec (pmin, pmax, root); }
void CutBoundary (const Box<3> & box) void CutBoundary (const Box<3> & box)
{ CutBoundaryRec (box.PMin(), box.PMax(), root); } { CutBoundaryRec (box.PMin(), box.PMax(), root); }
@ -117,7 +128,7 @@ namespace netgen
void Convexify (); void Convexify ();
/// ///
int GetNBoxes () { return boxes.Size(); } int GetNBoxes () { return boxes.Size(); }
const Box3d & GetBoundingBox () const const Box<3> & GetBoundingBox () const
{ return boundingbox; } { return boundingbox; }
/// ///
void PrintMemInfo (ostream & ost) const; void PrintMemInfo (ostream & ost) const;

View File

@ -2416,19 +2416,19 @@ namespace netgen
void Mesh :: SetLocalH (const Point3d & pmin, const Point3d & pmax, double grading) void Mesh :: SetLocalH (netgen::Point<3> pmin, netgen::Point<3> pmax, double grading)
{ {
Point3d c = Center (pmin, pmax); using netgen::Point;
double d = max3 (pmax.X()-pmin.X(), Point<3> c = Center (pmin, pmax);
pmax.Y()-pmin.Y(), double d = max3 (pmax(0)-pmin(0),
pmax.Z()-pmin.Z()); pmax(1)-pmin(1),
pmax(2)-pmin(2));
d /= 2; d /= 2;
Point3d pmin2 = c - Vec3d (d, d, d); Point<3> pmin2 = c - Vec<3> (d, d, d);
Point3d pmax2 = c + Vec3d (d, d, d); Point<3> pmax2 = c + Vec<3> (d, d, d);
delete lochfunc; delete lochfunc;
lochfunc = new LocalH (pmin2, pmax2, grading); lochfunc = new LocalH (pmin2, pmax2, grading, dimension);
} }
void Mesh :: RestrictLocalH (const Point3d & p, double hloc) void Mesh :: RestrictLocalH (const Point3d & p, double hloc)

View File

@ -376,7 +376,7 @@ namespace netgen
/// Calculates localh /// Calculates localh
DLL_HEADER void CalcLocalH (double grading); DLL_HEADER void CalcLocalH (double grading);
/// ///
DLL_HEADER void SetLocalH (const Point3d & pmin, const Point3d & pmax, double grading); DLL_HEADER void SetLocalH (netgen::Point<3> pmin, netgen::Point<3> pmax, double grading);
/// ///
DLL_HEADER void RestrictLocalH (const Point3d & p, double hloc); DLL_HEADER void RestrictLocalH (const Point3d & p, double hloc);
/// ///
@ -682,10 +682,8 @@ namespace netgen
area += Cross ( mesh[sel[1]]-mesh[sel[0]], area += Cross ( mesh[sel[1]]-mesh[sel[0]],
mesh[sel[2]]-mesh[sel[0]] ).Length() / 2; mesh[sel[2]]-mesh[sel[0]] ).Length() / 2;
else else
area += Cross (Vec3d (mesh.Point (sel.PNum(1)), area += Cross (Vec3d (mesh[sel.PNum(1)], mesh[sel.PNum(3)]),
mesh.Point (sel.PNum(3))), Vec3d (mesh[sel.PNum(1)], mesh[sel.PNum(4)])).Length() / 2;;
Vec3d (mesh.Point (sel.PNum(1)),
mesh.Point (sel.PNum(4)))).Length() / 2;;
} }
void ReCalc () void ReCalc ()
{ {

View File

@ -172,9 +172,12 @@ namespace netgen
mesh3d.FindOpenElements(k); mesh3d.FindOpenElements(k);
/*
for (PointIndex pi = mesh3d.Points().Begin(); pi < mesh3d.Points().End(); pi++) for (PointIndex pi = mesh3d.Points().Begin(); pi < mesh3d.Points().End(); pi++)
meshing.AddPoint (mesh3d[pi], pi); meshing.AddPoint (mesh3d[pi], pi);
*/
for (PointIndex pi : mesh3d.Points().Range())
meshing.AddPoint (mesh3d[pi], pi);
for (int i = 1; i <= mesh3d.GetNOpenElements(); i++) for (int i = 1; i <= mesh3d.GetNOpenElements(); i++)
meshing.AddBoundaryElement (mesh3d.OpenElement(i)); meshing.AddBoundaryElement (mesh3d.OpenElement(i));

View File

@ -23,28 +23,28 @@ namespace netgen
case 's': case 's':
{ // topological swap { // topological swap
MeshOptimize2d meshopt; MeshOptimize2d meshopt;
meshopt.SetMetricWeight (0); meshopt.SetMetricWeight (mp.elsizeweight);
meshopt.EdgeSwapping (mesh, 0); meshopt.EdgeSwapping (mesh, 0);
break; break;
} }
case 'S': case 'S':
{ // metric swap { // metric swap
MeshOptimize2d meshopt; MeshOptimize2d meshopt;
meshopt.SetMetricWeight (0); meshopt.SetMetricWeight (mp.elsizeweight);
meshopt.EdgeSwapping (mesh, 1); meshopt.EdgeSwapping (mesh, 1);
break; break;
} }
case 'm': case 'm':
{ {
MeshOptimize2d meshopt; MeshOptimize2d meshopt;
meshopt.SetMetricWeight (1); meshopt.SetMetricWeight (mp.elsizeweight);
meshopt.ImproveMesh(mesh, mp); meshopt.ImproveMesh(mesh, mp);
break; break;
} }
case 'c': case 'c':
{ {
MeshOptimize2d meshopt; MeshOptimize2d meshopt;
meshopt.SetMetricWeight (0.2); meshopt.SetMetricWeight (mp.elsizeweight);
meshopt.CombineImprove(mesh); meshopt.CombineImprove(mesh);
break; break;
} }

View File

@ -120,12 +120,7 @@ namespace netgen
PointIndex () { ; } PointIndex () { ; }
PointIndex (int ai) : i(ai) { ; } PointIndex (int ai) : i(ai) { ; }
PointIndex & operator= (const PointIndex &ai) { i = ai.i; return *this; } PointIndex & operator= (const PointIndex &ai) { i = ai.i; return *this; }
// PointIndex & operator= (int ai) { i = ai; return *this; }
operator int () const { return i; } operator int () const { return i; }
// int GetInt () const { return i; }
// PointIndex operator+ (int i2) { return PointIndex (i+i2); }
// PointIndex operator++ (int) { int hi = i; i++; return PointIndex(hi); }
// PointIndex operator-- (int) { int hi = i; i--; return PointIndex(hi); }
PointIndex operator++ (int) { PointIndex hi(*this); i++; return hi; } PointIndex operator++ (int) { PointIndex hi(*this); i++; return hi; }
PointIndex operator-- (int) { PointIndex hi(*this); i--; return hi; } PointIndex operator-- (int) { PointIndex hi(*this); i--; return hi; }
PointIndex operator++ () { i++; return *this; } PointIndex operator++ () { i++; return *this; }

View File

@ -743,7 +743,6 @@ namespace netgen
Array<SurfaceElementIndex> seia; Array<SurfaceElementIndex> seia;
mesh.GetSurfaceElementsOfFace (faceindex, seia); mesh.GetSurfaceElementsOfFace (faceindex, seia);
bool mixed = 0; bool mixed = 0;
for (int i = 0; i < seia.Size(); i++) for (int i = 0; i < seia.Size(); i++)
if (mesh[seia[i]].GetNP() != 3) if (mesh[seia[i]].GetNP() != 3)