Skip to content
Snippets Groups Projects
Forked from gmsh / gmsh
20321 commits behind the upstream repository.
1D_Mesh.cpp 6.59 KiB
// $Id: 1D_Mesh.cpp,v 1.24 2001-12-03 08:41:44 geuzaine Exp $

#include "Gmsh.h"
#include "Numeric.h"
#include "Geo.h"
#include "Mesh.h"
#include "Utils.h"
#include "Context.h"
#include "Interpolation.h"

extern Mesh      *THEM;
extern Context_T  CTX;

Curve *THEC;

// ipar[0] = nbpoints
// abs(ipar[1]) = method
// sign(ipar[1]) = orientation
// dpar[0] = parameter

double F_One (double t){
  Vertex der;
  double d;
  der = InterpolateCurve (THEC, t, 1);
  d = sqrt (der.Pos.X * der.Pos.X + der.Pos.Y * der.Pos.Y + der.Pos.Z * der.Pos.Z);
  return (d);
}


double F_Transfini (double t){
  Vertex der;
  double d, a, b, val;
  int i;

  der = InterpolateCurve (THEC, t, 1);
  d = sqrt (der.Pos.X * der.Pos.X + der.Pos.Y * der.Pos.Y +
            der.Pos.Z * der.Pos.Z);

  if (THEC->dpar[0] == 0.0 || THEC->dpar[0] == 1.0){
    val = d * (double) THEC->ipar[0] / THEC->l ;
  }
  else{
    switch (abs (THEC->ipar[1])){

    case 1: // Geometric progression ar^i; Sum of n terms = THEC->l = a (r^n-1)/(r-1)
      if(THEC->dpar[0] == 1.)
	a = THEC->l/(double)THEC->ipar[0];
      else
	a = THEC->l * (THEC->dpar[0]-1.)/(pow(THEC->dpar[0],THEC->ipar[0])-1.) ;
      i = (int)( log(t*THEC->l/a*(THEC->dpar[0]-1.)+1.) / log(THEC->dpar[0]) );
      val = d/(a*pow(THEC->dpar[0],(double)i));
      break;

    case 2: // Bump
      if (THEC->dpar[0] > 1.0){
        a = -4. * sqrt (THEC->dpar[0] - 1.) * 
          atan2 (1., sqrt (THEC->dpar[0] - 1.)) / 
          ((double) THEC->ipar[0] * THEC->l);
      }
      else{
        a = 2. * sqrt (1. - THEC->dpar[0]) * 
          log (fabs ((1. + 1. / sqrt (1. - THEC->dpar[0])) 
                     / (1. - 1. / sqrt (1. - THEC->dpar[0]))))
          / ((double) THEC->ipar[0] * THEC->l);
      }
      b = -a * THEC->l * THEC->l / (4. * (THEC->dpar[0] - 1.)) ;
      val = d / (-a * DSQR (t * THEC->l - (THEC->l) * 0.5) + b) ;
      break ;

    default:
      Msg(WARNING, "Unknown case in Transfinite Line mesh");
      val = 1. ;
    }
  }

  return val ;
}

double F_Lc (double t){
  Vertex  der, point;
  double  Lc, d;

  if (CTX.mesh.algo == DELAUNAY_ISO && THEM->BGM.Typ == ONFILE){
    der = InterpolateCurve(THEC, t, 1);
    point = InterpolateCurve(THEC, t, 0);  
    Lc = Lc_XYZ(point.Pos.X, point.Pos.Y, point.Pos.Z, THEM);
    d = sqrt(DSQR(der.Pos.X)+DSQR(der.Pos.Y)+DSQR(der.Pos.Z));
    if(!Lc){
      Msg(GERROR, "Null characteristic length in background mesh");
      return d;
    }
    if(CTX.mesh.constrained_bgmesh)
      return MAX(d/Lc,THEM->Metric->getLc(t, THEC));
    else
      return d/Lc;
  }
  else
    return THEM->Metric->getLc(t, THEC);
}

void Maillage_Curve (void *data, void *dummy){
  Curve **pc, *c;
  Simplex *s;
  double b, a, d, dt, dp, t;
  int i, N, count, NUMP;
  Vertex **v, **vexist, *pV, V, *v1, *v2;
  List_T *Points;
  IntPoint P1, P2;

  pc = (Curve **) data;
  c = *pc;
  THEC = c;

  if (c->Num < 0)
    return;

  if(c->Dirty){
    Msg(INFO, "Not meshing dirty Curve %d", c->Num);
    return;
  }

  Msg(STATUS3, "Meshing Curve %d", c->Num);

  Points = List_Create (10, 10, sizeof (IntPoint));
  c->l = Integration (c->ubeg, c->uend, F_One, Points, 1.e-4);
  List_Delete (Points);

  if(!c->l){
    Msg(GERROR, "Zero length Curve %d", c->Num);
    return;
  }

  if (c->Method == TRANSFINI || !Extrude_Mesh (c)){
    if (c->Method == TRANSFINI){
      Points = List_Create (10, 10, sizeof (IntPoint));
      a = Integration (c->ubeg, c->uend, F_Transfini, Points, 1.e-7);
      N = c->ipar[0];
    }
    else{
      Points = List_Create (10, 10, sizeof (IntPoint));
      a = Integration (c->ubeg, c->uend, F_Lc, Points, 1.e-4);
      N = IMAX (2, (int) (a + 1.));

      if (c->Typ == MSH_SEGM_CIRC ||
          c->Typ == MSH_SEGM_CIRC_INV ||
          c->Typ == MSH_SEGM_ELLI ||
          c->Typ == MSH_SEGM_ELLI_INV){
        N = IMAX (N, (int) (fabs (c->Circle.t1 - c->Circle.t2) *
                            (double)CTX.mesh.min_circ_points / Pi));
      }
      else if (c->Typ == MSH_SEGM_NURBS){
        N = IMAX (N, 2);
      }
    }
    b = a / (double) (N - 1);
    c->Vertices = List_Create (N, 2, sizeof (Vertex *));
    
    v = &c->beg;
    if ((vexist = (Vertex **) Tree_PQuery (THEM->Vertices, v))){
      (*vexist)->u = c->ubeg;
      Tree_Insert (THEM->Vertices, vexist);
      if ((*vexist)->ListCurves)
        List_Add ((*vexist)->ListCurves, &c);
      List_Add (c->Vertices, vexist);
    }
    else{
      pV = Create_Vertex ((*v)->Num, (*v)->Pos.X, (*v)->Pos.Y,
                          (*v)->Pos.Z, (*v)->lc, c->ubeg);
      pV->ListCurves = List_Create (1, 1, sizeof (Curve *));
      List_Add (pV->ListCurves, &c);
      Tree_Insert (THEM->Vertices, &pV);
      List_Add (c->Vertices, &pV);
    }

    count = NUMP = 1;
    while (NUMP < N - 1){
      List_Read (Points, count - 1, &P1);
      List_Read (Points, count, &P2);
      d = (double) NUMP *b;

      if ((fabs (P2.p) >= fabs (d)) && (fabs (P1.p) < fabs (d))){
        dt = P2.t - P1.t;
        dp = P2.p - P1.p;
        t = P1.t + dt / dp * (d - P1.p);
        V = InterpolateCurve (c, t, 0);
        pV = Create_Vertex (++THEM->MaxPointNum, V.Pos.X, V.Pos.Y, V.Pos.Z, V.lc, t);
        pV->w = V.w;
        pV->ListCurves = List_Create (1, 1, sizeof (Curve *));
        List_Add (pV->ListCurves, &c);
        Tree_Insert (THEM->Vertices, &pV);
        List_Add (c->Vertices, &pV);
        NUMP++;
      }
      else{
        count++;
      }
    }

    List_Delete(Points);

    v = &c->end;
    if ((vexist = (Vertex **) Tree_PQuery (THEM->Vertices, v))){
      (*vexist)->u = c->uend;
      Tree_Insert (THEM->Vertices, vexist);
      if ((*vexist)->ListCurves)
        List_Add ((*vexist)->ListCurves, &c);
      List_Add (c->Vertices, vexist);
    }
    else{
      pV = Create_Vertex ((*v)->Num, (*v)->Pos.X, (*v)->Pos.Y, 
                          (*v)->Pos.Z, (*v)->lc, c->uend);
      pV->ListCurves = List_Create (1, 1, sizeof (Curve *));
      List_Add (pV->ListCurves, &c);
      Tree_Insert (THEM->Vertices, &pV);
      List_Add (c->Vertices, &pV);
    }
  }

  for (i = 0; i < List_Nbr (c->Vertices) - 1; i++){
    List_Read (c->Vertices, i, &v1);
    List_Read (c->Vertices, i + 1, &v2);
    s = Create_Simplex (v1, v2, NULL, NULL);
    s->iEnt = c->Num;
    Tree_Add (c->Simplexes, &s);
    List_Add (c->TrsfSimplexes, &s);
  }

  if (CTX.mesh.degree == 2)
    Degre2 (THEM->Vertices, THEM->VertexEdges, c->Simplexes, c, NULL);

  THEM->Statistics[4] += List_Nbr (c->Vertices);

#if 0
  if(fabs(c->Num) != 41) return;
  printf("curve %d : ", c->Num);
  for (i = 0; i < List_Nbr (c->Vertices); i++){
    List_Read (c->Vertices, i, &v1);
    printf(" %d (%g %g %g)", v1->Num, v1->Pos.X, v1->Pos.Y, v1->Pos.Z);
  }
  printf("\n");
#endif
}