IPAL53401: BelongToGeom is very long on multiple lines

Use octree of classifiers
This commit is contained in:
eap 2016-06-21 19:42:11 +03:00
parent 79c2ca913c
commit afb6b35047
7 changed files with 262 additions and 98 deletions

View File

@ -39,6 +39,7 @@
#include <Basics_Utils.hxx> #include <Basics_Utils.hxx>
#include <BRepAdaptor_Surface.hxx> #include <BRepAdaptor_Surface.hxx>
#include <BRepBndLib.hxx>
#include <BRepClass_FaceClassifier.hxx> #include <BRepClass_FaceClassifier.hxx>
#include <BRep_Tool.hxx> #include <BRep_Tool.hxx>
#include <Geom_CylindricalSurface.hxx> #include <Geom_CylindricalSurface.hxx>
@ -4089,8 +4090,50 @@ bool ElementsOnSurface::isOnSurface( const SMDS_MeshNode* theNode )
ElementsOnShape ElementsOnShape
*/ */
ElementsOnShape::ElementsOnShape() struct ElementsOnShape::Classifier
: //myMesh(0), {
Classifier(const TopoDS_Shape& s, double tol) { Init(s,tol); }
void Init(const TopoDS_Shape& s, double tol);
bool IsOut(const gp_Pnt& p) { return myIsChecked = true, (this->*myIsOutFun)( p ); }
TopAbs_ShapeEnum ShapeType() const { return myShape.ShapeType(); }
Bnd_B3d* GetBndBox() { return & myBox; }
bool& IsChecked() { return myIsChecked; }
private:
bool isOutOfSolid (const gp_Pnt& p);
bool isOutOfBox (const gp_Pnt& p);
bool isOutOfFace (const gp_Pnt& p);
bool isOutOfEdge (const gp_Pnt& p);
bool isOutOfVertex(const gp_Pnt& p);
bool isBox (const TopoDS_Shape& s);
bool (Classifier::* myIsOutFun)(const gp_Pnt& p);
BRepClass3d_SolidClassifier mySolidClfr;
Bnd_B3d myBox;
GeomAPI_ProjectPointOnSurf myProjFace;
GeomAPI_ProjectPointOnCurve myProjEdge;
gp_Pnt myVertexXYZ;
TopoDS_Shape myShape;
double myTol;
bool myIsChecked;
};
struct ElementsOnShape::OctreeClassifier : public SMESH_Octree
{
OctreeClassifier( const std::vector< ElementsOnShape::Classifier* >& classifiers );
void GetClassifiersAtPoint( const gp_XYZ& p,
std::vector< ElementsOnShape::Classifier* >& classifiers );
protected:
OctreeClassifier() {}
SMESH_Octree* newChild() const { return new OctreeClassifier; }
void buildChildrenData();
Bnd_B3d* buildRootBox();
std::vector< ElementsOnShape::Classifier* > myClassifiers;
};
ElementsOnShape::ElementsOnShape():
myOctree(0),
myType(SMDSAbs_All), myType(SMDSAbs_All),
myToler(Precision::Confusion()), myToler(Precision::Confusion()),
myAllNodesFlag(false) myAllNodesFlag(false)
@ -4167,10 +4210,13 @@ void ElementsOnShape::setNodeIsOut( const SMDS_MeshNode* n, bool isOut )
void ElementsOnShape::SetShape (const TopoDS_Shape& theShape, void ElementsOnShape::SetShape (const TopoDS_Shape& theShape,
const SMDSAbs_ElementType theType) const SMDSAbs_ElementType theType)
{ {
bool shapeChanges = ( myShape != theShape );
myType = theType; myType = theType;
myShape = theShape; myShape = theShape;
if ( myShape.IsNull() ) return; if ( myShape.IsNull() ) return;
if ( shapeChanges )
{
TopTools_IndexedMapOfShape shapesMap; TopTools_IndexedMapOfShape shapesMap;
TopAbs_ShapeEnum shapeTypes[4] = { TopAbs_SOLID, TopAbs_FACE, TopAbs_EDGE, TopAbs_VERTEX }; TopAbs_ShapeEnum shapeTypes[4] = { TopAbs_SOLID, TopAbs_FACE, TopAbs_EDGE, TopAbs_VERTEX };
TopExp_Explorer sub; TopExp_Explorer sub;
@ -4187,7 +4233,8 @@ void ElementsOnShape::SetShape (const TopoDS_Shape& theShape,
clearClassifiers(); clearClassifiers();
myClassifiers.resize( shapesMap.Extent() ); myClassifiers.resize( shapesMap.Extent() );
for ( int i = 0; i < shapesMap.Extent(); ++i ) for ( int i = 0; i < shapesMap.Extent(); ++i )
myClassifiers[ i ] = new TClassifier( shapesMap( i+1 ), myToler ); myClassifiers[ i ] = new Classifier( shapesMap( i+1 ), myToler );
}
if ( theType == SMDSAbs_Node ) if ( theType == SMDSAbs_Node )
{ {
@ -4205,6 +4252,9 @@ void ElementsOnShape::clearClassifiers()
for ( size_t i = 0; i < myClassifiers.size(); ++i ) for ( size_t i = 0; i < myClassifiers.size(); ++i )
delete myClassifiers[ i ]; delete myClassifiers[ i ];
myClassifiers.clear(); myClassifiers.clear();
delete myOctree;
myOctree = 0;
} }
bool ElementsOnShape::IsSatisfy (long elemId) bool ElementsOnShape::IsSatisfy (long elemId)
@ -4219,6 +4269,11 @@ bool ElementsOnShape::IsSatisfy (long elemId)
gp_XYZ centerXYZ (0, 0, 0); gp_XYZ centerXYZ (0, 0, 0);
if ( !myOctree && myClassifiers.size() > 5 )
myOctree = new OctreeClassifier( myClassifiers );
std::vector< Classifier* >& classifiers = myOctree ? myWorkClassifiers : myClassifiers;
SMDS_ElemIteratorPtr aNodeItr = elem->nodesIterator(); SMDS_ElemIteratorPtr aNodeItr = elem->nodesIterator();
while (aNodeItr->more() && (isSatisfy == myAllNodesFlag)) while (aNodeItr->more() && (isSatisfy == myAllNodesFlag))
{ {
@ -4228,8 +4283,17 @@ bool ElementsOnShape::IsSatisfy (long elemId)
isNodeOut = true; isNodeOut = true;
if ( !getNodeIsOut( aPnt._node, isNodeOut )) if ( !getNodeIsOut( aPnt._node, isNodeOut ))
{ {
for ( size_t i = 0; i < myClassifiers.size() && isNodeOut; ++i ) if ( myOctree )
isNodeOut = myClassifiers[i]->IsOut( aPnt ); {
myWorkClassifiers.clear();
myOctree->GetClassifiersAtPoint( aPnt, myWorkClassifiers );
}
for ( size_t i = 0; i < classifiers.size(); ++i )
classifiers[i]->IsChecked() = false;
for ( size_t i = 0; i < classifiers.size() && isNodeOut; ++i )
if ( !classifiers[i]->IsChecked() )
isNodeOut = classifiers[i]->IsOut( aPnt );
setNodeIsOut( aPnt._node, isNodeOut ); setNodeIsOut( aPnt._node, isNodeOut );
} }
@ -4243,78 +4307,90 @@ bool ElementsOnShape::IsSatisfy (long elemId)
{ {
centerXYZ /= elem->NbNodes(); centerXYZ /= elem->NbNodes();
isSatisfy = false; isSatisfy = false;
for ( size_t i = 0; i < myClassifiers.size() && !isSatisfy; ++i ) for ( size_t i = 0; i < classifiers.size() && !isSatisfy; ++i )
isSatisfy = ! myClassifiers[i]->IsOut( centerXYZ ); isSatisfy = ! classifiers[i]->IsOut( centerXYZ );
} }
return isSatisfy; return isSatisfy;
} }
TopAbs_ShapeEnum ElementsOnShape::TClassifier::ShapeType() const void ElementsOnShape::Classifier::Init (const TopoDS_Shape& theShape, double theTol)
{
return myShape.ShapeType();
}
bool ElementsOnShape::TClassifier::IsOut(const gp_Pnt& p)
{
return (this->*myIsOutFun)( p );
}
void ElementsOnShape::TClassifier::Init (const TopoDS_Shape& theShape, double theTol)
{ {
myShape = theShape; myShape = theShape;
myTol = theTol; myTol = theTol;
bool isShapeBox = false;
switch ( myShape.ShapeType() ) switch ( myShape.ShapeType() )
{ {
case TopAbs_SOLID: { case TopAbs_SOLID:
if ( isBox( theShape ))
{ {
myIsOutFun = & ElementsOnShape::TClassifier::isOutOfBox; if (( isShapeBox = isBox( theShape )))
{
myIsOutFun = & ElementsOnShape::Classifier::isOutOfBox;
} }
else else
{ {
mySolidClfr.Load(theShape); mySolidClfr.Load(theShape);
myIsOutFun = & ElementsOnShape::TClassifier::isOutOfSolid; myIsOutFun = & ElementsOnShape::Classifier::isOutOfSolid;
} }
break; break;
} }
case TopAbs_FACE: { case TopAbs_FACE:
{
Standard_Real u1,u2,v1,v2; Standard_Real u1,u2,v1,v2;
Handle(Geom_Surface) surf = BRep_Tool::Surface( TopoDS::Face( theShape )); Handle(Geom_Surface) surf = BRep_Tool::Surface( TopoDS::Face( theShape ));
surf->Bounds( u1,u2,v1,v2 ); surf->Bounds( u1,u2,v1,v2 );
myProjFace.Init(surf, u1,u2, v1,v2, myTol ); myProjFace.Init(surf, u1,u2, v1,v2, myTol );
myIsOutFun = & ElementsOnShape::TClassifier::isOutOfFace; myIsOutFun = & ElementsOnShape::Classifier::isOutOfFace;
break; break;
} }
case TopAbs_EDGE: { case TopAbs_EDGE:
{
Standard_Real u1, u2; Standard_Real u1, u2;
Handle(Geom_Curve) curve = BRep_Tool::Curve( TopoDS::Edge( theShape ), u1, u2); Handle(Geom_Curve) curve = BRep_Tool::Curve( TopoDS::Edge( theShape ), u1, u2);
myProjEdge.Init(curve, u1, u2); myProjEdge.Init(curve, u1, u2);
myIsOutFun = & ElementsOnShape::TClassifier::isOutOfEdge; myIsOutFun = & ElementsOnShape::Classifier::isOutOfEdge;
break; break;
} }
case TopAbs_VERTEX:{ case TopAbs_VERTEX:
{
myVertexXYZ = BRep_Tool::Pnt( TopoDS::Vertex( theShape ) ); myVertexXYZ = BRep_Tool::Pnt( TopoDS::Vertex( theShape ) );
myIsOutFun = & ElementsOnShape::TClassifier::isOutOfVertex; myIsOutFun = & ElementsOnShape::Classifier::isOutOfVertex;
break; break;
} }
default: default:
throw SALOME_Exception("Programmer error in usage of ElementsOnShape::TClassifier"); throw SALOME_Exception("Programmer error in usage of ElementsOnShape::Classifier");
}
if ( !isShapeBox )
{
Bnd_Box box;
BRepBndLib::Add( myShape, box );
myBox.Clear();
myBox.Add( box.CornerMin() );
myBox.Add( box.CornerMax() );
gp_XYZ halfSize = 0.5 * ( box.CornerMax().XYZ() - box.CornerMin().XYZ() );
for ( int iDim = 1; iDim <= 3; ++iDim )
{
double x = halfSize.Coord( iDim );
halfSize.SetCoord( iDim, x + Max( myTol, 1e-2 * x ));
}
myBox.SetHSize( halfSize );
} }
} }
bool ElementsOnShape::TClassifier::isOutOfSolid (const gp_Pnt& p) bool ElementsOnShape::Classifier::isOutOfSolid (const gp_Pnt& p)
{ {
mySolidClfr.Perform( p, myTol ); mySolidClfr.Perform( p, myTol );
return ( mySolidClfr.State() != TopAbs_IN && mySolidClfr.State() != TopAbs_ON ); return ( mySolidClfr.State() != TopAbs_IN && mySolidClfr.State() != TopAbs_ON );
} }
bool ElementsOnShape::TClassifier::isOutOfBox (const gp_Pnt& p) bool ElementsOnShape::Classifier::isOutOfBox (const gp_Pnt& p)
{ {
return myBox.IsOut( p.XYZ() ); return myBox.IsOut( p.XYZ() );
} }
bool ElementsOnShape::TClassifier::isOutOfFace (const gp_Pnt& p) bool ElementsOnShape::Classifier::isOutOfFace (const gp_Pnt& p)
{ {
myProjFace.Perform( p ); myProjFace.Perform( p );
if ( myProjFace.IsDone() && myProjFace.LowerDistance() <= myTol ) if ( myProjFace.IsDone() && myProjFace.LowerDistance() <= myTol )
@ -4330,18 +4406,18 @@ bool ElementsOnShape::TClassifier::isOutOfFace (const gp_Pnt& p)
return true; return true;
} }
bool ElementsOnShape::TClassifier::isOutOfEdge (const gp_Pnt& p) bool ElementsOnShape::Classifier::isOutOfEdge (const gp_Pnt& p)
{ {
myProjEdge.Perform( p ); myProjEdge.Perform( p );
return ! ( myProjEdge.NbPoints() > 0 && myProjEdge.LowerDistance() <= myTol ); return ! ( myProjEdge.NbPoints() > 0 && myProjEdge.LowerDistance() <= myTol );
} }
bool ElementsOnShape::TClassifier::isOutOfVertex(const gp_Pnt& p) bool ElementsOnShape::Classifier::isOutOfVertex(const gp_Pnt& p)
{ {
return ( myVertexXYZ.Distance( p ) > myTol ); return ( myVertexXYZ.Distance( p ) > myTol );
} }
bool ElementsOnShape::TClassifier::isBox (const TopoDS_Shape& theShape) bool ElementsOnShape::Classifier::isBox (const TopoDS_Shape& theShape)
{ {
TopTools_IndexedMapOfShape vMap; TopTools_IndexedMapOfShape vMap;
TopExp::MapShapes( theShape, TopAbs_VERTEX, vMap ); TopExp::MapShapes( theShape, TopAbs_VERTEX, vMap );
@ -4368,6 +4444,67 @@ bool ElementsOnShape::TClassifier::isBox (const TopoDS_Shape& theShape)
return true; return true;
} }
ElementsOnShape::
OctreeClassifier::OctreeClassifier( const std::vector< ElementsOnShape::Classifier* >& classifiers )
:SMESH_Octree( new SMESH_TreeLimit )
{
myClassifiers = classifiers;
compute();
}
void ElementsOnShape::
OctreeClassifier::GetClassifiersAtPoint( const gp_XYZ& point,
std::vector< ElementsOnShape::Classifier* >& result )
{
if ( getBox()->IsOut( point ))
return;
if ( isLeaf() )
{
for ( size_t i = 0; i < myClassifiers.size(); ++i )
if ( !myClassifiers[i]->GetBndBox()->IsOut( point ))
result.push_back( myClassifiers[i] );
}
else
{
for (int i = 0; i < nbChildren(); i++)
((OctreeClassifier*) myChildren[i])->GetClassifiersAtPoint( point, result );
}
}
void ElementsOnShape::OctreeClassifier::buildChildrenData()
{
// distribute myClassifiers among myChildren
for ( size_t i = 0; i < myClassifiers.size(); ++i )
{
for (int j = 0; j < nbChildren(); j++)
{
if ( !myClassifiers[i]->GetBndBox()->IsOut( *myChildren[j]->getBox() ))
{
((OctreeClassifier*)myChildren[j])->myClassifiers.push_back( myClassifiers[i]);
}
}
}
SMESHUtils::FreeVector( myClassifiers );
// define if a child isLeaf()
for (int i = 0; i < nbChildren(); i++)
{
OctreeClassifier* child = static_cast<OctreeClassifier*>( myChildren[i]);
child->myIsLeaf = ( child->myClassifiers.size() <= 5 );
if ( child->myClassifiers.capacity() - child->myClassifiers.size() > 100 )
SMESHUtils::CompactVector( child->myClassifiers );
}
}
Bnd_B3d* ElementsOnShape::OctreeClassifier::buildRootBox()
{
Bnd_B3d* box = new Bnd_B3d;
for ( size_t i = 0; i < myClassifiers.size(); ++i )
box->Add( *myClassifiers[i]->GetBndBox() );
return box;
}
/* /*
Class : BelongToGeom Class : BelongToGeom
@ -4432,6 +4569,7 @@ void BelongToGeom::init()
//if (!myIsSubshape) // to be always ready to check an element not bound to geometry //if (!myIsSubshape) // to be always ready to check an element not bound to geometry
{ {
if ( !myElementsOnShapePtr )
myElementsOnShapePtr.reset( new ElementsOnShape() ); myElementsOnShapePtr.reset( new ElementsOnShape() );
myElementsOnShapePtr->SetTolerance( myTolerance ); myElementsOnShapePtr->SetTolerance( myTolerance );
myElementsOnShapePtr->SetAllNodes( true ); // "belong", while false means "lays on" myElementsOnShapePtr->SetAllNodes( true ); // "belong", while false means "lays on"
@ -4470,7 +4608,8 @@ bool BelongToGeom::IsSatisfy (long theId)
return myElementsOnShapePtr->IsSatisfy(theId); return myElementsOnShapePtr->IsSatisfy(theId);
} }
// Case of submesh // Case of sub-mesh
if (myType == SMDSAbs_Node) if (myType == SMDSAbs_Node)
{ {
if( const SMDS_MeshNode* aNode = myMeshDS->FindNode( theId ) ) if( const SMDS_MeshNode* aNode = myMeshDS->FindNode( theId ) )
@ -4605,8 +4744,9 @@ void LyingOnGeom::init()
mySubShapesIDs.Add( subID ); mySubShapesIDs.Add( subID );
} }
} }
else // else // to be always ready to check an element not bound to geometry
{ {
if ( !myElementsOnShapePtr )
myElementsOnShapePtr.reset( new ElementsOnShape() ); myElementsOnShapePtr.reset( new ElementsOnShape() );
myElementsOnShapePtr->SetTolerance( myTolerance ); myElementsOnShapePtr->SetTolerance( myTolerance );
myElementsOnShapePtr->SetAllNodes( false ); // lays on, while true means "belong" myElementsOnShapePtr->SetAllNodes( false ); // lays on, while true means "belong"

View File

@ -871,34 +871,15 @@ namespace SMESH{
private: private:
struct TClassifier struct Classifier;
{ struct OctreeClassifier;
TClassifier(const TopoDS_Shape& s, double tol) { Init(s,tol); }
void Init(const TopoDS_Shape& s, double tol);
bool IsOut(const gp_Pnt& p);
TopAbs_ShapeEnum ShapeType() const;
private:
bool isOutOfSolid (const gp_Pnt& p);
bool isOutOfBox (const gp_Pnt& p);
bool isOutOfFace (const gp_Pnt& p);
bool isOutOfEdge (const gp_Pnt& p);
bool isOutOfVertex(const gp_Pnt& p);
bool isBox (const TopoDS_Shape& s);
bool (TClassifier::* myIsOutFun)(const gp_Pnt& p);
BRepClass3d_SolidClassifier mySolidClfr;
Bnd_B3d myBox;
GeomAPI_ProjectPointOnSurf myProjFace;
GeomAPI_ProjectPointOnCurve myProjEdge;
gp_Pnt myVertexXYZ;
TopoDS_Shape myShape;
double myTol;
};
void clearClassifiers(); void clearClassifiers();
bool getNodeIsOut( const SMDS_MeshNode* n, bool& isOut ); bool getNodeIsOut( const SMDS_MeshNode* n, bool& isOut );
void setNodeIsOut( const SMDS_MeshNode* n, bool isOut ); void setNodeIsOut( const SMDS_MeshNode* n, bool isOut );
std::vector< TClassifier* > myClassifiers; std::vector< Classifier* > myClassifiers, myWorkClassifiers;
OctreeClassifier* myOctree;
SMDSAbs_ElementType myType; SMDSAbs_ElementType myType;
TopoDS_Shape myShape; TopoDS_Shape myShape;
double myToler; double myToler;

View File

@ -77,3 +77,20 @@ double SMESH_Octree::maxSize() const
} }
return 0.; return 0.;
} }
//================================================================================
/*!
* \brief Change size of a box by a factor; each dimension changes independently of others
*/
//================================================================================
void SMESH_Octree::enlargeByFactor( Bnd_B3d* box, double factor ) const
{
if ( !box->IsVoid() )
{
gp_XYZ halfSize = 0.5 * ( box->CornerMax() - box->CornerMin() );
for ( int iDim = 1; iDim <= 3; ++iDim )
halfSize.SetCoord( iDim, factor * halfSize.Coord( iDim ));
box->SetHSize( halfSize );
}
}

View File

@ -37,9 +37,9 @@
/*! /*!
* \brief 3D tree of anything. * \brief 3D tree of anything.
* Methods to implement in a descendant are: * Methods to implement in a descendant are:
* - Bnd_B3d* buildRootBox(); // box of the whole tree * - Bnd_B3d* buildRootBox(); // box of a tree
* - descendant* newChild() const; // a new child instance * - descendant* newChild() const; // a new child instance
* - void buildChildrenData(); // Fill in data of the children * - void buildChildrenData(); // distribute own data among children
*/ */
class SMESHUtils_EXPORT SMESH_Octree : public SMESH_Tree< Bnd_B3d, 8 > class SMESHUtils_EXPORT SMESH_Octree : public SMESH_Tree< Bnd_B3d, 8 >
{ {
@ -61,6 +61,9 @@ public:
// Allocate a bndbox according to childIndex. childIndex is zero based // Allocate a bndbox according to childIndex. childIndex is zero based
virtual Bnd_B3d* newChildBox(int childIndex) const; virtual Bnd_B3d* newChildBox(int childIndex) const;
// Change size of a box by a factor; each dimension changes independently of others
virtual void enlargeByFactor( Bnd_B3d* box, double factor ) const;
}; };
//================================================================================ //================================================================================

View File

@ -74,3 +74,20 @@ double SMESH_Quadtree::maxSize() const
} }
return 0.; return 0.;
} }
//================================================================================
/*!
* \brief Change size of a box by a factor; each dimension changes independently of others
*/
//================================================================================
void SMESH_Quadtree::enlargeByFactor( Bnd_B2d* box, double factor ) const
{
if ( !box->IsVoid() )
{
gp_XY halfSize = 0.5 * ( box->CornerMax() - box->CornerMin() );
for ( int iDim = 1; iDim <= 2; ++iDim )
halfSize.SetCoord( iDim, factor * halfSize.Coord( iDim ));
box->SetHSize( halfSize );
}
}

View File

@ -57,6 +57,9 @@ public:
// Allocate a bndbox according to childIndex. childIndex is zero based // Allocate a bndbox according to childIndex. childIndex is zero based
virtual Bnd_B2d* newChildBox(int childIndex) const; virtual Bnd_B2d* newChildBox(int childIndex) const;
// Change size of a box by a factor; each dimension changes independently of others
virtual void enlargeByFactor( Bnd_B2d* box, double factor ) const;
}; };
//================================================================================ //================================================================================

View File

@ -98,6 +98,9 @@ protected:
// Allocate a bndbox according to childIndex. childIndex is zero based // Allocate a bndbox according to childIndex. childIndex is zero based
virtual box_type* newChildBox(int childIndex) const = 0; virtual box_type* newChildBox(int childIndex) const = 0;
// Change size of a box by a factor; each dimension changes independently of others
virtual void enlargeByFactor( box_type* box, double factor ) const = 0;
// Fill in data of the children // Fill in data of the children
virtual void buildChildrenData() = 0; virtual void buildChildrenData() = 0;
@ -205,13 +208,13 @@ void SMESH_Tree<BND_BOX,NB_CHILDREN>::buildChildren()
myChildren = new SMESH_Tree*[NB_CHILDREN]; myChildren = new SMESH_Tree*[NB_CHILDREN];
// get the whole model size // get the whole model size
double rootSize = 0; // double rootSize = 0;
{ // {
SMESH_Tree* root = this; // SMESH_Tree* root = this;
while ( root->myLevel > 0 ) // while ( root->myLevel > 0 )
root = root->myFather; // root = root->myFather;
rootSize = root->maxSize(); // rootSize = root->maxSize();
} // }
for (int i = 0; i < NB_CHILDREN; i++) for (int i = 0; i < NB_CHILDREN; i++)
{ {
// The child is of the same type than its father (For instance, a SMESH_OctreeNode) // The child is of the same type than its father (For instance, a SMESH_OctreeNode)
@ -224,7 +227,7 @@ void SMESH_Tree<BND_BOX,NB_CHILDREN>::buildChildren()
myChildren[i]->myLimit = myLimit; myChildren[i]->myLimit = myLimit;
myChildren[i]->myLevel = myLevel + 1; myChildren[i]->myLevel = myLevel + 1;
myChildren[i]->myBox = newChildBox( i ); myChildren[i]->myBox = newChildBox( i );
myChildren[i]->myBox->Enlarge( rootSize * 1e-10 ); enlargeByFactor( myChildren[i]->myBox, 1. + 1e-10 );
if ( myLimit->myMinBoxSize > 0. && myChildren[i]->maxSize() <= myLimit->myMinBoxSize ) if ( myLimit->myMinBoxSize > 0. && myChildren[i]->maxSize() <= myLimit->myMinBoxSize )
myChildren[i]->myIsLeaf = true; myChildren[i]->myIsLeaf = true;
} }