2019-02-14 16:55:47 +05:00
|
|
|
// Copyright (C) 2007-2019 CEA/DEN, EDF R&D, OPEN CASCADE
|
2017-12-22 16:14:24 +05:00
|
|
|
//
|
|
|
|
// Copyright (C) 2003-2007 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, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// 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 : SMDS_ElementFactory.cxx
|
|
|
|
// Module : SMESH
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "SMDS_ElementFactory.hxx"
|
|
|
|
|
|
|
|
#include "ObjectPool.hxx"
|
|
|
|
#include "SMDS_EdgePosition.hxx"
|
|
|
|
#include "SMDS_FacePosition.hxx"
|
|
|
|
#include "SMDS_Mesh.hxx"
|
|
|
|
#include "SMDS_SpacePosition.hxx"
|
|
|
|
#include "SMDS_VertexPosition.hxx"
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
// nb of elements allocated by SMDS_ElementChunk at once
|
|
|
|
const int theChunkSize = 1024;
|
|
|
|
|
|
|
|
const int theDefaultShapeDim = 3;
|
|
|
|
|
|
|
|
// classes allowing to modify parameters of SMDS_Position stored in SMDS_ElementFactory
|
|
|
|
|
|
|
|
struct _EdgePosition : public SMDS_EdgePosition
|
|
|
|
{
|
|
|
|
TParam* myUParameter;
|
|
|
|
|
|
|
|
_EdgePosition( TParam* aUParam ) : myUParameter( aUParam )
|
|
|
|
{ SMDS_EdgePosition::SetUParameter( aUParam[0]); }
|
|
|
|
virtual void SetUParameter(double aUparam)
|
|
|
|
{ *myUParameter = (TParam) aUparam; SMDS_EdgePosition::SetUParameter( aUparam ); }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _FacePosition : public SMDS_FacePosition
|
|
|
|
{
|
|
|
|
TParam* myParameter;
|
|
|
|
|
|
|
|
_FacePosition(TParam* aParam) : myParameter( aParam )
|
|
|
|
{ SMDS_FacePosition::SetParameters( aParam[0], aParam[1] ); }
|
|
|
|
virtual void SetUParameter(double aUparam)
|
|
|
|
{ myParameter[0] = (TParam) aUparam; SMDS_FacePosition::SetUParameter( aUparam ); }
|
|
|
|
virtual void SetVParameter(double aVparam)
|
|
|
|
{ myParameter[1] = (TParam) aVparam; SMDS_FacePosition::SetVParameter( aVparam ); }
|
|
|
|
virtual void SetParameters(double aU, double aV)
|
|
|
|
{ myParameter[0] = aU; myParameter[1] = aV; SMDS_FacePosition::SetParameters( aU, aV ); }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Create a factory of cells or nodes in a given mesh
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_ElementFactory::SMDS_ElementFactory( SMDS_Mesh* mesh, const bool isNodal )
|
|
|
|
: myIsNodal( isNodal ), myMesh( mesh ), myNbUsedElements( 0 )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Destructor
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_ElementFactory::~SMDS_ElementFactory()
|
|
|
|
{
|
|
|
|
myChunksWithUnused.clear();
|
|
|
|
myChunks.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return a number of elements in a chunk
|
|
|
|
* \return int - chunk size
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementFactory::ChunkSize()
|
|
|
|
{
|
|
|
|
return theChunkSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return minimal ID of a non-used element
|
|
|
|
* \return int - minimal element ID
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementFactory::GetFreeID()
|
|
|
|
{
|
|
|
|
if ( myChunksWithUnused.empty() )
|
|
|
|
{
|
|
|
|
int id0 = myChunks.size() * theChunkSize + 1;
|
|
|
|
myChunks.push_back( new SMDS_ElementChunk( this, id0 ));
|
|
|
|
}
|
|
|
|
SMDS_ElementChunk * chunk = (*myChunksWithUnused.begin());
|
|
|
|
return chunk->GetUnusedID();
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return maximal ID of an used element
|
|
|
|
* \return int - element ID
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementFactory::GetMaxID()
|
|
|
|
{
|
|
|
|
int id = 0;
|
|
|
|
TIndexRanges usedRanges;
|
|
|
|
for ( int i = myChunks.size() - 1; i >= 0; --i )
|
|
|
|
if ( myChunks[i].GetUsedRanges().GetIndices( true, usedRanges ))
|
|
|
|
{
|
|
|
|
int index = usedRanges.back().second-1;
|
|
|
|
id = myChunks[i].Get1stID() + index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return minimal ID of an used element
|
|
|
|
* \return int - element ID
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementFactory::GetMinID()
|
|
|
|
{
|
|
|
|
int id = 0;
|
|
|
|
TIndexRanges usedRanges;
|
|
|
|
for ( size_t i = 0; i < myChunks.size(); ++i )
|
|
|
|
if ( myChunks[i].GetUsedRanges().GetIndices( true, usedRanges ))
|
|
|
|
{
|
|
|
|
int index = usedRanges[0].first;
|
|
|
|
id = myChunks[i].Get1stID() + index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return an element by ID. NULL if the element with the given ID is already used
|
|
|
|
* \param [in] id - element ID
|
|
|
|
* \return SMDS_MeshElement* - element pointer
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_MeshElement* SMDS_ElementFactory::NewElement( const int id )
|
|
|
|
{
|
|
|
|
int iChunk = ( id - 1 ) / theChunkSize;
|
|
|
|
int index = ( id - 1 ) % theChunkSize;
|
|
|
|
while ((int) myChunks.size() <= iChunk )
|
|
|
|
{
|
|
|
|
int id0 = myChunks.size() * theChunkSize + 1;
|
|
|
|
myChunks.push_back( new SMDS_ElementChunk( this, id0 ));
|
|
|
|
}
|
|
|
|
SMDS_MeshElement* e = myChunks[iChunk].Element( index );
|
|
|
|
if ( !e->IsNull() )
|
|
|
|
return 0; // element with given ID already exists
|
|
|
|
|
|
|
|
myChunks[iChunk].UseElement( index );
|
|
|
|
++myNbUsedElements;
|
|
|
|
|
|
|
|
e->myHolder = & myChunks[iChunk];
|
|
|
|
|
|
|
|
myMesh->setMyModified();
|
|
|
|
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return an used element by ID. NULL if the element with the given ID is not yet used
|
|
|
|
* \param [in] id - element ID
|
|
|
|
* \return const SMDS_MeshElement* - element pointer
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
const SMDS_MeshElement* SMDS_ElementFactory::FindElement( const int id ) const
|
|
|
|
{
|
|
|
|
if ( id > 0 )
|
|
|
|
{
|
|
|
|
int iChunk = ( id - 1 ) / theChunkSize;
|
|
|
|
int index = ( id - 1 ) % theChunkSize;
|
|
|
|
if ( iChunk < (int) myChunks.size() )
|
|
|
|
{
|
|
|
|
const SMDS_MeshElement* e = myChunks[iChunk].Element( index );
|
|
|
|
return e->IsNull() ? 0 : e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return an SMDS ID by a Vtk one
|
|
|
|
* \param [inout] vtkID - Vtk ID
|
|
|
|
* \return int - SMDS ID
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementFactory::FromVtkToSmds( vtkIdType vtkID )
|
|
|
|
{
|
|
|
|
if ( vtkID >= 0 && vtkID < (vtkIdType)mySmdsIDs.size() )
|
|
|
|
return mySmdsIDs[vtkID] + 1;
|
|
|
|
return vtkID + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Mark the element as non-used
|
|
|
|
* \param [in] e - element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementFactory::Free( const SMDS_MeshElement* e )
|
|
|
|
{
|
2019-04-08 15:40:11 +05:00
|
|
|
if ( e != FindElement( e->GetID() ))
|
|
|
|
SALOME_Exception("SMDS_ElementFactory::Free(): element of other mesh");
|
|
|
|
|
2017-12-22 16:14:24 +05:00
|
|
|
if ( !myVtkIDs.empty() )
|
|
|
|
{
|
|
|
|
size_t id = e->GetID() - 1;
|
|
|
|
size_t vtkID = e->GetVtkID();
|
|
|
|
if ( id < myVtkIDs.size() )
|
|
|
|
myVtkIDs[ id ] = -1;
|
|
|
|
if ( vtkID < mySmdsIDs.size() )
|
|
|
|
mySmdsIDs[ vtkID ] = -1;
|
|
|
|
}
|
|
|
|
e->myHolder->Free( e );
|
|
|
|
const_cast< SMDS_MeshElement*>( e )->myHolder = 0;
|
|
|
|
--myNbUsedElements;
|
|
|
|
|
|
|
|
myMesh->setMyModified();
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief De-allocate all elements
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementFactory::Clear()
|
|
|
|
{
|
|
|
|
myChunksWithUnused.clear();
|
|
|
|
clearVector( myChunks );
|
|
|
|
clearVector( myVtkIDs );
|
|
|
|
clearVector( mySmdsIDs );
|
|
|
|
myNbUsedElements = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Remove unused elements located not at the end of the last chunk.
|
|
|
|
* Minimize allocated memory
|
|
|
|
* \param [out] theVtkIDsNewToOld - theVtkIDsNewToOld[ new VtkID ] = old VtkID
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementFactory::Compact( std::vector<int>& theVtkIDsNewToOld )
|
|
|
|
{
|
|
|
|
int newNbCells = NbUsedElements();
|
|
|
|
int maxCellID = GetMaxID();
|
|
|
|
int newNbChunks = newNbCells / theChunkSize + bool ( newNbCells % theChunkSize );
|
|
|
|
|
|
|
|
theVtkIDsNewToOld.resize( newNbCells );
|
|
|
|
|
|
|
|
if ( newNbCells == 0 ) // empty mesh
|
|
|
|
{
|
|
|
|
clearVector( myChunks );
|
|
|
|
}
|
|
|
|
else if ( newNbCells == maxCellID ) // no holes
|
|
|
|
{
|
|
|
|
int newID, minLastID = std::min( myVtkIDs.size(), theVtkIDsNewToOld.size() );
|
|
|
|
for ( newID = 0; newID < minLastID; ++newID )
|
|
|
|
theVtkIDsNewToOld[ newID ] = myVtkIDs[ newID ];
|
|
|
|
for ( ; newID < newNbCells; ++newID )
|
|
|
|
theVtkIDsNewToOld[ newID ] = newID;
|
|
|
|
}
|
|
|
|
else // there are holes in SMDS IDs
|
|
|
|
{
|
|
|
|
int newVtkID = 0; // same as new smds ID (-1)
|
|
|
|
for ( int oldID = 1; oldID <= maxCellID; ++oldID ) // smds IDs
|
|
|
|
{
|
|
|
|
const SMDS_MeshElement* oldElem = FindElement( oldID );
|
|
|
|
if ( !oldElem ) continue;
|
|
|
|
theVtkIDsNewToOld[ newVtkID++ ] = oldElem->GetVtkID(); // now newVtkID == new smds ID
|
|
|
|
if ( oldID != newVtkID )
|
|
|
|
{
|
|
|
|
const SMDS_MeshElement* newElem = FindElement( newVtkID );
|
|
|
|
if ( !newElem )
|
|
|
|
newElem = NewElement( newVtkID );
|
|
|
|
if ( int shapeID = oldElem->GetShapeID() )
|
|
|
|
const_cast< SMDS_MeshElement* >( newElem )->setShapeID( shapeID );
|
|
|
|
if ( oldID > newNbCells )
|
|
|
|
Free( oldElem );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
myChunks.resize( newNbChunks );
|
|
|
|
|
|
|
|
myChunksWithUnused.clear();
|
|
|
|
if ( !myChunks.empty() && myChunks.back().GetUsedRanges().Size() > 1 )
|
|
|
|
myChunksWithUnused.insert( & myChunks.back() );
|
|
|
|
|
|
|
|
for ( size_t i = 0; i < myChunks.size(); ++i )
|
|
|
|
myChunks[i].Compact();
|
|
|
|
|
|
|
|
clearVector( myVtkIDs );
|
|
|
|
clearVector( mySmdsIDs );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return true if Compact() will change IDs of elements
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
bool SMDS_ElementFactory::CompactChangePointers()
|
|
|
|
{
|
|
|
|
// there can be VTK_EMPTY_CELL's in the VTK grid as well as "holes" in SMDS numeration
|
|
|
|
return ( NbUsedElements() != GetMaxID() );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Create a factory of nodes in a given mesh
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_NodeFactory::SMDS_NodeFactory( SMDS_Mesh* mesh )
|
|
|
|
: SMDS_ElementFactory( mesh, /*isNodal=*/true )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Destructor
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_NodeFactory::~SMDS_NodeFactory()
|
|
|
|
{
|
|
|
|
Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Remove unused nodes located not at the end of the last chunk.
|
|
|
|
* Minimize allocated memory
|
|
|
|
* \param [out] theVtkIDsOldToNew - vector storing change of vtk IDs
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_NodeFactory::Compact( std::vector<int>& theVtkIDsOldToNew )
|
|
|
|
{
|
|
|
|
// IDs of VTK nodes always correspond to SMDS IDs but there can be "holes"
|
|
|
|
// in the chunks. So we remove holes and report relocation in theVtkIDsOldToNew:
|
|
|
|
// theVtkIDsOldToNew[ old VtkID ] = new VtkID
|
|
|
|
|
|
|
|
int oldNbNodes = myMesh->GetGrid()->GetNumberOfPoints();
|
|
|
|
int newNbNodes = NbUsedElements();
|
|
|
|
int newNbChunks = newNbNodes / theChunkSize + bool ( newNbNodes % theChunkSize );
|
|
|
|
int maxNodeID = GetMaxID();
|
|
|
|
|
|
|
|
theVtkIDsOldToNew.resize( oldNbNodes, -1 );
|
|
|
|
|
|
|
|
if ( newNbNodes == 0 ) // empty mesh
|
|
|
|
{
|
|
|
|
clearVector( myChunks );
|
|
|
|
}
|
|
|
|
else if ( maxNodeID > newNbNodes ) // there are holes
|
|
|
|
{
|
|
|
|
size_t newID = 0;
|
|
|
|
for ( size_t oldID = 0; oldID < theVtkIDsOldToNew.size(); ++oldID )
|
|
|
|
{
|
|
|
|
const SMDS_MeshElement* oldNode = FindNode( oldID+1 );
|
|
|
|
if ( !oldNode )
|
|
|
|
continue;
|
|
|
|
theVtkIDsOldToNew[ oldID ] = newID;
|
|
|
|
if ( oldID != newID )
|
|
|
|
{
|
|
|
|
const SMDS_MeshElement* newNode = FindElement( newID+1 );
|
|
|
|
if ( !newNode )
|
|
|
|
newNode = NewElement( newID+1 );
|
|
|
|
int shapeID = oldNode->GetShapeID();
|
|
|
|
int shapeDim = GetShapeDim( shapeID );
|
|
|
|
int iChunk = newID / theChunkSize;
|
|
|
|
myChunks[ iChunk ].SetShapeID( newNode, shapeID );
|
|
|
|
if ( shapeDim == 2 || shapeDim == 1 )
|
|
|
|
{
|
|
|
|
int iChunkOld = oldID / theChunkSize;
|
|
|
|
TParam* oldPos = myChunks[ iChunkOld ].GetPositionPtr( oldNode );
|
|
|
|
TParam* newPos = myChunks[ iChunk ].GetPositionPtr( newNode, /*allocate=*/true );
|
|
|
|
if ( oldPos )
|
|
|
|
{
|
|
|
|
newPos[0] = oldPos[0];
|
|
|
|
newPos[1] = oldPos[1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( oldNode->GetID() > newNbNodes )
|
|
|
|
Free( oldNode );
|
|
|
|
}
|
|
|
|
++newID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else // no holes
|
|
|
|
{
|
|
|
|
for ( int i = 0; i < newNbNodes; ++i )
|
|
|
|
theVtkIDsOldToNew[ i ] = i;
|
|
|
|
}
|
|
|
|
myChunks.resize( newNbChunks );
|
|
|
|
|
|
|
|
myChunksWithUnused.clear();
|
|
|
|
if ( !myChunks.empty() && myChunks.back().GetUsedRanges().Size() > 1 )
|
|
|
|
myChunksWithUnused.insert( & myChunks.back() );
|
|
|
|
|
|
|
|
for ( size_t i = 0; i < myChunks.size(); ++i )
|
|
|
|
myChunks[i].Compact();
|
|
|
|
|
|
|
|
ASSERT( newNbNodes == GetMaxID() );
|
|
|
|
ASSERT( newNbNodes == NbUsedElements() );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return true if Compact() will change IDs of elements
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
bool SMDS_NodeFactory::CompactChangePointers()
|
|
|
|
{
|
|
|
|
// IDs of VTK nodes always correspond to SMDS IDs but there can be "holes" in SMDS numeration
|
|
|
|
return ( NbUsedElements() != GetMaxID() );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief De-allocate all nodes
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_NodeFactory::Clear()
|
|
|
|
{
|
|
|
|
SMDS_ElementFactory::Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Set a total number of sub-shapes in the main shape
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_NodeFactory::SetNbShapes( size_t nbShapes )
|
|
|
|
{
|
|
|
|
clearVector( myShapeDim );
|
|
|
|
myShapeDim.resize( nbShapes+1, theDefaultShapeDim );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return a dimension of a shape
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_NodeFactory::GetShapeDim( int shapeID ) const
|
|
|
|
{
|
|
|
|
return shapeID < (int)myShapeDim.size() ? myShapeDim[ shapeID ] : theDefaultShapeDim;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Set a dimension of a shape
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_NodeFactory::SetShapeDim( int shapeID, int dim )
|
|
|
|
{
|
|
|
|
if ( shapeID >= (int)myShapeDim.size() )
|
|
|
|
myShapeDim.resize( shapeID + 10, theDefaultShapeDim );
|
|
|
|
myShapeDim[ shapeID ] = dim;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief SMDS_ElementChunk constructor
|
|
|
|
* \param [in] factory - the factory
|
|
|
|
* \param [in] id0 - ID of the 1st element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_ElementChunk::SMDS_ElementChunk( SMDS_ElementFactory* factory, int id0 ):
|
|
|
|
myFactory( factory ),
|
2018-08-08 18:55:53 +05:00
|
|
|
my1stID( id0 )//,
|
|
|
|
//mySubIDSet( 0 )
|
|
|
|
// myMinSubID( std::numeric_limits<int>::max() ),
|
|
|
|
// myMaxSubID( 0 )
|
2017-12-22 16:14:24 +05:00
|
|
|
{
|
|
|
|
if ( !myFactory )
|
|
|
|
return;
|
|
|
|
if ( myFactory->myIsNodal )
|
|
|
|
myElements = new SMDS_MeshNode[ theChunkSize ];
|
|
|
|
else
|
|
|
|
myElements = new SMDS_MeshCell[ theChunkSize ];
|
|
|
|
|
|
|
|
myUsedRanges.mySet.reserve(2);
|
|
|
|
mySubIDRanges.mySet.insert( _ShapeIDRange( 0, 0 ));
|
|
|
|
myUsedRanges.mySet.insert( _UsedRange( 0, false ));
|
|
|
|
myFactory->myChunksWithUnused.insert( this );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief SMDS_ElementChunk destructor
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_ElementChunk::~SMDS_ElementChunk()
|
|
|
|
{
|
|
|
|
delete [] myElements;
|
|
|
|
myFactory->myChunksWithUnused.erase( this );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Mark an element as used
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::UseElement( const int index )
|
|
|
|
{
|
|
|
|
myUsedRanges.SetValue( index, true );
|
|
|
|
if ( myUsedRanges.Size() == 1 ) // all elements used
|
|
|
|
myFactory->myChunksWithUnused.erase( this );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return ID of the first non-used element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementChunk::GetUnusedID() const
|
|
|
|
{
|
|
|
|
TUsedRangeSet::set_iterator r = myUsedRanges.mySet.begin();
|
|
|
|
for ( ; r != myUsedRanges.mySet.end(); ++r )
|
|
|
|
if ( !IsUsed( *r ))
|
|
|
|
break;
|
|
|
|
|
|
|
|
return my1stID + r->my1st;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Mark an element as non-used
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::Free( const SMDS_MeshElement* e )
|
|
|
|
{
|
|
|
|
bool hasHoles = ( myUsedRanges.Size() > 1 );
|
|
|
|
myUsedRanges.SetValue( Index( e ), false );
|
|
|
|
SetShapeID( e, 0 ); // sub-mesh must do it?
|
|
|
|
SetIsMarked( e, false );
|
|
|
|
if ( !hasHoles )
|
|
|
|
myFactory->myChunksWithUnused.insert( this );
|
|
|
|
|
|
|
|
if ( myUsedRanges.Size() == 1 )
|
|
|
|
{
|
|
|
|
clearVector( myMarkedSet );
|
|
|
|
clearVector( myPositions );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return an SMDS ID of an element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementChunk::GetID( const SMDS_MeshElement* e ) const
|
|
|
|
{
|
|
|
|
return my1stID + Index( e );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Set a Vtk ID of an element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::SetVTKID( const SMDS_MeshElement* e, const vtkIdType vtkID )
|
|
|
|
{
|
|
|
|
if ( e->GetID() - 1 != vtkID )
|
|
|
|
{
|
|
|
|
if ((int) myFactory->myVtkIDs.size() <= e->GetID() - 1 )
|
|
|
|
{
|
|
|
|
size_t i = myFactory->myVtkIDs.size();
|
|
|
|
myFactory->myVtkIDs.resize( e->GetID() + 100 );
|
|
|
|
for ( ; i < myFactory->myVtkIDs.size(); ++i )
|
|
|
|
myFactory->myVtkIDs[i] = i;
|
|
|
|
}
|
|
|
|
myFactory->myVtkIDs[ e->GetID() - 1 ] = vtkID;
|
|
|
|
|
|
|
|
if ((vtkIdType) myFactory->mySmdsIDs.size() <= vtkID )
|
|
|
|
{
|
|
|
|
size_t i = myFactory->mySmdsIDs.size();
|
|
|
|
myFactory->mySmdsIDs.resize( vtkID + 100 );
|
|
|
|
for ( ; i < myFactory->mySmdsIDs.size(); ++i )
|
|
|
|
myFactory->mySmdsIDs[i] = i;
|
|
|
|
}
|
|
|
|
myFactory->mySmdsIDs[ vtkID ] = e->GetID() - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return a Vtk ID of an element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementChunk::GetVtkID( const SMDS_MeshElement* e ) const
|
|
|
|
{
|
|
|
|
size_t dfltVtkID = e->GetID() - 1;
|
|
|
|
return ( dfltVtkID < myFactory->myVtkIDs.size() ) ? myFactory->myVtkIDs[ dfltVtkID ] : dfltVtkID;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return ID of a shape an element is assigned to
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
int SMDS_ElementChunk::GetShapeID( const SMDS_MeshElement* e ) const
|
|
|
|
{
|
|
|
|
return mySubIDRanges.GetValue( Index( e ));
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Set ID of a shape an element is assigned to
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::SetShapeID( const SMDS_MeshElement* e, int shapeID ) const
|
|
|
|
{
|
2018-08-08 18:55:53 +05:00
|
|
|
//const size_t nbRanges = mySubIDRanges.Size();
|
2017-12-22 16:14:24 +05:00
|
|
|
|
|
|
|
SMDS_ElementChunk* me = const_cast<SMDS_ElementChunk*>( this );
|
|
|
|
int oldShapeID = me->mySubIDRanges.SetValue( Index( e ), shapeID );
|
|
|
|
if ( oldShapeID == shapeID ) return;
|
|
|
|
|
|
|
|
if ( const SMDS_MeshNode* n = dynamic_cast< const SMDS_MeshNode* >( e ))
|
|
|
|
if ( TParam* uv = me->GetPositionPtr( n ))
|
|
|
|
{
|
|
|
|
uv[0] = 0.;
|
|
|
|
uv[1] = 0.;
|
|
|
|
}
|
|
|
|
// update min/max
|
2018-08-08 18:55:53 +05:00
|
|
|
// if (( nbRanges > mySubIDRanges.Size() ) &&
|
|
|
|
// ( myMinSubID == oldShapeID || myMaxSubID == oldShapeID ))
|
|
|
|
// {
|
|
|
|
// me->myMinSubID = ( std::numeric_limits<int>::max() );
|
|
|
|
// me->myMaxSubID = 0;
|
|
|
|
// TSubIDRangeSet::set_iterator it;
|
|
|
|
// for ( it = mySubIDRanges.mySet.begin(); it < mySubIDRanges.mySet.end(); ++it )
|
|
|
|
// if ( it->myValue > 0 )
|
|
|
|
// {
|
|
|
|
// me->myMinSubID = std::min( myMinSubID, it->myValue );
|
|
|
|
// me->myMaxSubID = std::max( myMaxSubID, it->myValue );
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// else if ( shapeID > 0 )
|
|
|
|
// {
|
|
|
|
// me->myMinSubID = std::min( myMinSubID, shapeID );
|
|
|
|
// me->myMaxSubID = std::max( myMaxSubID, shapeID );
|
|
|
|
// }
|
2017-12-22 16:14:24 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Set isMarked flag of an element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
bool SMDS_ElementChunk::IsMarked( const SMDS_MeshElement* e ) const
|
|
|
|
{
|
|
|
|
return ( !myMarkedSet.empty() && myMarkedSet[ Index( e )]);
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return isMarked flag of an element
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::SetIsMarked( const SMDS_MeshElement* e, bool is )
|
|
|
|
{
|
|
|
|
if ( !is && myMarkedSet.empty() ) return;
|
|
|
|
if ( myMarkedSet.empty() ) myMarkedSet.resize( theChunkSize, false );
|
|
|
|
myMarkedSet[ Index( e )] = is;
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return SMDS_Position of a node on a shape
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
SMDS_PositionPtr SMDS_ElementChunk::GetPosition( const SMDS_MeshNode* n ) const
|
|
|
|
{
|
|
|
|
int shapeID = GetShapeID( n );
|
|
|
|
int shapeDim = static_cast< SMDS_NodeFactory* >( myFactory )->GetShapeDim( shapeID );
|
|
|
|
|
|
|
|
SMDS_ElementChunk* me = const_cast< SMDS_ElementChunk* >( this );
|
|
|
|
|
|
|
|
switch ( shapeDim ) {
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
return SMDS_PositionPtr( new _FacePosition( me->GetPositionPtr( n )));
|
|
|
|
}
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
return SMDS_PositionPtr( new _EdgePosition( me->GetPositionPtr( n )));
|
|
|
|
}
|
|
|
|
case 0:
|
|
|
|
return SMDS_VertexPosition::StaticPosition();
|
|
|
|
}
|
|
|
|
|
|
|
|
return SMDS_SpacePosition::originSpacePosition();
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Set SMDS_Position of a node on a shape
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::SetPosition( const SMDS_MeshNode* n, const SMDS_PositionPtr& pos, int shapeID )
|
|
|
|
{
|
|
|
|
int shapeDim = pos ? pos->GetDim() : theDefaultShapeDim;
|
|
|
|
if ( shapeID < 1 )
|
|
|
|
{
|
|
|
|
if ( shapeDim == theDefaultShapeDim )
|
|
|
|
return;
|
|
|
|
shapeID = GetShapeID( n );
|
|
|
|
if ( shapeID < 1 )
|
|
|
|
throw SALOME_Exception("SetPosition() No shape ID provided");
|
|
|
|
}
|
|
|
|
|
|
|
|
static_cast< SMDS_NodeFactory* >( myFactory )->SetShapeDim( shapeID, shapeDim );
|
|
|
|
|
|
|
|
switch ( shapeDim ) {
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
TParam* uv = GetPositionPtr( n, /*allocate=*/true );
|
|
|
|
uv[0] = (TParam) pos->GetParameters()[0];
|
|
|
|
uv[1] = (TParam) pos->GetParameters()[1];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
GetPositionPtr( n, /*allocate=*/true )[0] = (TParam) pos->GetParameters()[0];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Return pointer to on-shape-parameters of a node
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
TParam* SMDS_ElementChunk::GetPositionPtr( const SMDS_MeshElement* n, bool allocate )
|
|
|
|
{
|
|
|
|
if ( myPositions.empty() && !allocate )
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
myPositions.resize( theChunkSize * 2 );
|
|
|
|
return myPositions.data() + 2 * Index( n );
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Minimize allocated memory
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::Compact()
|
|
|
|
{
|
|
|
|
mySubIDRanges.mySet.shrink_to_fit();
|
|
|
|
if ( myUsedRanges.mySet.capacity() > 2 )
|
|
|
|
myUsedRanges.mySet.shrink_to_fit();
|
|
|
|
|
|
|
|
clearVector( myMarkedSet );
|
|
|
|
|
|
|
|
if ( !myPositions.empty() )
|
|
|
|
{
|
|
|
|
// look for the last position that must be kept
|
|
|
|
TSubIDRangeSet::set_t::reverse_iterator it;
|
|
|
|
for ( it = mySubIDRanges.mySet.rbegin(); it != mySubIDRanges.mySet.rend(); ++it )
|
|
|
|
{
|
|
|
|
int shapeDim = static_cast< SMDS_NodeFactory* >( myFactory )->GetShapeDim( it->myValue );
|
|
|
|
if ( shapeDim == 1 || shapeDim == 2 )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ( it == mySubIDRanges.mySet.rend() )
|
|
|
|
{
|
|
|
|
clearVector( myPositions );
|
|
|
|
}
|
|
|
|
else if ( it != mySubIDRanges.mySet.rbegin() )
|
|
|
|
{
|
|
|
|
int nbNodes = (it-1)->my1st;
|
|
|
|
myPositions.resize( nbNodes * 2 );
|
|
|
|
std::vector<TParam> newPos( myPositions.begin(), myPositions.end() );
|
|
|
|
myPositions.swap( newPos );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Print some data for debug purposes
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
void SMDS_ElementChunk::Dump() const
|
|
|
|
{
|
|
|
|
std::cout << "1stID: " << my1stID << std::endl;
|
|
|
|
|
2018-08-08 18:55:53 +05:00
|
|
|
//std::cout << "SubID min/max: " << myMinSubID << ", " << myMaxSubID << std::endl;
|
2017-12-22 16:14:24 +05:00
|
|
|
std::cout << "SubIDRanges: " << mySubIDRanges.Size() << " ";
|
|
|
|
{
|
|
|
|
TSubIDRangeSet::set_iterator i = mySubIDRanges.mySet.begin();
|
|
|
|
for ( int cnt = 0; i != mySubIDRanges.mySet.end(); ++i, ++cnt )
|
|
|
|
std::cout << "|" << cnt << " - (" << i->my1st << ", " << i->myValue << ") ";
|
|
|
|
std::cout << std::endl;
|
|
|
|
}
|
|
|
|
{
|
|
|
|
std::cout << "UsedRanges: " << myUsedRanges.Size() << " ";
|
|
|
|
TUsedRangeSet::set_iterator i = myUsedRanges.mySet.begin();
|
|
|
|
for ( int cnt = 0; i != myUsedRanges.mySet.end(); ++i, ++cnt )
|
|
|
|
std::cout << cnt << " - (" << i->my1st << ", " << i->myValue << ") ";
|
|
|
|
std::cout << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
/*!
|
|
|
|
* \brief Compare SMDS_ElementChunk's
|
|
|
|
*/
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
bool _ChunkCompare::operator () (const SMDS_ElementChunk* e1, const SMDS_ElementChunk* e2) const
|
|
|
|
{
|
|
|
|
return e1->Get1stID() < e2->Get1stID();
|
|
|
|
}
|
|
|
|
|