netgen/libsrc/interface/writeuser.cpp

829 lines
19 KiB
C++
Raw Normal View History

//
// Write user dependent output file
//
#include <mystdlib.h>
#include <myadt.hpp>
#include <linalg.hpp>
#include <csg.hpp>
#include <geometry2d.hpp>
#include <meshing.hpp>
#include "writeuser.hpp"
2021-10-13 21:24:05 +05:00
#include "../general/gzstream.h"
2016-07-10 21:07:36 +05:00
namespace netgen
{
extern MeshingParameters mparam;
2014-08-30 06:15:59 +06:00
2023-10-09 23:41:02 +05:00
Array<UserFormatRegister::UserFormatEntry> UserFormatRegister::entries;
std::map<string, int> UserFormatRegister::format_to_entry_index;
2019-07-09 13:39:16 +05:00
void RegisterUserFormats (NgArray<const char*> & names,
NgArray<const char*> & extensions)
{
2023-10-09 23:41:02 +05:00
for (const auto & entry : UserFormatRegister::entries)
{
2023-10-09 23:41:02 +05:00
names.Append (entry.format.c_str());
extensions.Append (entry.extensions[0].c_str());
}
}
2023-10-09 23:41:02 +05:00
bool WriteUserFormat (const string & format,
const Mesh & mesh,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
2023-10-09 23:41:02 +05:00
if(!UserFormatRegister::HaveFormat(format))
return true;
const auto entry = UserFormatRegister::Get(format);
2023-10-09 23:41:02 +05:00
if(!entry.write)
return true;
2023-10-09 23:41:02 +05:00
(*entry.write)(mesh, filename);
return false;
}
/*
* Neutral mesh format
* points, elements, surface elements
*/
void WriteNeutralFormat (const Mesh & mesh,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
cout << "write neutral, new" << endl;
int np = mesh.GetNP();
int ne = mesh.GetNE();
int nse = mesh.GetNSE();
int nseg = mesh.GetNSeg();
2024-12-27 17:12:59 +05:00
// int i, j;
int inverttets = mparam.inverttets;
int invertsurf = mparam.inverttrigs;
2022-02-17 20:52:07 +05:00
ofstream outfile (filename);
outfile.precision(6);
outfile.setf (ios::fixed, ios::floatfield);
outfile.setf (ios::showpoint);
outfile << np << "\n";
2024-12-27 17:12:59 +05:00
for (int i = 1; i <= np; i++)
{
const Point3d & p = mesh.Point(i);
outfile.width(10);
outfile << p.X() << " ";
outfile.width(9);
outfile << p.Y() << " ";
if (mesh.GetDimension() == 3)
{
outfile.width(9);
outfile << p.Z();
}
outfile << "\n";
}
if (mesh.GetDimension() == 3)
{
outfile << ne << "\n";
2025-01-01 01:26:04 +05:00
/*
2024-12-27 17:12:59 +05:00
for (int i = 1; i <= ne; i++)
{
Element el = mesh.VolumeElement(i);
2025-01-01 01:26:04 +05:00
*/
for (Element el : mesh.VolumeElements())
{
if (inverttets)
el.Invert();
outfile.width(4);
outfile << el.GetIndex() << " ";
2024-12-27 17:12:59 +05:00
for (int j = 1; j <= el.GetNP(); j++)
{
outfile << " ";
outfile.width(8);
outfile << el.PNum(j);
}
outfile << "\n";
}
}
outfile << nse << "\n";
2025-01-01 01:26:04 +05:00
/*
2024-12-27 17:12:59 +05:00
for (int i = 1; i <= nse; i++)
{
Element2d el = mesh.SurfaceElement(i);
2025-01-01 01:26:04 +05:00
*/
for (Element2d el : mesh.SurfaceElements())
{
if (invertsurf)
el.Invert();
outfile.width(4);
outfile << mesh.GetFaceDescriptor (el.GetIndex()).BCProperty() << " ";
2024-12-27 17:12:59 +05:00
for (int j = 1; j <= el.GetNP(); j++)
{
outfile << " ";
outfile.width(8);
outfile << el.PNum(j);
}
outfile << "\n";
}
if (mesh.GetDimension() == 2)
{
outfile << nseg << "\n";
for (int i = 1; i <= nseg; i++)
{
const Segment & seg = mesh.LineSegment(i);
outfile.width(4);
outfile << seg.si << " ";
for (int j = 0; j < seg.GetNP(); j++)
{
outfile << " ";
outfile.width(8);
outfile << seg[j];
}
/*
outfile << " ";
outfile.width(8);
outfile << seg[0];
outfile << " ";
outfile.width(8);
outfile << seg[1];
if (seg[2] != -1)
{
outfile.width(8);
outfile << seg[2];
}
*/
outfile << "\n";
}
}
}
void WriteSurfaceFormat (const Mesh & mesh,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
// surface mesh
int i, j;
cout << "Write Surface Mesh" << endl;
2022-02-17 20:52:07 +05:00
ofstream outfile (filename);
outfile << "surfacemesh" << endl;
outfile << mesh.GetNP() << endl;
for (i = 1; i <= mesh.GetNP(); i++)
{
for (j = 0; j < 3; j++)
{
outfile.width(10);
outfile << mesh.Point(i)(j) << " ";
}
outfile << endl;
}
outfile << mesh.GetNSE() << endl;
for (i = 1; i <= mesh.GetNSE(); i++)
{
for (j = 1; j <= 3; j++)
{
outfile.width(8);
outfile << mesh.SurfaceElement(i).PNum(j);
}
outfile << endl;
}
}
/*
* save surface mesh as STL file
*/
void WriteSTLFormat (const Mesh & mesh,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
cout << "\nWrite STL Surface Mesh" << endl;
2022-02-17 20:52:07 +05:00
auto ext = filename.extension();
ostream *outfile;
2022-02-17 20:52:07 +05:00
if(ext == ".gz")
outfile = new ogzstream(filename);
else
2022-02-17 20:52:07 +05:00
outfile = new ofstream(filename);
int i;
outfile->precision(10);
*outfile << "solid" << endl;
for (i = 1; i <= mesh.GetNSE(); i++)
{
*outfile << "facet normal ";
const Point3d& p1 = mesh.Point(mesh.SurfaceElement(i).PNum(1));
const Point3d& p2 = mesh.Point(mesh.SurfaceElement(i).PNum(2));
const Point3d& p3 = mesh.Point(mesh.SurfaceElement(i).PNum(3));
Vec3d normal = Cross(p2-p1,p3-p1);
if (normal.Length() != 0)
{
normal /= (normal.Length());
}
*outfile << normal.X() << " " << normal.Y() << " " << normal.Z() << "\n";
*outfile << "outer loop\n";
*outfile << "vertex " << p1.X() << " " << p1.Y() << " " << p1.Z() << "\n";
*outfile << "vertex " << p2.X() << " " << p2.Y() << " " << p2.Z() << "\n";
*outfile << "vertex " << p3.X() << " " << p3.Y() << " " << p3.Z() << "\n";
*outfile << "endloop\n";
*outfile << "endfacet\n";
}
*outfile << "endsolid" << endl;
}
/*
* Philippose - 16 August 2010
* Save surface mesh as STL file
* with a separate solid definition
* for each face
* - This helps in splitting up the
* STL into named boundary faces
* when using a third-party mesher
*/
void WriteSTLExtFormat (const Mesh & mesh,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
cout << "\nWrite STL Surface Mesh (with separated boundary faces)" << endl;
2022-02-17 20:52:07 +05:00
auto ext = filename.extension();
ostream *outfile;
2022-02-17 20:52:07 +05:00
if(ext == ".gz")
outfile = new ogzstream(filename);
else
2022-02-17 20:52:07 +05:00
outfile = new ofstream(filename);
outfile->precision(10);
int numBCs = 0;
2019-07-09 13:39:16 +05:00
NgArray<int> faceBCs;
TABLE<int> faceBCMapping;
faceBCs.SetSize(mesh.GetNFD());
faceBCMapping.SetSize(mesh.GetNFD());
faceBCs = -1;
// Collect the BC numbers used in the mesh
for(int faceNr = 1; faceNr <= mesh.GetNFD(); faceNr++)
{
int bcNum = mesh.GetFaceDescriptor(faceNr).BCProperty();
if(faceBCs.Pos(bcNum) < 0)
{
numBCs++;
faceBCs.Set(numBCs,bcNum);
faceBCMapping.Add1(numBCs,faceNr);
}
else
{
faceBCMapping.Add1(faceBCs.Pos(bcNum)+1,faceNr);
}
}
faceBCs.SetSize(numBCs);
faceBCMapping.ChangeSize(numBCs);
// Now actually write the data to file
for(int bcInd = 1; bcInd <= faceBCs.Size(); bcInd++)
{
*outfile << "solid Boundary_" << faceBCs.Elem(bcInd) << "\n";
for(int faceNr = 1;faceNr <= faceBCMapping.EntrySize(bcInd); faceNr++)
{
Array<SurfaceElementIndex> faceSei;
mesh.GetSurfaceElementsOfFace(faceBCMapping.Get(bcInd,faceNr),faceSei);
for (int i = 0; i < faceSei.Size(); i++)
{
*outfile << "facet normal ";
2019-08-09 12:02:50 +05:00
const Point3d& p1 = mesh.Point(mesh[faceSei[i]].PNum(1));
const Point3d& p2 = mesh.Point(mesh[faceSei[i]].PNum(2));
const Point3d& p3 = mesh.Point(mesh[faceSei[i]].PNum(3));
Vec3d normal = Cross(p2-p1,p3-p1);
if (normal.Length() != 0)
{
normal /= (normal.Length());
}
*outfile << normal.X() << " " << normal.Y() << " " << normal.Z() << "\n";
*outfile << "outer loop\n";
*outfile << "vertex " << p1.X() << " " << p1.Y() << " " << p1.Z() << "\n";
*outfile << "vertex " << p2.X() << " " << p2.Y() << " " << p2.Z() << "\n";
*outfile << "vertex " << p3.X() << " " << p3.Y() << " " << p3.Z() << "\n";
*outfile << "endloop\n";
*outfile << "endfacet\n";
}
}
*outfile << "endsolid Boundary_" << faceBCs.Elem(bcInd) << "\n";
}
}
/*
*
* write surface mesh as VRML file
*
*/
void WriteVRMLFormat (const Mesh & mesh,
bool faces,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
if (faces)
{
// Output in VRML, IndexedFaceSet is used
// Bartosz Sawicki <sawickib@ee.pw.edu.pl>
int np = mesh.GetNP();
int nse = mesh.GetNSE();
int i, j;
2022-02-17 20:52:07 +05:00
ofstream outfile (filename);
outfile.precision(6);
outfile.setf (ios::fixed, ios::floatfield);
outfile.setf (ios::showpoint);
outfile << "#VRML V2.0 utf8 \n"
"Background {\n"
" skyColor [1 1 1]\n"
" groundColor [1 1 1]\n"
"}\n"
"Group{ children [\n"
"Shape{ \n"
"appearance Appearance { material Material { }} \n"
"geometry IndexedFaceSet { \n"
"coord Coordinate { point [ \n";
for (i = 1; i <= np; i++)
{
const Point3d & p = mesh.Point(i);
outfile.width(10);
outfile << p.X() << " ";
outfile << p.Y() << " ";
outfile << p.Z() << " \n";
}
outfile << " ] } \n"
"coordIndex [ \n";
for (i = 1; i <= nse; i++)
{
const Element2d & el = mesh.SurfaceElement(i);
for (j = 1; j <= 3; j++)
{
outfile.width(8);
outfile << el.PNum(j)-1;
}
outfile << " -1 \n";
}
outfile << " ] \n";
//define number and RGB definitions of colors
outfile << "color Color { color [1 0 0, 0 1 0, 0 0 1, 1 1 0]} \n"
"colorIndex [\n";
for (i = 1; i <= nse; i++)
{
outfile << mesh.GetFaceDescriptor(mesh.SurfaceElement(i).GetIndex ()).BCProperty();
outfile << endl;
}
outfile << " ] \n"
"colorPerVertex FALSE \n"
"creaseAngle 0 \n"
"solid FALSE \n"
"ccw FALSE \n"
"convex TRUE \n"
"} } # end of Shape\n"
"] }\n";
} /* end of VRMLFACES */
else
{
// Output in VRML, IndexedLineSet is used
// Bartosz Sawicki <sawickib@ee.pw.edu.pl>
int np = mesh.GetNP();
int nse = mesh.GetNSE();
int i, j;
2022-02-17 20:52:07 +05:00
ofstream outfile (filename);
outfile.precision(6);
outfile.setf (ios::fixed, ios::floatfield);
outfile.setf (ios::showpoint);
outfile << "#VRML V2.0 utf8 \n"
"Background {\n"
" skyColor [1 1 1]\n"
" groundColor [1 1 1]\n"
"}\n"
"Group{ children [\n"
"Shape{ \n"
"appearance Appearance { material Material { }} \n"
"geometry IndexedLineSet { \n"
"coord Coordinate { point [ \n";
for (i = 1; i <= np; i++)
{
const Point3d & p = mesh.Point(i);
outfile.width(10);
outfile << p.X() << " ";
outfile << p.Y() << " ";
outfile << p.Z() << " \n";
}
outfile << " ] } \n"
"coordIndex [ \n";
for (i = 1; i <= nse; i++)
{
const Element2d & el = mesh.SurfaceElement(i);
for (j = 1; j <= 3; j++)
{
outfile.width(8);
outfile << el.PNum(j)-1;
}
outfile.width(8);
outfile << el.PNum(1)-1;
outfile << " -1 \n";
}
outfile << " ] \n";
/* Uncomment if you want color mesh
outfile << "color Color { color [1 1 1, 0 1 0, 0 0 1, 1 1 0]} \n"
"colorIndex [\n";
for (i = 1; i <= nse; i++)
{
outfile << mesh.GetFaceDescriptor(mesh.SurfaceElement(i).GetIndex ()).BCProperty();
outfile << endl;
}
outfile << " ] \n"
*/
outfile << "colorPerVertex FALSE \n"
"} } #end of Shape\n"
"] } \n";
}
}
2023-10-09 23:41:02 +05:00
void WriteVRMLFormatLineset (const Mesh & mesh, const filesystem::path & filename)
{
WriteVRMLFormat(mesh, false, filename);
}
void WriteVRMLFormatFaceset (const Mesh & mesh, const filesystem::path & filename)
{
WriteVRMLFormat(mesh, true, filename);
}
/*
* FEPP .. a finite element package developed at University Linz, Austria
*/
void WriteFEPPFormat (const Mesh & mesh,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
2022-02-17 20:52:07 +05:00
ofstream outfile (filename);
if (mesh.GetDimension() == 3)
{
// output for FEPP
int np = mesh.GetNP();
int ne = mesh.GetNE();
int nse = mesh.GetNSE();
2014-12-05 21:00:01 +05:00
// int ns = mesh.GetNFD();
int i, j;
outfile.precision(5);
outfile.setf (ios::fixed, ios::floatfield);
outfile.setf (ios::showpoint);
outfile << "volumemesh4" << endl;
outfile << nse << endl;
for (i = 1; i <= nse; i++)
{
const Element2d & el = mesh.SurfaceElement(i);
// int facenr = mesh.facedecoding.Get(el.GetIndex()).surfnr;
outfile.width(4);
outfile << el.GetIndex() << " ";
outfile.width(4);
// outfile << mesh.GetFaceDescriptor(el.GetIndex()).BCProperty() << " ";
outfile << mesh.GetFaceDescriptor(el.GetIndex()).BCProperty() << " ";
outfile.width(4);
outfile << el.GetNP() << " ";
for (j = 1; j <= el.GetNP(); j++)
{
outfile.width(8);
outfile << el.PNum(j);
}
outfile << "\n";
}
outfile << ne << "\n";
2025-01-01 01:26:04 +05:00
/*
for (i = 1; i <= ne; i++)
{
const Element & el = mesh.VolumeElement(i);
2025-01-01 01:26:04 +05:00
*/
for (const Element & el : mesh.VolumeElements())
{
outfile.width(4);
outfile << el.GetIndex() << " ";
outfile.width(4);
outfile << el.GetNP() << " ";
for (j = 1; j <= el.GetNP(); j++)
{
outfile.width(8);
outfile << el.PNum(j);
}
outfile << "\n";
}
outfile << np << "\n";
for (i = 1; i <= np; i++)
{
const Point3d & p = mesh.Point(i);
outfile.width(10);
outfile << p.X() << " ";
outfile.width(9);
outfile << p.Y() << " ";
outfile.width(9);
outfile << p.Z() << "\n";
}
/*
if (typ == WRITE_FEPPML)
{
int nbn = mesh.mlbetweennodes.Size();
outfile << nbn << "\n";
for (i = 1; i <= nbn; i++)
outfile << mesh.mlbetweennodes.Get(i).I1() << " "
<< mesh.mlbetweennodes.Get(i).I2() << "\n";
// int ncon = mesh.connectedtonode.Size();
// outfile << ncon << "\n";
// for (i = 1; i <= ncon; i++)
// outfile << i << " " << mesh.connectedtonode.Get(i) << endl;
}
*/
2014-08-30 06:15:59 +06:00
/*
// write CSG surfaces
if (&geom && geom.GetNSurf() >= ns)
{
outfile << ns << endl;
for (i = 1; i <= ns; i++)
geom.GetSurface(mesh.GetFaceDescriptor(i).SurfNr())->Print(outfile);
}
else
2014-08-30 06:15:59 +06:00
*/
outfile << "0" << endl;
}
else
{ // 2D fepp format
;
/*
extern SplineGeometry2d * geometry2d;
if (geometry2d)
Save2DMesh (mesh, &geometry2d->GetSplines(), outfile);
else
Save2DMesh (mesh, 0, outfile);
*/
}
}
/*
* Edge element mesh format
* points, elements, edges
*/
void WriteEdgeElementFormat (const Mesh & mesh,
2022-02-17 20:52:07 +05:00
const filesystem::path & filename)
{
cout << "write edge element format" << endl;
const MeshTopology * top = &mesh.GetTopology();
int npoints = mesh.GetNP();
int nelements = mesh.GetNE();
int nsurfelem = mesh.GetNSE();
int nedges = top->GetNEdges();
int inverttets = mparam.inverttets;
int invertsurf = mparam.inverttrigs;
2019-07-09 13:39:16 +05:00
NgArray<int> edges;
2022-02-17 20:52:07 +05:00
ofstream outfile (filename);
outfile.precision(6);
outfile.setf (ios::fixed, ios::floatfield);
outfile.setf (ios::showpoint);
// vertices with coordinates
outfile << npoints << "\n";
2025-01-01 01:26:04 +05:00
for (int i = 1; i <= npoints; i++)
{
const Point3d & p = mesh.Point(i);
outfile.width(10);
outfile << p.X() << " ";
outfile.width(9);
outfile << p.Y() << " ";
outfile.width(9);
outfile << p.Z() << "\n";
}
// element - edge - list
outfile << nelements << " " << nedges << "\n";
2025-01-01 01:26:04 +05:00
/*
for (i = 1; i <= nelements; i++)
{
Element el = mesh.VolumeElement(i);
2025-01-01 01:26:04 +05:00
*/
for (ElementIndex ei : Range(mesh.VolumeElements()))
{
int i = ei-IndexBASE(ei)+1;
Element el = mesh.VolumeElement(ei);
if (inverttets)
el.Invert();
outfile.width(4);
outfile << el.GetIndex() << " ";
outfile.width(8);
outfile << el.GetNP();
2025-01-01 01:26:04 +05:00
for (int j = 1; j <= el.GetNP(); j++)
{
outfile << " ";
outfile.width(8);
outfile << el.PNum(j);
}
2023-07-31 01:29:54 +05:00
// top->GetElementEdges(i,edges);
2025-01-01 01:26:04 +05:00
auto eledges = top->GetEdges(ei);
outfile << endl << " ";
outfile.width(8);
2023-08-05 15:01:01 +05:00
outfile << eledges.Size();
2025-01-01 01:26:04 +05:00
for (int j=1; j <= eledges.Size(); j++)
{
outfile << " ";
outfile.width(8);
2023-08-05 15:01:01 +05:00
outfile << eledges[j-1]+1;
}
outfile << "\n";
// orientation:
top->GetElementEdgeOrientations(i,edges);
outfile << " ";
2025-01-01 01:26:04 +05:00
for (int j=1; j <= edges.Size(); j++)
{
outfile << " ";
outfile.width(8);
outfile << edges[j-1];
}
outfile << "\n";
}
// surface element - edge - list (with boundary conditions)
outfile << nsurfelem << "\n";
2025-01-01 01:26:04 +05:00
for (int i = 1; i <= nsurfelem; i++)
{
2025-01-05 22:19:21 +05:00
SurfaceElementIndex sei(i-1);
Element2d el = mesh[sei];
if (invertsurf)
el.Invert();
outfile.width(4);
outfile << mesh.GetFaceDescriptor (el.GetIndex()).BCProperty() << " ";
outfile.width(8);
outfile << el.GetNP();
2025-01-01 01:26:04 +05:00
for (int j = 1; j <= el.GetNP(); j++)
{
outfile << " ";
outfile.width(8);
outfile << el.PNum(j);
}
2025-01-05 22:19:21 +05:00
// top->GetSurfaceElementEdges(i,edges);
auto edges = top->GetEdges(sei);
outfile << endl << " ";
outfile.width(8);
outfile << edges.Size();
2025-01-05 22:19:21 +05:00
for (int j=0; j < edges.Size(); j++)
{
outfile << " ";
outfile.width(8);
2025-01-05 22:19:21 +05:00
outfile << edges[j]+1;
}
outfile << "\n";
}
2023-07-31 01:29:54 +05:00
// int v1, v2;
// edge - vertex - list
outfile << nedges << "\n";
2025-01-01 01:26:04 +05:00
for (int i=1; i <= nedges; i++)
{
2023-07-31 01:29:54 +05:00
// top->GetEdgeVertices(i,v1,v2);
auto [v1,v2] = top->GetEdgeVertices(i-1);
outfile.width(4);
outfile << v1;
outfile << " ";
outfile.width(8);
outfile << v2 << endl;
}
}
2023-10-09 23:41:02 +05:00
static RegisterUserFormat reg_neutral("Neutral Format", {".mesh"}, ReadFile, WriteNeutralFormat);
static RegisterUserFormat reg_surface("Surface Mesh Format", {".mesh", ".surf"} ,ReadFile, WriteSurfaceFormat);
static RegisterUserFormat reg_stl ("STL Format", {".stl", ".stlb"}, ReadFile, WriteSTLFormat);
static RegisterUserFormat reg_stlx ("STL Extended Format", {".stl", ".stlb"}, nullopt, WriteSTLExtFormat);
static RegisterUserFormat reg_vrml ("VRML Format", {".*"}, nullopt, WriteVRMLFormatLineset);
static RegisterUserFormat reg_vrml_faces ("VRML Format Faces", {".*"}, nullopt, WriteVRMLFormatFaceset);
static RegisterUserFormat reg_fepp ("Fepp Format", {"*"}, nullopt, WriteFEPPFormat);
static RegisterUserFormat reg_edgeelement ("EdgeElement Format", {"*"}, nullopt, WriteEdgeElementFormat);
}