From 8856c3aabbc0b025510d37243787ebda9feaaf36 Mon Sep 17 00:00:00 2001 From: Christophe Geuzaine <cgeuzaine@ulg.ac.be> Date: Sat, 12 Feb 2011 10:15:38 +0000 Subject: [PATCH] fix extrude with scalar view --- Common/SmoothData.h | 3 + Mesh/BoundaryLayers.cpp | 65 ++++++++++++------- benchmarks/2d/world.geo | 3 + .../sphere_boundary_layer_from_view.geo | 10 +-- 4 files changed, 52 insertions(+), 29 deletions(-) diff --git a/Common/SmoothData.h b/Common/SmoothData.h index 80a28e13d5..741d5d43f6 100644 --- a/Common/SmoothData.h +++ b/Common/SmoothData.h @@ -48,6 +48,9 @@ class smooth_data{ private: std::set<xyzv, lessthanxyzv> c; public: + typedef std::set<xyzv, lessthanxyzv>::iterator iter; + iter begin(){ return c.begin(); } + iter end(){ return c.end(); } smooth_data() {} void add(double x, double y, double z, int n, double *vals); bool get(double x, double y, double z, int n, double *vals); diff --git a/Mesh/BoundaryLayers.cpp b/Mesh/BoundaryLayers.cpp index 8d1fe40c94..992008360d 100644 --- a/Mesh/BoundaryLayers.cpp +++ b/Mesh/BoundaryLayers.cpp @@ -56,13 +56,6 @@ static void addExtrudeNormals(std::vector<T*> &elements, int invert, double nn[3] = {n[0], n[1], n[2]}; for(int k = 0; k < ele->getNumVertices(); k++){ MVertex *v = ele->getVertex(k); - if(octree){ -#if defined(HAVE_POST) - double d; - octree->searchScalarWithTol(v->x(), v->y(), v->z(), &d, 0); - for(int i = 0; i < 3; i++) nn[i] *= d; -#endif - } ExtrudeParams::normals[index]->add(v->x(), v->y(), v->z(), 3, nn); } } @@ -75,6 +68,9 @@ template<class T> static void addExtrudeNormals(std::set<T*> &entities, std::map<int, infoset> &infos) { + bool normalize = true; + std::vector<OctreePost*> octrees; + for(typename std::set<T*>::iterator it = entities.begin(); it != entities.end(); it++){ T *ge = *it; infoset info = infos[ge->tag()]; @@ -90,6 +86,7 @@ static void addExtrudeNormals(std::set<T*> &entities, octree = new OctreePost(PView::list[view]); if(PView::list[view]->getData()->getNumVectors()) gouraud = false; + octrees.push_back(octree); } else Msg::Error("Unknown View[%d]: using normals instead", view); @@ -101,8 +98,44 @@ static void addExtrudeNormals(std::set<T*> &entities, addExtrudeNormals(((GFace*)ge)->triangles, invert, octree, gouraud, index); addExtrudeNormals(((GFace*)ge)->quadrangles, invert, octree, gouraud, index); } + if(!gouraud) normalize = false; } } + + // enforce coherent normals at some points if necessary + for(int i = 0; i < ExtrudeParams::normalsCoherence.size(); i++){ + SPoint3 &p(ExtrudeParams::normalsCoherence[i]); + double n0[3], n1[3]; + ExtrudeParams::normals[0]->get(p.x(), p.y(), p.z(), 3, n0); + ExtrudeParams::normals[1]->get(p.x(), p.y(), p.z(), 3, n1); + ExtrudeParams::normals[0]->add(p.x(), p.y(), p.z(), 3, n1); + ExtrudeParams::normals[1]->add(p.x(), p.y(), p.z(), 3, n0); + } + + // normalize extrusion directions if not using explicit vector + // post-processing views + if(normalize){ + for(int i = 0; i < 2; i++){ + ExtrudeParams::normals[i]->normalize(); +#if defined(HAVE_POST) + if(octrees.size()){ // scale normals by scalar views + for(smooth_data::iter it = ExtrudeParams::normals[i]->begin(); + it != ExtrudeParams::normals[i]->end(); it++){ + for(unsigned int j = 0; j < octrees.size(); j++){ + double d; + if(octrees[j]->searchScalarWithTol(it->x, it->y, it->z, &d, 0)){ + for(int k = 0; k < 3; k++) it->vals[k] *= d; + break; + } + } + } + } +#endif + } + } + + for(unsigned int i = 0; i < octrees.size(); i++) + delete octrees[i]; } int Mesh2DWithBoundaryLayers(GModel *m) @@ -110,7 +143,6 @@ int Mesh2DWithBoundaryLayers(GModel *m) std::set<GFace*> sourceFaces, otherFaces; std::set<GEdge*> sourceEdges, otherEdges; std::map<int, infoset> sourceFaceInfo, sourceEdgeInfo; - bool normalize = true; // 2D boundary layers for(GModel::eiter it = m->firstEdge(); it != m->lastEdge(); it++){ @@ -124,7 +156,6 @@ int Mesh2DWithBoundaryLayers(GModel *m) Msg::Error("Unknown source curve %d for boundary layer", ep->geo.Source); return 0; } - if(ep->mesh.ViewIndex >= 0) normalize = false; std::pair<bool, std::pair<int, int> > tags (ep->geo.Source < 0, std::pair<int, int> (ep->mesh.BoundaryLayerIndex, ep->mesh.ViewIndex)); @@ -146,7 +177,6 @@ int Mesh2DWithBoundaryLayers(GModel *m) Msg::Error("Unknown source face %d for boundary layer", ep->geo.Source); return 0; } - if(ep->mesh.ViewIndex >= 0) normalize = false; std::pair<bool, std::pair<int, int> > tags (ep->geo.Source < 0, std::pair<int, int> (ep->mesh.BoundaryLayerIndex, ep->mesh.ViewIndex)); @@ -186,21 +216,6 @@ int Mesh2DWithBoundaryLayers(GModel *m) else addExtrudeNormals(sourceFaces, sourceFaceInfo); - // enforce coherent normals at some points if necessary - for(int i = 0; i < ExtrudeParams::normalsCoherence.size(); i++){ - SPoint3 &p(ExtrudeParams::normalsCoherence[i]); - double n0[3], n1[3]; - ExtrudeParams::normals[0]->get(p.x(), p.y(), p.z(), 3, n0); - ExtrudeParams::normals[1]->get(p.x(), p.y(), p.z(), 3, n1); - ExtrudeParams::normals[0]->add(p.x(), p.y(), p.z(), 3, n1); - ExtrudeParams::normals[1]->add(p.x(), p.y(), p.z(), 3, n0); - } - - // normalize extrusion directions if not using post-processing views - if(normalize) - for(int i = 0; i < 2; i++) - ExtrudeParams::normals[i]->normalize(); - // set the position of boundary layer points using the smooth normal // field for(GModel::eiter it = m->firstEdge(); it != m->lastEdge(); it++){ diff --git a/benchmarks/2d/world.geo b/benchmarks/2d/world.geo index 4bcc6d42da..a7d51c1dcd 100644 --- a/benchmarks/2d/world.geo +++ b/benchmarks/2d/world.geo @@ -1,7 +1,10 @@ lc = 0.1; Point(1) = {0,0,0,lc}; Point(2) = {1,0,0,lc}; + +// Fabien - enleve ceci: PolarSphere (1) = {1,2}; + Point(3) = {0.335046,-1.96526,0,lc}; Point(4) = {0.319083,-1.86011,0,lc}; Point(5) = {0.298206,-1.76125,0,lc}; diff --git a/benchmarks/extrude/sphere_boundary_layer_from_view.geo b/benchmarks/extrude/sphere_boundary_layer_from_view.geo index 323a83609b..ac010f628f 100644 --- a/benchmarks/extrude/sphere_boundary_layer_from_view.geo +++ b/benchmarks/extrude/sphere_boundary_layer_from_view.geo @@ -1,4 +1,7 @@ +// merge vector view (View[0]) Merge "sphere_boundary_layer_from_view.pos"; + +// create scalar view View[1] for testing Plugin(MathEval).Run; lc = 0.2; @@ -41,10 +44,9 @@ Line Loop(27) = {-4,12,-6}; Ruled Surface(28) = {27}; tmp[] = Extrude { - Surface{14:28:2}; Layers{5, 0.2}; Recombine; Using Index[0]; - // can use scalar or vectpr view - //Using View[0]; - Using View[1]; + Surface{14:28:2}; Layers{5, 0.2}; Recombine; + //Using View[0]; // provide extrude directions using vector field + Using View[1]; // use mesh normals, but scaled by scalar field }; // test 2nd bnd layer -- GitLab