diff --git a/Mesh/meshGFace.cpp b/Mesh/meshGFace.cpp index 5409a01ca6b7fd566b83bb3182b9e5dd17ef00fa..6d41195dfe48ba37109ff8a2f45bd482a5840bae 100644 --- a/Mesh/meshGFace.cpp +++ b/Mesh/meshGFace.cpp @@ -1425,7 +1425,11 @@ static bool buildConsecutiveListOfVertices(GFace *gf, GEdgeLoop &gel, return true; } -static void printParamGrid(GFace *gf, std::vector<MVertex*> vert1, std::vector<MVertex*> vert2) +static void printParamGrid(GFace *gf, std::vector<MVertex*> vert1, std::vector<MVertex*> vert2, + std::vector<MVertex*> e01, std::vector<MVertex*> e10, + std::vector<MVertex*> e23, std::vector<MVertex*> e32, + std::vector<MVertex*> e02, std::vector<MVertex*> e13, + std::vector<MQuadrangle*> quads) { std::vector<SPoint2> p1,p2; @@ -1440,6 +1444,7 @@ static void printParamGrid(GFace *gf, std::vector<MVertex*> vert1, std::vector<M p2.push_back(pj); } + char name[234]; sprintf(name,"paramGrid_%d.pos", gf->tag()); FILE *f = fopen(name,"w"); @@ -1454,6 +1459,55 @@ static void printParamGrid(GFace *gf, std::vector<MVertex*> vert1, std::vector<M fprintf(f,"};\n"); fclose(f); + char name2[234]; + sprintf(name2,"paramEdges_%d.pos", gf->tag()); + FILE *f2 = fopen(name2,"w"); + fprintf(f2,"View \"%s\" {\n",name2); + + for (unsigned int i = 0; i < e01.size(); i++){ + SPoint2 pi; reparamMeshVertexOnFace(e01[i], gf, pi); + fprintf(f2,"SP(%g,%g,%g) {%d};\n", pi.x(), pi.y(), 0.0, 1); + } + + for (unsigned int i = 0; i < e10.size(); i++){ + SPoint2 pi; reparamMeshVertexOnFace(e10[i], gf, pi); + fprintf(f2,"SP(%g,%g,%g) {%d};\n", pi.x(), pi.y(), 0.0, 10); + } + + for (unsigned int i = 0; i < e23.size(); i++){ + SPoint2 pi; reparamMeshVertexOnFace(e23[i], gf, pi); + fprintf(f2,"SP(%g,%g,%g) {%d};\n", pi.x(), pi.y(), 0.0, 23); + } + + for (unsigned int i = 0; i < e32.size(); i++){ + SPoint2 pi; reparamMeshVertexOnFace(e32[i], gf, pi); + fprintf(f2,"SP(%g,%g,%g) {%d};\n", pi.x(), pi.y(), 0.0, 32); + } + + for (unsigned int i = 0; i < e02.size(); i++){ + SPoint2 pi; reparamMeshVertexOnFace(e02[i], gf, pi); + fprintf(f2,"SP(%g,%g,%g) {%d};\n", pi.x(), pi.y(), 0.0, 2); + } + + for (unsigned int i = 0; i < e13.size(); i++){ + SPoint2 pi; reparamMeshVertexOnFace(e13[i], gf, pi); + fprintf(f2,"SP(%g,%g,%g) {%d};\n", pi.x(), pi.y(), 0.0, 13); + } + fprintf(f2,"};\n"); + fclose(f2); + + + char name3[234]; + sprintf(name3,"quad_%d.pos", gf->tag()); + FILE *f3 = fopen(name3,"w"); + fprintf(f3,"View \"%s\" {\n",name3); + for (unsigned int i = 0; i < quads.size(); i++){ + quads[i]->writePOS(f3,true,false,false,false,false,false); + } + fprintf(f3,"};\n"); + fclose(f3); + + return; } @@ -1461,48 +1515,79 @@ static void createRegularTwoCircleGrid (Centerline *center, GFace *gf) { int nbBoundaries = gf->edges().size(); - printf("nb boundaries =%d \n", nbBoundaries ); - + std::list<GEdge*> bedges = gf->edges(); std::list<GEdge*>::iterator itb = bedges.begin(); std::list<GEdge*>::iterator ite = bedges.end(); ite--; GEdge *ge1 = (*itb)->getCompound(); GEdge *ge2 = (*ite)->getCompound(); - int N1 = ge1->mesh_vertices.size() + 1; + int N = ge1->mesh_vertices.size() + 1; int N2 = ge2->mesh_vertices.size() + 1; - printf("nb mesh vertices N = %d \n",N1 ); - if (N1 != N2 || N1%2 != 0){ + if (N != N2 || N%2 != 0){ Msg::Error("You should have a pair number of points specified in centerline field \n"); } - std::vector<MVertex*> vert1, vert2; + //vert1 is the outer circle + //vert2 is the inner circle + // - - - - + // - - + // - - - + // v0- v2- -v3 -v1 + // - - - + // - - + // - - - - + + std::vector<MVertex*> vert1, vert2, vert_temp; vert1.push_back(ge1->getBeginVertex()->mesh_vertices[0]); vert2.push_back(ge2->getBeginVertex()->mesh_vertices[0]); - for(unsigned int i = 0; i < N1-1; i++) { + for(unsigned int i = 0; i < N-1; i++) { vert1.push_back(ge1->mesh_vertices[i]); vert2.push_back(ge2->mesh_vertices[i]); } + SPoint2 pv1; reparamMeshVertexOnFace(vert1[0], gf, pv1); + SPoint2 pv2; reparamMeshVertexOnFace(vert2[0], gf, pv2); + SPoint2 pv1b; reparamMeshVertexOnFace(vert1[1], gf, pv1b); + SPoint2 pv2b; reparamMeshVertexOnFace(vert2[1], gf, pv2b); + SPoint2 pv1c; reparamMeshVertexOnFace(vert1[2], gf, pv1c); + SPoint2 pv2c; reparamMeshVertexOnFace(vert2[2], gf, pv2c); + SVector3 vec1(pv1b.x()-pv1.x(),pv1b.y()-pv1.y() , 0.0); + SVector3 vec1b(pv1c.x()-pv1b.x(),pv1c.y()-pv1b.y() ,0.0); + SVector3 vec2(pv2b.x()-pv2.x(),pv2b.y()-pv2.y() ,0.0); + SVector3 vec2b(pv2c.x()-pv2b.x(),pv2c.y()-pv2b.y() , 0.0); + int sign2 = (dot(crossprod(vec1,vec1b),crossprod(vec2,vec2b)) < 0) ? -1 : +1 ; + double n1 = sqrt(pv1.x()*pv1.x()+pv1.y()*pv1.y()); + double n2 = sqrt(pv2.x()*pv2.x()+pv2.y()*pv2.y()); + if (n2 > n1) { + vert_temp = vert1; + vert1.clear(); + vert1 = vert2; + vert2.clear(); + vert2 = vert_temp; + } #if defined(HAVE_ANN) - ANNpointArray nodes = annAllocPts(N1, 3); + ANNpointArray nodes = annAllocPts(N, 3); ANNidxArray index = new ANNidx[1]; ANNdistArray dist = new ANNdist[1]; - for (int ind = 0; ind < N1; ind++){ - nodes[ind][0] = vert2[ind]->x(); - nodes[ind][1] = vert2[ind]->y(); - nodes[ind][2] = vert2[ind]->z(); - } - ANNkd_tree *kdtree = new ANNkd_tree(nodes, N1, 3); - double xyz[3] = {vert1[0]->x(),vert1[0]->y(),vert1[0]->z()}; + for (int ind = 0; ind < N; ind++){ + SPoint2 pp2; reparamMeshVertexOnFace(vert2[ind], gf, pp2); + nodes[ind][0] = pp2.x(); + nodes[ind][1] = pp2.y(); + nodes[ind][2] = 0.0; + } + ANNkd_tree *kdtree = new ANNkd_tree(nodes, N, 3); + SPoint2 pp1; reparamMeshVertexOnFace(vert1[0], gf, pp1); + double xyz[3] = {pp1.x(), pp1.y(),0.0}; kdtree->annkSearch(xyz, 1, index, dist); int close_ind = index[0]; - double length = sqrt(dist[0]); + double length = sqrt((vert1[0]->x()-vert2[close_ind]->x())*(vert1[0]->x()-vert2[close_ind]->x())+ + (vert1[0]->y()-vert2[close_ind]->y())*(vert1[0]->y()-vert2[close_ind]->y())+ + (vert1[0]->z()-vert2[close_ind]->z())*(vert1[0]->z()-vert2[close_ind]->z())); double rad = center->operator()(vert1[0]->x(), vert1[0]->y(), vert1[0]->z()); - double lc = 2*M_PI*rad/N1; + double lc = 2*M_PI*rad/N; int M = length/lc; - - printf("dist =%g ind =%d \n", length, close_ind); + printf("length =%g arc=%g \n", length, 2*M_PI*rad); delete kdtree; delete[]index; @@ -1510,62 +1595,53 @@ static void createRegularTwoCircleGrid (Centerline *center, GFace *gf) annDeallocPts(nodes); #endif; - printParamGrid(gf, vert1, vert2); - - // std::vector<std::vector<MVertex*> > tab(M+2); - // for(int i = 0; i <= M+1; i++) tab[i].resize(N+2); - - // tab[0][0] = v1; - // tab[0][N+1] = v2; - // tab[M+1][N+1] = v3; - // tab[M+1][0] = v4; - // for (int i=0;i<N;i++){ - // tab[0][i+1] = sign12 > 0 ? e12 [i] : e12 [N-i-1]; - // tab[M+1][i+1] = sign34 < 0 ? e34 [i] : e34 [N-i-1]; - // } - // for (int i=0;i<M;i++){ - // tab[i+1][N+1] = sign23 > 0 ? e23 [i] : e23 [M-i-1]; - // tab[i+1][0] = sign41 < 0 ? e41 [i] : e41 [M-i-1]; - // } - - // for (int i=0;i<N;i++){ - // for (int j=0;j<M;j++){ - // double u = (double) (i+1)/ ((double)(N+1)); - // double v = (double) (j+1)/ ((double)(M+1)); - // MVertex *v12 = (sign12 >0) ? e12[i] : e12 [N-1-i]; - // MVertex *v23 = (sign23 >0) ? e23[j] : e23 [M-1-j]; - // MVertex *v34 = (sign34 <0) ? e34[i] : e34 [N-1-i]; - // MVertex *v41 = (sign41 <0) ? e41[j] : e41 [M-1-j]; - // SPoint2 p12; reparamMeshVertexOnFace(v12, gf, p12); - // SPoint2 p23; reparamMeshVertexOnFace(v23, gf, p23); - // SPoint2 p34; reparamMeshVertexOnFace(v34, gf, p34); - // SPoint2 p41; reparamMeshVertexOnFace(v41, gf, p41); - // double Up = TRAN_QUA(p12.x(), p23.x(), p34.x(), p41.x(), - // c1.x(),c2.x(),c3.x(),c4.x(),u,v); - // double Vp = TRAN_QUA(p12.y(), p23.y(), p34.y(), p41.y(), - // c1.y(),c2.y(),c3.y(),c4.y(),u,v); - // // printf("v1=%d v2=%d v3=%d v4=%d \n", v1->getNum(), v2->getNum(), v3->getNum(), v4->getNum()); - // // printf("c1=%g %g, c2=%g %g, c3=%g %g, c4=%g,%g \n", c1.x(),c1.y(),c2.x(),c2.y(),c3.x(),c3.y(),c4.x(),c4.y()); - // // printf("p1=%g %g, p2=%g %g, p3=%g %g, p4=%g,%g \n", p12.x(),p12.x(),p23.x(),p23.y(),p34.x(),p34.y(),p41.x(),p41.y()); - // if ((p12.x() && p12.y() == -1.0) || - // (p23.x() && p23.y() == -1.0) || - // (p34.x() && p34.y() == -1.0) || - // (p41.x() && p41.y() == -1.0)) { - // Msg::Error("Wrong param -1"); - // return; - // } - - // MVertex *vnew = createNewVertex (gf, SPoint2(Up,Vp)); - // tab[j+1][i+1] = vnew; - // } - // } - // // create quads - // for (int i=0;i<=N;i++){ - // for (int j=0;j<=M;j++){ - // MQuadrangle *qnew = new MQuadrangle (tab[j][i],tab[j][i+1],tab[j+1][i+1],tab[j+1][i]); - // q.push_back(qnew); - // } - // } + MVertex *v0 = vert1[0]; SPoint2 p0; reparamMeshVertexOnFace(v0, gf, p0); + MVertex *v1 = vert1[N/2]; SPoint2 p1; reparamMeshVertexOnFace(v1, gf, p1); + MVertex *v2 = vert2[close_ind]; SPoint2 p2; reparamMeshVertexOnFace(v2, gf, p2); + MVertex *v3 = vert2[(close_ind+N/2)%N]; SPoint2 p3; reparamMeshVertexOnFace(v3, gf, p3); + + printf("grid N = %d M = %d\n", N , M); + printf("v2 =%d v3=%d \n", close_ind, (close_ind+N/2)%N); + std::vector<MVertex*> e01,e10,e23,e32; + for (int i=0;i<N/2;i++) e01.push_back(vert1[i]); + for (int i=N/2;i<N;i++) e10.push_back(vert1[i]); + for (int i=0;i<N/2;i++) e23.push_back(vert2[(close_ind+i)%N]); + for (int i=N/2;i<N;i++) e32.push_back(vert2[(close_ind+i)%N]); + + std::vector<MVertex*> e02 = saturateEdge (gf,p0,p2,M); + std::vector<MVertex*> e13 = saturateEdge (gf,p1,p3,M); + + std::vector<MQuadrangle*> q; + std::vector<MVertex*> e_inner1 = e23; + std::vector<MVertex*> e_inner2 = e32; + if (sign2 <0){ + e_inner1 = e32; + e_inner2 = e23; + } + + createRegularMesh (gf, + v0, p0, + e01, +1, + v1,p1, + e13, +1, + v3,p3, + e_inner1, -sign2, + v2,p2, + e02, -1, + q); + + createRegularMesh (gf, + v0,p0, + e02, +1, + v2,p2, + e_inner2, -sign2, + v3,p3, + e13, -1, + v1,p1, + e10, +1, + q); + + printParamGrid(gf, vert1, vert2, e01,e10,e23,e32,e02,e13, q); } diff --git a/Mesh/meshGFaceOptimize.cpp b/Mesh/meshGFaceOptimize.cpp index 8b8f53ed97562fc456948577fbebd782905a31bc..64b09c5ae2552af63846215b8b50813bf0850013 100644 --- a/Mesh/meshGFaceOptimize.cpp +++ b/Mesh/meshGFaceOptimize.cpp @@ -765,12 +765,12 @@ static std::vector<MVertex*> closestPathBetweenTwoDefects (v2t_cont &adj, static MVertex * createNewVertex (GFace *gf, SPoint2 p){ GPoint gp = gf->point(p); if (!gp.succeeded()) { - //printf("**** ARRG new vertex not created \n"); + printf("******* ARRG new vertex not created p=%g %g \n", p.x(), p.y()); return NULL; } return new MFaceVertex(gp.x(),gp.y(),gp.z(),gf,gp.u(),gp.v()); } -static std::vector<MVertex*> saturateEdge (GFace *gf, SPoint2 p1, SPoint2 p2, int n){ +std::vector<MVertex*> saturateEdge (GFace *gf, SPoint2 p1, SPoint2 p2, int n){ std::vector<MVertex*> pts; for (int i=1;i<n;i++){ SPoint2 p = p1 + (p2-p1)*((double)i/((double)(n))); diff --git a/Mesh/meshGFaceOptimize.h b/Mesh/meshGFaceOptimize.h index 2a44576ea573721792f29c6c7237cf0aa96fd513..0918ce4335f2158179af9a0b6fbc8639548ec8cf 100644 --- a/Mesh/meshGFaceOptimize.h +++ b/Mesh/meshGFaceOptimize.h @@ -57,6 +57,19 @@ void buildListOfEdgeAngle(e2t_cont adj, std::vector<edge_angle> &edges_detected, void buildEdgeToElements(std::vector<MElement*> &tris, e2t_cont &adj); void laplaceSmoothing(GFace *gf, int niter=1); + +void createRegularMesh (GFace *gf, + MVertex *v1, SPoint2 &c1, + std::vector<MVertex*> &e12, int sign12, + MVertex *v2, SPoint2 &c2, + std::vector<MVertex*> &e23, int sign23, + MVertex *v3, SPoint2 &c3, + std::vector<MVertex*> &e34, int sign34, + MVertex *v4, SPoint2 &c4, + std::vector<MVertex*> &e41, int sign41, + std::vector<MQuadrangle*> &q); +std::vector<MVertex*> saturateEdge (GFace *gf, SPoint2 p1, SPoint2 p2, int n); + /* void edgeSwappingLawson(GFace *gf); */