// SMESH DriverMED : driver to read and write 'med' files // // Copyright (C) 2003 OPEN CASCADE, EADS/CCR, LIP6, CEA/DEN, // CEDRAT, EDF R&D, LEG, PRINCIPIA R&D, BUREAU VERITAS // // This library is free software; you can redistribute it and/or // modify it under the terms of the GNU Lesser General Public // License as published by the Free Software Foundation; either // version 2.1 of the License. // // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, write to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA // // See http://www.opencascade.org/SALOME/ or email : webmaster.salome@opencascade.org // // // // File : DriverMED_R_SMESHDS_Mesh.cxx // Module : SMESH #include "DriverMED_R_SMESHDS_Mesh.h" #include "DriverMED_R_SMDS_Mesh.h" #include "utilities.h" #include "DriverMED_Family.h" #include "SMESHDS_Group.hxx" #include "MEDA_Wrapper.hxx" #include "MED_Utilities.hxx" #include #define _EDF_NODE_IDS_ DriverMED_R_SMESHDS_Mesh::DriverMED_R_SMESHDS_Mesh() : myMesh (NULL), myFile (""), myFileId (-1), myMeshId (-1) { } DriverMED_R_SMESHDS_Mesh::~DriverMED_R_SMESHDS_Mesh() { // map::iterator aFamsIter = myFamilies.begin(); // for (; aFamsIter != myFamilies.end(); aFamsIter++) // { // delete (*aFamsIter).second; // } } void DriverMED_R_SMESHDS_Mesh::SetMesh(SMDS_Mesh * aMesh) { myMesh = aMesh; } void DriverMED_R_SMESHDS_Mesh::SetFile(string aFile) { myFile = aFile; } void DriverMED_R_SMESHDS_Mesh::SetFileId(med_idt aFileId) { myFileId = aFileId; } void DriverMED_R_SMESHDS_Mesh::SetMeshId(int aMeshId) { myMeshId = aMeshId; } void DriverMED_R_SMESHDS_Mesh::SetMeshName(string theMeshName) { myMeshName = theMeshName; } void DriverMED_R_SMESHDS_Mesh::Read() { string myClass = string("SMDS_Mesh"); string myExtension = string("MED"); DriverMED_R_SMDS_Mesh *myReader = new DriverMED_R_SMDS_Mesh; myReader->SetMesh(myMesh); myReader->SetMeshId(myMeshId); myReader->SetFile(myFile); myReader->SetFileId(-1); myReader->Read(); } void DriverMED_R_SMESHDS_Mesh::Add() { string myClass = string("SMDS_Mesh"); string myExtension = string("MED"); DriverMED_R_SMDS_Mesh *myReader = new DriverMED_R_SMDS_Mesh; myReader->SetMesh(myMesh); myReader->SetMeshId(myMeshId); SCRUTE(myFileId); myReader->SetFileId(myFileId); myReader->Read(); } static const SMDS_MeshNode* FindNode(const SMDS_Mesh* theMesh, med_int theId){ const SMDS_MeshNode* aNode = theMesh->FindNode(theId); if(aNode) return aNode; EXCEPTION(runtime_error,"SMDS_Mesh::FindNode - cannot find a SMDS_MeshNode for ID = "< med_float GetCoord(MEDA::PNodeInfo& thePNodeInfo, med_int theElemId){ return thePNodeInfo->GetNodeCoord(theElemId,TheCoordId); } template<> med_float GetCoord(MEDA::PNodeInfo& thePNodeInfo, med_int theElemId){ return 0.0; } static TGetCoord aXYZGetCoord[3] = { &GetCoord, &GetCoord, &GetCoord }; static TGetCoord aXYGetCoord[3] = { &GetCoord, &GetCoord, &GetCoord }; static TGetCoord aYZGetCoord[3] = { &GetCoord, &GetCoord, &GetCoord }; static TGetCoord aXZGetCoord[3] = { &GetCoord, &GetCoord, &GetCoord }; static TGetCoord aXGetCoord[3] = { &GetCoord, &GetCoord, &GetCoord }; static TGetCoord aYGetCoord[3] = { &GetCoord, &GetCoord, &GetCoord }; static TGetCoord aZGetCoord[3] = { &GetCoord, &GetCoord, &GetCoord }; class TCoordHelper{ MEDA::PNodeInfo myPNodeInfo; TGetCoord* myGetCoord; public: TCoordHelper(const MEDA::PNodeInfo& thePNodeInfo, TGetCoord* theGetCoord): myPNodeInfo(thePNodeInfo), myGetCoord(theGetCoord) {} virtual ~TCoordHelper(){} med_float GetCoord(med_int theElemId, med_int theCoodId){ return (*myGetCoord[theCoodId])(myPNodeInfo,theElemId); } }; typedef boost::shared_ptr TCoordHelperPtr; DriverMED_R_SMESHDS_Mesh::ReadStatus DriverMED_R_SMESHDS_Mesh::ReadMySelf() { ReadStatus result = DRS_FAIL; try{ using namespace MEDA; myFamilies.clear(); MESSAGE("ReadMySelf - myFile : "<GetName()); if(aMeshName != aMeshInfo->GetName()) continue; result = DRS_OK; // Reading MED families to the temporary structure //------------------------------------------------ med_int aNbFams = aMed.GetNbFamilies(aMeshInfo); MESSAGE("Read " << aNbFams << " families"); for (med_int iFam = 0; iFam < aNbFams; iFam++) { PFamilyInfo aFamilyInfo = aMed.GetFamilyInfo(aMeshInfo, iFam); med_int aFamId = aFamilyInfo->GetId(); MESSAGE("Family " << aFamId << " :"); DriverMED_FamilyPtr aFamily (new DriverMED_Family); med_int aNbGrp = aFamilyInfo->GetNbGroup(); MESSAGE("belong to " << aNbGrp << " groups"); for (med_int iGr = 0; iGr < aNbGrp; iGr++) { string aGroupName = aFamilyInfo->GetGroupName(iGr); MESSAGE(aGroupName); aFamily->AddGroupName(aGroupName); } myFamilies[aFamId] = aFamily; } // Reading MED nodes to the corresponding SMDS structure //------------------------------------------------------ PNodeInfo aNodeInfo = aMed.GetNodeInfo(aMeshInfo); TCoordHelperPtr aCoordHelperPtr; { med_int aMeshDimension = aMeshInfo->GetDim(); bool anIsDimPresent[3] = {false, false, false}; for(med_int iDim = 0; iDim < aMeshDimension; iDim++){ string aDimName = aNodeInfo->GetCoordName(iDim); if(aDimName == "x" || aDimName == "X") anIsDimPresent[eX] = true; else if(aDimName == "y" || aDimName == "Y") anIsDimPresent[eY] = true; else if(aDimName == "z" || aDimName == "Z") anIsDimPresent[eZ] = true; } switch(aMeshDimension){ case 3: aCoordHelperPtr.reset(new TCoordHelper(aNodeInfo,aXYZGetCoord)); break; case 2: if(anIsDimPresent[eY] && anIsDimPresent[eZ]) aCoordHelperPtr.reset(new TCoordHelper(aNodeInfo,aYZGetCoord)); else if(anIsDimPresent[eX] && anIsDimPresent[eZ]) aCoordHelperPtr.reset(new TCoordHelper(aNodeInfo,aXZGetCoord)); else aCoordHelperPtr.reset(new TCoordHelper(aNodeInfo,aXYGetCoord)); break; case 1: if(anIsDimPresent[eY]) aCoordHelperPtr.reset(new TCoordHelper(aNodeInfo,aYGetCoord)); else if(anIsDimPresent[eZ]) aCoordHelperPtr.reset(new TCoordHelper(aNodeInfo,aZGetCoord)); else aCoordHelperPtr.reset(new TCoordHelper(aNodeInfo,aXGetCoord)); break; } } med_booleen anIsNodeNum = aNodeInfo->IsElemNum(); med_int aNbElems = aNodeInfo->GetNbElem(); MESSAGE("ReadMySelf - aNodeInfo->GetNbElem() = "<X()<<", "<Y()<<", "<Z()<GetFamNum(iElem); if (myFamilies.find(aFamNum) != myFamilies.end()) { myFamilies[aFamNum]->AddElement(aNode); myFamilies[aFamNum]->SetType(SMDSAbs_Node); } } // Reading pre information about all MED cells //-------------------------------------------- bool takeNumbers = true; // initially we trust the numbers from file MED::TEntityInfo aEntityInfo = aMed.GetEntityInfo(aMeshInfo); MED::TEntityInfo::iterator anEntityIter = aEntityInfo.begin(); for(; anEntityIter != aEntityInfo.end(); anEntityIter++){ const med_entite_maillage& anEntity = anEntityIter->first; if(anEntity == MED_NOEUD) continue; // Reading MED cells to the corresponding SMDS structure //------------------------------------------------------ const MED::TGeom& aTGeom = anEntityIter->second; MED::TGeom::const_iterator anTGeomIter = aTGeom.begin(); for(; anTGeomIter != aTGeom.end(); anTGeomIter++){ const med_geometrie_element& aGeom = anTGeomIter->first; if(aGeom == MED_POINT1) continue; PCellInfo aCellInfo = aMed.GetCellInfo(aMeshInfo,anEntity,aGeom); med_booleen anIsElemNum = takeNumbers ? aCellInfo->IsElemNum() : MED_FAUX; med_int aNbElems = aCellInfo->GetNbElem(); MESSAGE("ReadMySelf - anEntity = "<AddElement(anElement); myFamilies[aFamNum]->SetType(anElement->GetType()); } } } } } break; } } }catch(const std::exception& exc){ INFOS("Follow exception was cought:\n\t"<GetName()); } } }catch(const std::exception& exc){ INFOS("Follow exception was cought:\n\t"< DriverMED_R_SMESHDS_Mesh::GetGroupNames() { list aResult; set aResGroupNames; map::iterator aFamsIter = myFamilies.begin(); for (; aFamsIter != myFamilies.end(); aFamsIter++) { DriverMED_FamilyPtr aFamily = (*aFamsIter).second; const MED::TStringSet& aGroupNames = aFamily->GetGroupNames(); set::iterator aGrNamesIter = aGroupNames.begin(); for (; aGrNamesIter != aGroupNames.end(); aGrNamesIter++) { string aName = *aGrNamesIter; // Check, if this is a Group or SubMesh name //if (aName.substr(0, 5) == string("Group")) { if (aResGroupNames.find(aName) == aResGroupNames.end()) { aResGroupNames.insert(aName); aResult.push_back(aName); } // } } } return aResult; } void DriverMED_R_SMESHDS_Mesh::GetGroup(SMESHDS_Group* theGroup) { string aGroupName (theGroup->GetStoreName()); MESSAGE("Get Group " << aGroupName); map::iterator aFamsIter = myFamilies.begin(); for (; aFamsIter != myFamilies.end(); aFamsIter++) { DriverMED_FamilyPtr aFamily = (*aFamsIter).second; if (aFamily->MemberOf(aGroupName)) { const set& anElements = aFamily->GetElements(); set::iterator anElemsIter = anElements.begin(); for (; anElemsIter != anElements.end(); anElemsIter++) { theGroup->SMDS_MeshGroup::Add(*anElemsIter); } } } } void DriverMED_R_SMESHDS_Mesh::GetSubMesh (SMESHDS_SubMesh* theSubMesh, const int theId) { char submeshGrpName[ 30 ]; sprintf( submeshGrpName, "SubMesh %d", theId ); string aName (submeshGrpName); map::iterator aFamsIter = myFamilies.begin(); for (; aFamsIter != myFamilies.end(); aFamsIter++) { DriverMED_FamilyPtr aFamily = (*aFamsIter).second; if (aFamily->MemberOf(aName)) { const set& anElements = aFamily->GetElements(); set::iterator anElemsIter = anElements.begin(); if (aFamily->GetType() == SMDSAbs_Node) { for (; anElemsIter != anElements.end(); anElemsIter++) { const SMDS_MeshNode* node = static_cast(*anElemsIter); theSubMesh->AddNode(node); } } else { for (; anElemsIter != anElements.end(); anElemsIter++) { theSubMesh->AddElement(*anElemsIter); } } } } } void DriverMED_R_SMESHDS_Mesh::CreateAllSubMeshes () { SMESHDS_Mesh* aSMESHDSMesh = dynamic_cast(myMesh); if (!aSMESHDSMesh) { EXCEPTION(runtime_error,"Can not cast SMDS_Mesh to SMESHDS_Mesh"); } map::iterator aFamsIter = myFamilies.begin(); for (; aFamsIter != myFamilies.end(); aFamsIter++) { DriverMED_FamilyPtr aFamily = (*aFamsIter).second; MED::TStringSet aGroupNames = aFamily->GetGroupNames(); set::iterator aGrNamesIter = aGroupNames.begin(); for (; aGrNamesIter != aGroupNames.end(); aGrNamesIter++) { string aName = *aGrNamesIter; // Check, if this is a Group or SubMesh name if (aName.substr(0, 7) == string("SubMesh")) { int Id = atoi(string(aName).substr(7).c_str()); set anElements = aFamily->GetElements(); set::iterator anElemsIter = anElements.begin(); if (aFamily->GetType() == SMDSAbs_Node) { for (; anElemsIter != anElements.end(); anElemsIter++) { const SMDS_MeshNode* node = static_cast(*anElemsIter); aSMESHDSMesh->SetNodeInVolume(node, Id); // aSMESHDSMesh->SetNodeOnFace(node, Id); // aSMESHDSMesh->SetNodeOnEdge(node, Id); // aSMESHDSMesh->SetNodeOnVertex(node, Id); } } else { for (; anElemsIter != anElements.end(); anElemsIter++) { aSMESHDSMesh->SetMeshElementOnShape(*anElemsIter, Id); } } } } } }