52546: Viscous Layers construction fails

Correct finding normal at a degenerated vertex
This commit is contained in:
eap 2014-10-23 16:40:33 +04:00
parent 3ba242025d
commit ddab513d85
3 changed files with 197 additions and 82 deletions

View File

@ -226,7 +226,7 @@ SMESH_Mesh::~SMESH_Mesh()
bool SMESH_Mesh::MeshExists( int meshId ) const bool SMESH_Mesh::MeshExists( int meshId ) const
{ {
return _myDocument ? _myDocument->GetMesh( meshId ) : false; return _myDocument ? bool( _myDocument->GetMesh( meshId )) : false;
} }
//============================================================================= //=============================================================================

View File

@ -4917,7 +4917,7 @@ void SMESHGUI::createPreferences()
addPreference( tr( "PREF_PRECISION_USE" ), qaGroup, LightApp_Preferences::Bool, "SMESH", "use_precision" ); addPreference( tr( "PREF_PRECISION_USE" ), qaGroup, LightApp_Preferences::Bool, "SMESH", "use_precision" );
int prec = addPreference( tr( "PREF_PRECISION_VALUE" ), qaGroup, LightApp_Preferences::IntSpin, "SMESH", "controls_precision" ); int prec = addPreference( tr( "PREF_PRECISION_VALUE" ), qaGroup, LightApp_Preferences::IntSpin, "SMESH", "controls_precision" );
setPreferenceProperty( prec, "min", 0 ); setPreferenceProperty( prec, "min", 0 );
setPreferenceProperty( prec, "max", 16 ); setPreferenceProperty( prec, "max", 100 );
int doubleNodesTol = addPreference( tr( "PREF_EQUAL_NODES_TOL" ), qaGroup, LightApp_Preferences::DblSpin, "SMESH", "equal_nodes_tolerance" ); int doubleNodesTol = addPreference( tr( "PREF_EQUAL_NODES_TOL" ), qaGroup, LightApp_Preferences::DblSpin, "SMESH", "equal_nodes_tolerance" );
setPreferenceProperty( doubleNodesTol, "precision", 10 ); setPreferenceProperty( doubleNodesTol, "precision", 10 );
setPreferenceProperty( doubleNodesTol, "min", 0.0000000001 ); setPreferenceProperty( doubleNodesTol, "min", 0.0000000001 );

View File

@ -44,6 +44,7 @@
#include "SMESH_subMeshEventListener.hxx" #include "SMESH_subMeshEventListener.hxx"
#include "StdMeshers_FaceSide.hxx" #include "StdMeshers_FaceSide.hxx"
#include <Adaptor3d_HSurface.hxx>
#include <BRepAdaptor_Curve2d.hxx> #include <BRepAdaptor_Curve2d.hxx>
#include <BRepAdaptor_Surface.hxx> #include <BRepAdaptor_Surface.hxx>
#include <BRepLProp_SLProps.hxx> #include <BRepLProp_SLProps.hxx>
@ -75,6 +76,8 @@
#include <TopoDS_Face.hxx> #include <TopoDS_Face.hxx>
#include <TopoDS_Vertex.hxx> #include <TopoDS_Vertex.hxx>
#include <gp_Ax1.hxx> #include <gp_Ax1.hxx>
#include <gp_Cone.hxx>
#include <gp_Sphere.hxx>
#include <gp_Vec.hxx> #include <gp_Vec.hxx>
#include <gp_XY.hxx> #include <gp_XY.hxx>
@ -84,7 +87,7 @@
#include <limits> #include <limits>
#ifdef _DEBUG_ #ifdef _DEBUG_
//#define __myDEBUG #define __myDEBUG
//#define __NOT_INVALIDATE_BAD_SMOOTH //#define __NOT_INVALIDATE_BAD_SMOOTH
#endif #endif
@ -699,9 +702,13 @@ namespace VISCOUS_3D
SMESH_MesherHelper& helper, SMESH_MesherHelper& helper,
bool& isOK, bool& isOK,
bool shiftInside=false); bool shiftInside=false);
gp_XYZ getWeigthedNormal( const SMDS_MeshNode* n, bool getFaceNormalAtSingularity(const gp_XY& uv,
std::pair< TGeomID, gp_XYZ > fId2Normal[], const TopoDS_Face& face,
int nbFaces ); SMESH_MesherHelper& helper,
gp_Dir& normal );
gp_XYZ getWeigthedNormal( const SMDS_MeshNode* n,
std::pair< TopoDS_Face, gp_XYZ > fId2Normal[],
int nbFaces );
bool findNeiborsOnEdge(const _LayerEdge* edge, bool findNeiborsOnEdge(const _LayerEdge* edge,
const SMDS_MeshNode*& n1, const SMDS_MeshNode*& n1,
const SMDS_MeshNode*& n2, const SMDS_MeshNode*& n2,
@ -1262,6 +1269,43 @@ namespace VISCOUS_3D
} }
return done; return done;
} }
//================================================================================
/*!
* \brief Return direction of axis or revolution of a surface
*/
//================================================================================
bool getRovolutionAxis( const Adaptor3d_Surface& surface,
gp_Dir & axis )
{
switch ( surface.GetType() ) {
case GeomAbs_Cone:
{
gp_Cone cone = surface.Cone();
axis = cone.Axis().Direction();
break;
}
case GeomAbs_Sphere:
{
gp_Sphere sphere = surface.Sphere();
axis = sphere.Position().Direction();
break;
}
case GeomAbs_SurfaceOfRevolution:
{
axis = surface.AxeOfRevolution().Direction();
break;
}
//case GeomAbs_SurfaceOfExtrusion:
case GeomAbs_OffsetSurface:
{
Handle(Adaptor3d_HSurface) base = surface.BasisSurface();
return getRovolutionAxis( base->Surface(), axis );
}
default: return false;
}
return true;
}
//-------------------------------------------------------------------------------- //--------------------------------------------------------------------------------
// DEBUG. Dump intermediate node positions into a python script // DEBUG. Dump intermediate node positions into a python script
@ -1963,14 +2007,16 @@ bool _ViscousBuilder::makeLayer(_SolidData& data)
subIds = data._noShrinkShapes; subIds = data._noShrinkShapes;
TopExp_Explorer exp( data._solid, TopAbs_FACE ); TopExp_Explorer exp( data._solid, TopAbs_FACE );
for ( ; exp.More(); exp.Next() ) for ( ; exp.More(); exp.Next() )
{
SMESH_subMesh* fSubM = _mesh->GetSubMesh( exp.Current() );
if ( ! data._ignoreFaceIds.count( fSubM->GetId() ))
{ {
SMESH_subMesh* fSubM = _mesh->GetSubMesh( exp.Current() ); faceIds.insert( fSubM->GetId() );
if ( ! data._ignoreFaceIds.count( fSubM->GetId() ))
faceIds.insert( fSubM->GetId() );
SMESH_subMeshIteratorPtr subIt = fSubM->getDependsOnIterator(/*includeSelf=*/true); SMESH_subMeshIteratorPtr subIt = fSubM->getDependsOnIterator(/*includeSelf=*/true);
while ( subIt->more() ) while ( subIt->more() )
subIds.insert( subIt->next()->GetId() ); subIds.insert( subIt->next()->GetId() );
} }
}
// make a map to find new nodes on sub-shapes shared with other SOLID // make a map to find new nodes on sub-shapes shared with other SOLID
map< TGeomID, TNode2Edge* >::iterator s2ne; map< TGeomID, TNode2Edge* >::iterator s2ne;
@ -2614,7 +2660,7 @@ bool _ViscousBuilder::setEdgeData(_LayerEdge& edge,
SMESH_MeshEditor editor(_mesh); SMESH_MeshEditor editor(_mesh);
const SMDS_MeshNode* node = edge._nodes[0]; // source node const SMDS_MeshNode* node = edge._nodes[0]; // source node
SMDS_TypeOfPosition posType = node->GetPosition()->GetTypeOfPosition(); const SMDS_TypeOfPosition posType = node->GetPosition()->GetTypeOfPosition();
edge._len = 0; edge._len = 0;
edge._2neibors = 0; edge._2neibors = 0;
@ -2628,13 +2674,41 @@ bool _ViscousBuilder::setEdgeData(_LayerEdge& edge,
edge._normal.SetCoord(0,0,0); edge._normal.SetCoord(0,0,0);
int totalNbFaces = 0; int totalNbFaces = 0;
TopoDS_Face F;
std::pair< TopoDS_Face, gp_XYZ > face2Norm[20];
gp_Vec geomNorm; gp_Vec geomNorm;
bool normOK = true; bool normOK = true;
// get geom FACEs the node lies on
{
set<TGeomID> faceIds;
if ( posType == SMDS_TOP_FACE )
{
faceIds.insert( node->getshapeId() );
}
else
{
SMDS_ElemIteratorPtr fIt = node->GetInverseElementIterator(SMDSAbs_Face);
while ( fIt->more() )
faceIds.insert( editor.FindShape(fIt->next()));
}
set<TGeomID>::iterator id = faceIds.begin();
for ( ; id != faceIds.end(); ++id )
{
const TopoDS_Shape& s = getMeshDS()->IndexToShape( *id );
if ( s.IsNull() || s.ShapeType() != TopAbs_FACE || !subIds.count( *id ))
continue;
F = TopoDS::Face( s );
face2Norm[ totalNbFaces ].first = F;
totalNbFaces++;
}
}
const TGeomID shapeInd = node->getshapeId(); const TGeomID shapeInd = node->getshapeId();
map< TGeomID, TopoDS_Shape >::const_iterator s2s = data._shrinkShape2Shape.find( shapeInd ); map< TGeomID, TopoDS_Shape >::const_iterator s2s = data._shrinkShape2Shape.find( shapeInd );
const bool onShrinkShape ( s2s != data._shrinkShape2Shape.end() ); const bool onShrinkShape ( s2s != data._shrinkShape2Shape.end() );
// find _normal
if ( onShrinkShape ) // one of faces the node is on has no layers if ( onShrinkShape ) // one of faces the node is on has no layers
{ {
TopoDS_Shape vertEdge = getMeshDS()->IndexToShape( s2s->first ); // vertex or edge TopoDS_Shape vertEdge = getMeshDS()->IndexToShape( s2s->first ); // vertex or edge
@ -2658,54 +2732,35 @@ bool _ViscousBuilder::setEdgeData(_LayerEdge& edge,
} }
else // layers are on all faces of SOLID the node is on else // layers are on all faces of SOLID the node is on
{ {
// find indices of geom faces the node lies on int nbOkNorms = 0;
set<TGeomID> faceIds; for ( int iF = 0; iF < totalNbFaces; ++iF )
if ( posType == SMDS_TOP_FACE )
{ {
faceIds.insert( node->getshapeId() ); F = TopoDS::Face( face2Norm[ iF ].first );
}
else
{
SMDS_ElemIteratorPtr fIt = node->GetInverseElementIterator(SMDSAbs_Face);
while ( fIt->more() )
faceIds.insert( editor.FindShape(fIt->next()));
}
set<TGeomID>::iterator id = faceIds.begin();
TopoDS_Face F;
std::pair< TGeomID, gp_XYZ > id2Norm[20];
for ( ; id != faceIds.end(); ++id )
{
const TopoDS_Shape& s = getMeshDS()->IndexToShape( *id );
if ( s.IsNull() || s.ShapeType() != TopAbs_FACE || !subIds.count( *id ))
continue;
F = TopoDS::Face( s );
geomNorm = getFaceNormal( node, F, helper, normOK ); geomNorm = getFaceNormal( node, F, helper, normOK );
if ( !normOK ) continue; if ( !normOK ) continue;
nbOkNorms++;
if ( helper.GetSubShapeOri( data._solid, F ) != TopAbs_REVERSED ) if ( helper.GetSubShapeOri( data._solid, F ) != TopAbs_REVERSED )
geomNorm.Reverse(); geomNorm.Reverse();
id2Norm[ totalNbFaces ].first = *id; face2Norm[ iF ].second = geomNorm.XYZ();
id2Norm[ totalNbFaces ].second = geomNorm.XYZ();
totalNbFaces++;
edge._normal += geomNorm.XYZ(); edge._normal += geomNorm.XYZ();
} }
if ( totalNbFaces == 0 ) if ( nbOkNorms == 0 )
return error(SMESH_Comment("Can't get normal to node ") << node->GetID(), data._index); return error(SMESH_Comment("Can't get normal to node ") << node->GetID(), data._index);
if ( normOK && edge._normal.Modulus() < 1e-3 && totalNbFaces > 1 ) if ( edge._normal.Modulus() < 1e-3 && nbOkNorms > 1 )
{ {
// opposite normals, re-get normals at shifted positions (IPAL 52426) // opposite normals, re-get normals at shifted positions (IPAL 52426)
edge._normal.SetCoord( 0,0,0 ); edge._normal.SetCoord( 0,0,0 );
for ( int i = 0; i < totalNbFaces; ++i ) for ( int iF = 0; iF < totalNbFaces; ++iF )
{ {
const TopoDS_Face& F = TopoDS::Face( getMeshDS()->IndexToShape( id2Norm[i].first )); const TopoDS_Face& F = face2Norm[iF].first;
geomNorm = getFaceNormal( node, F, helper, normOK, /*shiftInside=*/true ); geomNorm = getFaceNormal( node, F, helper, normOK, /*shiftInside=*/true );
if ( helper.GetSubShapeOri( data._solid, F ) != TopAbs_REVERSED ) if ( helper.GetSubShapeOri( data._solid, F ) != TopAbs_REVERSED )
geomNorm.Reverse(); geomNorm.Reverse();
if ( normOK ) if ( normOK )
id2Norm[ i ].second = geomNorm.XYZ(); face2Norm[ iF ].second = geomNorm.XYZ();
edge._normal += id2Norm[ i ].second; edge._normal += face2Norm[ iF ].second;
} }
} }
@ -2715,44 +2770,46 @@ bool _ViscousBuilder::setEdgeData(_LayerEdge& edge,
} }
else else
{ {
edge._normal = getWeigthedNormal( node, id2Norm, totalNbFaces ); edge._normal = getWeigthedNormal( node, face2Norm, totalNbFaces );
} }
}
switch ( posType ) // set _cosin
{ switch ( posType )
case SMDS_TOP_FACE: {
edge._cosin = 0; break; case SMDS_TOP_FACE: {
edge._cosin = 0;
case SMDS_TOP_EDGE: { break;
TopoDS_Edge E = TopoDS::Edge( helper.GetSubShapeByNode( node, getMeshDS())); }
gp_Vec inFaceDir = getFaceDir( F, E, node, helper, normOK ); case SMDS_TOP_EDGE: {
double angle = inFaceDir.Angle( edge._normal ); // [0,PI] TopoDS_Edge E = TopoDS::Edge( helper.GetSubShapeByNode( node, getMeshDS()));
edge._cosin = Cos( angle ); gp_Vec inFaceDir = getFaceDir( F, E, node, helper, normOK );
//cout << "Cosin on EDGE " << edge._cosin << " node " << node->GetID() << endl; double angle = inFaceDir.Angle( edge._normal ); // [0,PI]
break; edge._cosin = Cos( angle );
} //cout << "Cosin on EDGE " << edge._cosin << " node " << node->GetID() << endl;
case SMDS_TOP_VERTEX: { break;
TopoDS_Vertex V = TopoDS::Vertex( helper.GetSubShapeByNode( node, getMeshDS())); }
gp_Vec inFaceDir = getFaceDir( F, V, node, helper, normOK ); case SMDS_TOP_VERTEX: {
double angle = inFaceDir.Angle( edge._normal ); // [0,PI] TopoDS_Vertex V = TopoDS::Vertex( helper.GetSubShapeByNode( node, getMeshDS()));
edge._cosin = Cos( angle ); gp_Vec inFaceDir = getFaceDir( F, V, node, helper, normOK );
if ( totalNbFaces > 2 || helper.IsSeamShape( node->getshapeId() )) double angle = inFaceDir.Angle( edge._normal ); // [0,PI]
for ( int iF = totalNbFaces-2; iF >=0; --iF ) edge._cosin = Cos( angle );
{ if ( totalNbFaces > 2 || helper.IsSeamShape( node->getshapeId() ))
F = TopoDS::Face( getMeshDS()->IndexToShape( id2Norm[ iF ].first )); for ( int iF = totalNbFaces-2; iF >=0; --iF )
inFaceDir = getFaceDir( F, V, node, helper, normOK ); {
if ( normOK ) { F = face2Norm[ iF ].first;
double angle = inFaceDir.Angle( edge._normal ); inFaceDir = getFaceDir( F, V, node, helper, normOK );
edge._cosin = Max( edge._cosin, Cos( angle )); if ( normOK ) {
} double angle = inFaceDir.Angle( edge._normal );
edge._cosin = Max( edge._cosin, Cos( angle ));
} }
//cout << "Cosin on VERTEX " << edge._cosin << " node " << node->GetID() << endl; }
break; //cout << "Cosin on VERTEX " << edge._cosin << " node " << node->GetID() << endl;
} break;
default: }
return error(SMESH_Comment("Invalid shape position of node ")<<node, data._index); default:
} return error(SMESH_Comment("Invalid shape position of node ")<<node, data._index);
} // case _sWOL.IsNull() }
double normSize = edge._normal.SquareModulus(); double normSize = edge._normal.SquareModulus();
if ( normSize < numeric_limits<double>::min() ) if ( normSize < numeric_limits<double>::min() )
@ -2887,6 +2944,15 @@ gp_XYZ _ViscousBuilder::getFaceNormal(const SMDS_MeshNode* node,
isOK = false; isOK = false;
Handle(Geom_Surface) surface = BRep_Tool::Surface( face ); Handle(Geom_Surface) surface = BRep_Tool::Surface( face );
if ( !shiftInside &&
helper.IsDegenShape( node->getshapeId() ) &&
getFaceNormalAtSingularity( uv, face, helper, normal ))
{
isOK = true;
return normal.XYZ();
}
int pointKind = GeomLib::NormEstim( surface, uv, 1e-5, normal ); int pointKind = GeomLib::NormEstim( surface, uv, 1e-5, normal );
enum { REGULAR = 0, QUASYSINGULAR, CONICAL, IMPOSSIBLE }; enum { REGULAR = 0, QUASYSINGULAR, CONICAL, IMPOSSIBLE };
@ -2935,6 +3001,55 @@ gp_XYZ _ViscousBuilder::getFaceNormal(const SMDS_MeshNode* node,
return normal.XYZ(); return normal.XYZ();
} }
//================================================================================
/*!
* \brief Try to get normal at a singularity of a surface basing on it's nature
*/
//================================================================================
bool _ViscousBuilder::getFaceNormalAtSingularity( const gp_XY& uv,
const TopoDS_Face& face,
SMESH_MesherHelper& helper,
gp_Dir& normal )
{
BRepAdaptor_Surface surface( face );
gp_Dir axis;
if ( !getRovolutionAxis( surface, axis ))
return false;
double f,l, d, du, dv;
f = surface.FirstUParameter();
l = surface.LastUParameter();
d = ( uv.X() - f ) / ( l - f );
du = ( d < 0.5 ? +1. : -1 ) * 1e-5 * ( l - f );
f = surface.FirstVParameter();
l = surface.LastVParameter();
d = ( uv.Y() - f ) / ( l - f );
dv = ( d < 0.5 ? +1. : -1 ) * 1e-5 * ( l - f );
gp_Dir refDir;
gp_Pnt2d testUV = uv;
enum { REGULAR = 0, QUASYSINGULAR, CONICAL, IMPOSSIBLE };
double tol = 1e-5;
Handle(Geom_Surface) geomsurf = surface.Surface().Surface();
for ( int iLoop = 0; true ; ++iLoop )
{
testUV.SetCoord( testUV.X() + du, testUV.Y() + dv );
if ( GeomLib::NormEstim( geomsurf, testUV, tol, refDir ) == REGULAR )
break;
if ( iLoop > 20 )
return false;
tol /= 10.;
}
if ( axis * refDir < 0. )
axis.Reverse();
normal = axis;
return true;
}
//================================================================================ //================================================================================
/*! /*!
* \brief Return a normal at a node weighted with angles taken by FACEs * \brief Return a normal at a node weighted with angles taken by FACEs
@ -2945,9 +3060,9 @@ gp_XYZ _ViscousBuilder::getFaceNormal(const SMDS_MeshNode* node,
*/ */
//================================================================================ //================================================================================
gp_XYZ _ViscousBuilder::getWeigthedNormal( const SMDS_MeshNode* n, gp_XYZ _ViscousBuilder::getWeigthedNormal( const SMDS_MeshNode* n,
std::pair< TGeomID, gp_XYZ > fId2Normal[], std::pair< TopoDS_Face, gp_XYZ > fId2Normal[],
int nbFaces ) int nbFaces )
{ {
gp_XYZ resNorm(0,0,0); gp_XYZ resNorm(0,0,0);
TopoDS_Shape V = SMESH_MesherHelper::GetSubShapeByNode( n, getMeshDS() ); TopoDS_Shape V = SMESH_MesherHelper::GetSubShapeByNode( n, getMeshDS() );
@ -2959,13 +3074,13 @@ gp_XYZ _ViscousBuilder::getWeigthedNormal( const SMDS_MeshNode* n,
} }
// exclude equal normals // exclude equal normals
int nbUniqNorms = nbFaces; //int nbUniqNorms = nbFaces;
for ( int i = 0; i < nbFaces; ++i ) for ( int i = 0; i < nbFaces; ++i )
for ( int j = i+1; j < nbFaces; ++j ) for ( int j = i+1; j < nbFaces; ++j )
if ( fId2Normal[i].second.IsEqual( fId2Normal[j].second, 0.1 )) if ( fId2Normal[i].second.IsEqual( fId2Normal[j].second, 0.1 ))
{ {
fId2Normal[i].second.SetCoord( 0,0,0 ); fId2Normal[i].second.SetCoord( 0,0,0 );
--nbUniqNorms; //--nbUniqNorms;
break; break;
} }
//if ( nbUniqNorms < 3 ) //if ( nbUniqNorms < 3 )
@ -2978,7 +3093,7 @@ gp_XYZ _ViscousBuilder::getWeigthedNormal( const SMDS_MeshNode* n,
double angles[30]; double angles[30];
for ( int i = 0; i < nbFaces; ++i ) for ( int i = 0; i < nbFaces; ++i )
{ {
const TopoDS_Face& F = TopoDS::Face( getMeshDS()->IndexToShape( fId2Normal[i].first )); const TopoDS_Face& F = fId2Normal[i].first;
// look for two EDGEs shared by F and other FACEs within fId2Normal // look for two EDGEs shared by F and other FACEs within fId2Normal
TopoDS_Edge ee[2]; TopoDS_Edge ee[2];
@ -2992,7 +3107,7 @@ gp_XYZ _ViscousBuilder::getWeigthedNormal( const SMDS_MeshNode* n,
for ( int j = 0; j < nbFaces && !isSharedEdge; ++j ) for ( int j = 0; j < nbFaces && !isSharedEdge; ++j )
{ {
if ( i == j ) continue; if ( i == j ) continue;
const TopoDS_Shape& otherF = getMeshDS()->IndexToShape( fId2Normal[j].first ); const TopoDS_Shape& otherF = fId2Normal[j].first;
isSharedEdge = SMESH_MesherHelper::IsSubShape( *E, otherF ); isSharedEdge = SMESH_MesherHelper::IsSubShape( *E, otherF );
} }
if ( !isSharedEdge ) if ( !isSharedEdge )