From cd0705f6600306acdf53ac4f596ac6c4f77a766b Mon Sep 17 00:00:00 2001 From: emv Date: Thu, 3 Sep 2020 14:37:29 +0300 Subject: [PATCH] 0024789: BOPAlgo_Builder produces invalid result on planar, conical and spherical surfaces BOPAlgo_PaveFiller::MakeBlocks() - Improve Face/Face post treatment procedure by: * Adding more existing edge for intersection with section edges to resolve all possible intersections. * Using tree for edges selection. * Avoiding intersection of the existing edges among themselves. IntPatch_ImpImpIntersection::CyCyNoGeometric - add more points to wline to make it more smooth. --- src/BOPAlgo/BOPAlgo_PaveFiller.cxx | 3 +- src/BOPAlgo/BOPAlgo_PaveFiller.hxx | 43 +- src/BOPAlgo/BOPAlgo_PaveFiller_5.cxx | 23 +- src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx | 394 ++++++++++++------ src/BOPAlgo/BOPAlgo_WireSplitter_1.cxx | 2 +- src/BOPTools/BOPTools_AlgoTools.cxx | 48 +-- .../IntPatch_ImpImpIntersection_4.gxx | 10 +- src/IntTools/IntTools_BeanFaceIntersector.cxx | 28 +- src/IntTools/IntTools_BeanFaceIntersector.hxx | 14 +- src/IntTools/IntTools_EdgeFace.cxx | 6 +- src/IntTools/IntTools_EdgeFace.hxx | 6 + tests/boolean/bopfuse_complex/F1 | 3 +- tests/boolean/boptuc_complex/B5 | 9 +- tests/bugs/modalg_7/bug24789 | 32 ++ tests/bugs/modalg_7/bug31662 | 4 +- 15 files changed, 427 insertions(+), 198 deletions(-) create mode 100644 tests/bugs/modalg_7/bug24789 diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller.cxx b/src/BOPAlgo/BOPAlgo_PaveFiller.cxx index a12e527c4d..faf4ce3ee9 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller.cxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller.cxx @@ -57,7 +57,8 @@ BOPAlgo_PaveFiller::BOPAlgo_PaveFiller BOPAlgo_Algo(theAllocator), myFPBDone(1, theAllocator), myIncreasedSS(1, theAllocator), - myVertsToAvoidExtension(1, theAllocator) + myVertsToAvoidExtension(1, theAllocator), + myDistances(1, theAllocator) { myDS = NULL; myIterator = NULL; diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller.hxx b/src/BOPAlgo/BOPAlgo_PaveFiller.hxx index b31a11a79d..307f2e5031 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller.hxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller.hxx @@ -35,6 +35,7 @@ #include #include #include +#include #include #include #include @@ -319,7 +320,9 @@ protected: //! created the section curve. Standard_EXPORT Standard_Boolean IsExistingPaveBlock (const Handle(BOPDS_PaveBlock)& thePB, const BOPDS_Curve& theNC, - const Standard_Real theTolR3D, const BOPDS_IndexedMapOfPaveBlock& theMPB, + const Standard_Real theTolR3D, + const BOPDS_IndexedMapOfPaveBlock& theMPB, + BOPTools_BoxTree& thePBTree, const BOPDS_MapOfPaveBlock& theMPBCommon, Handle(BOPDS_PaveBlock)& thePBOut, Standard_Real& theTolNew); @@ -412,6 +415,21 @@ protected: TColStd_DataMapOfIntegerListOfInteger& aDMVLV, const Standard_Integer aType = 0); + //! Adds the existing edges for intersection with section edges + //! by checking the possible intersection with the faces comparing + //! pre-saved E-F distances with new tolerances. + Standard_EXPORT void ProcessExistingPaveBlocks (const Standard_Integer theInt, + const Standard_Integer theCur, + const Standard_Integer nF1, + const Standard_Integer nF2, + const TopoDS_Edge& theES, + const BOPDS_IndexedMapOfPaveBlock& theMPBOnIn, + BOPTools_BoxTree& thePBTree, + BOPDS_IndexedDataMapOfShapeCoupleOfPaveBlocks& theMSCPB, + TopTools_DataMapOfShapeInteger& theMVI, + BOPDS_ListOfPaveBlock& theLPBC, + BOPAlgo_DataMapOfPaveBlockListOfInteger& thePBFacesMap, + BOPDS_MapOfPaveBlock& theMPB); //! Adds the existing edges from the map which interfere //! with the vertices from map to the post treatment of section edges. @@ -419,12 +437,12 @@ protected: const Standard_Integer nF1, const Standard_Integer nF2, const BOPDS_IndexedMapOfPaveBlock& theMPBOnIn, + BOPTools_BoxTree& thePBTree, const TColStd_DataMapOfIntegerListOfInteger& theDMBV, BOPDS_IndexedDataMapOfShapeCoupleOfPaveBlocks& theMSCPB, TopTools_DataMapOfShapeInteger& theMVI, BOPAlgo_DataMapOfPaveBlockListOfInteger& thePBFacesMap, BOPDS_MapOfPaveBlock& theMPB); - //! Replaces existing pave block with new pave blocks . //! The list contains images of which were created in @@ -599,6 +617,22 @@ protected: //! Check all edges on the micro status and remove the positive ones Standard_EXPORT void RemoveMicroEdges(); + //! Auxiliary structure to hold the edge distance to the face + struct EdgeRangeDistance + { + Standard_Real First; + Standard_Real Last; + Standard_Real Distance; + + EdgeRangeDistance (const Standard_Real theFirst = 0.0, + const Standard_Real theLast = 0.0, + const Standard_Real theDistance = RealLast()) + : First (theFirst), Last (theLast), Distance (theDistance) + {} + }; + +protected: //! Fields + TopTools_ListOfShape myArguments; BOPDS_PDS myDS; BOPDS_PIterator myIterator; @@ -615,6 +649,11 @@ protected: //! which has already been extended to cover the real intersection //! points, and should not be extended any longer to be put //! on a section curve. + + NCollection_DataMap , + BOPDS_PairMapHasher> myDistances; //!< Map to store minimal distances between shapes + //! which have no real intersections }; diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller_5.cxx b/src/BOPAlgo/BOPAlgo_PaveFiller_5.cxx index f06e5a5860..550744b742 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller_5.cxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller_5.cxx @@ -300,12 +300,6 @@ void BOPAlgo_PaveFiller::PerformEF() continue; } // - const IntTools_SequenceOfCommonPrts& aCPrts=aEdgeFace.CommonParts(); - aNbCPrts = aCPrts.Length(); - if (!aNbCPrts) { - continue; - } - // aEdgeFace.Indices(nE, nF); // const TopoDS_Edge& aE=aEdgeFace.Edge(); @@ -313,6 +307,23 @@ void BOPAlgo_PaveFiller::PerformEF() // aTolE=BRep_Tool::Tolerance(aE); aTolF=BRep_Tool::Tolerance(aF); + // + const IntTools_SequenceOfCommonPrts& aCPrts=aEdgeFace.CommonParts(); + aNbCPrts = aCPrts.Length(); + if (!aNbCPrts) { + if (aEdgeFace.MinimalDistance() < RealLast() && + aEdgeFace.MinimalDistance() > aTolE + aTolF) + { + const Handle(BOPDS_PaveBlock)& aPB=aEdgeFace.PaveBlock(); + aPB->Range(aT1, aT2); + NCollection_List* pList = myDistances.ChangeSeek (BOPDS_Pair (nE, nF)); + if (!pList) + pList = myDistances.Bound (BOPDS_Pair (nE, nF), NCollection_List()); + pList->Append (EdgeRangeDistance (aT1, aT2, aEdgeFace.MinimalDistance())); + } + continue; + } + // const IntTools_Range& anewSR=aEdgeFace.NewSR(); Handle(BOPDS_PaveBlock)& aPB=aEdgeFace.PaveBlock(); // diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx b/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx index e5689c50c9..799709a2bf 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx @@ -38,7 +38,9 @@ #include #include #include +#include #include +#include #include #include #include @@ -500,7 +502,6 @@ void BOPAlgo_PaveFiller::MakeBlocks() myDS->SubShapesOnIn(nF1, nF2, aMVOnIn, aMVCommon, aMPBOnIn, aMPBCommon); myDS->SharedEdges(nF1, nF2, aLSE, aAllocator); // - Standard_Boolean bHasRealSectionEdge = Standard_False; // 1. Treat Points for (j=0; jHasEdge()) + continue; + + if (myDS->ShapeInfo (aPB->OriginalEdge()).HasFlag()) + continue; + + aPBTree.Add (iPB, Bnd_Tools::Bnd2BVH (myDS->ShapeInfo (aPB->Edge()).Box())); + } + } + // // 3. Make section edges for (j=0; jIsEmpty() || !pFaces->Contains(nF)) pFaces->Append(nF); + // Try fusing the vertices of the existing pave block + // with the vertices put on the real section curve (except + // for technological vertices, which will be removed) + Standard_Integer nVOut1, nVOut2; + aPBOut->Indices(nVOut1, nVOut2); + if (nV1 != nVOut1 && nV1 != nVOut2 && !aMVBounds.Contains(nV1)) + { + aVertsOnRejectedPB.Add(aV1); + } + if (nV2 != nVOut1 && nV2 != nVOut2 && !aMVBounds.Contains(nV2)) + { + aVertsOnRejectedPB.Add(aV2); + } + if (aMPBAdd.Add(aPBOut)) { // Add edge for processing as the section edge PreparePostTreatFF(i, j, aPBOut, aMSCPB, aMVI, aLPBC); - // Try fusing the vertices of the existing pave block - // with the vertices put on the real section curve (except - // for technological vertices, which will be removed) - Standard_Integer nVOut1, nVOut2; - aPBOut->Indices(nVOut1, nVOut2); - if (nV1 != nVOut1 && nV1 != nVOut2 && !aMVBounds.Contains(nV1)) - { - aVertsOnRejectedPB.Add(aV1); - } - if (nV2 != nVOut1 && nV2 != nVOut2 && !aMVBounds.Contains(nV2)) - { - aVertsOnRejectedPB.Add(aV2); - } } } continue; @@ -715,8 +738,10 @@ void BOPAlgo_PaveFiller::MakeBlocks() // aMVTol.UnBind(nV1); aMVTol.UnBind(nV2); - // - bHasRealSectionEdge = Standard_True; + + // Add existing pave blocks for post treatment + ProcessExistingPaveBlocks (i, j, nF1, nF2, aES, aMPBOnIn, aPBTree, + aMSCPB, aMVI, aLPBC, aPBFacesMap, aMPBAdd); } // aLPBC.RemoveFirst(); @@ -744,29 +769,7 @@ void BOPAlgo_PaveFiller::MakeBlocks() aDMVLV.UnBind(nV1); } // - ProcessExistingPaveBlocks(i, nF1, nF2, aMPBOnIn, aDMBV, aMSCPB, aMVI, aPBFacesMap, aMPBAdd); - // - // If the pair of faces has produced any real section edges - // it is necessary to check if these edges do not intersect - // any common IN edges of the faces. For that, all such edges - // are added for Post Treatment along with sections edges. - if (bHasRealSectionEdge) { - const BOPDS_IndexedMapOfPaveBlock& aMPBIn1 = aFI1.PaveBlocksIn(); - const BOPDS_IndexedMapOfPaveBlock& aMPBIn2 = aFI2.PaveBlocksIn(); - // - // For simplicity add all IN edges into the first set of section curves. - // These existing edges will be removed from the set on the post treatment - // stage in the UpdateFaceInfo function. - BOPDS_ListOfPaveBlock& aLPBC = aVC.ChangeValue(0).ChangePaveBlocks(); - // - Standard_Integer aNbIn1 = aMPBIn1.Extent(); - for (j = 1; j <= aNbIn1; ++j) { - const Handle(BOPDS_PaveBlock)& aPB = aMPBIn1(j); - if (aMPBIn2.Contains(aPB) && aMPBAdd.Add(aPB)) { - PreparePostTreatFF(i, 0, aPB, aMSCPB, aMVI, aLPBC); - } - } - } + ProcessExistingPaveBlocks(i, nF1, nF2, aMPBOnIn, aPBTree, aDMBV, aMSCPB, aMVI, aPBFacesMap, aMPBAdd); }//for (i=0; i 0; --k) { const TopoDS_Shape& aS=theMSCPB.FindKey(k); - aLS.Append(aS); + const Handle(BOPDS_PaveBlock)& aPB = theMSCPB (k).PaveBlock1(); + if (!aPB.IsNull() && aPB->HasEdge()) + BRep_Builder().Add (anExistingEdges, aS); + else + aLS.Append(aS); // add vertices-candidates for SD from the map aDMNewSD, // so that they took part in fuse operation. TopoDS_Iterator itV(aS); @@ -948,6 +958,8 @@ void BOPAlgo_PaveFiller::PostTreatFF } } } + if (anExistingEdges.NbChildren() > 0) + aLS.Append (anExistingEdges); // // The section edges considered as a micro should be // specially treated - their vertices should be united and @@ -1029,6 +1041,12 @@ void BOPAlgo_PaveFiller::PostTreatFF aItLS.Initialize(aLS); for (; aItLS.More(); aItLS.Next()) { const TopoDS_Shape& aSx=aItLS.Value(); + if (aSx.ShapeType() == TopAbs_COMPOUND) + { + for (TopoDS_Iterator itC (aSx); itC.More(); itC.Next()) + aLS.Append (itC.Value()); + continue; + } nSx=aPDS->Index(aSx); const BOPDS_ShapeInfo& aSIx=aPDS->ShapeInfo(nSx); // @@ -1596,108 +1614,124 @@ Standard_Boolean BOPAlgo_PaveFiller::IsExistingPaveBlock const BOPDS_Curve& theNC, const Standard_Real theTolR3D, const BOPDS_IndexedMapOfPaveBlock& theMPBOnIn, + BOPTools_BoxTree& thePBTree, const BOPDS_MapOfPaveBlock& theMPBCommon, Handle(BOPDS_PaveBlock)& aPBOut, Standard_Real& theTolNew) { - Standard_Boolean bRet; - Standard_Real aT1, aT2, aTm, aTx, aTolCheck; - Standard_Integer nSp, iFlag1, iFlag2, nV11, nV12, nV21, nV22, i, aNbPB; - gp_Pnt aP1, aPm, aP2; - Bnd_Box aBoxP1, aBoxPm, aBoxP2, aBoxTmp; - // - bRet=Standard_False; - aTolCheck = theTolR3D + myFuzzyValue; const IntTools_Curve& aIC=theNC.Curve(); - // + + Standard_Real aT1, aT2; thePB->Range(aT1, aT2); - thePB->Indices(nV11, nV12); - const Standard_Real aTolV11 = BRep_Tool::Tolerance(TopoDS::Vertex(myDS->Shape(nV11))); - const Standard_Real aTolV12 = BRep_Tool::Tolerance(TopoDS::Vertex(myDS->Shape(nV12))); - const Standard_Real aTolV1 = Max(aTolV11, aTolV12) + myFuzzyValue; + + Standard_Integer nV11, nV12; + thePB->Indices (nV11, nV12); //first point - aIC.D0(aT1, aP1); - aBoxP1.Add(aP1); - aBoxP1.Enlarge(aTolV11); + Bnd_Box aBoxP1; + gp_Pnt aP1; + aIC.D0 (aT1, aP1); + aBoxP1.Add (aP1); + const Standard_Real aTolV11 = BRep_Tool::Tolerance (TopoDS::Vertex (myDS->Shape (nV11))); + aBoxP1.Enlarge (aTolV11); + + // Find edges intersecting by AABB with the first point + BOPTools_BoxTreeSelector aSelector; + aSelector.SetBox (Bnd_Tools::Bnd2BVH (aBoxP1)); + aSelector.SetBVHSet (&thePBTree); + if (!aSelector.Select()) + return Standard_False; + //intermediate point - aTm=IntTools_Tools::IntermediatePoint (aT1, aT2); + Bnd_Box aBoxPm; + Standard_Real aTm = IntTools_Tools::IntermediatePoint (aT1, aT2); + gp_Pnt aPm; aIC.D0(aTm, aPm); - aBoxPm.Add(aPm); - //last point - aIC.D0(aT2, aP2); - aBoxP2.Add(aP2); - aBoxP2.Enlarge(aTolV12); - // + aBoxPm.Add (aPm); + + // last point + Bnd_Box aBoxP2; + gp_Pnt aP2; + aIC.D0 (aT2, aP2); + aBoxP2.Add (aP2); + const Standard_Real aTolV12 = BRep_Tool::Tolerance (TopoDS::Vertex (myDS->Shape (nV12))); + aBoxP2.Enlarge (aTolV12); + + const Standard_Real aTolV1 = Max(aTolV11, aTolV12) + myFuzzyValue; + + Standard_Real aTolCheck = theTolR3D + myFuzzyValue; + // Look for the existing pave block closest to the section curve + Standard_Boolean bFound = Standard_False; theTolNew = ::RealLast(); - aNbPB = theMPBOnIn.Extent(); - for (i = 1; i <= aNbPB; ++i) { - const Handle(BOPDS_PaveBlock)& aPB = theMPBOnIn(i); - aPB->Indices(nV21, nV22); - const Standard_Real aTolV21 = BRep_Tool::Tolerance(TopoDS::Vertex(myDS->Shape(nV21))); - const Standard_Real aTolV22 = BRep_Tool::Tolerance(TopoDS::Vertex(myDS->Shape(nV22))); - const Standard_Real aTolV2 = Max(aTolV21, aTolV22) + myFuzzyValue; - nSp=aPB->Edge(); - if (nSp < 0) + for (TColStd_ListOfInteger::Iterator it (aSelector.Indices()); it.More(); it.Next()) + { + const Handle (BOPDS_PaveBlock)& aPB = theMPBOnIn (it.Value()); + + Standard_Integer nV21, nV22; + aPB->Indices (nV21, nV22); + + const Standard_Real aTolV21 = BRep_Tool::Tolerance (TopoDS::Vertex (myDS->Shape (nV21))); + const Standard_Real aTolV22 = BRep_Tool::Tolerance (TopoDS::Vertex (myDS->Shape (nV22))); + const Standard_Real aTolV2 = Max (aTolV21, aTolV22) + myFuzzyValue; + + const BOPDS_ShapeInfo& aSISp = myDS->ChangeShapeInfo (aPB->Edge()); + const TopoDS_Edge& aSp = (*(TopoDS_Edge *)(&aSISp.Shape())); + const Bnd_Box& aBoxSp = aSISp.Box(); + + Standard_Integer iFlag1 = (nV11 == nV21 || nV11 == nV22) ? 2 : 1; + Standard_Integer iFlag2 = (nV12 == nV21 || nV12 == nV22) ? 2 : (!aBoxSp.IsOut (aBoxP2) ? 1 : 0); + if (!iFlag2) continue; - const BOPDS_ShapeInfo& aSISp=myDS->ChangeShapeInfo(nSp); - const TopoDS_Edge& aSp=(*(TopoDS_Edge *)(&aSISp.Shape())); - const Bnd_Box& aBoxSp=aSISp.Box(); - // - iFlag1 = (nV11 == nV21 || nV11 == nV22) ? 2 : - (!aBoxSp.IsOut(aBoxP1) ? 1 : 0); - iFlag2 = (nV12 == nV21 || nV12 == nV22) ? 2 : - (!aBoxSp.IsOut(aBoxP2) ? 1 : 0); - if (iFlag1 && iFlag2) { - Standard_Real aDist = 0.; - Standard_Real aRealTol = aTolCheck; - if (myDS->IsCommonBlock(aPB)) - { - aRealTol = Max(aRealTol, Max(aTolV1, aTolV2)); - if (theMPBCommon.Contains(aPB)) - // for an edge, which is a common block with a face, - // increase the chance to coincide with section curve - aRealTol *= 2.; - } + Standard_Real aDist = 0.; + + Standard_Real aRealTol = aTolCheck; + if (myDS->IsCommonBlock(aPB)) + { + aRealTol = Max(aRealTol, Max(aTolV1, aTolV2)); + if (theMPBCommon.Contains(aPB)) + // for an edge, which is a common block with a face, + // increase the chance to coincide with section curve + aRealTol *= 2.; + } - aBoxTmp = aBoxPm; - aBoxTmp.Enlarge(aRealTol); + Bnd_Box aBoxTmp = aBoxPm; + aBoxTmp.Enlarge(aRealTol); - Standard_Real aDistToSp = 0.; - if (aBoxSp.IsOut(aBoxTmp) || myContext->ComputePE(aPm, - aRealTol, - aSp, - aTx, aDistToSp)) { - continue; - } - // - if (iFlag1 == 1) { - iFlag1 = !myContext->ComputePE(aP1, aRealTol, aSp, aTx, aDist); - if (iFlag1 && aDistToSp < aDist) - aDistToSp = aDist; - } - // - if (iFlag2 == 1) { - iFlag2 = !myContext->ComputePE(aP2, aRealTol, aSp, aTx, aDist); - if (iFlag2 && aDistToSp < aDist) - aDistToSp = aDist; - } - // - if (iFlag1 && iFlag2) + Standard_Real aDistToSp = 0.; + Standard_Real aTx; + if (aBoxSp.IsOut(aBoxTmp) || myContext->ComputePE(aPm, + aRealTol, + aSp, + aTx, aDistToSp)) { + continue; + } + // + if (iFlag1 == 1) { + iFlag1 = !myContext->ComputePE(aP1, aRealTol, aSp, aTx, aDist); + if (iFlag1 && aDistToSp < aDist) + aDistToSp = aDist; + } + // + if (iFlag2 == 1) { + iFlag2 = !myContext->ComputePE(aP2, aRealTol, aSp, aTx, aDist); + if (iFlag2 && aDistToSp < aDist) + aDistToSp = aDist; + } + // + if (iFlag1 && iFlag2) + { + if (aDistToSp < theTolNew) { - if (aDistToSp < theTolNew) - { - aPBOut = aPB; - theTolNew = aDistToSp; - bRet = Standard_True; - } + aPBOut = aPB; + theTolNew = aDistToSp; + bFound = Standard_True; } } } - return bRet; + return bFound; } //======================================================================= @@ -2476,6 +2510,96 @@ void BOPAlgo_PaveFiller::PutPaveOnCurve } } +//======================================================================= +//function : ProcessExistingPaveBlocks +//purpose : +//======================================================================= +void BOPAlgo_PaveFiller::ProcessExistingPaveBlocks (const Standard_Integer theInt, + const Standard_Integer theCur, + const Standard_Integer nF1, + const Standard_Integer nF2, + const TopoDS_Edge& theES, + const BOPDS_IndexedMapOfPaveBlock& theMPBOnIn, + BOPTools_BoxTree& thePBTree, + BOPDS_IndexedDataMapOfShapeCoupleOfPaveBlocks& theMSCPB, + TopTools_DataMapOfShapeInteger& theMVI, + BOPDS_ListOfPaveBlock& theLPBC, + BOPAlgo_DataMapOfPaveBlockListOfInteger& thePBFacesMap, + BOPDS_MapOfPaveBlock& theMPB) +{ + Bnd_Box aBoxES; + BRepBndLib::Add (theES, aBoxES, false); + + BOPTools_BoxTreeSelector aSelector; + aSelector.SetBox (Bnd_Tools::Bnd2BVH (aBoxES)); + aSelector.SetBVHSet (&thePBTree); + if (!aSelector.Select()) + return; + + const Standard_Real aTolES = BRep_Tool::Tolerance (theES); + + const BOPDS_FaceInfo& aFI1 = myDS->FaceInfo (nF1); + const BOPDS_FaceInfo& aFI2 = myDS->FaceInfo (nF2); + + for (TColStd_ListOfInteger::Iterator itPB (aSelector.Indices()); itPB.More(); itPB.Next()) + { + const Handle(BOPDS_PaveBlock)& aPBF = theMPBOnIn (itPB.Value()); + if (theMPB.Contains (aPBF)) + continue; + + Standard_Boolean bInF1 = (aFI1.PaveBlocksOn().Contains(aPBF) || + aFI1.PaveBlocksIn().Contains(aPBF)); + Standard_Boolean bInF2 = (aFI2.PaveBlocksOn().Contains(aPBF) || + aFI2.PaveBlocksIn().Contains(aPBF)); + if (bInF1 && bInF2) + { + // Add all common edges for post treatment + theMPB.Add (aPBF); + PreparePostTreatFF (theInt, theCur, aPBF, theMSCPB, theMVI, theLPBC); + continue; + } + + const Standard_Integer nF = bInF1 ? nF2 : nF1; + const NCollection_List* pList = myDistances.Seek (BOPDS_Pair (aPBF->OriginalEdge(), nF)); + if (!pList) + continue; + + Standard_Real aT1, aT2; + aPBF->Range (aT1, aT2); + + Standard_Real aDist = RealLast(); + for (NCollection_List::Iterator itR (*pList); itR.More(); itR.Next()) + { + const EdgeRangeDistance& aRangeDist = itR.Value(); + if ((aT1 <= aRangeDist.First && aRangeDist.First <= aT2) || + (aT1 <= aRangeDist.Last && aRangeDist.Last <= aT2) || + (aRangeDist.First <= aT1 && aT1 <= aRangeDist.Last) || + (aRangeDist.First <= aT2 && aT2 <= aRangeDist.Last)) + { + aDist = aRangeDist.Distance; + break; + } + } + if (aDist < RealLast()) + { + const TopoDS_Edge& aEF = TopoDS::Edge (myDS->Shape (aPBF->Edge())); + const Standard_Real aTolSum = aTolES + BRep_Tool::Tolerance (aEF); + + if (aDist <= aTolSum) + { + theMPB.Add (aPBF); + PreparePostTreatFF (theInt, theCur, aPBF, theMSCPB, theMVI, theLPBC); + + TColStd_ListOfInteger* pFaces = thePBFacesMap.ChangeSeek(aPBF); + if (!pFaces) + pFaces = thePBFacesMap.Bound (aPBF, TColStd_ListOfInteger()); + if (pFaces->IsEmpty() || !pFaces->Contains (nF)) + pFaces->Append (nF); + } + } + } +} + //======================================================================= //function : ProcessExistingPaveBlocks //purpose : @@ -2485,6 +2609,7 @@ void BOPAlgo_PaveFiller::ProcessExistingPaveBlocks const Standard_Integer nF1, const Standard_Integer nF2, const BOPDS_IndexedMapOfPaveBlock& aMPBOnIn, + BOPTools_BoxTree& thePBTree, const TColStd_DataMapOfIntegerListOfInteger& aDMBV, BOPDS_IndexedDataMapOfShapeCoupleOfPaveBlocks& aMSCPB, TopTools_DataMapOfShapeInteger& aMVI, @@ -2496,7 +2621,7 @@ void BOPAlgo_PaveFiller::ProcessExistingPaveBlocks } // Standard_Real aT, dummy; - Standard_Integer i, nV, nE, iC, aNbPB, iFlag; + Standard_Integer nV, nE, iC, iFlag; TColStd_ListIteratorOfListOfInteger aItLI; TColStd_DataMapIteratorOfDataMapOfIntegerListOfInteger aItBV; // @@ -2507,8 +2632,6 @@ void BOPAlgo_PaveFiller::ProcessExistingPaveBlocks const BOPDS_FaceInfo& aFI1 = myDS->FaceInfo(nF1); const BOPDS_FaceInfo& aFI2 = myDS->FaceInfo(nF2); // - aNbPB = aMPBOnIn.Extent(); - // aItBV.Initialize(aDMBV); for (; aItBV.More(); aItBV.Next()) { iC = aItBV.Key(); @@ -2527,8 +2650,15 @@ void BOPAlgo_PaveFiller::ProcessExistingPaveBlocks continue; } // - for (i = 1; i <= aNbPB; ++i) { - const Handle(BOPDS_PaveBlock)& aPB = aMPBOnIn(i); + BOPTools_BoxTreeSelector aSelector; + aSelector.SetBox (Bnd_Tools::Bnd2BVH (aBoxV)); + aSelector.SetBVHSet (&thePBTree); + if (!aSelector.Select()) + continue; + + for (TColStd_ListOfInteger::Iterator it (aSelector.Indices()); it.More(); it.Next()) + { + const Handle(BOPDS_PaveBlock)& aPB = aMPBOnIn (it.Value()); if (aPB->Pave1().Index() == nV || aPB->Pave2().Index() == nV) { continue; } @@ -2536,18 +2666,8 @@ void BOPAlgo_PaveFiller::ProcessExistingPaveBlocks if (aMPB.Contains(aPB)) { continue; } - if (myDS->ShapeInfo(aPB->OriginalEdge()).HasFlag()) { // skip degenerated edges - continue; - } - // nE = aPB->Edge(); const BOPDS_ShapeInfo& aSIE = myDS->ShapeInfo(nE); - const Bnd_Box& aBoxE = aSIE.Box(); - // - if (aBoxV.IsOut(aBoxE)) { - continue; - } - // const TopoDS_Edge& aE = *(TopoDS_Edge*)&aSIE.Shape(); // iFlag = myContext->ComputeVE(aV, aE, aT, dummy, myFuzzyValue); diff --git a/src/BOPAlgo/BOPAlgo_WireSplitter_1.cxx b/src/BOPAlgo/BOPAlgo_WireSplitter_1.cxx index 0fd3e771b0..37ed067bce 100644 --- a/src/BOPAlgo/BOPAlgo_WireSplitter_1.cxx +++ b/src/BOPAlgo/BOPAlgo_WireSplitter_1.cxx @@ -786,7 +786,7 @@ Standard_Integer NbWaysOut(const BOPAlgo_ListOfEdgeInfo& aLEInfo) //for case chl/927/r9 aTX=0.05*(aLast - aFirst);//aTX=0.25*(aLast - aFirst); if (aTX < 5.e-5) { - aTX = 5.e-5; + aTX = Min (5.e-5, (aLast - aFirst) / 2.); } if(dt > aTX) { // to save direction of the curve as much as it possible diff --git a/src/BOPTools/BOPTools_AlgoTools.cxx b/src/BOPTools/BOPTools_AlgoTools.cxx index 3a037f4761..bf1891b02b 100644 --- a/src/BOPTools/BOPTools_AlgoTools.cxx +++ b/src/BOPTools/BOPTools_AlgoTools.cxx @@ -825,7 +825,8 @@ Standard_Boolean BOPTools_AlgoTools::IsInternalFace const TopoDS_Face& aF2=(*(TopoDS_Face*)(&aLF.Last())); iRet=BOPTools_AlgoTools::IsInternalFace(theFace, aE, aF1, aF2, theContext); - break; + if (iRet != 2) + break; } }//for(; aExp.More(); aExp.Next()) { // @@ -901,8 +902,6 @@ Standard_Integer BOPTools_AlgoTools::IsInternalFace const TopoDS_Face& theFace2, const Handle(IntTools_Context)& theContext) { - Standard_Boolean bRet; - Standard_Integer iRet; TopoDS_Edge aE1, aE2; TopoDS_Face aFOff; BOPTools_ListOfCoupleOfShape theLCSOff; @@ -931,17 +930,15 @@ Standard_Integer BOPTools_AlgoTools::IsInternalFace aCS2.SetShape2(theFace2); theLCSOff.Append(aCS2); // - bRet=GetFaceOff(aE1, theFace1, theLCSOff, aFOff, theContext); - // - iRet=0; // theFace is not internal - if (theFace.IsEqual(aFOff)) { + Standard_Integer iRet = 0; // theFace is not internal + Standard_Boolean isDone = GetFaceOff (aE1, theFace1, theLCSOff, aFOff, theContext); + if (!isDone) + // error, unable to classify face by this edge + iRet = 2; + else if (theFace.IsEqual (aFOff)) // theFace is internal - iRet=1; - if (!bRet) { - // theFace seems to be internal - iRet=2; - } - } + iRet = 1; + return iRet; } //======================================================================= @@ -957,7 +954,7 @@ Standard_Boolean BOPTools_AlgoTools::GetFaceOff { Standard_Boolean bRet, bIsComputed; Standard_Real aT, aT1, aT2, aAngle, aTwoPI, aAngleMin, aDt3D; - Standard_Real aUmin, aUsup, aVmin, aVsup, aPA; + Standard_Real aUmin, aUsup, aVmin, aVsup; gp_Pnt aPn1, aPn2, aPx; gp_Dir aDN1, aDN2, aDBF, aDBF2, aDTF; gp_Vec aVTgt; @@ -967,7 +964,6 @@ Standard_Boolean BOPTools_AlgoTools::GetFaceOff BOPTools_ListIteratorOfListOfCoupleOfShape aIt; GeomAPI_ProjectPointOnSurf aProjPL; // - aPA=Precision::Angular(); aAngleMin=100.; aTwoPI=M_PI+M_PI; aC3D =BRep_Tool::Curve(theE1, aT1, aT2); @@ -994,6 +990,10 @@ Standard_Boolean BOPTools_AlgoTools::GetFaceOff // aDTF=aDN1^aDBF; // + // The difference between faces should be obvious enough + // to guarantee the correctness of the classification + Standard_Real anAngleCriteria = Precision::Confusion(); + bRet=Standard_True; aIt.Initialize(theLCSOff); for (; aIt.More(); aIt.Next()) { @@ -1012,11 +1012,7 @@ Standard_Boolean BOPTools_AlgoTools::GetFaceOff //Angle aAngle=AngleWithRef(aDBF, aDBF2, aDTF); // - if(aAngle<0.) { - aAngle=aTwoPI+aAngle; - } - // - if (aAngle= theMinNbPoints) - { - return; - } Standard_Real aMinDeltaParam = theTol2D; @@ -2192,7 +2188,7 @@ static void SeekAdditionalPoints( const IntSurf_Quadric& theQuad1, Standard_Real U1prec = 0.0, V1prec = 0.0, U2prec = 0.0, V2prec = 0.0; Standard_Integer aNbPointsPrev = 0; - while(aNbPoints < theMinNbPoints && (aNbPoints != aNbPointsPrev)) + do { aNbPointsPrev = aNbPoints; for(Standard_Integer fp = theStartPointOnLine, lp = 0; fp < aLastPointIndex; fp = lp + 1) @@ -2269,7 +2265,7 @@ static void SeekAdditionalPoints( const IntSurf_Quadric& theQuad1, { return; } - } + } while(aNbPoints < theMinNbPoints && (aNbPoints != aNbPointsPrev)); } //======================================================================= @@ -3564,7 +3560,7 @@ static IntPatch_ImpImpIntersection::IntStatus } #ifdef INTPATCH_IMPIMPINTERSECTION_DEBUG - //aWLine[i]->Dump(); + aWLine[i]->Dump(0); #endif } } diff --git a/src/IntTools/IntTools_BeanFaceIntersector.cxx b/src/IntTools/IntTools_BeanFaceIntersector.cxx index 323e249f88..513e3e5c2a 100644 --- a/src/IntTools/IntTools_BeanFaceIntersector.cxx +++ b/src/IntTools/IntTools_BeanFaceIntersector.cxx @@ -125,7 +125,8 @@ myVMinParameter(0.), myVMaxParameter(0.), myBeanTolerance(0.), myFaceTolerance(0.), -myIsDone(Standard_False) +myIsDone(Standard_False), +myMinSqDistance(RealLast()) { myCriteria = Precision::Confusion(); myCurveResolution = Precision::PConfusion(); @@ -146,7 +147,8 @@ IntTools_BeanFaceIntersector::IntTools_BeanFaceIntersector(const TopoDS_Edge& th myVMaxParameter(0.), myBeanTolerance(0.), myFaceTolerance(0.), - myIsDone(Standard_False) + myIsDone(Standard_False), + myMinSqDistance(RealLast()) { Init(theEdge, theFace); } @@ -165,7 +167,8 @@ IntTools_BeanFaceIntersector::IntTools_BeanFaceIntersector(const BRepAdaptor_Cur myUMaxParameter(0.), myVMinParameter(0.), myVMaxParameter(0.), - myIsDone(Standard_False) + myIsDone(Standard_False), + myMinSqDistance(RealLast()) { Init(theCurve, theSurface, theBeanTolerance, theFaceTolerance); } @@ -192,7 +195,8 @@ IntTools_BeanFaceIntersector::IntTools_BeanFaceIntersector(const BRepAdaptor_Cur myVMaxParameter(theVMaxParameter), myBeanTolerance(theBeanTolerance), myFaceTolerance(theFaceTolerance), - myIsDone(Standard_False) + myIsDone(Standard_False), + myMinSqDistance(RealLast()) { myCurve = theCurve; @@ -651,9 +655,14 @@ void IntTools_BeanFaceIntersector::ComputeAroundExactIntersection() ComputeRangeFromStartPoint(Standard_False, aPoint.W(), U, V); ComputeRangeFromStartPoint(Standard_True, aPoint.W(), U, V); - if(aNbRanges == myRangeManager.Length()) { + if(aNbRanges == myRangeManager.Length()) + { SetEmptyResultRange(aPoint.W(), myRangeManager); - } // end if(aNbRanges == myRangeManager.Length()) + } + else + { + myMinSqDistance = 0.0; + } } } @@ -669,6 +678,7 @@ void IntTools_BeanFaceIntersector::ComputeAroundExactIntersection() ComputeRangeFromStartPoint(Standard_False, aPoint1.W(), aPoint1.U(), aPoint1.V()); ComputeRangeFromStartPoint(Standard_True, aPoint2.W(), aPoint2.U(), aPoint2.V()); + myMinSqDistance = 0.0; } } } @@ -940,6 +950,9 @@ void IntTools_BeanFaceIntersector::ComputeUsingExtremum() if (myExtrema.IsParallel()) { + if (myMinSqDistance > myExtrema.SquareDistance (1)) + myMinSqDistance = myExtrema.SquareDistance (1); + if(myExtrema.SquareDistance(1) < myCriteria * myCriteria) { Standard_Real U1, V1, U2, V2; Standard_Real adistance1 = Distance(anarg1, U1, V1); @@ -1022,6 +1035,9 @@ void IntTools_BeanFaceIntersector::ComputeUsingExtremum() SetEmptyResultRange(p1.Parameter(), myRangeManager); } } + + if (myMinSqDistance > myExtrema.SquareDistance (j)) + myMinSqDistance = myExtrema.SquareDistance (j); } //end for if(!solutionfound) { diff --git a/src/IntTools/IntTools_BeanFaceIntersector.hxx b/src/IntTools/IntTools_BeanFaceIntersector.hxx index da5f2f4d0a..23284c0020 100644 --- a/src/IntTools/IntTools_BeanFaceIntersector.hxx +++ b/src/IntTools/IntTools_BeanFaceIntersector.hxx @@ -130,17 +130,14 @@ public: Standard_EXPORT void Result (IntTools_SequenceOfRanges& theResults) const; - - - -protected: - - - + //! Returns the minimal distance found between edge and face + Standard_Real MinimalSquareDistance() const + { + return myMinSqDistance; + } private: - Standard_EXPORT void ComputeAroundExactIntersection(); @@ -191,6 +188,7 @@ private: Handle(IntTools_Context) myContext; IntTools_SequenceOfRanges myResults; Standard_Boolean myIsDone; + Standard_Real myMinSqDistance; }; diff --git a/src/IntTools/IntTools_EdgeFace.cxx b/src/IntTools/IntTools_EdgeFace.cxx index 46fe699b27..dffdb25c2c 100644 --- a/src/IntTools/IntTools_EdgeFace.cxx +++ b/src/IntTools/IntTools_EdgeFace.cxx @@ -71,6 +71,7 @@ static myIsDone=Standard_False; myErrorStatus=1; myQuickCoincidenceCheck=Standard_False; + myMinDistance = RealLast(); } //======================================================================= //function : IsCoincident @@ -535,7 +536,10 @@ void IntTools_EdgeFace::Perform() anIntersector.SetContext(myContext); // anIntersector.Perform(); - + + if (anIntersector.MinimalSquareDistance() < RealLast()) + myMinDistance = Sqrt (anIntersector.MinimalSquareDistance()); + if(!anIntersector.IsDone()) { return; } diff --git a/src/IntTools/IntTools_EdgeFace.hxx b/src/IntTools/IntTools_EdgeFace.hxx index 12c25926e9..8acc5c190e 100644 --- a/src/IntTools/IntTools_EdgeFace.hxx +++ b/src/IntTools/IntTools_EdgeFace.hxx @@ -177,6 +177,11 @@ public: //! @name Obtaining results return mySeqOfCommonPrts; } + //! Returns the minimal distance found between edge and face + Standard_Real MinimalDistance() const + { + return myMinDistance; + } protected: //! @name Protected methods performing the intersection @@ -210,6 +215,7 @@ protected: IntTools_SequenceOfCommonPrts mySeqOfCommonPrts; IntTools_Range myRange; Standard_Boolean myQuickCoincidenceCheck; + Standard_Real myMinDistance; //!< Minimal distance found }; #endif // _IntTools_EdgeFace_HeaderFile diff --git a/tests/boolean/bopfuse_complex/F1 b/tests/boolean/bopfuse_complex/F1 index 4004cd81a7..74f78f5da3 100644 --- a/tests/boolean/bopfuse_complex/F1 +++ b/tests/boolean/bopfuse_complex/F1 @@ -4,5 +4,6 @@ restore [locate_data_file b60] b bop a b bopfuse result -checkprops result -s 18750 +checkprops result -s 18750 -v 156250 +checknbshapes result -wire 36 -face 36 -shell 1 -solid 1 checkview -display result -2d -otherwise { a b } -s -path ${imagedir}/${test_image}.png diff --git a/tests/boolean/boptuc_complex/B5 b/tests/boolean/boptuc_complex/B5 index 95afbfe9b5..b0cf508d1a 100755 --- a/tests/boolean/boptuc_complex/B5 +++ b/tests/boolean/boptuc_complex/B5 @@ -4,5 +4,12 @@ restore [locate_data_file b60] b bop a b boptuc result -checkprops result -s 15000 +checkprops result -s 15000 -v 31250 +checknbshapes result -wire 24 -face 24 -shell 6 -solid 6 +foreach sol [explode result so] { + if {![regexp "This shape seems to be OK" [bopcheck $sol]]} { + puts "Error: the result solid $sol shape is self-interfered" + } +} + checkview -display result -2d -otherwise { a b } -s -path ${imagedir}/${test_image}.png diff --git a/tests/bugs/modalg_7/bug24789 b/tests/bugs/modalg_7/bug24789 new file mode 100644 index 0000000000..5477a4ca19 --- /dev/null +++ b/tests/bugs/modalg_7/bug24789 @@ -0,0 +1,32 @@ +puts "=============================================================================================" +puts "0024789: BOPAlgo_Builder produces invalid result on planar, conical and spherical surfaces" +puts "=============================================================================================" +puts "" + +restore [locate_data_file bug24789.brep] c1 +tcopy c1 c2 + +bclearobjects +bcleartools +eval baddobjects [explode c1] +bfillds +bbuild result1 + +checkshape result1 +checknbshapes result1 -shell 16 -solid 16 +checkprops result1 -s 4.91964e+06 -v 6.9691e+08 +if {![regexp "This shape seems to be OK" [bopcheck result1]]} { + puts "Error: GF result is self-interfered shape" +} + + +mkvolume result c2 -c + +checkshape result +checknbshapes result -ref [nbshapes result1] +checkprops result -equal result1 +if {![regexp "This shape seems to be OK" [bopcheck result]]} { + puts "Error: GF result is self-interfered shape" +} + +checkview -display result -2d -path ${imagedir}/${test_image}.png diff --git a/tests/bugs/modalg_7/bug31662 b/tests/bugs/modalg_7/bug31662 index 82cfa10e3e..b24071ee93 100644 --- a/tests/bugs/modalg_7/bug31662 +++ b/tests/bugs/modalg_7/bug31662 @@ -21,12 +21,12 @@ checkprops r4 -l 70.3856 bbuild rgf checkshape rgf -checknbshapes rgf -wire 363 -face 363 -shell 2 -solid 0 +checknbshapes rgf -shell 2 -solid 0 mkvolume result rgf -ni checkshape result -checknbshapes result -wire 254 -face 254 -shell 1 -solid 1 +checknbshapes result -shell 1 -solid 1 checkprops result -s 668.352 -v 774.749 checkview -display result -2d -path ${imagedir}/${test_image}.png