diff --git a/Fltk/GUI_Classifier.cpp b/Fltk/GUI_Classifier.cpp index 6ed4038825f5a7ae93ecf8def21064601e4bb9a3..1a445d0d59b5e31afd48965022279d073c19af7b 100644 --- a/Fltk/GUI_Classifier.cpp +++ b/Fltk/GUI_Classifier.cpp @@ -4,6 +4,7 @@ // bugs and problems to <gmsh@geuz.org>. #include "GUI_Classifier.h" +#include "Numeric.h" #include "Draw.h" #include "Options.h" #include "Context.h" diff --git a/Geo/GModel.cpp b/Geo/GModel.cpp index b969cbdcca34f30c78e381ad2b2626be11a7bb10..e87abab66952f006f200f3de3c08093d9d71c5c6 100644 --- a/Geo/GModel.cpp +++ b/Geo/GModel.cpp @@ -17,6 +17,7 @@ #else #include "gmshSurface.h" #include "Octree.h" +#include "SmoothData.h" #include "Field.h" #include "Generator.h" #include "Context.h" @@ -98,15 +99,15 @@ void GModel::destroy() delete *it; vertices.clear(); - if(normals) delete normals; - normals = 0; - destroyMeshCaches(); MVertex::resetGlobalNumber(); MElement::resetGlobalNumber(); #if !defined(HAVE_GMSH_EMBEDDED) + if(normals) delete normals; + normals = 0; + _fields->reset(); gmshSurface::reset(); #endif diff --git a/Mesh/BackgroundMesh.cpp b/Mesh/BackgroundMesh.cpp index 0d531988c06b123ddc93b5adeb57ae71fea08eda..7851be3552049055279739c487fec8ac0aac998d 100644 --- a/Mesh/BackgroundMesh.cpp +++ b/Mesh/BackgroundMesh.cpp @@ -153,7 +153,7 @@ double BGM_MeshSize(GEntity *ge, double U, double V, double X, double Y, double FieldManager *fields = GModel::current()->getFields(); if(fields->background_field > 0){ Field *f = fields->get(fields->background_field); - if(f) l4 = (*f)(X, Y, Z); + if(f) l4 = (*f)(X, Y, Z, ge); } // take the minimum, then contrain by lc_min and lc_max diff --git a/Mesh/Field.cpp b/Mesh/Field.cpp index d9e23e1926e8dfd9792d0321d83db128f75e6295..a965eb547dce224f78959e2408faa709496b7ac7 100644 --- a/Mesh/Field.cpp +++ b/Mesh/Field.cpp @@ -220,7 +220,8 @@ class StructuredField : public Field &update_needed); data = 0; } - std::string get_description(){ + std::string get_description() + { return "Linearly interpolate between data provided on a 3D rectangular structured grid. " "The format of the input file is : \n" "Ox Oy Oz \n" @@ -240,11 +241,11 @@ class StructuredField : public Field { return "Structured"; } - virtual ~ StructuredField() { - if(data) - delete[]data; + virtual ~StructuredField() + { + if(data) delete[]data; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { if(update_needed) { error_status = false; @@ -320,10 +321,11 @@ class UTMField : public Field double a, b, n, n2, n3, n4, n5, e, e2, e1, e12, e13, e14, J1, J2, J3, J4, Ap, Bp, Cp, Dp, Ep, e4, e6, ep, ep2, ep4, k0, mu_fact; public: - std::string get_description(){ - return "Evaluate Field[IField] in Universal Transverse Mercator coordinates. " - "The formulas for the coordinates transformation are taken from " - "http://www.uwgb.edu/dutchs/UsefulData/UTMFormulas.HTM\n"; + std::string get_description() + { + return "Evaluate Field[IField] in Universal Transverse Mercator coordinates. " + "The formulas for the coordinates transformation are taken from " + "http://www.uwgb.edu/dutchs/UsefulData/UTMFormulas.HTM\n"; } UTMField() { @@ -367,7 +369,7 @@ class UTMField : public Field { return "UTM"; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { double r = sqrt(x * x + y * y + z * z); double lon = atan2(y, x); @@ -403,9 +405,10 @@ class LonLatField : public Field { int field_id; public: - std::string get_description(){ + std::string get_description() + { return "Evaluate Field[IField] in geographic coordinates (longitude,latitude). \n" - "F = Field[IField](arctan(y/x),arcsin(z/sqrt(x^2+y^2+z^2))"; + "F = Field[IField](arctan(y/x),arcsin(z/sqrt(x^2+y^2+z^2))"; } LonLatField() { @@ -416,7 +419,7 @@ class LonLatField : public Field { return "LonLat"; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { Field *field = GModel::current()->getFields()->get(field_id); if(!field) return MAX_LC; @@ -428,9 +431,10 @@ class BoxField : public Field { double v_in, v_out, x_min, x_max, y_min, y_max, z_min, z_max; public: - std::string get_description(){ - return "The value of this field is VIn inside the box, VOut outside the box. \n" - "The box is given by Xmin<=x<=XMax && YMin<=y<=YMax && ZMin<=z<=ZMax"; + std::string get_description() + { + return "The value of this field is VIn inside the box, VOut outside the box. \n" + "The box is given by Xmin<=x<=XMax && YMin<=y<=YMax && ZMin<=z<=ZMax"; } BoxField() { @@ -448,7 +452,7 @@ class BoxField : public Field { return "Box"; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { return (x <= x_max && x >= x_min && y <= y_max && y >= y_min && z <= z_max && z >= z_min) ? v_in : v_out; @@ -465,10 +469,11 @@ class ThresholdField : public Field { return "Threshold"; } - std::string get_description(){ - return "F = LCMin if Field[IField] <= DistMin\n" - "F = LCMax if Field[IField] >= DistMax\n" - "F = Interpolation between LcMin and LcMax if DistMin<Field[IField]<DistMax"; + std::string get_description() + { + return "F = LCMin if Field[IField] <= DistMin\n" + "F = LCMax if Field[IField] >= DistMax\n" + "F = Interpolation between LcMin and LcMax if DistMin<Field[IField]<DistMax"; } ThresholdField() { @@ -490,9 +495,11 @@ class ThresholdField : public Field "and LcMax using a sigmoid, false to " "interpolate linearly"); options["StopAtDistMax"] = new FieldOptionBool(stopAtDistMax, "True to not impose " - "element size outside DistMax (i.e. F = a very big value if Field[IField]>DistMax)"); + "element size outside DistMax (i.e. " + "F = a very big value if " + "Field[IField]>DistMax)"); } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { Field *field = GModel::current()->getFields()->get(iField); if(!field) return MAX_LC; @@ -522,7 +529,8 @@ class GradientField : public Field { return "Gradient"; } - std::string get_description(){ + std::string get_description() + { return "Compute the finite difference gradient of Field[IField].\n " "F = (Field[IField](X + Delta/2) - Field[IField](X - Delta/2))/Delta"; } @@ -532,10 +540,11 @@ class GradientField : public Field kind = 0; delta = 0.; options["IField"] = new FieldOptionInt(iField, "Field index"); - options["Kind"] = new FieldOptionInt(kind, "Component of the gradient to evaluate : 0 for X, 1 for Y, 2 for Z, 3 for the norm"); + options["Kind"] = new FieldOptionInt(kind, "Component of the gradient to evaluate :" + " 0 for X, 1 for Y, 2 for Z, 3 for the norm"); options["Delta"] = new FieldOptionDouble(delta, "Finite difference step"); } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { Field *field = GModel::current()->getFields()->get(iField); if(!field) return MAX_LC; @@ -578,7 +587,8 @@ class CurvatureField : public Field { return "Curvature"; } - std::string get_description(){ + std::string get_description() + { return "Compute the curvature of Field[IField]. \n" "F = divergence( || grad( Field[IField] ) || )"; } @@ -599,7 +609,7 @@ class CurvatureField : public Field g[1] /= n; g[2] /= n; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { Field *field = GModel::current()->getFields()->get(iField); if(!field) return MAX_LC; @@ -630,11 +640,12 @@ class MaxEigenHessianField : public Field { return "MaxEigenHessian"; } - std::string get_description(){ - return "Compute the maximum eigen value of the Hessian matrix of Field[IField]. " - "Gradients are evaluated by finite differences, " - "eigenvalues are computed using the GSL library."; - "F = max ( eigenvalues ( grad ( grad ( Field[IField] ) ) ) ) "; + std::string get_description() + { + return "Compute the maximum eigen value of the Hessian matrix of Field[IField]. " + "Gradients are evaluated by finite differences, " + "eigenvalues are computed using the GSL library." + "F = max ( eigenvalues ( grad ( grad ( Field[IField] ) ) ) ) "; } MaxEigenHessianField() : iField(0), delta(CTX.lc / 1e4) { @@ -652,7 +663,7 @@ class MaxEigenHessianField : public Field gsl_vector_free(eigenvalues); gsl_matrix_free(gslmat); } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { Field *field = GModel::current()->getFields()->get(iField); if(!field) return MAX_LC; @@ -696,12 +707,13 @@ class LaplacianField : public Field { return "Laplacian"; } - std::string get_description(){ - return "Compute finite difference the Laplacian of Field[IField].\n" - "F = divergence(gradient(Field[IField])) \n" - "F = G(x+d,y,z)+G(x-d,y,z)+G(x,y+d,z)+G(x,y-d,z)+ " - "+G(x,y,z+d)+G(x,y,z-d)-6*G(x,y,z) " - "where G=Field[IField] and d=delta\n"; + std::string get_description() + { + return "Compute finite difference the Laplacian of Field[IField].\n" + "F = divergence(gradient(Field[IField])) \n" + "F = G(x+d,y,z)+G(x-d,y,z)+G(x,y+d,z)+G(x,y-d,z)+ " + "+G(x,y,z+d)+G(x,y,z-d)-6*G(x,y,z) " + "where G=Field[IField] and d=delta\n"; } LaplacianField() : iField(0), delta(CTX.lc / 1e4) { @@ -710,7 +722,7 @@ class LaplacianField : public Field options["IField"] = new FieldOptionInt(iField, "Field index"); options["Delta"] = new FieldOptionDouble(delta, "Finite difference step"); } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { Field *field = GModel::current()->getFields()->get(iField); if(!field) return MAX_LC; @@ -730,20 +742,21 @@ class MeanField : public Field { return "Mean"; } - std::string get_description(){ - return "Very simple smoother.\n" - "F = (G(x+delta,y,z)+G(x-delta,y,z) " - "+G(x,y+delta,z)+G(x,y-delta,z) " - "+G(x,y,z+delta)+G(x,y,z-delta) " - "+G(x,y,z))/7 " - "where G=Field[IField]"; + std::string get_description() + { + return "Very simple smoother.\n" + "F = (G(x+delta,y,z)+G(x-delta,y,z) " + "+G(x,y+delta,z)+G(x,y-delta,z) " + "+G(x,y,z+delta)+G(x,y,z-delta) " + "+G(x,y,z))/7 " + "where G=Field[IField]"; } MeanField() : iField(0), delta(CTX.lc / 1e4) { options["IField"] = new FieldOptionInt(iField, "Field index"); options["Delta"] = new FieldOptionDouble(delta, "Distance used to compute the mean value"); } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { Field *field = GModel::current()->getFields()->get(iField); if(!field) return MAX_LC; @@ -853,10 +866,11 @@ class MathEvalField : public Field public: MathEvalField() { - options["F"] = new FieldOptionString(f, "Mathematical function to evaluate.", &update_needed); - f="F2 + Sin(z)"; + options["F"] = new FieldOptionString(f, "Mathematical function to evaluate.", + &update_needed); + f = "F2 + Sin(z)"; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { if(update_needed) { if(!expr.set_function(f)) @@ -870,10 +884,11 @@ class MathEvalField : public Field { return "MathEval"; } - std::string get_description(){ + std::string get_description() + { return "Evaluate a mathematical expression. " - "The expression can contains x, y, z for spatial coordinates, F0, F1, ... for field values, " - "and mathematical functions. " + "The expression can contains x, y, z for spatial coordinates, " + "F0, F1, ... for field values, and mathematical functions. " "This evaluator is based on a modified version of the GNU libmatheval library. \n" "Example : F2 + Sin(z)"; } @@ -896,12 +911,13 @@ class ParametricField : public Field options["FZ"] = new FieldOptionString(f[2], "Z component of parametric function", &update_needed); } - std::string get_description(){ + std::string get_description() + { return "Evaluate Field IField in parametric coordinate. " "See MathEval Field help to get a description of valid FX, FY and FZ expressions.\n" "F = Field[IField](FX,FY,FZ) "; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { if(update_needed) { for(int i = 0; i < 3; i++) { @@ -930,7 +946,7 @@ class PostViewField : public Field OctreePost *octree; public: int view_index; - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { // FIXME: should test unique view num instead, but that would be slower if(view_index < 0 || view_index >= (int)PView::list.size()) @@ -972,8 +988,9 @@ class PostViewField : public Field { return "PostView"; } - std::string get_description(){ - return "Evaluate the post processing view IView."; + std::string get_description() + { + return "Evaluate the post processing view IView."; } PostViewField() { @@ -984,8 +1001,7 @@ class PostViewField : public Field } ~PostViewField() { - if(octree) - delete octree; + if(octree) delete octree; } }; #endif @@ -999,15 +1015,16 @@ class MinField : public Field options["FieldsList"] = new FieldOptionList(idlist, "Field indices", &update_needed); } - std::string get_description(){ + std::string get_description() + { return "Take the minimum value of a list of fields. "; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { double v = MAX_LC; for(std::list<int>::iterator it = idlist.begin(); it != idlist.end(); it++) { Field *f = (GModel::current()->getFields()->get(*it)); - if(f) v = std::min(v, (*f) (x, y, z)); + if(f) v = std::min(v, (*f) (x, y, z, ge)); } return v; } @@ -1026,15 +1043,16 @@ class MaxField : public Field options["FieldsList"] = new FieldOptionList(idlist, "Field indices", &update_needed); } - std::string get_description(){ + std::string get_description() + { return "Take the maximum value of a list of fields."; } - double operator() (double x, double y, double z) + double operator() (double x, double y, double z, GEntity *ge=0) { double v = -MAX_LC; for(std::list<int>::iterator it = idlist.begin(); it != idlist.end(); it++) { Field *f = (GModel::current()->getFields()->get(*it)); - if(f) v = std::max(v, (*f) (x, y, z)); + if(f) v = std::max(v, (*f) (x, y, z, ge)); } return v; } @@ -1044,6 +1062,42 @@ class MaxField : public Field } }; +class RestrictField : public Field +{ + int ifield; + std::list<int> edges, faces, regions; + public: + RestrictField() + { + ifield = 1; + options["IField"] = new FieldOptionInt(ifield, "Field index"); + options["EdgesList"] = new FieldOptionList(edges, "Curve indices"); + options["FacesList"] = new FieldOptionList(faces, "Surface indices"); + options["RegionsList"] = new FieldOptionList(regions, "Volume indices"); + } + std::string get_description() + { + return "Restrict the application of a field to a given list of geometrical " + "curves, surfaces or volumes. "; + } + double operator() (double x, double y, double z, GEntity *ge=0) + { + Field *f = (GModel::current()->getFields()->get(ifield)); + if(!f) return MAX_LC; + if(!ge) return (*f) (x, y, z); + if((ge->dim() == 0) || + (ge->dim() == 1 && std::find(edges.begin(), edges.end(), ge->tag()) != edges.end()) || + (ge->dim() == 2 && std::find(faces.begin(), faces.end(), ge->tag()) != faces.end()) || + (ge->dim() == 3 && std::find(regions.begin(), regions.end(), ge->tag()) != regions.end())) + return (*f) (x, y, z); + return MAX_LC; + } + const char *get_name() + { + return "Restrict"; + } +}; + #ifdef HAVE_ANN class AttractorField : public Field { @@ -1072,10 +1126,8 @@ class AttractorField : public Field } ~AttractorField() { - if(kdtree) - delete kdtree; - if(zeronodes) - annDeallocPts(zeronodes); + if(kdtree) delete kdtree; + if(zeronodes) annDeallocPts(zeronodes); delete[]index; delete[]dist; } @@ -1083,13 +1135,16 @@ class AttractorField : public Field { return "Attractor"; } - std::string get_description(){ - return "Compute the distance from the nearest node in a list. " - "It can also be used to compute distance from curves, in this case each curve is replaced " - "by NNodesByEdge equidistant nodes and the distance from those nodes is computed. \n" - "The ANN library is used to find the nearest node : http://www.cs.umd.edu/~mount/ANN/ "; - } - virtual double operator() (double X, double Y, double Z) + std::string get_description() + { + return "Compute the distance from the nearest node in a list. " + "It can also be used to compute distance from curves, in this case each " + "curve is replaced by NNodesByEdge equidistant nodes and the distance " + "from those nodes is computed. \n" + "The ANN library is used to find the nearest node: " + "http://www.cs.umd.edu/~mount/ANN/ "; + } + virtual double operator() (double X, double Y, double Z, GEntity *ge=0) { if(update_needed) { if(zeronodes) { @@ -1156,9 +1211,7 @@ class AttractorField : public Field template<class F> class FieldFactoryT : public FieldFactory { public: - Field * operator()() { - return new F; - }; + Field * operator()() { return new F; } }; template<class F> Field *field_factory() @@ -1176,6 +1229,7 @@ FieldManager::FieldManager() map_type_name["PostView"] = new FieldFactoryT<PostViewField>(); #endif map_type_name["Gradient"] = new FieldFactoryT<GradientField>(); + map_type_name["Restrict"] = new FieldFactoryT<RestrictField>(); map_type_name["Min"] = new FieldFactoryT<MinField>(); map_type_name["Max"] = new FieldFactoryT<MaxField>(); map_type_name["UTM"] = new FieldFactoryT<UTMField>(); diff --git a/Mesh/Field.h b/Mesh/Field.h index af6966a25db484c4166b313082e7ba0922a53730..4660c17473e82e880a77d016e04c9b01abd98e3a 100644 --- a/Mesh/Field.h +++ b/Mesh/Field.h @@ -6,16 +6,16 @@ #ifndef _FIELD_H_ #define _FIELD_H_ +#include <string> #include <map> #include <list> -#include <string.h> -#include "Geo.h" #if !defined(HAVE_NO_POST) #include "PView.h" #endif class Field; +class GEntity; typedef enum { FIELD_OPTION_DOUBLE = 0, @@ -37,9 +37,7 @@ class FieldOption { virtual ~FieldOption() {} virtual FieldOptionType get_type() = 0; virtual void get_text_representation(std::string & v_str) = 0; - virtual std::string get_description(){ - return _help; - } + virtual std::string get_description(){ return _help; } std::string get_type_name(){ switch(get_type()){ case FIELD_OPTION_INT: return "integer"; break; @@ -62,7 +60,7 @@ class Field { public: int id; std::map<std::string, FieldOption *> options; - virtual double operator() (double x, double y, double z) = 0; + virtual double operator() (double x, double y, double z, GEntity *ge=0) = 0; virtual ~Field() {} bool update_needed; Field(); @@ -70,9 +68,7 @@ class Field { #if !defined(HAVE_NO_POST) void put_on_view(PView * view, int comp = -1); #endif - virtual std::string get_description(){ - return ""; - } + virtual std::string get_description(){ return ""; } }; class FieldFactory { diff --git a/README b/README index 203680b521679c501cbf43b1cfe8d3721fc0ff34..e02ae898d6abf94ce2f1c9f7ef6870ba3cdcb0d5 100644 --- a/README +++ b/README @@ -25,7 +25,7 @@ For a description of all other configuration options, type ./configure --help -To build Gmsh with Microsoft Visual Studio, follow the intructions in +To build Gmsh with Microsoft Visual C++, follow the intructions in doc/README.msvc. Gmsh is distributed under the terms of the GNU General Public License, diff --git a/benchmarks/2d/field_restrict.geo b/benchmarks/2d/field_restrict.geo new file mode 100644 index 0000000000000000000000000000000000000000..6c7beaff8fe29e7f60864957c3cf843d6ed68d14 --- /dev/null +++ b/benchmarks/2d/field_restrict.geo @@ -0,0 +1,54 @@ +/* -------------------------------------------------------------------------- + This is a gmsh geometry file + --------------------------------------------------------------------------*/ + +res1 = 1e-1; +x = 1.0; +y = 0.1; + +a0 = newp; Point(a0) = {0,0,0,res1}; +a1 = newp; Point(a1) = {-x,0,0,res1}; +a2 = newp; Point(a2) = {x,0,0,res1}; + +la1 = newl; Circle(la1) = {a1,a0,a2}; +la2 = newl; Circle(la2) = {a2,a0,a1}; + +loop = newll; Line Loop(loop) = {la1,la2}; + +sa = news; Plane Surface(sa) = {loop}; + +Point(4) = {-2, -2, 0}; +Point(5) = {2, -2, 0}; +Point(6) = {2, 2, 0}; +Point(7) = {-2, 2, 0}; +Line(5) = {4, 5}; +Line(6) = {5, 6}; +Line(7) = {6, 7}; +Line(8) = {7, 4}; +Line Loop(9) = {7, 8, 5, 6}; +Plane Surface(10) = {9, 3}; + +// specifying explicit element sizes as a function of the distance to +// the point a0 (square function here) +Field[1] = Attractor; +Field[1].EdgesList = {la1,la2,6}; +Field[1].NNodesByEdge = 100; +Field[2] = MathEval; +Field[2].F = Sprintf("F1^2 + %g", res1/4); +Field[3] = Restrict; +Field[3].IField = 2; +Field[3].FacesList = {10}; +Field[3].EdgesList = {la1,la2, 5:8}; + +Field[4] = MathEval; +Field[4].F = "0.2"; +Field[5] = Restrict; +Field[5].IField = 4; +Field[5].FacesList = {sa}; + +Field[6] = Min; +Field[6].FieldsList = {3,5}; + +Background Field = 6; + +Mesh.CharacteristicLengthExtendFromBoundary = 0; diff --git a/doc/README.msvc b/doc/README.msvc index 643365eeef76e896a155e0fcdd1a6e2d1f81e899..2b72804c5a7bf2280be041b7fcc2a50cfe86add5 100644 --- a/doc/README.msvc +++ b/doc/README.msvc @@ -2,13 +2,13 @@ To compile Gmsh with Microsoft Visual C++ 1) launch the Visual Studio command prompt -2) go to the gmsh source directory +2) go to the Gmsh source directory 4) rename "utils\misc\variables.msvc" into "variables" (in the root - of the gmsh source directory) + of the Gmsh source directory) -5) edit "variables" to match your local installation of gmsh and - Visual C++ (and to configure which optional gmsh features to build) +5) edit "variables" to match your local installation of Gmsh and + Visual C++ (and to configure which optional Gmsh features to build) 6) type "utils\misc\gmake.exe" diff --git a/doc/TODO.txt b/doc/TODO.txt index 862e076b2c6f79ccd31e4fff030563d8c0142938..f178815b829b6ba8844a5e1f201e8cbfa699adc8 100644 --- a/doc/TODO.txt +++ b/doc/TODO.txt @@ -1,13 +1,9 @@ -$Id: TODO.txt,v 1.5 2008-10-01 07:06:48 geuzaine Exp $ +$Id: TODO.txt,v 1.6 2008-10-01 19:18:14 geuzaine Exp $ ******************************************************************** -Add option to Fields to apply them only on certain elementary -entities. (From Lars) - -Add Attractor on surfaces: specfify a mesh size on the boundary of a -volume, and the elements should get smaller/larger (following e.g. a -quadratic formula) when you go away from the boundary +Add Attractor field on surfaces (use the surface mesh points as point +attractors?) ********************************************************************