diff --git a/Applications/DataExplorer/mainwindow.cpp b/Applications/DataExplorer/mainwindow.cpp index 9ea909a42079866fa8d7b1efd2b039a18d7989f4..61c7062a3493f636572cc3a486582d1d50b30945 100644 --- a/Applications/DataExplorer/mainwindow.cpp +++ b/Applications/DataExplorer/mainwindow.cpp @@ -60,14 +60,15 @@ #include "InSituLib/VtkMappedMeshSource.h" // FileIO includes -#include "FileIO/FEFLOWInterface.h" #include "FileIO/GMSInterface.h" +#include "MeshLib/IO/FEFLOW/FEFLOWMeshInterface.h" #include "MeshLib/IO/Legacy/MeshIO.h" #include "MeshLib/IO/readMeshFromFile.h" #include "FileIO/GMSHInterface.h" #include "FileIO/TetGenInterface.h" #include "FileIO/PetrelInterface.h" #include "FileIO/XmlIO/Qt/XmlGspInterface.h" +#include "GeoLib/IO/FEFLOW/FEFLOWGeoInterface.h" #include "GeoLib/IO/XmlIO/Qt/XmlGmlInterface.h" #include "GeoLib/IO/XmlIO/Qt/XmlStnInterface.h" @@ -521,13 +522,15 @@ void MainWindow::loadFile(ImportFileType::type t, const QString &fileName) { if (fi.suffix().toLower() == "fem") // FEFLOW model files { - FileIO::FEFLOWInterface feflowIO(&_project.getGEOObjects()); + MeshLib::IO::FEFLOWMeshInterface feflowMeshIO; std::unique_ptr<MeshLib::Mesh> mesh( - feflowIO.readFEFLOWFile(fileName.toStdString())); + feflowMeshIO.readFEFLOWFile(fileName.toStdString())); if (mesh) _meshModel->addMesh(std::move(mesh)); else OGSError::box("Failed to load a FEFLOW mesh."); + GeoLib::IO::FEFLOWGeoInterface feflowGeoIO; + feflowGeoIO.readFEFLOWFile(fileName.toStdString(), _project.getGEOObjects()); } settings.setValue("lastOpenedFileDirectory", dir.absolutePath()); diff --git a/Applications/Utils/FileConverter/FEFLOW2OGS.cpp b/Applications/Utils/FileConverter/FEFLOW2OGS.cpp index 6c288f8981e0de381996198fd6b5629b9edce01f..d550b73808e36df567017c24a91190d67a6caa7d 100644 --- a/Applications/Utils/FileConverter/FEFLOW2OGS.cpp +++ b/Applications/Utils/FileConverter/FEFLOW2OGS.cpp @@ -23,7 +23,7 @@ #endif // FileIO -#include "FileIO/FEFLOWInterface.h" +#include "MeshLib/IO/FEFLOW/FEFLOWMeshInterface.h" #include "MeshLib/IO/writeMeshToFile.h" #include "MeshLib/IO/Legacy/MeshIO.h" @@ -64,7 +64,7 @@ int main (int argc, char* argv[]) #endif BaseLib::RunTime run_time; run_time.start(); - FileIO::FEFLOWInterface feflowIO(nullptr); + MeshLib::IO::FEFLOWMeshInterface feflowIO; std::unique_ptr<MeshLib::Mesh const> mesh( feflowIO.readFEFLOWFile(feflow_mesh_arg.getValue())); diff --git a/FileIO/FEFLOWInterface.h b/FileIO/FEFLOWInterface.h deleted file mode 100644 index c924c51f752fa051304f48a8d186076f0c2b5b1d..0000000000000000000000000000000000000000 --- a/FileIO/FEFLOWInterface.h +++ /dev/null @@ -1,166 +0,0 @@ -/** - * \copyright - * Copyright (c) 2012-2016, OpenGeoSys Community (http://www.opengeosys.org) - * Distributed under a Modified BSD License. - * See accompanying file LICENSE.txt or - * http://www.opengeosys.org/project/license - */ - -#ifndef FEFLOWINTERFACE_H_ -#define FEFLOWINTERFACE_H_ - -#include <iosfwd> -#include <memory> -#include <string> -#include <vector> - -//#include "MeshLib/MeshEnums.h" - -class QDomElement; -class QString; - -namespace GeoLib -{ - class GEOObjects; - class Point; - class Polyline; -} - -namespace MeshLib -{ - class Mesh; - class Element; - class Node; - enum class MeshElemType; -} - -namespace FileIO -{ - -/** - * Read FEFLOW model files (*.fem) into OGS data structure. Currently this class supports - * only import of mesh data and some geometry given in Supermesh section. - */ -class FEFLOWInterface -{ -public: - /// Constructor - explicit FEFLOWInterface(GeoLib::GEOObjects* obj = nullptr) - : _geoObjects(obj) - { - } - - /** - * read a FEFLOW Model file (*.fem) in ASCII format (Version 5.4) - * - * This function reads mesh data in addition to geometry data given in Supermesh. - * - * @param filename FEFLOW file name - * @return a pointer to a created OGS mesh - */ - MeshLib::Mesh* readFEFLOWFile(const std::string &filename); - -private: - // CLASS - struct FEM_CLASS - { - unsigned problem_class; - unsigned time_mode; - unsigned orientation; - unsigned dimension; - unsigned n_layers3d; - unsigned saturation_flag; - unsigned save_fsize_rreal; - unsigned save_fsize_creal; - - FEM_CLASS() - { - problem_class = 0; - time_mode = 0; - orientation = 0; - dimension = 0; - n_layers3d = 0; - saturation_flag = 0; - save_fsize_rreal = 0; - save_fsize_creal = 0; - } - }; - - // DIMENSION - struct FEM_DIM - { - std::size_t n_nodes; - std::size_t n_elements; - std::size_t obs; - std::size_t np_cor; - unsigned n_nodes_of_element; - unsigned n_steps; - unsigned icrank; - unsigned upwind; - unsigned optim; - unsigned aquifer_type; - unsigned nwca; - unsigned adaptive_mesh; - unsigned sp_fem_pcs_id; - unsigned sorption_type; - unsigned reaction_type; - unsigned dispersion_type; - - FEM_DIM() - { - n_nodes = 0; - n_elements = 0; - n_nodes_of_element = 0; - n_steps = 0; - icrank = 0; - upwind = 0; - obs = 0; - optim = 0; - aquifer_type = 0; - nwca = 0; - np_cor = 0; - adaptive_mesh = 0; - sp_fem_pcs_id = 0; - sorption_type = 0; - reaction_type = 0; - dispersion_type = 0; - } - }; - - /// read node indices and create a mesh element - MeshLib::Element* readElement(const FEM_DIM &fem_dim, const MeshLib::MeshElemType elem_type, const std::string& line, const std::vector<MeshLib::Node*> &nodes); - - /// read node coordinates - void readNodeCoordinates(std::ifstream &in, const FEM_CLASS &fem_class, const FEM_DIM &fem_dim, std::vector<MeshLib::Node*> &nodes); - - /// read elevation data - void readElevation(std::ifstream &in, const FEM_CLASS &fem_class, const FEM_DIM &fem_dim, std::vector<MeshLib::Node*> &vec_nodes); - - //// parse node lists - std::vector<std::size_t> getIndexList(const std::string &str_ranges); - - /// parse ELEMENTALSETS - void readELEMENTALSETS(std::ifstream &in, std::vector<std::vector<std::size_t>> &vec_elementsets); - - /// read Supermesh data - /// - /// A super mesh is a collection of polygons, lines and points in the 2D plane - /// and will be used for mesh generation and to define the modeling region - void readSuperMesh(std::ifstream &feflow_file, const FEM_CLASS &fem_class, std::vector<GeoLib::Point*>** points, std::vector<GeoLib::Polyline*>** lines); - - //// read point data in Supermesh - void readPoints(QDomElement &nodesEle, const std::string &tag, int dim, std::vector<GeoLib::Point*> &points); - - void setMaterialIDs(FEM_CLASS const& fem_class, - FEM_DIM const& fem_dim, - std::vector<GeoLib::Polyline*>* const& lines, - std::vector<std::vector<std::size_t>> const& vec_elementsets, - std::vector<MeshLib::Element*> const& vec_elements, - std::vector<int> & material_ids); - - //// Geometric objects - GeoLib::GEOObjects* _geoObjects; -}; -} // end namespace FileIO - -#endif /* FEFLOWINTERFACE_H_ */ diff --git a/GeoLib/CMakeLists.txt b/GeoLib/CMakeLists.txt index ce9bfc4975c081dda7836eec31d9c242544ab6a1..2748a2e7a26e58d12a74e80d84f00124b005b001 100644 --- a/GeoLib/CMakeLists.txt +++ b/GeoLib/CMakeLists.txt @@ -13,6 +13,8 @@ set(SOURCES ${SOURCES} ${SOURCES_IO_BASE_XML} ${SOURCES_IO_BOOST_XML}) if(QT4_FOUND) GET_SOURCE_FILES(SOURCES_IO_QT_XML IO/XmlIO/Qt) set(SOURCES ${SOURCES} ${SOURCES_IO_QT_XML}) + GET_SOURCE_FILES(SOURCES_IO_FEFLOW IO/FEFLOW) + SET(SOURCES ${SOURCES} ${SOURCES_IO_FEFLOW}) endif() # Create the library diff --git a/GeoLib/IO/FEFLOW/FEFLOWGeoInterface.cpp b/GeoLib/IO/FEFLOW/FEFLOWGeoInterface.cpp new file mode 100644 index 0000000000000000000000000000000000000000..3275c0fc2c90f1fa2aa7cdcb585e67f3e30b62f8 --- /dev/null +++ b/GeoLib/IO/FEFLOW/FEFLOWGeoInterface.cpp @@ -0,0 +1,220 @@ +/** + * \copyright + * Copyright (c) 2012-2016, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/project/license + */ + +#include "FEFLOWGeoInterface.h" + +#include <cctype> +#include <memory> + +#include <QDomElement> +#include <QString> +#include <QtXml> + +#include <logog/include/logog.hpp> + +#include "BaseLib/FileTools.h" +#include "BaseLib/StringTools.h" + +#include "GeoLib/GEOObjects.h" +#include "GeoLib/Point.h" +#include "GeoLib/Polygon.h" + +namespace GeoLib +{ +namespace IO +{ +void FEFLOWGeoInterface::readFEFLOWFile(const std::string& filename, + GeoLib::GEOObjects& geoObjects) +{ + std::ifstream in(filename.c_str()); + if (!in) + { + ERR("FEFLOWGeoInterface::readFEFLOWFile(): Could not open file %s.", + filename.c_str()); + return; + } + + unsigned dimension = 0; + std::vector<GeoLib::Point*>* points = nullptr; + std::vector<GeoLib::Polyline*>* lines = nullptr; + + bool isXZplane = false; + + std::string line_string; + std::stringstream line_stream; + while (!in.eof()) + { + getline(in, line_string); + //.................................................................... + // CLASS: the version number follows afterward, e.g. CLASS (v.5.313) + if (line_string.find("CLASS") != std::string::npos) + { + getline(in, line_string); + line_stream.str(line_string); + // problem class, time mode, problem orientation, dimension, ... + unsigned dummy = 0; + for (int i = 0; i < 3; i++) + line_stream >> dummy; + line_stream >> dimension; + line_stream.clear(); + } + //.................................................................... + // GRAVITY + else if (line_string.compare("GRAVITY") == 0) + { + getline(in, line_string); + line_stream.str(line_string); + double vec[3] = {}; + line_stream >> vec[0] >> vec[1] >> vec[2]; + if (vec[0] == 0.0 && vec[1] == -1.0 && vec[2] == 0.0) + // x-z plane + isXZplane = true; + line_stream.clear(); + } + //.................................................................... + // SUPERMESH + else if (line_string.compare("SUPERMESH") == 0) + { + readSuperMesh(in, dimension, points, lines); + } + //.................................................................... + } + in.close(); + + std::string project_name( + BaseLib::extractBaseNameWithoutExtension(filename)); + if (points) + geoObjects.addPointVec( + std::unique_ptr<std::vector<GeoLib::Point*>>(points), project_name); + if (lines) + geoObjects.addPolylineVec( + std::unique_ptr<std::vector<GeoLib::Polyline*>>(lines), + project_name); + + if (isXZplane && points) + { + for (auto* pt : *points) + { + (*pt)[2] = (*pt)[1]; + (*pt)[1] = .0; + } + } +} + +void FEFLOWGeoInterface::readPoints(QDomElement& nodesEle, + const std::string& tag, + int dim, + std::vector<GeoLib::Point*>& points) +{ + QDomElement xmlEle = + nodesEle.firstChildElement(QString::fromStdString(tag)); + if (xmlEle.isNull()) + return; + QString str_pt_list1 = xmlEle.text(); + std::istringstream ss(str_pt_list1.toStdString()); + std::string line_str; + while (!ss.eof()) + { + std::getline(ss, line_str); + BaseLib::trim(line_str, ' '); + if (line_str.empty()) + continue; + std::istringstream line_ss(line_str); + std::size_t pt_id = 0; + std::array<double, 3> pt_xyz; + line_ss >> pt_id; + for (int i = 0; i < dim; i++) + line_ss >> pt_xyz[i]; + points[pt_id - 1] = new GeoLib::Point(pt_xyz, pt_id); + } +} + +void FEFLOWGeoInterface::readSuperMesh(std::ifstream& in, + unsigned dimension, + std::vector<GeoLib::Point*>*& points, + std::vector<GeoLib::Polyline*>*& lines) +{ + // get XML strings + std::ostringstream oss; + std::string line_string; + while (true) + { + getline(in, line_string); + BaseLib::trim(line_string); + oss << line_string << "\n"; + if (line_string.find("</supermesh>") != std::string::npos) + break; + } + const QString strXML(oss.str().c_str()); + + // convert string to XML + QDomDocument doc; + if (!doc.setContent(strXML)) + { + ERR("FEFLOWGeoInterface::readSuperMesh(): Illegal XML format error"); + return; + } + + // get geometry data from XML + QDomElement docElem = doc.documentElement(); // #supermesh + // #nodes + points = new std::vector<GeoLib::Point*>(); + QDomElement nodesEle = docElem.firstChildElement("nodes"); + if (nodesEle.isNull()) + return; + + { + const QString str = nodesEle.attribute("count"); + const long n_points = str.toLong(); + points->resize(n_points); + // fixed + readPoints(nodesEle, "fixed", dimension, *points); + readPoints(nodesEle, "linear", dimension, *points); + readPoints(nodesEle, "parabolic", dimension, *points); + } + + // #polygons + lines = new std::vector<GeoLib::Polyline*>(); + QDomElement polygonsEle = docElem.firstChildElement("polygons"); + if (polygonsEle.isNull()) + return; + + { + QDomNode child = polygonsEle.firstChild(); + while (!child.isNull()) + { + if (child.nodeName() != "polygon") + { + child = child.nextSibling(); + continue; + } + QDomElement xmlEle = child.firstChildElement("nodes"); + if (xmlEle.isNull()) + continue; + const QString str = xmlEle.attribute("count"); + const std::size_t n_points = str.toLong(); + QString str_ptId_list = xmlEle.text().simplified(); + { + GeoLib::Polyline* line = new GeoLib::Polyline(*points); + lines->push_back(line); + std::istringstream ss(str_ptId_list.toStdString()); + for (std::size_t i = 0; i < n_points; i++) + { + int pt_id = 0; + ss >> pt_id; + line->addPoint(pt_id - 1); + } + line->addPoint(line->getPointID(0)); + } + child = child.nextSibling(); + } + } +} + +} // IO +} // GeoLib diff --git a/GeoLib/IO/FEFLOW/FEFLOWGeoInterface.h b/GeoLib/IO/FEFLOW/FEFLOWGeoInterface.h new file mode 100644 index 0000000000000000000000000000000000000000..fb0603361c822bdd9b673483c46cc68ec9e2091c --- /dev/null +++ b/GeoLib/IO/FEFLOW/FEFLOWGeoInterface.h @@ -0,0 +1,64 @@ +/** + * \copyright + * Copyright (c) 2012-2016, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/project/license + */ + +#ifndef FEFLOWGEOINTERFACE_H_ +#define FEFLOWGEOINTERFACE_H_ + +#include <iosfwd> +#include <string> +#include <vector> + +class QDomElement; + +namespace GeoLib +{ +class GEOObjects; +class Point; +class Polyline; +} + +namespace GeoLib +{ +namespace IO +{ +/** + * Interface to geometric data in FEFLOW files + */ +class FEFLOWGeoInterface +{ +public: + /** + * read a FEFLOW Model file (*.fem) in ASCII format (Version 5.4) + * + * This function reads geometry data given in Supermesh. + * + * @param filename FEFLOW file name + * @param geo_objects Geometric objects where imported geometry data are + * added + */ + void readFEFLOWFile(const std::string& filename, + GeoLib::GEOObjects& geo_objects); + + /// read points and polylines in Supermesh section + /// + /// A super mesh is a collection of polygons, lines and points in the 2D + /// plane and will be used for mesh generation and to define the modeling + /// region + static void readSuperMesh(std::ifstream& feflow_file, unsigned dimension, + std::vector<GeoLib::Point*>*& points, + std::vector<GeoLib::Polyline*>*& lines); + +private: + //// read point data in Supermesh + static void readPoints(QDomElement& nodesEle, const std::string& tag, + int dim, std::vector<GeoLib::Point*>& points); +}; +} // IO +} // GeoLib + +#endif /* FEFLOWGEOINTERFACE_H_ */ diff --git a/MeshLib/CMakeLists.txt b/MeshLib/CMakeLists.txt index 643a6926a67e2fa7e9d2e51bce9c2f19d1e19476..1b23db93238608f23a042117ff9019c0061cca93 100644 --- a/MeshLib/CMakeLists.txt +++ b/MeshLib/CMakeLists.txt @@ -20,6 +20,11 @@ set(SOURCES ${SOURCES_MESHLIB} ${SOURCES_ELEMENTS} ${SOURCES_EDITING} ${SOURCES_GENERATORS} ${SOURCES_QUALITY} ${SOURCES_SEARCH} ${SOURCES_IO} ${SOURCES_IO_LEGACY} ${SOURCES_IO_VTKIO}) +if(QT4_FOUND) + GET_SOURCE_FILES(SOURCES_IO_FEFLOW IO/FEFLOW) + SET(SOURCES ${SOURCES} ${SOURCES_IO_FEFLOW}) +endif() + # It could be used for other MPI based DDC approach in future. if(OGS_USE_PETSC) GET_SOURCE_FILES(SOURCES_MPI_IO IO/MPI_IO) diff --git a/FileIO/FEFLOWInterface.cpp b/MeshLib/IO/FEFLOW/FEFLOWMeshInterface.cpp similarity index 58% rename from FileIO/FEFLOWInterface.cpp rename to MeshLib/IO/FEFLOW/FEFLOWMeshInterface.cpp index ff8d1f286e7705974d3294b9efd3221b0b2e75ed..4d38d038c95c0e49f4ddc526f4d165f45837ab2c 100644 --- a/FileIO/FEFLOWInterface.cpp +++ b/MeshLib/IO/FEFLOW/FEFLOWMeshInterface.cpp @@ -6,17 +6,17 @@ * http://www.opengeosys.org/project/license */ -#include "FEFLOWInterface.h" +#include "FEFLOWMeshInterface.h" #include <cctype> -#include <QtXml> +#include <memory> #include <logog/include/logog.hpp> #include "BaseLib/FileTools.h" #include "BaseLib/StringTools.h" -#include "GeoLib/GEOObjects.h" +#include "GeoLib/IO/FEFLOW/FEFLOWGeoInterface.h" #include "GeoLib/Point.h" #include "GeoLib/Polygon.h" @@ -24,22 +24,24 @@ #include "MeshLib/Mesh.h" #include "MeshLib/Node.h" -namespace FileIO +namespace MeshLib { - -MeshLib::Mesh* FEFLOWInterface::readFEFLOWFile(const std::string &filename) +namespace IO +{ +MeshLib::Mesh* FEFLOWMeshInterface::readFEFLOWFile(const std::string& filename) { std::ifstream in(filename.c_str()); if (!in) { - ERR("FEFLOWInterface::readFEFLOWFile(): Could not open file %s.", filename.c_str()); + ERR("FEFLOWMeshInterface::readFEFLOWFile(): Could not open file %s.", + filename.c_str()); return nullptr; } FEM_CLASS fem_class; FEM_DIM fem_dim; - std::vector<GeoLib::Point*>* points = NULL; - std::vector<GeoLib::Polyline*>* lines = NULL; + std::vector<GeoLib::Point*>* points = nullptr; + std::vector<GeoLib::Polyline*>* lines = nullptr; bool isXZplane = false; @@ -54,13 +56,17 @@ MeshLib::Mesh* FEFLOWInterface::readFEFLOWFile(const std::string &filename) { getline(in, line_string); //.................................................................... - // CLASS - if (line_string.find("CLASS") != std::string::npos) // the version number follows afterward, e.g. CLASS (v.5.313) + // CLASS: the version number follows afterward, e.g. CLASS (v.5.313) + if (line_string.find("CLASS") != std::string::npos) { getline(in, line_string); line_stream.str(line_string); - // problem class, time mode, problem orientation, dimension, nr. layers for 3D, saturation switch, precision of results, precision of coordinates - line_stream >> fem_class.problem_class >> fem_class.time_mode >> fem_class.orientation >> fem_class.dimension >> fem_class.n_layers3d; + // problem class, time mode, problem orientation, dimension, nr. + // layers for 3D, saturation switch, precision of results, precision + // of coordinates + line_stream >> fem_class.problem_class >> fem_class.time_mode >> + fem_class.orientation >> fem_class.dimension >> + fem_class.n_layers3d; line_stream.clear(); } //.................................................................... @@ -70,13 +76,18 @@ MeshLib::Mesh* FEFLOWInterface::readFEFLOWFile(const std::string &filename) // DIMENS getline(in, line_string); line_stream.str(line_string); - line_stream >> fem_dim.n_nodes >> fem_dim.n_elements >> fem_dim.n_nodes_of_element >> std::ws; - // create node pointers with dummy coordinates to create element objects. + line_stream >> fem_dim.n_nodes >> fem_dim.n_elements >> + fem_dim.n_nodes_of_element >> std::ws; + // create node pointers with dummy coordinates to create element + // objects. // True coordinates are set later in COOR and ELEV_I. vec_nodes.resize(fem_dim.n_nodes); std::size_t count = 0; double dummy_coords[3] = {}; - std::generate(vec_nodes.begin(), vec_nodes.end(), [&]() { return new MeshLib::Node(dummy_coords, count++); }); + std::generate(vec_nodes.begin(), vec_nodes.end(), [&]() + { + return new MeshLib::Node(dummy_coords, count++); + }); line_stream.clear(); } //.................................................................... @@ -90,27 +101,39 @@ MeshLib::Mesh* FEFLOWInterface::readFEFLOWFile(const std::string &filename) eleType = MeshLib::MeshElemType::LINE; else if (fem_dim.n_nodes_of_element == 3) eleType = MeshLib::MeshElemType::TRIANGLE; - else if (fem_dim.n_nodes_of_element == 4 && fem_class.dimension == 2) + else if (fem_dim.n_nodes_of_element == 4 && + fem_class.dimension == 2) eleType = MeshLib::MeshElemType::QUAD; - else if (fem_dim.n_nodes_of_element == 4 && fem_class.dimension == 3) + else if (fem_dim.n_nodes_of_element == 4 && + fem_class.dimension == 3) eleType = MeshLib::MeshElemType::TETRAHEDRON; - else if (fem_dim.n_nodes_of_element == 6 && fem_class.dimension == 3) + else if (fem_dim.n_nodes_of_element == 6 && + fem_class.dimension == 3) eleType = MeshLib::MeshElemType::PRISM; - else if (fem_dim.n_nodes_of_element == 8 && fem_class.dimension == 3) + else if (fem_dim.n_nodes_of_element == 8 && + fem_class.dimension == 3) eleType = MeshLib::MeshElemType::HEXAHEDRON; - if (eleType == MeshLib::MeshElemType::INVALID) { - ERR("FEFLOWInterface::readFEFLOWFile(): Unsupported element type with the number of node = %d and dim = %d", fem_dim.n_nodes_of_element, fem_class.dimension); - std::for_each(vec_nodes.begin(), vec_nodes.end(), [](MeshLib::Node* nod) { delete nod;}); + if (eleType == MeshLib::MeshElemType::INVALID) + { + ERR("FEFLOWInterface::readFEFLOWFile(): Unsupported element " + "type with the number of node = %d and dim = %d", + fem_dim.n_nodes_of_element, fem_class.dimension); + std::for_each(vec_nodes.begin(), vec_nodes.end(), + [](MeshLib::Node* nod) + { + delete nod; + }); vec_nodes.clear(); return nullptr; } vec_elements.reserve(fem_dim.n_elements); - for (std::size_t i=0; i<fem_dim.n_elements; i++) + for (std::size_t i = 0; i < fem_dim.n_elements; i++) { getline(in, line_string); - vec_elements.push_back(readElement(fem_dim, eleType, line_string, vec_nodes)); + vec_elements.push_back( + readElement(fem_dim, eleType, line_string, vec_nodes)); } } //.................................................................... @@ -133,7 +156,7 @@ MeshLib::Mesh* FEFLOWInterface::readFEFLOWFile(const std::string &filename) { getline(in, line_string); line_stream.str(line_string); - double vec[3] = { }; + double vec[3] = {}; line_stream >> vec[0] >> vec[1] >> vec[2]; if (vec[0] == 0.0 && vec[1] == -1.0 && vec[2] == 0.0) // x-z plane @@ -150,34 +173,29 @@ MeshLib::Mesh* FEFLOWInterface::readFEFLOWFile(const std::string &filename) // SUPERMESH else if (line_string.compare("SUPERMESH") == 0) { - readSuperMesh(in, fem_class, &points, &lines); + GeoLib::IO::FEFLOWGeoInterface::readSuperMesh( + in, fem_class.dimension, points, lines); } //.................................................................... } in.close(); + INFO("Create mesh"); std::string project_name( BaseLib::extractBaseNameWithoutExtension(filename)); - if (_geoObjects && points) - _geoObjects->addPointVec( - std::unique_ptr<std::vector<GeoLib::Point*>>(points), project_name); - if (_geoObjects && lines) - _geoObjects->addPolylineVec( - std::unique_ptr<std::vector<GeoLib::Polyline*>>(lines), - project_name); - - INFO("Create mesh"); auto mesh(std::unique_ptr<MeshLib::Mesh>( new MeshLib::Mesh(project_name, vec_nodes, vec_elements))); INFO("Set values for material property."); auto opt_material_ids(mesh->getProperties().createNewPropertyVector<int>( - "MaterialIDs", MeshLib::MeshItemType::Cell, 1) - ); - if (!opt_material_ids) { + "MaterialIDs", MeshLib::MeshItemType::Cell, 1)); + if (!opt_material_ids) + { WARN("Could not create PropertyVector for MaterialIDs in Mesh."); - } else { + } + else + { setMaterialIDs(fem_class, fem_dim, lines, vec_elementsets, vec_elements, - *opt_material_ids); + *opt_material_ids); } if (isXZplane) @@ -200,17 +218,22 @@ MeshLib::Mesh* FEFLOWInterface::readFEFLOWFile(const std::string &filename) return mesh.release(); } - -void FEFLOWInterface::readNodeCoordinates(std::ifstream &in, const FEM_CLASS &fem_class, const FEM_DIM &fem_dim, std::vector<MeshLib::Node*> &vec_nodes) +void FEFLOWMeshInterface::readNodeCoordinates( + std::ifstream& in, const FEM_CLASS& fem_class, const FEM_DIM& fem_dim, + std::vector<MeshLib::Node*>& vec_nodes) { - const std::size_t no_nodes_per_layer = (fem_class.dimension == 2) ? fem_dim.n_nodes : fem_dim.n_nodes / (fem_class.n_layers3d + 1); - assert(no_nodes_per_layer>0); - const std::size_t n_lines = (no_nodes_per_layer-1) / 12 + 1; - const std::size_t n_layers = (fem_class.dimension == 3) ? fem_class.n_layers3d + 1 : 1; + const std::size_t no_nodes_per_layer = + (fem_class.dimension == 2) + ? fem_dim.n_nodes + : fem_dim.n_nodes / (fem_class.n_layers3d + 1); + assert(no_nodes_per_layer > 0); + const std::size_t n_lines = (no_nodes_per_layer - 1) / 12 + 1; + const std::size_t n_layers = + (fem_class.dimension == 3) ? fem_class.n_layers3d + 1 : 1; std::string line_string; std::stringstream line_stream; double x; - char dummy_char; // for comma(,) + char dummy_char; // for comma(,) // x, y for (unsigned k = 0; k < 2; k++) { @@ -239,27 +262,34 @@ void FEFLOWInterface::readNodeCoordinates(std::ifstream &in, const FEM_CLASS &fe } } -std::vector<std::size_t> FEFLOWInterface::getIndexList(const std::string &str_ranges) +std::vector<std::size_t> FEFLOWMeshInterface::getIndexList( + const std::string& str_ranges) { std::vector<std::size_t> vec_node_IDs; // insert space before and after minus for splitting - std::string str_ranges2(BaseLib::replaceString("-", " # ", str_ranges)); + std::string str_ranges2(BaseLib::replaceString("-", " # ", str_ranges)); BaseLib::trim(str_ranges2); auto splitted_str = BaseLib::splitString(str_ranges2, ' '); bool is_range = false; for (auto str : splitted_str) { - if (str.empty()) continue; - if (str[0]=='#') { + if (str.empty()) + continue; + if (str[0] == '#') + { is_range = true; - } else if (is_range) { + } + else if (is_range) + { const std::size_t start = vec_node_IDs.back(); const std::size_t end = BaseLib::str2number<std::size_t>(str); - for (std::size_t i=start+1; i<end+1; i++) + for (std::size_t i = start + 1; i < end + 1; i++) vec_node_IDs.push_back(i); is_range = false; - } else { + } + else + { BaseLib::trim(str); vec_node_IDs.push_back(BaseLib::str2number<std::size_t>(str)); } @@ -268,15 +298,20 @@ std::vector<std::size_t> FEFLOWInterface::getIndexList(const std::string &str_ra return vec_node_IDs; } -void FEFLOWInterface::readElevation(std::ifstream &in, const FEM_CLASS &fem_class, const FEM_DIM &fem_dim, std::vector<MeshLib::Node*> &vec_nodes) +void FEFLOWMeshInterface::readElevation(std::ifstream& in, + const FEM_CLASS& fem_class, + const FEM_DIM& fem_dim, + std::vector<MeshLib::Node*>& vec_nodes) { - const std::size_t no_nodes_per_layer = fem_dim.n_nodes / (fem_class.n_layers3d + 1); + const std::size_t no_nodes_per_layer = + fem_dim.n_nodes / (fem_class.n_layers3d + 1); double z = .0; std::string str_nodeList; std::string line_string; std::stringstream line_stream; std::size_t l = 0; - unsigned mode = 0; // 0: exit, 1: slice no, 2: elevation value, 3: continued line of mode 2 + unsigned mode = 0; // 0: exit, 1: slice no, 2: elevation value, 3: + // continued line of mode 2 int pos_prev_line = 0; while (true) { @@ -289,15 +324,16 @@ void FEFLOWInterface::readElevation(std::ifstream &in, const FEM_CLASS &fem_clas mode = 0; else if (line_string.empty()) continue; - else if (line_string[0]=='\t') + else if (line_string[0] == '\t') mode = 3; - else if (columns.size()==1) + else if (columns.size() == 1) mode = 1; - else // columns.size()>1 + else // columns.size()>1 mode = 2; // process stocked data - if (mode != 3 && !str_nodeList.empty()) { + if (mode != 3 && !str_nodeList.empty()) + { // process previous lines auto vec_nodeIDs = getIndexList(str_nodeList); for (auto n0 : vec_nodeIDs) @@ -308,20 +344,27 @@ void FEFLOWInterface::readElevation(std::ifstream &in, const FEM_CLASS &fem_clas str_nodeList.clear(); } - if (mode == 0) { + if (mode == 0) + { break; - } else if (mode == 1) { + } + else if (mode == 1) + { // slice number l++; - assert(l+1==BaseLib::str2number<std::size_t>(columns.front())); - } else if (mode == 2) { + assert(l + 1 == BaseLib::str2number<std::size_t>(columns.front())); + } + else if (mode == 2) + { // parse current line line_stream.str(line_string); line_stream >> z; getline(line_stream, str_nodeList); BaseLib::trim(str_nodeList, '\t'); line_stream.clear(); - } else if (mode == 3) { + } + else if (mode == 3) + { // continue reading node range BaseLib::trim(line_string, '\t'); str_nodeList += " " + line_string; @@ -333,9 +376,9 @@ void FEFLOWInterface::readElevation(std::ifstream &in, const FEM_CLASS &fem_clas in.seekg(pos_prev_line); } -MeshLib::Element* FEFLOWInterface::readElement(const FEM_DIM &fem_dim, - const MeshLib::MeshElemType elem_type, const std::string& line, - const std::vector<MeshLib::Node*> &nodes) +MeshLib::Element* FEFLOWMeshInterface::readElement( + const FEM_DIM& fem_dim, const MeshLib::MeshElemType elem_type, + const std::string& line, const std::vector<MeshLib::Node*>& nodes) { std::stringstream ss(line); @@ -348,14 +391,15 @@ MeshLib::Element* FEFLOWInterface::readElement(const FEM_DIM &fem_dim, { default: for (unsigned k(0); k < fem_dim.n_nodes_of_element; ++k) - ele_nodes[k] = nodes[idx[k]-1]; + ele_nodes[k] = nodes[idx[k] - 1]; break; case MeshLib::MeshElemType::HEXAHEDRON: case MeshLib::MeshElemType::PRISM: - const unsigned n_half_nodes = fem_dim.n_nodes_of_element/2; - for (unsigned k(0); k < n_half_nodes; ++k) { - ele_nodes[k] = nodes[idx[k+n_half_nodes]-1]; - ele_nodes[k+n_half_nodes] = nodes[idx[k]-1]; + const unsigned n_half_nodes = fem_dim.n_nodes_of_element / 2; + for (unsigned k(0); k < n_half_nodes; ++k) + { + ele_nodes[k] = nodes[idx[k + n_half_nodes] - 1]; + ele_nodes[k + n_half_nodes] = nodes[idx[k] - 1]; } break; } @@ -380,36 +424,16 @@ MeshLib::Element* FEFLOWInterface::readElement(const FEM_DIM &fem_dim, } } -void FEFLOWInterface::readPoints(QDomElement &nodesEle, const std::string &tag, int dim, std::vector<GeoLib::Point*> &points) +void FEFLOWMeshInterface::readELEMENTALSETS( + std::ifstream& in, std::vector<std::vector<std::size_t>>& vec_elementsets) { - QDomElement xmlEle = nodesEle.firstChildElement(QString::fromStdString(tag)); - if (xmlEle.isNull()) - return; - QString str_pt_list1 = xmlEle.text(); - std::istringstream ss(str_pt_list1.toStdString()); - std::string line_str; - while (!ss.eof()) + auto compressSpaces = [](std::string const& str) { - std::getline(ss, line_str); - BaseLib::trim(line_str, ' '); - if (line_str.empty()) continue; - std::istringstream line_ss(line_str); - std::size_t pt_id = 0; - std::array<double,3> pt_xyz; - line_ss >> pt_id; - for (int i = 0; i < dim; i++) - line_ss >> pt_xyz[i]; - points[pt_id - 1] = new GeoLib::Point(pt_xyz, pt_id); - } -} - -void FEFLOWInterface::readELEMENTALSETS(std::ifstream &in, std::vector<std::vector<std::size_t>> &vec_elementsets) -{ - auto compressSpaces = [](std::string const& str) { std::stringstream ss(str); std::string new_str; std::string word; - while (ss) { + while (ss) + { ss >> word; new_str += " " + word; } @@ -426,45 +450,55 @@ void FEFLOWInterface::readELEMENTALSETS(std::ifstream &in, std::vector<std::vect unsigned mode = 0; if (!in) - mode = 0; // reached the end of the file + mode = 0; // reached the end of the file else if (line_string.empty()) - continue; // skip and see what comes next + continue; // skip and see what comes next else if (std::isalpha(line_string[0])) - mode = 0; // reached the next section + mode = 0; // reached the next section else if (line_string[0] == ' ') - mode = 1; // start of the element set definition + mode = 1; // start of the element set definition else if (line_string[0] == '\t') - mode = 2; // continue the definition + mode = 2; // continue the definition else { - ERR("Failed during parsing of an ELEMENTALSETS section in a FEFLOW file"); + ERR("Failed during parsing of an ELEMENTALSETS section in a FEFLOW " + "file"); break; } - if (mode!=2 && !str_idList.empty()) { + if (mode != 2 && !str_idList.empty()) + { vec_elementsets.push_back(getIndexList(str_idList)); str_idList.clear(); } - if (mode == 0) { + if (mode == 0) + { break; - } else if (mode == 1) { + } + else if (mode == 1) + { // starting a new set std::string set_name; std::string ids; BaseLib::trim(line_string, ' '); - if (line_string[0]=='"') { // multiple words + if (line_string[0] == '"') + { // multiple words auto pos = line_string.find_last_of('"'); - set_name = line_string.substr(1, pos-1); // without quotation - ids = line_string.substr(pos+1); - } else { // single word + set_name = line_string.substr(1, pos - 1); // without quotation + ids = line_string.substr(pos + 1); + } + else + { // single word auto pos = line_string.find_first_of(' '); set_name = line_string.substr(0, pos); - ids = line_string.substr(pos+1); + ids = line_string.substr(pos + 1); } INFO("Found an element group - %s", set_name.data()); str_idList += compressSpaces(ids); - } else { + } + else + { // continue reading a element ids BaseLib::trim(line_string, '\t'); str_idList += compressSpaces(line_string); @@ -473,105 +507,28 @@ void FEFLOWInterface::readELEMENTALSETS(std::ifstream &in, std::vector<std::vect // move stream position to previous line if (std::isalpha(line_string[0])) in.seekg(pos_prev_line); - -} - -// -void FEFLOWInterface::readSuperMesh(std::ifstream &in, const FEM_CLASS &fem_class, std::vector<GeoLib::Point*>** p_points, std::vector<GeoLib::Polyline*>** p_lines) -{ - // get XML strings - std::ostringstream oss; - std::string line_string; - while (true) - { - getline(in, line_string); - BaseLib::trim(line_string); - oss << line_string << "\n"; - if (line_string.find("</supermesh>") != std::string::npos) - break; - } - const QString strXML(oss.str().c_str()); - - // convert string to XML - QDomDocument doc; - if (!doc.setContent(strXML)) - { - ERR("FEFLOWInterface::readSuperMesh(): Illegal XML format error"); - return; - } - - // get geometry data from XML - QDomElement docElem = doc.documentElement(); // #supermesh - // #nodes - *p_points = new std::vector<GeoLib::Point*>(); - std::vector<GeoLib::Point*>* points = *p_points; - QDomElement nodesEle = docElem.firstChildElement("nodes"); - if (nodesEle.isNull()) - return; - - { - const QString str = nodesEle.attribute("count"); - const long n_points = str.toLong(); - points->resize(n_points); - //fixed - readPoints(nodesEle, "fixed", fem_class.dimension, *points); - readPoints(nodesEle, "linear", fem_class.dimension, *points); - readPoints(nodesEle, "parabolic", fem_class.dimension, *points); - } - - // #polygons - *p_lines = new std::vector<GeoLib::Polyline*>(); - std::vector<GeoLib::Polyline*>* lines = *p_lines; - QDomElement polygonsEle = docElem.firstChildElement("polygons"); - if (polygonsEle.isNull()) - return; - - { - QDomNode child = polygonsEle.firstChild(); - while (!child.isNull()) - { - if (child.nodeName() != "polygon") - { - child = child.nextSibling(); - continue; - } - QDomElement xmlEle = child.firstChildElement("nodes"); - if (xmlEle.isNull()) - continue; - const QString str = xmlEle.attribute("count"); - const std::size_t n_points = str.toLong(); - QString str_ptId_list = xmlEle.text().simplified(); - { - GeoLib::Polyline* line = new GeoLib::Polyline(*points); - lines->push_back(line); - std::istringstream ss(str_ptId_list.toStdString()); - for (std::size_t i = 0; i < n_points; i++) - { - int pt_id = 0; - ss >> pt_id; - line->addPoint(pt_id - 1); - } - line->addPoint(line->getPointID(0)); - } - child = child.nextSibling(); - } - } } -void FEFLOWInterface::setMaterialIDs(FEM_CLASS const& fem_class, +void FEFLOWMeshInterface::setMaterialIDs( + FEM_CLASS const& fem_class, FEM_DIM const& fem_dim, std::vector<GeoLib::Polyline*>* const& lines, std::vector<std::vector<std::size_t>> const& vec_elementsets, std::vector<MeshLib::Element*> const& vec_elements, - std::vector<int> & material_ids) + std::vector<int>& material_ids) { - if (!vec_elementsets.empty()) { - for (std::size_t matid=0; matid<vec_elementsets.size(); ++matid) { - auto &eids = vec_elementsets[matid]; + if (!vec_elementsets.empty()) + { + for (std::size_t matid = 0; matid < vec_elementsets.size(); ++matid) + { + auto& eids = vec_elementsets[matid]; for (auto eid : eids) - material_ids[eid-1] = matid; // Element IDs given by FEFLOW starts from one! + material_ids[eid - 1] = + matid; // Element IDs given by FEFLOW starts from one! } - } else if (lines && !lines->empty()) { + } + else if (lines && !lines->empty()) + { for (std::size_t i = 0; i < vec_elements.size(); ++i) { MeshLib::Element const* e = vec_elements[i]; @@ -592,13 +549,16 @@ void FEFLOWInterface::setMaterialIDs(FEM_CLASS const& fem_class, } material_ids[i] = matId; } - } else if (fem_class.n_layers3d>0) { - const std::size_t no_nodes_per_layer = fem_dim.n_nodes / (fem_class.n_layers3d + 1); + } + else if (fem_class.n_layers3d > 0) + { + const std::size_t no_nodes_per_layer = + fem_dim.n_nodes / (fem_class.n_layers3d + 1); for (std::size_t i = 0; i < vec_elements.size(); i++) { MeshLib::Element* e = vec_elements[i]; unsigned e_min_nodeID = std::numeric_limits<unsigned>::max(); - for (std::size_t j=0; j<e->getNBaseNodes(); j++) + for (std::size_t j = 0; j < e->getNBaseNodes(); j++) e_min_nodeID = std::min(e_min_nodeID, e->getNodeIndex(j)); std::size_t layer_id = e_min_nodeID / no_nodes_per_layer; material_ids[i] = layer_id; @@ -606,4 +566,5 @@ void FEFLOWInterface::setMaterialIDs(FEM_CLASS const& fem_class, } } -} // end namespace FileIO +} // IO +} // MeshLib diff --git a/MeshLib/IO/FEFLOW/FEFLOWMeshInterface.h b/MeshLib/IO/FEFLOW/FEFLOWMeshInterface.h new file mode 100644 index 0000000000000000000000000000000000000000..59654d57fcbd047ab08aa3854965914854af8b51 --- /dev/null +++ b/MeshLib/IO/FEFLOW/FEFLOWMeshInterface.h @@ -0,0 +1,124 @@ +/** + * \copyright + * Copyright (c) 2012-2016, OpenGeoSys Community (http://www.opengeosys.org) + * Distributed under a Modified BSD License. + * See accompanying file LICENSE.txt or + * http://www.opengeosys.org/project/license + */ + +#ifndef FEFLOWMESHINTERFACE_H_ +#define FEFLOWMESHINTERFACE_H_ + +#include <iosfwd> +#include <string> +#include <vector> + +namespace GeoLib +{ +class Point; +class Polyline; +} + +namespace MeshLib +{ +class Mesh; +class Element; +class Node; +enum class MeshElemType; +} + +namespace MeshLib +{ +namespace IO +{ +/** + * Read FEFLOW model files (*.fem) into OGS data structure. Currently this class + * supports only import of mesh data and some geometry given in Supermesh section. + */ +class FEFLOWMeshInterface +{ +public: + /** + * read a FEFLOW Model file (*.fem) in ASCII format (Version 5.4) + * + * This function reads mesh data in addition to geometry data given in + * Supermesh. + * + * @param filename FEFLOW file name + * @return a pointer to a created OGS mesh + */ + MeshLib::Mesh* readFEFLOWFile(const std::string& filename); + +private: + // CLASS + struct FEM_CLASS + { + unsigned problem_class = 0; + unsigned time_mode = 0; + unsigned orientation = 0; + unsigned dimension = 0; + unsigned n_layers3d = 0; + unsigned saturation_flag = 0; + unsigned save_fsize_rreal = 0; + unsigned save_fsize_creal = 0; + }; + + // DIMENSION + struct FEM_DIM + { + std::size_t n_nodes = 0; + std::size_t n_elements = 0; + std::size_t obs = 0; + std::size_t np_cor = 0; + unsigned n_nodes_of_element = 0; + unsigned n_steps = 0; + unsigned icrank = 0; + unsigned upwind = 0; + unsigned optim = 0; + unsigned aquifer_type = 0; + unsigned nwca = 0; + unsigned adaptive_mesh = 0; + unsigned sp_fem_pcs_id = 0; + unsigned sorption_type = 0; + unsigned reaction_type = 0; + unsigned dispersion_type = 0; + }; + + /// read node indices and create a mesh element + MeshLib::Element* readElement(const FEM_DIM& fem_dim, + const MeshLib::MeshElemType elem_type, + const std::string& line, + const std::vector<MeshLib::Node*>& nodes); + + /// read node coordinates + void readNodeCoordinates(std::ifstream& in, + const FEM_CLASS& fem_class, + const FEM_DIM& fem_dim, + std::vector<MeshLib::Node*>& nodes); + + /// read elevation data + void readElevation(std::ifstream& in, + const FEM_CLASS& fem_class, + const FEM_DIM& fem_dim, + std::vector<MeshLib::Node*>& vec_nodes); + + //// parse node lists + std::vector<std::size_t> getIndexList(const std::string& str_ranges); + + /// parse ELEMENTALSETS + void readELEMENTALSETS( + std::ifstream& in, + std::vector<std::vector<std::size_t>>& vec_elementsets); + + void setMaterialIDs( + FEM_CLASS const& fem_class, + FEM_DIM const& fem_dim, + std::vector<GeoLib::Polyline*>* const& lines, + std::vector<std::vector<std::size_t>> const& vec_elementsets, + std::vector<MeshLib::Element*> const& vec_elements, + std::vector<int>& material_ids); +}; +} // IO +} // MeshLib + +#endif /* FEFLOWMESHINTERFACE_H_ */