// SMESH SMESH : implementaion of SMESH idl descriptions // // 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.salome-platform.org/ or email : webmaster.salome@opencascade.com // // // // File : StdMeshers_RadialPrism_3D.cxx // Module : SMESH // Created : Fri Oct 20 11:37:07 2006 // Author : Edward AGAPOV (eap) #include "StdMeshers_RadialPrism_3D.hxx" #include "StdMeshers_ProjectionUtils.hxx" #include "StdMeshers_NumberOfLayers.hxx" #include "StdMeshers_LayerDistribution.hxx" #include "StdMeshers_Prism_3D.hxx" #include "StdMeshers_Regular_1D.hxx" #include "SMDS_MeshNode.hxx" #include "SMESHDS_SubMesh.hxx" #include "SMESH_Gen.hxx" #include "SMESH_Mesh.hxx" #include "SMESH_MeshEditor.hxx" #include "SMESH_MesherHelper.hxx" #include "SMESH_subMesh.hxx" #include "utilities.h" #include #include #include #include #include #include #include using namespace std; #define RETURN_BAD_RESULT(msg) { MESSAGE(msg); return false; } #define gpXYZ(n) gp_XYZ(n->X(),n->Y(),n->Z()) typedef StdMeshers_ProjectionUtils TAssocTool; //======================================================================= //function : StdMeshers_RadialPrism_3D //purpose : //======================================================================= StdMeshers_RadialPrism_3D::StdMeshers_RadialPrism_3D(int hypId, int studyId, SMESH_Gen* gen) :SMESH_3D_Algo(hypId, studyId, gen) { _name = "RadialPrism_3D"; _shapeType = (1 << TopAbs_SOLID); // 1 bit per shape type _compatibleHypothesis.push_back("LayerDistribution"); _compatibleHypothesis.push_back("NumberOfLayers"); myNbLayerHypo = 0; myDistributionHypo = 0; } //================================================================================ /*! * \brief Destructor */ //================================================================================ StdMeshers_RadialPrism_3D::~StdMeshers_RadialPrism_3D() {} //======================================================================= //function : CheckHypothesis //purpose : //======================================================================= bool StdMeshers_RadialPrism_3D::CheckHypothesis(SMESH_Mesh& aMesh, const TopoDS_Shape& aShape, SMESH_Hypothesis::Hypothesis_Status& aStatus) { // check aShape that must have 2 shells if ( TAssocTool::Count( aShape, TopAbs_SOLID, 0 ) != 1 || TAssocTool::Count( aShape, TopAbs_SHELL, 0 ) != 2 ) { aStatus = HYP_BAD_GEOMETRY; return false; } myNbLayerHypo = 0; myDistributionHypo = 0; list ::const_iterator itl; const list &hyps = GetUsedHypothesis(aMesh, aShape); if ( hyps.size() == 0 ) { aStatus = SMESH_Hypothesis::HYP_MISSING; return false; // can't work with no hypothesis } if ( hyps.size() > 1 ) { aStatus = SMESH_Hypothesis::HYP_ALREADY_EXIST; return false; } const SMESHDS_Hypothesis *theHyp = hyps.front(); string hypName = theHyp->GetName(); if (hypName == "NumberOfLayers") { myNbLayerHypo = static_cast(theHyp); aStatus = SMESH_Hypothesis::HYP_OK; return true; } if (hypName == "LayerDistribution") { myDistributionHypo = static_cast(theHyp); aStatus = SMESH_Hypothesis::HYP_OK; return true; } aStatus = SMESH_Hypothesis::HYP_INCOMPATIBLE; return true; } //======================================================================= //function : Compute //purpose : //======================================================================= bool StdMeshers_RadialPrism_3D::Compute(SMESH_Mesh& aMesh, const TopoDS_Shape& aShape) { TopExp_Explorer exp; SMESHDS_Mesh * meshDS = aMesh.GetMeshDS(); myHelper = new SMESH_MesherHelper( aMesh ); myHelper->IsQuadraticSubMesh( aShape ); // to delete helper at exit from Compute() std::auto_ptr helperDeleter( myHelper ); // get 2 shells TopoDS_Solid solid = TopoDS::Solid( aShape ); TopoDS_Shell outerShell = BRepTools::OuterShell( solid ); TopoDS_Shape innerShell; int nbShells = 0; for ( TopoDS_Iterator It (solid); It.More(); It.Next(), ++nbShells ) if ( !outerShell.IsSame( It.Value() )) innerShell = It.Value(); if ( nbShells != 2 ) RETURN_BAD_RESULT("Must be 2 shells"); // ---------------------------------- // Associate subshapes of the shells // ---------------------------------- TAssocTool::TShapeShapeMap shape2ShapeMap; if ( !TAssocTool::FindSubShapeAssociation( outerShell, &aMesh, innerShell, &aMesh, shape2ShapeMap) ) RETURN_BAD_RESULT("FindSubShapeAssociation failed"); // ------------------ // Make mesh // ------------------ TNode2ColumnMap node2columnMap; myLayerPositions.clear(); for ( exp.Init( outerShell, TopAbs_FACE ); exp.More(); exp.Next() ) { // Corresponding subshapes TopoDS_Face outFace = TopoDS::Face( exp.Current() ); TopoDS_Face inFace; if ( !shape2ShapeMap.IsBound( outFace )) { RETURN_BAD_RESULT("Association not found for face " << meshDS->ShapeToIndex( outFace )); } else { inFace = TopoDS::Face( shape2ShapeMap( outFace )); } // Find matching nodes of in and out faces TNodeNodeMap nodeIn2OutMap; if ( ! TAssocTool::FindMatchingNodesOnFaces( inFace, &aMesh, outFace, &aMesh, shape2ShapeMap, nodeIn2OutMap )) RETURN_BAD_RESULT("Different mesh on corresponding out and in faces: " << meshDS->ShapeToIndex( outFace ) << " and " << meshDS->ShapeToIndex( inFace )); // Create volumes SMDS_ElemIteratorPtr faceIt = meshDS->MeshElements( inFace )->GetElements(); while ( faceIt->more() ) // loop on faces on inFace { const SMDS_MeshElement* face = faceIt->next(); if ( !face || face->GetType() != SMDSAbs_Face ) continue; int nbNodes = face->NbNodes(); if ( face->IsQuadratic() ) nbNodes /= 2; // find node columns for each node vector< const TNodeColumn* > columns( nbNodes ); for ( int i = 0; i < nbNodes; ++i ) { const SMDS_MeshNode* n = face->GetNode( i ); TNode2ColumnMap::iterator n_col = node2columnMap.find( n ); if ( n_col != node2columnMap.end() ) columns[ i ] = & n_col->second; else columns[ i ] = makeNodeColumn( node2columnMap, n, nodeIn2OutMap[ n ] ); } StdMeshers_Prism_3D::AddPrisms( columns, myHelper ); } } // loop on faces of out shell return true; } //================================================================================ /*! * \brief Create a column of nodes from outNode to inNode * \param n2ColMap - map of node columns to add a created column * \param outNode - botton node of a column * \param inNode - top node of a column * \retval const TNodeColumn* - a new column pointer */ //================================================================================ TNodeColumn* StdMeshers_RadialPrism_3D::makeNodeColumn( TNode2ColumnMap& n2ColMap, const SMDS_MeshNode* outNode, const SMDS_MeshNode* inNode) { SMESHDS_Mesh * meshDS = myHelper->GetMeshDS(); int shapeID = myHelper->GetSubShapeID(); if ( myLayerPositions.empty() ) computeLayerPositions( gpXYZ( inNode ), gpXYZ( outNode )); int nbSegments = myLayerPositions.size() + 1; TNode2ColumnMap::iterator n_col = n2ColMap.insert( make_pair( outNode, TNodeColumn() )).first; TNodeColumn & column = n_col->second; column.resize( nbSegments + 1 ); column.front() = outNode; column.back() = inNode; gp_XYZ p1 = gpXYZ( outNode ); gp_XYZ p2 = gpXYZ( inNode ); for ( int z = 1; z < nbSegments; ++z ) { double r = myLayerPositions[ z - 1 ]; gp_XYZ p = ( 1 - r ) * p1 + r * p2; SMDS_MeshNode* n = meshDS->AddNode( p.X(), p.Y(), p.Z() ); meshDS->SetNodeInVolume( n, shapeID ); column[ z ] = n; } return & column; } //================================================================================ //================================================================================ /*! * \brief Class computing layers distribution using data of * StdMeshers_LayerDistribution hypothesis */ //================================================================================ //================================================================================ class TNodeDistributor: private StdMeshers_Regular_1D { list myUsedHyps; public: // ----------------------------------------------------------------------------- static TNodeDistributor* GetDistributor(SMESH_Mesh& aMesh) { const int myID = -1000; map < int, SMESH_1D_Algo * > & algoMap = aMesh.GetGen()->_map1D_Algo; map < int, SMESH_1D_Algo * >::iterator id_algo = algoMap.find( myID ); if ( id_algo == algoMap.end() ) return new TNodeDistributor( myID, 0, aMesh.GetGen() ); return static_cast< TNodeDistributor* >( id_algo->second ); } // ----------------------------------------------------------------------------- bool Compute( vector< double > & positions, gp_Pnt pIn, gp_Pnt pOut, SMESH_Mesh& aMesh, const StdMeshers_LayerDistribution* hyp) { double len = pIn.Distance( pOut ); if ( len <= DBL_MIN ) RETURN_BAD_RESULT("Bad points"); if ( !hyp || !hyp->GetLayerDistribution() ) RETURN_BAD_RESULT("Bad StdMeshers_LayerDistribution hypothesis"); myUsedHyps.clear(); myUsedHyps.push_back( hyp->GetLayerDistribution() ); TopoDS_Edge edge = BRepBuilderAPI_MakeEdge( pIn, pOut ); SMESH_Hypothesis::Hypothesis_Status aStatus; if ( !StdMeshers_Regular_1D::CheckHypothesis( aMesh, edge, aStatus )) RETURN_BAD_RESULT("StdMeshers_Regular_1D::CheckHypothesis() failed with status "< params; if ( !StdMeshers_Regular_1D::computeInternalParameters( edge, params, false )) RETURN_BAD_RESULT("StdMeshers_Regular_1D::computeInternalParameters() failed"); positions.clear(); positions.reserve( params.size() ); for (list::iterator itU = params.begin(); itU != params.end(); itU++) positions.push_back( *itU / len ); return true; } protected: // ----------------------------------------------------------------------------- TNodeDistributor( int hypId, int studyId, SMESH_Gen* gen) : StdMeshers_Regular_1D( hypId, studyId, gen) { } // ----------------------------------------------------------------------------- virtual const list & GetUsedHypothesis(SMESH_Mesh &, const TopoDS_Shape &, const bool) { return myUsedHyps; } // ----------------------------------------------------------------------------- }; //================================================================================ /*! * \brief Compute positions of nodes between the internal and the external surfaces * \retval bool - is a success */ //================================================================================ bool StdMeshers_RadialPrism_3D::computeLayerPositions(gp_Pnt pIn, gp_Pnt pOut) { if ( myNbLayerHypo ) { int nbSegments = myNbLayerHypo->GetNumberOfLayers(); myLayerPositions.resize( nbSegments - 1 ); for ( int z = 1; z < nbSegments; ++z ) myLayerPositions[ z - 1 ] = double( z )/ double( nbSegments ); return true; } if ( myDistributionHypo ) { SMESH_Mesh * mesh = myHelper->GetMesh(); return TNodeDistributor::GetDistributor(*mesh)->Compute( myLayerPositions, pIn, pOut, *mesh, myDistributionHypo ); } RETURN_BAD_RESULT("Bad hypothesis"); }