From 492b997f32006312edd9ad8854363e21c3b9da46 Mon Sep 17 00:00:00 2001
From: Christophe Geuzaine <cgeuzaine@ulg.ac.be>
Date: Sun, 6 Aug 2006 22:58:49 +0000
Subject: [PATCH] *** empty log message ***

---
 Common/Makefile          |  12 +-
 Common/Options.cpp       |  12 +-
 Fltk/Callbacks.cpp       |  45 +------
 Fltk/GUI.cpp             |  19 ++-
 Fltk/Main.cpp            |  10 +-
 Fltk/Message.cpp         |   4 +-
 Geo/GEdge.h              |  16 +--
 Geo/GEntity.h            |  55 ++++----
 Geo/GFace.cpp            |   4 +-
 Geo/GModel.cpp           |  57 ++++++++
 Geo/GModel.h             |  14 +-
 Geo/GModelIO.cpp         | 279 ++++++++++++++++++++++++++++++++-------
 Geo/GVertex.cpp          |  10 +-
 Geo/GVertex.h            |  15 ++-
 Geo/MElement.cpp         |  97 +++++++++++++-
 Geo/MElement.h           | 167 ++++++++++++++++++++++-
 Geo/MVertex.cpp          |  35 +++++
 Geo/MVertex.h            |  20 +--
 Geo/Makefile             |   6 +-
 Geo/gmshVertex.h         |  27 +++-
 Graphics/CreateFile.cpp  |  18 ++-
 Graphics/Geom.cpp        |   4 +-
 Mesh/DiscreteSurface.cpp |   4 +-
 23 files changed, 730 insertions(+), 200 deletions(-)

diff --git a/Common/Makefile b/Common/Makefile
index 5af303006a..8c3794da9b 100644
--- a/Common/Makefile
+++ b/Common/Makefile
@@ -1,4 +1,4 @@
-# $Id: Makefile,v 1.96 2006-08-04 14:28:01 geuzaine Exp $
+# $Id: Makefile,v 1.97 2006-08-06 22:58:47 geuzaine Exp $
 #
 # Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 #
@@ -133,7 +133,15 @@ CommandLine.o: CommandLine.cpp Gmsh.h Message.h ../DataStr/Malloc.h \
   ../Common/SmoothNormals.h ../Common/GmshDefines.h ../Mesh/Metric.h \
   ../Mesh/Vertex.h ../Mesh/Simplex.h ../Mesh/Mesh.h ../Mesh/Matrix.h \
   Views.h ColorTable.h VertexArray.h SmoothNormals.h GmshMatrix.h \
-  AdaptiveViews.h ../Parser/OpenFile.h ../Parser/Parser.h OS.h
+  AdaptiveViews.h ../Parser/OpenFile.h ../Graphics/CreateFile.h \
+  ../Parser/Parser.h ../Geo/GModel.h ../Geo/GVertex.h ../Geo/GEntity.h \
+  ../Geo/Range.h ../Geo/SPoint3.h ../Geo/SBoundingBox3d.h \
+  ../Geo/SPoint3.h ../Geo/MVertex.h ../Geo/MVertex.h ../Geo/GPoint.h \
+  ../Geo/GEdge.h ../Geo/GEntity.h ../Geo/GVertex.h ../Geo/SVector3.h \
+  ../Geo/SPoint3.h ../Geo/SPoint3.h ../Geo/SPoint2.h ../Geo/MElement.h \
+  ../Geo/MVertex.h ../Geo/GFace.h ../Geo/GPoint.h ../Geo/GEntity.h \
+  ../Geo/MElement.h ../Geo/SPoint2.h ../Geo/SVector3.h ../Geo/Pair.h \
+  ../Geo/GRegion.h ../Geo/GEntity.h ../Geo/MElement.h OS.h
 # 1 "/Users/geuzaine/.gmsh/Common//"
 OS.o: OS.cpp Message.h
 # 1 "/Users/geuzaine/.gmsh/Common//"
diff --git a/Common/Options.cpp b/Common/Options.cpp
index 3dbf968f2f..7d2cde1fd8 100644
--- a/Common/Options.cpp
+++ b/Common/Options.cpp
@@ -1,4 +1,4 @@
-// $Id: Options.cpp,v 1.282 2006-08-05 13:31:28 geuzaine Exp $
+// $Id: Options.cpp,v 1.283 2006-08-06 22:58:47 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -2777,11 +2777,11 @@ double opt_general_orthographic(OPT_ARGS_NUM)
   if(WID && (action & GMSH_GUI)) {
     if(CTX.ortho){
       WID->gen_choice[2]->value(0);
-      Msg(STATUS2N, "Orthographic projection");
+      if(!CTX.batch) Msg(STATUS2N, "Orthographic projection");
     }
     else{
       WID->gen_choice[2]->value(1);
-      Msg(STATUS2N, "Perspective projection");
+      if(!CTX.batch) Msg(STATUS2N, "Perspective projection");
     }
   }
 #endif
@@ -2795,15 +2795,15 @@ double opt_general_mouse_selection(OPT_ARGS_NUM)
 #if defined(HAVE_FLTK)
   if(WID && (action & GMSH_GUI)) {
     if(CTX.enable_mouse_selection == 0){
-      Msg(STATUS2N, "Mouse selection OFF");
+      if(!CTX.batch) Msg(STATUS2N, "Mouse selection OFF");
       WID->g_status_butt[9]->color(FL_RED);
     }
     else if(CTX.enable_mouse_selection == 1){
-      Msg(STATUS2N, "Mouse hover OFF");
+      if(!CTX.batch) Msg(STATUS2N, "Mouse hover OFF");
       WID->g_status_butt[9]->color(FL_GREEN);
     }
     else{
-      Msg(STATUS2N, "Mouse selection ON");
+      if(!CTX.batch) Msg(STATUS2N, "Mouse selection ON");
       WID->g_status_butt[9]->color(FL_BACKGROUND_COLOR);
     }
     WID->g_status_butt[9]->redraw();
diff --git a/Fltk/Callbacks.cpp b/Fltk/Callbacks.cpp
index 615e755d7c..23f6e550ad 100644
--- a/Fltk/Callbacks.cpp
+++ b/Fltk/Callbacks.cpp
@@ -1,4 +1,4 @@
-// $Id: Callbacks.cpp,v 1.421 2006-08-05 10:05:44 geuzaine Exp $
+// $Id: Callbacks.cpp,v 1.422 2006-08-06 22:58:47 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -955,7 +955,6 @@ void general_options_rotation_center_select_cb(CALLBACK_ARGS)
   opt_geometry_points(0, GMSH_SET | GMSH_GUI, 1);
   Draw();
 
-  Msg(STATUS2N, "Setting rotation center");
   Msg(ONSCREEN, "Select point\n[Press 'q' to abort]");
   char ib = SelectEntity(ENT_POINT, &ne, v, c, s);
   if(ib == 'l') {
@@ -972,7 +971,6 @@ void general_options_rotation_center_select_cb(CALLBACK_ARGS)
   }
   ZeroHighlight();
   Draw();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
@@ -1891,7 +1889,6 @@ void geometry_elementary_add_new_point_cb(CALLBACK_ARGS)
   WID->create_geometry_context_window(1);
 
   while(1) {
-    Msg(STATUS2N, "Creating point");
     WID->g_opengl_window->AddPointMode = true;
     Msg(ONSCREEN, "Move mouse and/or enter coordinates\n"
 	"[Press 'Shift' to hold position, 'e' to add point or 'q' to abort]");
@@ -1915,7 +1912,6 @@ void geometry_elementary_add_new_point_cb(CALLBACK_ARGS)
   }
 
   WID->reset_visibility();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
@@ -1932,7 +1928,6 @@ static void _new_multiline(int type)
 
   n = 0;
   while(1) {
-    Msg(STATUS2N, "Creating curve");
     if(n == 0)
       Msg(ONSCREEN, "Select control points\n"
 	  "[Press 'e' to end selection or 'q' to abort]");
@@ -1983,7 +1978,6 @@ static void _new_multiline(int type)
   }
 
   WID->reset_visibility();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
@@ -2005,7 +1999,6 @@ void geometry_elementary_add_new_line_cb(CALLBACK_ARGS)
 
   n = 0;
   while(1) {
-    Msg(STATUS2N, "Creating straight line");
     if(n == 0)
       Msg(ONSCREEN, "Select start point\n"
 	  "[Press 'q' to abort]");
@@ -2040,7 +2033,6 @@ void geometry_elementary_add_new_line_cb(CALLBACK_ARGS)
   }
 
   WID->reset_visibility();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
@@ -2067,7 +2059,6 @@ void geometry_elementary_add_new_circle_cb(CALLBACK_ARGS)
 
   n = 0;
   while(1) {
-    Msg(STATUS2N, "Creating circle");
     if(n == 0)
       Msg(ONSCREEN, "Select start point\n"
 	  "[Press 'q' to abort]");
@@ -2105,7 +2096,6 @@ void geometry_elementary_add_new_circle_cb(CALLBACK_ARGS)
   }
 
   WID->reset_visibility();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
@@ -2122,7 +2112,6 @@ void geometry_elementary_add_new_ellipse_cb(CALLBACK_ARGS)
 
   n = 0;
   while(1) {
-    Msg(STATUS2N, "Creating ellipse");
     if(n == 0)
       Msg(ONSCREEN, "Select start point\n"
 	  "[Press 'q' to abort]");
@@ -2163,7 +2152,6 @@ void geometry_elementary_add_new_ellipse_cb(CALLBACK_ARGS)
   }
 
   WID->reset_visibility();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
@@ -2195,7 +2183,6 @@ static void _new_surface_volume(int mode)
 
     while(1) {
       if(type == ENT_LINE){
-	Msg(STATUS2N, "Creating surface");
 	if(!List_Nbr(List1))
 	  Msg(ONSCREEN, "Select surface boundary\n"
 	      "[Press 'q' to abort]");
@@ -2204,7 +2191,6 @@ static void _new_surface_volume(int mode)
 	      "[Press 'u' to undo last selection or 'q' to abort]");
       }
       else{
-	Msg(STATUS2N, "Creating volume");
 	if(!List_Nbr(List1))
 	  Msg(ONSCREEN, "Select volume boundary\n"
 	      "[Press 'q' to abort]");
@@ -2305,7 +2291,6 @@ stopall:;
   List_Delete(List2);
 
   WID->reset_visibility();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
@@ -2375,19 +2360,6 @@ static void _action_point_line_surface_volume(int action, int mode, char *what)
     
   List_T *List1 = List_Create(5, 5, sizeof(int));
   while(1) {
-    if(action == 10)
-      Msg(STATUS2N, "Adding new model edges");
-    else if(action == 9)
-      Msg(STATUS2N, "Defining surfaces to recombine");
-    else if(action == 8)
-      Msg(STATUS2N, "Setting characteristic length");
-    else if(action == 7)
-      Msg(STATUS2N, "Creating physical %s", str);
-    else if(action == 6)
-      Msg(STATUS2N, "Deleting %s", str);
-    else
-      Msg(STATUS2N, "Transforming %s", str);
-
     if(!List_Nbr(List1))
       Msg(ONSCREEN, "Select %s\n"
 	  "[Press 'e' to end selection or 'q' to abort]", str);
@@ -2572,7 +2544,6 @@ static void _action_point_line_surface_volume(int action, int mode, char *what)
   List_Delete(List1);
 
   WID->reset_visibility();
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
   
@@ -2890,21 +2861,21 @@ void mesh_1d_cb(CALLBACK_ARGS)
 {
   mai3d(1);
   Draw();
-  Msg(STATUS2N, "");
+  Msg(STATUS2N, " ");
 }
 
 void mesh_2d_cb(CALLBACK_ARGS)
 {
   mai3d(2);
   Draw();
-  Msg(STATUS2N, "");
+  Msg(STATUS2N, " ");
 }
 
 void mesh_3d_cb(CALLBACK_ARGS)
 {
   mai3d(3);
   Draw();
-  Msg(STATUS2N, "");
+  Msg(STATUS2N, " ");
 }
 
 void mesh_stl_cb(CALLBACK_ARGS)
@@ -2925,7 +2896,7 @@ void mesh_degree_cb(CALLBACK_ARGS)
   }
   CTX.mesh.changed = 1;
   Draw();
-  Msg(STATUS2N, "");
+  Msg(STATUS2N, " ");
 }
 
 void mesh_optimize_cb(CALLBACK_ARGS)
@@ -2940,14 +2911,14 @@ void mesh_optimize_cb(CALLBACK_ARGS)
 
   CTX.mesh.changed = 1;
   Draw();
-  Msg(STATUS2N, "");
+  Msg(STATUS2N, " ");
 }
 
 void mesh_remesh_cb(CALLBACK_ARGS)
 {
   ReMesh(THEM);
   Draw();
-  Msg(STATUS2N, "");
+  Msg(STATUS2N, " ");
 }
 
 void mesh_update_edges_cb(CALLBACK_ARGS)
@@ -2999,7 +2970,6 @@ static void _add_transfinite_elliptic(int type, int dim)
 
   n = 0;
   while(1) {
-    Msg(STATUS2N, "Setting structured mesh contraints");
     switch (dim) {
     case 1:
       if(n == 0)
@@ -3113,7 +3083,6 @@ static void _add_transfinite_elliptic(int type, int dim)
   }
 
 stopall:
-  Msg(STATUS2N, "");
   Msg(ONSCREEN, "");
 }
 
diff --git a/Fltk/GUI.cpp b/Fltk/GUI.cpp
index 8389fa3a1a..b093172867 100644
--- a/Fltk/GUI.cpp
+++ b/Fltk/GUI.cpp
@@ -1,4 +1,4 @@
-// $Id: GUI.cpp,v 1.503 2006-08-05 10:05:44 geuzaine Exp $
+// $Id: GUI.cpp,v 1.504 2006-08-06 22:58:48 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -1337,8 +1337,6 @@ int GUI::get_context()
 
 void GUI::create_graphic_window()
 {
-  int i, x;
-
   if(g_window) {
     g_window->show();
     return;
@@ -1358,7 +1356,7 @@ void GUI::create_graphic_window()
   Fl_Box *bottom = new Fl_Box(0, glheight, width, sh);
   bottom->box(FL_THIN_UP_BOX);
 
-  x = 2;
+  int x = 2;
 
   g_status_butt[0] = new Fl_Button(x, glheight + 2, sw, sh - 4, "X");
   x += sw;
@@ -1421,18 +1419,19 @@ void GUI::create_graphic_window()
   stop_bmp = new Fl_Bitmap(stop_bits, stop_width, stop_height);
   g_status_butt[7]->deactivate();
 
-  for(i = 0; i < 10; i++) {
+  for(int i = 0; i < 10; i++) {
     g_status_butt[i]->box(FL_FLAT_BOX);
     g_status_butt[i]->selection_color(FL_WHITE);
     g_status_butt[i]->align(FL_ALIGN_CENTER | FL_ALIGN_INSIDE | FL_ALIGN_CLIP);
   }
 
+  x += 2;
   g_status_label[0] = new Fl_Box(x, glheight + 2, 
-				 (width - x) / 2, sh - 4);
-  g_status_label[1] = new Fl_Box(x + (width - x) / 2, glheight + 2, 
-				 (width - x) / 2, sh - 4);
-  for(i = 0; i < 2; i++) {
-    g_status_label[i]->box(FL_FLAT_BOX);
+				 (width - x) / 3, sh - 4);
+  g_status_label[1] = new Fl_Box(x + (width - x) / 3, glheight + 2, 
+				 2 * (width - x - 1) / 3, sh - 4);
+  for(int i = 0; i < 2; i++) {
+    g_status_label[i]->box(FL_THIN_DOWN_BOX);
     g_status_label[i]->align(FL_ALIGN_LEFT | FL_ALIGN_INSIDE | FL_ALIGN_CLIP);
   }
 
diff --git a/Fltk/Main.cpp b/Fltk/Main.cpp
index 47eb418429..1022dbeb64 100644
--- a/Fltk/Main.cpp
+++ b/Fltk/Main.cpp
@@ -1,4 +1,4 @@
-// $Id: Main.cpp,v 1.92 2006-08-05 10:05:45 geuzaine Exp $
+// $Id: Main.cpp,v 1.93 2006-08-06 22:58:49 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -165,6 +165,10 @@ int main(int argc, char *argv[])
 
   CTX.batch = 0;
 
+  // Say welcome!
+
+  Msg(STATUS2N, "Gmsh %s", Get_GmshVersion());
+
   // Log the following for bug reports
 
   Msg(INFO, "-------------------------------------------------------");
@@ -231,10 +235,6 @@ int main(int argc, char *argv[])
       Msg(GERROR, "Invalid background mesh (no view)");
   }
 
-  // Say welcome!
-
-  Msg(STATUS2N, "Gmsh %s", Get_GmshVersion());
-
   // Draw the actual scene
   Draw();
 
diff --git a/Fltk/Message.cpp b/Fltk/Message.cpp
index 92445f7a4e..b86fdce7cc 100644
--- a/Fltk/Message.cpp
+++ b/Fltk/Message.cpp
@@ -1,4 +1,4 @@
-// $Id: Message.cpp,v 1.74 2006-08-05 10:05:45 geuzaine Exp $
+// $Id: Message.cpp,v 1.75 2006-08-06 22:58:49 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -139,7 +139,7 @@ void Msg(int level, char *fmt, ...)
 	 level != DEBUG2 &&
 	 level != DEBUG3 &&
 	 level != STATUS1N &&
-	 level != STATUS1N){
+	 level != STATUS2N){
 	WID->check();
       }
     }
diff --git a/Geo/GEdge.h b/Geo/GEdge.h
index e7d87cf53f..8e242869dd 100644
--- a/Geo/GEdge.h
+++ b/Geo/GEdge.h
@@ -11,6 +11,10 @@
 // A model edge.
 
 class GEdge : public GEntity {
+ protected:
+  GVertex *v0, *v1;
+  std::list<GFace *> l_faces;
+
  public:
   GEdge(GModel *model, 
 	int tag, 
@@ -44,8 +48,8 @@ class GEdge : public GEntity {
   void addFace(GFace *f);
   void delFace(GFace *f);
 
-  GVertex * getBeginVertex () const { return v0; }
-  GVertex * getEndVertex () const { return v1; }
+  GVertex *getBeginVertex () const { return v0; }
+  GVertex *getEndVertex () const { return v1; }
   
   struct {
     char   Method;
@@ -59,18 +63,12 @@ class GEdge : public GEntity {
   virtual int minimumMeshSegments () const {return 1;}
   virtual int minimumDrawSegments () const {return 1;}
 
-  virtual std::string getSpecificInfoString()
+  virtual std::string getAdditionalInfoString()
   {
     char tmp[256];
     sprintf(tmp, "{%d,%d}", v0->tag(), v1->tag());
     return std::string(tmp);
   }
-
- protected:
-
-  GVertex *v0,*v1;
-  std::list<GFace *> l_faces;
-
 };
 
 #endif
diff --git a/Geo/GEntity.h b/Geo/GEntity.h
index 681b47197e..c811ce0410 100644
--- a/Geo/GEntity.h
+++ b/Geo/GEntity.h
@@ -74,34 +74,6 @@ class GEntity {
       return name[type];
   }
 
-  virtual std::string getSpecificInfoString() 
-  {
-    return "";
-  }
-
-  virtual std::string getInfoString() 
-  {
-    char str[256];
-    sprintf(str, "%s %d", getTypeString().c_str(), tag());
-    std::string specific = getSpecificInfoString();
-    if(specific.size()){
-      strcat(str, " ");
-      strcat(str, specific.c_str());
-    }
-    if(physicals.size()){
-      char str2[256] = " [";
-      for(unsigned int i = 0; i < physicals.size(); i++){
-	char str3[256];
-	sprintf(str3, "%d", physicals[i]);
-	if(i) strcat(str2, " ");
-	strcat(str2, str3);
-      }
-      strcat(str2, "]");
-      strcat(str, str2);
-    }
-    return std::string(str);
-  }
-
   GEntity(GModel *m, int t) : _model(m), _tag(t)
   {
     drawAttributes.Visible = VIS_GEOM | VIS_MESH; 
@@ -177,6 +149,33 @@ class GEntity {
 
   // Returns a renderable representation of the mesh
   virtual MRep *meshRep(){ return _mesh; }
+
+  // Returns an information string for the entity
+  virtual std::string getInfoString()
+  {
+    char str[256];
+    sprintf(str, "%s %d", getTypeString().c_str(), tag());
+    std::string info = getAdditionalInfoString();
+    if(info.size()){
+      strcat(str, " ");
+      strcat(str, info.c_str());
+    }
+    if(physicals.size()){
+      char str2[256] = " [";
+      for(unsigned int i = 0; i < physicals.size(); i++){
+	char str3[256];
+	sprintf(str3, "%d", physicals[i]);
+	if(i) strcat(str2, " ");
+	strcat(str2, str3);
+      }
+      strcat(str2, "]");
+      strcat(str, str2);
+    }
+    return std::string(str);
+  }
+
+  // Returns a type-specific additional information string
+  virtual std::string getAdditionalInfoString() { return std::string(""); }
 };
 
 // A minimal, non-abstract entity that can be used for sorting
diff --git a/Geo/GFace.cpp b/Geo/GFace.cpp
index f33fd87a0a..1073024d7e 100644
--- a/Geo/GFace.cpp
+++ b/Geo/GFace.cpp
@@ -30,9 +30,9 @@ std::list<GVertex*> GFace::vertices() const
   while (it != l_edges.end()){
     GVertex *v1 = (*it)->getBeginVertex();
     GVertex *v2 = (*it)->getEndVertex();
-    if(std::find (ret.begin(),ret.end(),v1) == ret.end())
+    if(std::find(ret.begin(), ret.end(), v1) == ret.end())
       ret.push_back(v1);
-    if(std::find (ret.begin(),ret.end(),v2) == ret.end())
+    if(std::find(ret.begin(), ret.end(), v2) == ret.end())
       ret.push_back(v2);
     ++it;
   }
diff --git a/Geo/GModel.cpp b/Geo/GModel.cpp
index 8edfab243b..5b891496c7 100644
--- a/Geo/GModel.cpp
+++ b/Geo/GModel.cpp
@@ -58,3 +58,60 @@ GVertex * GModel::vertexByTag(int n) const
   else
     return 0;
 }
+
+int GModel::renumberMeshVertices()
+{
+  int numVertices = 0;
+  for(viter it = firstVertex(); it != lastVertex(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->setNum(++numVertices);
+  for(GModel::eiter it = firstEdge(); it != lastEdge(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->setNum(++numVertices);
+  for(GModel::fiter it = firstFace(); it != lastFace(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->setNum(++numVertices);
+  for(GModel::riter it = firstRegion(); it != lastRegion(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->setNum(++numVertices);
+  return numVertices;
+}
+
+bool GModel::noPhysicals()
+{
+  bool somePhysicals = false;
+  for(viter it = firstVertex(); it != lastVertex(); ++it)
+    if((*it)->physicals.size()){ somePhysicals = true; break; }
+  if(!somePhysicals)
+    for(eiter it = firstEdge(); it != lastEdge(); ++it)
+      if((*it)->physicals.size()){ somePhysicals = true; break; }
+  if(!somePhysicals)
+    for(fiter it = firstFace(); it != lastFace(); ++it)
+      if((*it)->physicals.size()){ somePhysicals = true; break; }
+  if(!somePhysicals)
+    for(riter it = firstRegion(); it != lastRegion(); ++it)
+      if((*it)->physicals.size()){ somePhysicals = true; break; }
+
+  return !somePhysicals;
+}
+
+static void addInGroup(GEntity* ge, std::map<int, std::vector<GEntity*> > &group)
+{
+  for(unsigned int i = 0; i < ge->physicals.size(); i++){
+    int p = ge->physicals[i];
+    if(std::find(group[p].begin(), group[p].end(), ge) == group[p].end())
+      group[p].push_back(ge);
+  }
+}
+
+void GModel::getPhysicalGroups(std::map<int, std::vector<GEntity*> > groups[4])
+{
+  for(viter it = firstVertex(); it != lastVertex(); ++it)
+    addInGroup(*it, groups[0]);
+  for(eiter it = firstEdge(); it != lastEdge(); ++it)
+    addInGroup(*it, groups[1]);
+  for(fiter it = firstFace(); it != lastFace(); ++it)
+    addInGroup(*it, groups[2]);
+  for(riter it = firstRegion(); it != lastRegion(); ++it)
+    addInGroup(*it, groups[3]);
+}
diff --git a/Geo/GModel.h b/Geo/GModel.h
index 77d30989b9..1d98c17bcd 100644
--- a/Geo/GModel.h
+++ b/Geo/GModel.h
@@ -56,11 +56,23 @@ public:
   void remove(GEdge *e){edges.erase(std::find(firstEdge(),lastEdge(),e));}
   void remove(GVertex *v){vertices.erase(std::find(firstVertex(),lastVertex(),v));}
 
+  // Renumber all the mesh vertices in a continuous sequence
+  int renumberMeshVertices();
+
+  // Checks if there are no physical entities in the model
+  bool noPhysicals();
+
+  // Returns all physical groups (one map per dimension: 0-D to 3-D)
+  void getPhysicalGroups(std::map<int, std::vector<GEntity*> > groups[4]);
+
   // IO routines
   int readMSH(const std::string &name);
   int writeMSH(const std::string &name, double version=1.0, bool saveAll=false,
 	       double scalingFactor=1.0);
-  int writePOS(const std::string &name);
+  int writePOS(const std::string &name, double scalingFactor=1.0);
+  int writeSTL(const std::string &name, double scalingFactor=1.0);
+  int writeVRML(const std::string &name, double scalingFactor=1.0);
+  int writeUNV(const std::string &name, double scalingFactor=1.0);
 
 protected:
   std::string modelName;
diff --git a/Geo/GModelIO.cpp b/Geo/GModelIO.cpp
index 09f0509a2d..9ba813951a 100644
--- a/Geo/GModelIO.cpp
+++ b/Geo/GModelIO.cpp
@@ -8,7 +8,7 @@
 #include "gmshEdge.h"
 #include "MElement.h"
 
-static int getNumVerticesForElementType(int type)
+static int getNumVerticesForElementTypeMSH(int type)
 {
   switch (type) {
   case PNT : return 1;
@@ -176,7 +176,7 @@ int GModel::readMSH(const std::string &name)
 	int num, type, physical = 1, elementary = 1, partition = 1, numVertices;
 	if(version <= 1.0){
 	  fscanf(fp, "%d %d %d %d %d", &num, &type, &physical, &elementary, &numVertices);
-	  int check = getNumVerticesForElementType(type);
+	  int check = getNumVerticesForElementTypeMSH(type);
 	  if(!check){
 	    Msg(GERROR, "Unknown type for element %d", num); 
 	    continue;
@@ -197,7 +197,7 @@ int GModel::readMSH(const std::string &name)
 	    else if(j == 2) partition = tag;
 	    // ignore any other tags for now
 	  }
-	  numVertices = getNumVerticesForElementType(type);
+	  numVertices = getNumVerticesForElementTypeMSH(type);
 	  if(!numVertices){
 	    Msg(GERROR, "Unknown type (%d) for element %d", type, num); 
 	    continue;
@@ -341,39 +341,6 @@ int GModel::readMSH(const std::string &name)
   return 1;
 }
 
-static void numVerticesAndElementsInModel(GModel *m, bool saveAll, 
-					  int &numVertices, int &numElements)
-{
-  // get the number of vertices and elements in the msh file and
-  // renumber the nodes in a continuous sequence
-  numVertices = numElements = 0;
-  for(GModel::viter it = m->firstVertex(); it != m->lastVertex(); ++it){
-    int n = (*it)->mesh_vertices.size();
-    numElements += (saveAll ? 1 : (*it)->physicals.size()) * n;
-    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
-      (*it)->mesh_vertices[i]->setNum(++numVertices);
-  }
-  for(GModel::eiter it = m->firstEdge(); it != m->lastEdge(); ++it){
-    int n = (*it)->lines.size();
-    numElements += (saveAll ? 1 : (*it)->physicals.size()) * n;
-    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
-      (*it)->mesh_vertices[i]->setNum(++numVertices);
-  }
-  for(GModel::fiter it = m->firstFace(); it != m->lastFace(); ++it){
-    int n = (*it)->triangles.size() + (*it)->quadrangles.size();
-    numElements += (saveAll ? 1 : (*it)->physicals.size()) * n;
-    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
-      (*it)->mesh_vertices[i]->setNum(++numVertices);
-  }
-  for(GModel::riter it = m->firstRegion(); it != m->lastRegion(); ++it){
-    int n = (*it)->tetrahedra.size() + (*it)->hexahedra.size() +
-      (*it)->prisms.size() + (*it)->pyramids.size();
-    numElements += (saveAll ? 1 : (*it)->physicals.size()) * n;
-    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
-      (*it)->mesh_vertices[i]->setNum(++numVertices);
-  }
-}
-
 template<class T>
 static void writeElementsMSH(FILE *fp, const std::vector<T*> &ele, int saveAll, 
 			     double version, int &num, int elementary, 
@@ -396,6 +363,29 @@ int GModel::writeMSH(const std::string &name, double version, bool saveAll,
     return 0;
   }
 
+  // if there are no physicals we save all the elements
+  if(noPhysicals()) saveAll = true;
+
+  // get the number of vertices and renumber the vertices in a
+  // continuous sequence
+  int numVertices = renumberMeshVertices();
+  
+  // get the number of elements
+  int numElements = 0;
+  for(viter it = firstVertex(); it != lastVertex(); ++it)
+    numElements += (saveAll ? 1 : (*it)->physicals.size()) * 
+      (*it)->mesh_vertices.size();
+  for(eiter it = firstEdge(); it != lastEdge(); ++it)
+    numElements += (saveAll ? 1 : (*it)->physicals.size()) * 
+      (*it)->lines.size();
+  for(fiter it = firstFace(); it != lastFace(); ++it)
+    numElements += (saveAll ? 1 : (*it)->physicals.size()) * 
+      ((*it)->triangles.size() + (*it)->quadrangles.size());
+  for(riter it = firstRegion(); it != lastRegion(); ++it)
+    numElements += (saveAll ? 1 : (*it)->physicals.size()) * 
+      ((*it)->tetrahedra.size() + (*it)->hexahedra.size() +
+       (*it)->prisms.size() + (*it)->pyramids.size());
+
   if(version > 2.0){
     fprintf(fp, "$MeshFormat\n");
     fprintf(fp, "%g %d %d\n", version, 0, (int)sizeof(double));
@@ -404,10 +394,7 @@ int GModel::writeMSH(const std::string &name, double version, bool saveAll,
   }
   else
     fprintf(fp, "$NOD\n");
-
-  int numVertices, numElements;
-  numVerticesAndElementsInModel(this, saveAll, numVertices, numElements);
-
+ 
   fprintf(fp, "%d\n", numVertices);
   for(viter it = firstVertex(); it != lastVertex(); ++it)
     for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
@@ -469,7 +456,7 @@ int GModel::writeMSH(const std::string &name, double version, bool saveAll,
   return 1;
 }
 
-int GModel::writePOS(const std::string &name)
+int GModel::writePOS(const std::string &name, double scalingFactor)
 {
   FILE *fp = fopen(name.c_str(), "w");
   if(!fp){
@@ -483,13 +470,13 @@ int GModel::writePOS(const std::string &name)
 	    "\"Gamma\", \"Eta\", \"Rho\"};\n", (1<<16)|(4<<8));
     for(riter it = firstRegion(); it != lastRegion(); ++it) {
       for(unsigned int i = 0; i < (*it)->tetrahedra.size(); i++)
-	(*it)->tetrahedra[i]->writePOS(fp);
+	(*it)->tetrahedra[i]->writePOS(fp, scalingFactor);
       for(unsigned int i = 0; i < (*it)->hexahedra.size(); i++)
-	(*it)->hexahedra[i]->writePOS(fp);
+	(*it)->hexahedra[i]->writePOS(fp, scalingFactor);
       for(unsigned int i = 0; i < (*it)->prisms.size(); i++)
-	(*it)->prisms[i]->writePOS(fp);
+	(*it)->prisms[i]->writePOS(fp, scalingFactor);
       for(unsigned int i = 0; i < (*it)->pyramids.size(); i++)
-	(*it)->pyramids[i]->writePOS(fp);
+	(*it)->pyramids[i]->writePOS(fp, scalingFactor);
     }
     fprintf(fp, "};\n");
   }
@@ -500,9 +487,9 @@ int GModel::writePOS(const std::string &name)
 	    "\"Gamma\", \"Eta\", \"Rho\"};\n", (1<<16)|(4<<8));
     for(fiter it = firstFace(); it != lastFace(); ++it) {
       for(unsigned int i = 0; i < (*it)->triangles.size(); i++)
-	(*it)->triangles[i]->writePOS(fp);
+	(*it)->triangles[i]->writePOS(fp, scalingFactor);
       for(unsigned int i = 0; i < (*it)->quadrangles.size(); i++)
-	(*it)->quadrangles[i]->writePOS(fp);
+	(*it)->quadrangles[i]->writePOS(fp, scalingFactor);
     }
     fprintf(fp, "};\n");
   }
@@ -513,7 +500,7 @@ int GModel::writePOS(const std::string &name)
 	    "\"Gamma\", \"Eta\", \"Rho\"};\n", (1<<16)|(4<<8));
     for(eiter it = firstEdge(); it != lastEdge(); ++it) {
       for(unsigned int i = 0; i < (*it)->lines.size(); i++)
- 	(*it)->lines[i]->writePOS(fp);
+ 	(*it)->lines[i]->writePOS(fp, scalingFactor);
     }
     fprintf(fp, "};\n");
   }
@@ -521,3 +508,199 @@ int GModel::writePOS(const std::string &name)
   fclose(fp);
   return 1;
 }
+
+int GModel::writeSTL(const std::string &name, double scalingFactor)
+{
+  FILE *fp = fopen(name.c_str(), "w");
+  if(!fp){
+    Msg(GERROR, "Unable to open file '%s'", name.c_str());
+    return 0;
+  }
+
+  fprintf(fp, "solid Created by Gmsh\n");
+  for(fiter it = firstFace(); it != lastFace(); ++it) {
+    for(unsigned int i = 0; i < (*it)->triangles.size(); i++)
+      (*it)->triangles[i]->writeSTL(fp, scalingFactor);
+    for(unsigned int i = 0; i < (*it)->quadrangles.size(); i++)
+      (*it)->quadrangles[i]->writeSTL(fp, scalingFactor);
+  }
+  fprintf(fp, "endsolid Created by Gmsh\n");
+
+  fclose(fp);
+  return 1;
+}
+
+int GModel::writeVRML(const std::string &name, double scalingFactor)
+{
+  FILE *fp = fopen(name.c_str(), "w");
+  if(!fp){
+    Msg(GERROR, "Unable to open file '%s'", name.c_str());
+    return 0;
+  }
+
+  renumberMeshVertices();
+
+  fprintf(fp, "#VRML V1.0 ascii\n");
+  fprintf(fp, "#created by Gmsh\n");
+  fprintf(fp, "Coordinate3 {\n");
+  fprintf(fp, "  point [\n");
+
+  for(viter it = firstVertex(); it != lastVertex(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->writeVRML(fp, scalingFactor);
+  for(eiter it = firstEdge(); it != lastEdge(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++)
+      (*it)->mesh_vertices[i]->writeVRML(fp, scalingFactor);
+  for(fiter it = firstFace(); it != lastFace(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->writeVRML(fp, scalingFactor);
+
+  fprintf(fp, "  ]\n");
+  fprintf(fp, "}\n");
+
+  for(eiter it = firstEdge(); it != lastEdge(); ++it){
+    fprintf(fp, "DEF Curve%d IndexedLineSet {\n", (*it)->tag());
+    fprintf(fp, "  coordIndex [\n");
+    for(unsigned int i = 0; i < (*it)->lines.size(); i++)
+      (*it)->lines[i]->writeVRML(fp);
+    fprintf(fp, "  ]\n");
+    fprintf(fp, "}\n");
+  }
+
+  for(fiter it = firstFace(); it != lastFace(); ++it){
+    fprintf(fp, "DEF Surface%d IndexedFaceSet {\n", (*it)->tag());
+    fprintf(fp, "  coordIndex [\n");
+    for(unsigned int i = 0; i < (*it)->triangles.size(); i++)
+      (*it)->triangles[i]->writeVRML(fp);
+    for(unsigned int i = 0; i < (*it)->quadrangles.size(); i++)
+      (*it)->quadrangles[i]->writeVRML(fp);
+    fprintf(fp, "  ]\n");
+    fprintf(fp, "}\n");
+  }
+  
+  fclose(fp);
+  return 1;
+}
+
+static void addInGroupOfNodesUNV(FILE *fp, std::set<int> &nodes, int num)
+{
+  std::set<int>::iterator it = nodes.find(num);
+  if(it == nodes.end()){
+    nodes.insert(num);
+    fprintf(fp, "%10d%10d%2d%2d%2d%2d%2d%2d\n", num, 1, 0, 1, 0, 0, 0, 0);
+    fprintf(fp, "   0.0000000000000000D+00   1.0000000000000000D+00"
+	    "   0.0000000000000000D+00\n");
+    fprintf(fp, "   0.0000000000000000D+00   0.0000000000000000D+00"
+	    "   0.0000000000000000D+00\n");
+    fprintf(fp, "%10d%10d%10d%10d%10d%10d\n", 0, 0, 0, 0, 0, 0);
+  }
+}
+
+template<class T>
+static void addInGroupOfNodesUNV(FILE *fp, std::set<int> &nodes, 
+				 std::vector<T*> &elements)
+{
+  for(unsigned int i = 0; i < elements.size(); i++)
+    for(int j = 0; j < elements[i]->getNumVertices(); j++)
+      addInGroupOfNodesUNV(fp, nodes, elements[i]->getVertex(j)->getNum());
+}
+
+int GModel::writeUNV(const std::string &name, double scalingFactor)
+{
+  FILE *fp = fopen(name.c_str(), "w");
+  if(!fp){
+    Msg(GERROR, "Unable to open file '%s'", name.c_str());
+    return 0;
+  }
+
+  // IDEAS records
+  const int NODES=2411, ELEMENTS=2412, GROUPOFNODES=790;
+
+  // IDEAS elements
+  const int BEAM=21, THINSHLL=91, QUAD=94, SOLIDFEM=111, WEDGE=112, BRICK=115;
+  //const int BEAM2=24, THINSHLL2=92, QUAD2=95/*?*/, SOLIDFEM2=118;
+
+  renumberMeshVertices();
+  
+  fprintf(fp, "%6d\n", -1);
+  fprintf(fp, "%6d\n", NODES);
+  for(viter it = firstVertex(); it != lastVertex(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->writeUNV(fp, scalingFactor);
+  for(eiter it = firstEdge(); it != lastEdge(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++)
+      (*it)->mesh_vertices[i]->writeUNV(fp, scalingFactor);
+  for(fiter it = firstFace(); it != lastFace(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->writeUNV(fp, scalingFactor);
+  for(riter it = firstRegion(); it != lastRegion(); ++it)
+    for(unsigned int i = 0; i < (*it)->mesh_vertices.size(); i++) 
+      (*it)->mesh_vertices[i]->writeUNV(fp, scalingFactor);
+  fprintf(fp, "%6d\n", -1);  
+
+  fprintf(fp, "%6d\n", -1);
+  fprintf(fp, "%6d\n", ELEMENTS);
+  for(eiter it = firstEdge(); it != lastEdge(); ++it){
+    for(unsigned int i = 0; i < (*it)->lines.size(); i++)
+      (*it)->lines[i]->writeUNV(fp, BEAM, (*it)->tag());
+  }
+  for(fiter it = firstFace(); it != lastFace(); ++it){
+    for(unsigned int i = 0; i < (*it)->triangles.size(); i++)
+      (*it)->triangles[i]->writeUNV(fp, THINSHLL, (*it)->tag());
+    for(unsigned int i = 0; i < (*it)->quadrangles.size(); i++)
+      (*it)->quadrangles[i]->writeUNV(fp, QUAD, (*it)->tag());
+  }
+  for(riter it = firstRegion(); it != lastRegion(); ++it){
+    for(unsigned int i = 0; i < (*it)->tetrahedra.size(); i++)
+      (*it)->tetrahedra[i]->writeUNV(fp, SOLIDFEM, (*it)->tag());
+    for(unsigned int i = 0; i < (*it)->hexahedra.size(); i++)
+      (*it)->hexahedra[i]->writeUNV(fp, BRICK, (*it)->tag());
+    for(unsigned int i = 0; i < (*it)->prisms.size(); i++)
+      (*it)->prisms[i]->writeUNV(fp, WEDGE, (*it)->tag());
+  }
+  fprintf(fp, "%6d\n", -1);
+
+  std::map<int, std::vector<GEntity*> > physicals[4];
+  getPhysicalGroups(physicals);
+
+  for(int dim = 0; dim < 4; dim++){
+    std::map<int, std::vector<GEntity*> >::const_iterator it = physicals[dim].begin();
+    std::map<int, std::vector<GEntity*> >::const_iterator ite = physicals[dim].end();
+    for(; it != ite; ++it){
+      fprintf(fp, "%6d\n", -1);
+      fprintf(fp, "%6d\n", GROUPOFNODES);
+      fprintf(fp, "%10d%10d\n", it->first, 1);
+      fprintf(fp, "LOAD SET %2d\n", 1);
+      std::set<int> nodes;
+      for(unsigned int i = 0; i < it->second.size(); i++){
+	// we could also do this using the mesh_vertices of the entity
+	// and all the entities in the closure
+	GVertex *v;
+	switch(dim){
+	case 0: 
+	  v = (GVertex*)it->second[i];
+	  for(unsigned int j = 0; j < v->mesh_vertices.size(); j++)
+	    addInGroupOfNodesUNV(fp, nodes, v->mesh_vertices[j]->getNum());
+	  break;
+	case 1: 
+	  addInGroupOfNodesUNV(fp, nodes, ((GEdge*)it->second[i])->lines);
+	  break;
+	case 2: 
+	  addInGroupOfNodesUNV(fp, nodes, ((GFace*)it->second[i])->triangles);
+	  addInGroupOfNodesUNV(fp, nodes, ((GFace*)it->second[i])->quadrangles);
+	  break;
+	case 3: 
+	  addInGroupOfNodesUNV(fp, nodes, ((GRegion*)it->second[i])->tetrahedra);
+	  addInGroupOfNodesUNV(fp, nodes, ((GRegion*)it->second[i])->hexahedra);
+	  addInGroupOfNodesUNV(fp, nodes, ((GRegion*)it->second[i])->prisms);
+	  addInGroupOfNodesUNV(fp, nodes, ((GRegion*)it->second[i])->pyramids);
+	  break;
+	}
+      }
+      fprintf(fp, "%6d\n", -1);
+    }
+  }
+
+  fclose(fp);
+  return 1;
+}
diff --git a/Geo/GVertex.cpp b/Geo/GVertex.cpp
index aa6526e6d1..80f878273c 100644
--- a/Geo/GVertex.cpp
+++ b/Geo/GVertex.cpp
@@ -1,10 +1,12 @@
 #include "GVertex.h"
 #include <algorithm>
-void GVertex::addEdge ( GEdge *e )
+
+void GVertex::addEdge(GEdge *e)
 { 
-  l_edges.push_back (e);  
+  l_edges.push_back(e);
 }
-void GVertex::delEdge ( GEdge *e )
+
+void GVertex::delEdge(GEdge *e)
 { 
-  l_edges.erase(std::find(l_edges.begin(),l_edges.end(),e));  
+  l_edges.erase(std::find(l_edges.begin(), l_edges.end(), e));
 }
diff --git a/Geo/GVertex.h b/Geo/GVertex.h
index 1d53a12ee2..80d8787554 100644
--- a/Geo/GVertex.h
+++ b/Geo/GVertex.h
@@ -26,11 +26,18 @@ public:
   virtual int dim() const {return 0;}
   virtual GeomType geomType() const {return Point;}
   virtual double prescribedMeshSizeAtVertex() const {return 0;}
-  virtual std::string getSpecificInfoString()
+  virtual std::string getAdditionalInfoString()
   {
-    char tmp[256];
-    sprintf(tmp, "{%g,%g,%g}", x(), y(), z());
-    return std::string(tmp);
+    char str[256];
+    sprintf(str, "{%g,%g,%g", x(), y(), z());
+    double lc = prescribedMeshSizeAtVertex();
+    if(lc > 0.){
+      char str2[256];
+      sprintf(str2, ",%g", lc);
+      strcat(str, str2);
+    }
+    strcat(str, "}");
+    return std::string(str);
   }
 
  protected:
diff --git a/Geo/MElement.cpp b/Geo/MElement.cpp
index 94e630fac9..dca8bb17c5 100644
--- a/Geo/MElement.cpp
+++ b/Geo/MElement.cpp
@@ -1,6 +1,7 @@
 #include <math.h>
 #include "MElement.h"
 #include "GEntity.h"
+#include "Numeric.h"
 
 int MElement::_globalNum = 0;
 
@@ -63,19 +64,41 @@ void MElement::writeMSH(FILE *fp, double version, int num, int elementary,
 			int physical)
 {
   int n = getNumVertices();
+  int type = getTypeForMSH();
 
-  fprintf(fp, "%d %d", num ? num : _num, getTypeForMSH());
+  // if necessary, change the ordering of the vertices to get positive
+  // volume
+  setVolumePositive();
+
+  fprintf(fp, "%d %d", num ? num : _num, type);
   if(version < 2.0)
     fprintf(fp, " %d %d %d", physical, elementary, n);
   else
     fprintf(fp, " 3 %d %d %d", physical, elementary, _partition);
-  for(int i = 0; i < n; i++){
-    fprintf(fp, " %d", getVertex(i)->getNum());
+  
+  if(physical >= 0){
+    for(int i = 0; i < n; i++)
+      fprintf(fp, " %d", getVertex(i)->getNum());
+  }
+  else{
+    int nn = n - getNumEdgeVertices() - getNumFaceVertices() - getNumVolumeVertices();
+    for(int i = 0; i < nn; i++)
+      fprintf(fp, " %d", getVertex(nn - i - 1)->getNum());
+    int ne = getNumEdgeVertices();
+    for(int i = 0; i < ne; i++)
+      fprintf(fp, " %d", getVertex(nn + ne - i - 1)->getNum());
+    int nf = getNumFaceVertices();
+    for(int i = 0; i < nf; i++)
+      fprintf(fp, " %d", getVertex(nn + ne + nf - i - 1)->getNum());
+    int nv = getNumVolumeVertices();
+    for(int i = 0; i < nv; i++)
+      fprintf(fp, " %d", getVertex(n - i - 1)->getNum());
   }
+
   fprintf(fp, " \n");
 }
 
-void MElement::writePOS(FILE *fp)
+void MElement::writePOS(FILE *fp, double scalingFactor)
 {
   int n = getNumVertices();
   double gamma = gammaShapeMeasure();
@@ -85,7 +108,8 @@ void MElement::writePOS(FILE *fp)
   fprintf(fp, "%s(", getStringForPOS());
   for(int i = 0; i < n; i++){
     if(i) fprintf(fp, ",");
-    fprintf(fp, "%g,%g,%g", getVertex(i)->x(), getVertex(i)->y(), getVertex(i)->z());
+    fprintf(fp, "%g,%g,%g", getVertex(i)->x() * scalingFactor, 
+	    getVertex(i)->y() * scalingFactor, getVertex(i)->z() * scalingFactor);
   }
   fprintf(fp, "){");
   for(int i = 0; i < n; i++)
@@ -105,3 +129,66 @@ void MElement::writePOS(FILE *fp)
   fprintf(fp, "};\n");
 }
 
+void MElement::writeSTL(FILE *fp, double scalingFactor)
+{
+  int n = getNumVertices();
+  if(n < 3 || n > 4) return;
+
+  MVertex *v0 = getVertex(0);
+  MVertex *v1 = getVertex(1);
+  MVertex *v2 = getVertex(2);
+  double N[3];
+  normal3points(v0->x(), v0->y(), v0->z(), 
+		v1->x(), v1->y(), v1->z(), 
+		v2->x(), v2->y(), v2->z(), N);
+  fprintf(fp, "facet normal %g %g %g\n", N[0], N[1], N[2]);
+  fprintf(fp, "  outer loop\n");
+  fprintf(fp, "    vertex %g %g %g\n", v0->x() * scalingFactor, 
+	  v0->y() * scalingFactor, v0->z() * scalingFactor);
+  fprintf(fp, "    vertex %g %g %g\n", v1->x() * scalingFactor, 
+	  v1->y() * scalingFactor, v1->z() * scalingFactor);
+  fprintf(fp, "    vertex %g %g %g\n", v2->x() * scalingFactor, 
+	  v2->y() * scalingFactor, v2->z() * scalingFactor);
+  fprintf(fp, "  endloop\n");
+  fprintf(fp, "endfacet\n");
+  if(n == 4){
+    MVertex *v3 = getVertex(3);
+    fprintf(fp, "facet normal %g %g %g\n", N[0], N[1], N[2]);
+    fprintf(fp, "  outer loop\n");
+    fprintf(fp, "    vertex %g %g %g\n", v0->x() * scalingFactor, 
+	    v0->y() * scalingFactor, v0->z() * scalingFactor);
+    fprintf(fp, "    vertex %g %g %g\n", v2->x() * scalingFactor, 
+	    v2->y() * scalingFactor, v2->z() * scalingFactor);
+    fprintf(fp, "    vertex %g %g %g\n", v3->x() * scalingFactor, 
+	    v3->y() * scalingFactor, v3->z() * scalingFactor);
+    fprintf(fp, "  endloop\n");
+    fprintf(fp, "endfacet\n");
+  }
+}
+
+void MElement::writeVRML(FILE *fp)
+{
+  for(int i = 0; i < getNumVertices(); i++)
+    fprintf(fp, "%d,", getVertex(i)->getNum() - 1);
+  fprintf(fp, "-1,\n");
+}
+
+void MElement::writeUNV(FILE *fp, int type, int elementary)
+{
+  // if necessary, change the ordering of the vertices to get positive
+  // volume
+  setVolumePositive();
+
+  int n = getNumVertices();
+  fprintf(fp, "%10d%10d%10d%10d%10d%10d\n",
+	  _num, type, elementary, elementary, 7, n);
+  if(type == 21 || type == 24) // BEAM or BEAM2
+    fprintf(fp, "%10d%10d%10d\n", 0, 0, 0);
+  for(int k = 0; k < n; k++) {
+    fprintf(fp, "%10d", getVertex(k)->getNum());
+    if(k % 8 == 7)
+      fprintf(fp, "\n");
+  }
+  if(n - 1 % 8 != 7)
+    fprintf(fp, "\n");
+}
diff --git a/Geo/MElement.h b/Geo/MElement.h
index e287e6d9ba..e464bb8185 100644
--- a/Geo/MElement.h
+++ b/Geo/MElement.h
@@ -5,6 +5,7 @@
 #include <algorithm>
 #include "GmshDefines.h"
 #include "MVertex.h"
+#include "Numeric.h"
 
 // the reference topology is defined in Mesh/{Edge,Face}.cpp
 extern int edges_tetra[6][2];
@@ -57,6 +58,12 @@ class MElement
   virtual int getNumVertices() = 0;
   virtual MVertex *getVertex(int num) = 0;
 
+  // get the number of vertices associated with edges, faces and
+  // volumes (nonzero only for higher order elements)
+  virtual int getNumEdgeVertices(){ return 0; }
+  virtual int getNumFaceVertices(){ return 0; }
+  virtual int getNumVolumeVertices(){ return 0; }
+
   // get the edges
   virtual int getNumEdges() = 0;
   virtual void getEdge(int num, MVertex *v[2]) = 0;
@@ -85,10 +92,18 @@ class MElement
   // computes the barycenter
   virtual void cog(double &x, double &y, double &z);
 
+  // compute and change the orientation of 3D elements to get
+  // positive volume
+  virtual int getVolumeSign(){ return 1; }
+  virtual void setVolumePositive(){}
+
   // IO routines
   virtual void writeMSH(FILE *fp, double version=1.0, int num=0, 
 			int elementary=1, int physical=1);
-  virtual void writePOS(FILE *fp);
+  virtual void writePOS(FILE *fp, double scalingFactor=1.0);
+  virtual void writeSTL(FILE *fp, double scalingFactor=1.0);
+  virtual void writeVRML(FILE *fp);
+  virtual void writeUNV(FILE *fp, int type, int elementary);
   virtual char *getStringForPOS() = 0;
   virtual int getTypeForMSH() = 0;
 };
@@ -131,6 +146,7 @@ class MLine2 : public MLine {
   ~MLine2(){}
   inline int getNumVertices(){ return 3; }
   inline MVertex *getVertex(int num){ return num < 2 ? _v[num] : _vs[num - 2]; }
+  inline int getNumEdgeVertices(){ return 1; }
   int getNumEdgesRep(){ return 2; }
   void getEdgeRep(int num, MVertex *v[2])
   { 
@@ -187,6 +203,7 @@ class MTriangle2 : public MTriangle {
   ~MTriangle2(){}
   inline int getNumVertices(){ return 6; }
   inline MVertex *getVertex(int num){ return num < 3 ? _v[num] : _vs[num - 3]; }
+  inline int getNumEdgeVertices(){ return 3; }
   int getNumEdgesRep(){ return 6; }
   void getEdgeRep(int num, MVertex *v[2])
   { 
@@ -240,8 +257,8 @@ class MQuadrangle : public MElement {
     v[0] = _v[edges_quad[num][0]];
     v[1] = _v[edges_quad[num][1]];
   }
-  int getNumFaces(){ return 1; }
-  void getFace(int num, MVertex *v[4])
+  virtual int getNumFaces(){ return 1; }
+  virtual void getFace(int num, MVertex *v[4])
   {
     v[0] = _v[0]; v[1] = _v[1]; v[2] = _v[2]; v[3] = _v[3];
   }
@@ -263,6 +280,8 @@ class MQuadrangle2 : public MQuadrangle {
   ~MQuadrangle2(){}
   inline int getNumVertices(){ return 9; }
   inline MVertex *getVertex(int num){ return num < 4 ? _v[num] : _vs[num - 4]; }
+  inline int getNumEdgeVertices(){ return 4; }
+  inline int getNumFaceVertices(){ return 1; }
   int getTypeForMSH(){ return QUA2; }
   char *getStringForPOS(){ return "SQ2"; }
 };
@@ -286,8 +305,8 @@ class MTetrahedron : public MElement {
     v[0] = _v[edges_tetra[num][0]];
     v[1] = _v[edges_tetra[num][1]];
   }
-  int getNumFaces(){ return 4; }
-  void getFace(int num, MVertex *v[4])
+  virtual int getNumFaces(){ return 4; }
+  virtual void getFace(int num, MVertex *v[4])
   {
     v[0] = _v[trifaces_tetra[num][0]];
     v[1] = _v[trifaces_tetra[num][1]];
@@ -296,8 +315,40 @@ class MTetrahedron : public MElement {
   }
   int getTypeForMSH(){ return TET1; }
   char *getStringForPOS(){ return "SS"; }
+  virtual int getVolumeSign()
+  { 
+    double mat[3][3];
+    mat[0][0] = _v[1]->x() - _v[0]->x();
+    mat[0][1] = _v[2]->x() - _v[0]->x();
+    mat[0][2] = _v[3]->x() - _v[0]->x();
+    mat[1][0] = _v[1]->y() - _v[0]->y();
+    mat[1][1] = _v[2]->y() - _v[0]->y();
+    mat[1][2] = _v[3]->y() - _v[0]->y();
+    mat[2][0] = _v[1]->z() - _v[0]->z();
+    mat[2][1] = _v[2]->z() - _v[0]->z();
+    mat[2][2] = _v[3]->z() - _v[0]->z();
+    return sign(det3x3(mat));
+  }
+  void setVolumePositive()
+  {
+    if(getVolumeSign() < 0){
+      MVertex *tmp;
+      tmp = _v[0]; _v[0] = _v[1]; _v[1] = tmp;
+    }
+  }
 };
 
+// TODO: for MTetrahedron2
+// void setVolumePositive()
+// {
+//   if(getVolumeSign() < 0){
+//     MVertex *tmp;
+//     tmp = _v[0]; _v[0] = _v[1]; _v[1] = tmp;
+//     tmp = _vs[1]; _vs[1] = _vs[2]; _vs[2] = temp;
+//     tmp = _vs[5]; _vs[5] = _vs[3]; _vs[3] = temp;
+//   }
+// }
+
 class MHexahedron : public MElement {
  protected:
   MVertex *_v[8];
@@ -329,8 +380,46 @@ class MHexahedron : public MElement {
   }
   int getTypeForMSH(){ return HEX1; }
   char *getStringForPOS(){ return "SH"; }
+  virtual int getVolumeSign()
+  { 
+    double mat[3][3];
+    mat[0][0] = _v[1]->x() - _v[0]->x();
+    mat[0][1] = _v[3]->x() - _v[0]->x();
+    mat[0][2] = _v[4]->x() - _v[0]->x();
+    mat[1][0] = _v[1]->y() - _v[0]->y();
+    mat[1][1] = _v[3]->y() - _v[0]->y();
+    mat[1][2] = _v[4]->y() - _v[0]->y();
+    mat[2][0] = _v[1]->z() - _v[0]->z();
+    mat[2][1] = _v[3]->z() - _v[0]->z();
+    mat[2][2] = _v[4]->z() - _v[0]->z();
+    return sign(det3x3(mat));
+  }
+  void setVolumePositive()
+  {
+    if(getVolumeSign() < 0){
+      MVertex *tmp;
+      tmp = _v[0]; _v[0] = _v[2]; _v[2] = tmp;
+      tmp = _v[4]; _v[4] = _v[6]; _v[6] = tmp;
+    }
+  }
 };
 
+// TODO: for MHexahedron2
+// void setVolumePositive()
+// {
+//   if(getVolumeSign() < 0){
+//     MVertex *tmp;
+//     tmp = _v[0]; _v[0] = _v[2]; _v[2] = tmp;
+//     tmp = _v[4]; _v[4] = _v[6]; _v[6] = tmp;
+//     MVertex *old[12];
+//     for(int i = 0; i < 12; i++) old[i] = _vs[i];
+//     _vs[0] = old[3]; _vs[1] = old[5]; _vs[2] = old[6];
+//     _vs[3] = old[0]; _vs[4] = old[4]; _vs[5] = old[1];
+//     _vs[6] = old[2]; _vs[7] = old[7]; _vs[8] = old[10];
+//     _vs[9] = old[11]; _vs[10] = old[8]; _vs[11] = old[9];
+//   }
+// }
+
 class MPrism : public MElement {
  protected:
   MVertex *_v[6];
@@ -369,8 +458,43 @@ class MPrism : public MElement {
   }
   int getTypeForMSH(){ return PRI1; }
   char *getStringForPOS(){ return "SI"; }
+  virtual int getVolumeSign()
+  { 
+    double mat[3][3];
+    mat[0][0] = _v[1]->x() - _v[0]->x();
+    mat[0][1] = _v[2]->x() - _v[0]->x();
+    mat[0][2] = _v[3]->x() - _v[0]->x();
+    mat[1][0] = _v[1]->y() - _v[0]->y();
+    mat[1][1] = _v[2]->y() - _v[0]->y();
+    mat[1][2] = _v[3]->y() - _v[0]->y();
+    mat[2][0] = _v[1]->z() - _v[0]->z();
+    mat[2][1] = _v[2]->z() - _v[0]->z();
+    mat[2][2] = _v[3]->z() - _v[0]->z();
+    return sign(det3x3(mat));
+  }
+  void setVolumePositive()
+  {
+    if(getVolumeSign() < 0){
+      MVertex *tmp;
+      tmp = _v[0]; _v[0] = _v[1]; _v[1] = tmp;
+      tmp = _v[3]; _v[3] = _v[4]; _v[4] = tmp;
+    }
+  }
 };
 
+// TODO: for MPrism2
+// void setVolumePositive()
+// {
+//   if(getVolumeSign() < 0){
+//      MVertex *tmp;
+//      tmp = _v[0]; _v[0] = _v[1]; _v[1] = tmp;
+//      tmp = _v[3]; _v[3] = _v[4]; _v[4] = tmp;
+//      tmp = _vs[1]; _vs[1] = _vs[3]; _vs[3] = tmp;
+//      tmp = _vs[2]; _vs[2] = _vs[4]; _vs[4] = tmp;
+//      tmp = _vs[7]; _vs[7] = _vs[8]; _vs[8] = tmp;
+//   }
+// }
+
 class MPyramid : public MElement {
  protected:
   MVertex *_v[5];
@@ -408,6 +532,39 @@ class MPyramid : public MElement {
   }
   int getTypeForMSH(){ return PYR1; }
   char *getStringForPOS(){ return "SY"; }
+  virtual int getVolumeSign()
+  { 
+    double mat[3][3];
+    mat[0][0] = _v[1]->x() - _v[0]->x();
+    mat[0][1] = _v[3]->x() - _v[0]->x();
+    mat[0][2] = _v[4]->x() - _v[0]->x();
+    mat[1][0] = _v[1]->y() - _v[0]->y();
+    mat[1][1] = _v[3]->y() - _v[0]->y();
+    mat[1][2] = _v[4]->y() - _v[0]->y();
+    mat[2][0] = _v[1]->z() - _v[0]->z();
+    mat[2][1] = _v[3]->z() - _v[0]->z();
+    mat[2][2] = _v[4]->z() - _v[0]->z();
+    return sign(det3x3(mat));
+  }
+  void setVolumePositive()
+  {
+    if(getVolumeSign() < 0){
+      MVertex *tmp;
+      tmp = _v[0]; _v[0] = _v[2]; _v[2] = tmp;
+    }
+  }
 };
 
+// TODO: for MPyramid2
+// void setVolumePositive()
+// {
+//   if(getVolumeSign() < 0){
+//      MVertex *tmp;
+//      tmp = _v[0]; _v[0] = _v[2]; _v[2] = tmp;
+//      tmp = _vs[0]; _vs[0] = _vs[3]; _vs[3] = tmp;
+//      tmp = _vs[1]; _vs[1] = _vs[5]; _vs[5] = tmp;
+//      tmp = _vs[2]; _vs[2] = _vs[6]; _vs[6] = tmp;
+//   }
+// }
+
 #endif
diff --git a/Geo/MVertex.cpp b/Geo/MVertex.cpp
index 538ab681f1..32a44fed3f 100644
--- a/Geo/MVertex.cpp
+++ b/Geo/MVertex.cpp
@@ -3,3 +3,38 @@
 
 int MVertex::_globalNum = 0;
 
+void MVertex::writeMSH(FILE *fp, double scalingFactor)
+{
+  fprintf(fp, "%d %.16g %.16g %.16g\n", _num, x() * scalingFactor, 
+	  y() * scalingFactor, z() * scalingFactor);
+}
+
+void MVertex::writeMSH(FILE *fp, double version, int num, 
+		       int elementary, int physical)
+{
+  fprintf(fp, "%d 15", num);
+  if(version < 2.0)
+    fprintf(fp, " %d %d 1", physical, elementary);
+  else
+    fprintf(fp, " 2 %d %d", physical, elementary);
+  fprintf(fp, " %d\n", _num);
+}
+
+void MVertex::writeVRML(FILE *fp, double scalingFactor)
+{
+  fprintf(fp, "%.16g %.16g %.16g,\n",
+	  x() * scalingFactor, y() * scalingFactor, z() * scalingFactor);
+}
+
+void MVertex::writeUNV(FILE *fp, double scalingFactor)
+{
+  fprintf(fp, "%10d%10d%10d%10d\n", _num, 1, 1, 11);
+  // hack to print the numbers with "D+XX" exponents
+  char tmp[128];
+  sprintf(tmp, "%25.16E%25.16E%25.16E\n", x() * scalingFactor, 
+	  y() * scalingFactor, z() * scalingFactor);
+  tmp[21] = 'D';
+  tmp[46] = 'D';
+  tmp[71] = 'D';
+  fprintf(fp, tmp);
+}
diff --git a/Geo/MVertex.h b/Geo/MVertex.h
index 53ef43b0af..4510bad51e 100644
--- a/Geo/MVertex.h
+++ b/Geo/MVertex.h
@@ -35,20 +35,12 @@ class MVertex{
   inline void setEntity(GEntity *ge) { _ge = ge; }
   inline int getNum() const {return _num;}
   inline void setNum(int num) { _num = num; }
-  void writeMSH(FILE *fp, double scalingFactor=1.0)
-  {
-    fprintf(fp, "%d %.16g %.16g %.16g\n", getNum(), 
-	    x() * scalingFactor, y() * scalingFactor, z() * scalingFactor);
-  }
-  void writeMSH(FILE *fp, double version, int num, int elementary, int physical)
-  {
-    fprintf(fp, "%d 15", num);
-    if(version < 2.0)
-      fprintf(fp, " %d %d 1", physical, elementary);
-    else
-      fprintf(fp, " 2 %d %d", physical, elementary);
-    fprintf(fp, " %d\n", _num);
-  }
+
+  // IO routines
+  void writeMSH(FILE *fp, double scalingFactor=1.0);
+  void writeMSH(FILE *fp, double version, int num, int elementary, int physical);
+  void writeVRML(FILE *fp, double scalingFactor=1.0);
+  void writeUNV(FILE *fp, double scalingFactor=1.0);
 };
 
 class MEdgeVertex : public MVertex{
diff --git a/Geo/Makefile b/Geo/Makefile
index cb7f187959..69ab0e8c9a 100644
--- a/Geo/Makefile
+++ b/Geo/Makefile
@@ -1,4 +1,4 @@
-# $Id: Makefile,v 1.81 2006-08-04 14:28:02 geuzaine Exp $
+# $Id: Makefile,v 1.82 2006-08-06 22:58:49 geuzaine Exp $
 #
 # Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 #
@@ -166,8 +166,8 @@ GModelIO.o: GModelIO.cpp ../Common/Message.h ../Common/GmshDefines.h \
 # 1 "/Users/geuzaine/.gmsh/Geo//"
 MVertex.o: MVertex.cpp MVertex.h
 # 1 "/Users/geuzaine/.gmsh/Geo//"
-MElement.o: MElement.cpp MElement.h MVertex.h GEntity.h Range.h SPoint3.h \
-  SBoundingBox3d.h ../Common/GmshDefines.h
+MElement.o: MElement.cpp MElement.h ../Common/GmshDefines.h MVertex.h \
+  GEntity.h Range.h SPoint3.h SBoundingBox3d.h
 # 1 "/Users/geuzaine/.gmsh/Geo//"
 gmshModel.o: gmshModel.cpp gmshModel.h GModel.h GVertex.h GEntity.h \
   Range.h SPoint3.h SBoundingBox3d.h MVertex.h ../Common/GmshDefines.h \
diff --git a/Geo/gmshVertex.h b/Geo/gmshVertex.h
index b3057c3412..0ca9540863 100644
--- a/Geo/gmshVertex.h
+++ b/Geo/gmshVertex.h
@@ -17,14 +17,29 @@ public:
   virtual ~gmshVertex() {}
   virtual GPoint point() const 
   {
-    if(!v) return GPoint(0., 0., 0., this);
-    return GPoint(v->Pos.X, v->Pos.Y, v->Pos.Z, this);
+    if(v)
+      return GPoint(v->Pos.X, v->Pos.Y, v->Pos.Z, this);
+    else if(mesh_vertices.size())
+      return GPoint(mesh_vertices[0]->x(),
+		    mesh_vertices[0]->y(),
+		    mesh_vertices[0]->z(), this);
+    else
+      return GPoint(0., 0., 0., this);
+  }
+  virtual double x() const 
+  {
+    return v ? v->Pos.X : mesh_vertices.size() ? mesh_vertices[0]->x() : 0.;
+  }
+  virtual double y() const 
+  {
+    return v ? v->Pos.Y : mesh_vertices.size() ? mesh_vertices[0]->y() : 0.;
+  }
+  virtual double z() const 
+  {
+    return v ? v->Pos.Z : mesh_vertices.size() ? mesh_vertices[0]->z() : 0.;
   }
-  virtual double x() const {return v ? v->Pos.X : 0.;}
-  virtual double y() const {return v ? v->Pos.Y : 0.;}
-  virtual double z() const {return v ? v->Pos.Z : 0.;}
   void * getNativePtr() const {return v;}
-  virtual double prescribedMeshSizeAtVertex () const {return v ? v->lc : 0.;}
+  virtual double prescribedMeshSizeAtVertex() const {return v ? v->lc : 0.;}
  protected:
   Vertex *v;
 };
diff --git a/Graphics/CreateFile.cpp b/Graphics/CreateFile.cpp
index a6b7e6ae22..21aa08bac5 100644
--- a/Graphics/CreateFile.cpp
+++ b/Graphics/CreateFile.cpp
@@ -1,4 +1,4 @@
-// $Id: CreateFile.cpp,v 1.85 2006-08-05 10:05:45 geuzaine Exp $
+// $Id: CreateFile.cpp,v 1.86 2006-08-06 22:58:49 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -153,15 +153,25 @@ void CreateOutputFile(char *filename, int format)
     break;
 
   case FORMAT_MSH:
-    GMODEL->writeMSH(name);
+    GMODEL->writeMSH(name, CTX.mesh.msh_file_version, CTX.mesh.save_all,
+		     CTX.mesh.scaling_factor);
+    break;
+
+  case FORMAT_STL:
+    GMODEL->writeSTL(name, CTX.mesh.scaling_factor);
+    break;
+
+  case FORMAT_VRML:
+    GMODEL->writeVRML(name, CTX.mesh.scaling_factor);
     break;
 
   case FORMAT_UNV:
+    GMODEL->writeUNV(name, CTX.mesh.scaling_factor);
+    break;
+
   case FORMAT_P3D:
   case FORMAT_DMG:
-  case FORMAT_STL:
   case FORMAT_GREF:
-  case FORMAT_VRML:
     Print_Mesh(name, format);
     break;
 
diff --git a/Graphics/Geom.cpp b/Graphics/Geom.cpp
index dee0ea5daa..f63c4289f1 100644
--- a/Graphics/Geom.cpp
+++ b/Graphics/Geom.cpp
@@ -1,4 +1,4 @@
-// $Id: Geom.cpp,v 1.103 2006-08-05 10:05:45 geuzaine Exp $
+// $Id: Geom.cpp,v 1.104 2006-08-06 22:58:49 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -288,7 +288,7 @@ void HighlightEntity(GVertex *v, GEdge *c, GFace *s, int permanent)
   if(v) HighlightEntity(v, permanent);
   else if(c) HighlightEntity(c, permanent);
   else if(s) HighlightEntity(s, permanent);
-  else if(!permanent) Msg(STATUS2N, "");
+  else if(!permanent) Msg(STATUS2N, " ");
 }
 
 void HighlightEntityNum(int v, int c, int s, int permanent)
diff --git a/Mesh/DiscreteSurface.cpp b/Mesh/DiscreteSurface.cpp
index 3b1ce9061a..ce8db9341d 100644
--- a/Mesh/DiscreteSurface.cpp
+++ b/Mesh/DiscreteSurface.cpp
@@ -1,4 +1,4 @@
-// $Id: DiscreteSurface.cpp,v 1.41 2006-08-05 10:05:45 geuzaine Exp $
+// $Id: DiscreteSurface.cpp,v 1.42 2006-08-06 22:58:49 geuzaine Exp $
 //
 // Copyright (C) 1997-2006 C. Geuzaine, J.-F. Remacle
 //
@@ -309,7 +309,7 @@ void BDS_To_Mesh_2(Mesh *M)
     }
   }
 
-  Msg(STATUS2N, "");
+  Msg(STATUS2N, " ");
 }
 
 void BDS_To_Mesh(Mesh *M)
-- 
GitLab