From d476fc37f83e205ccbaa3e348466f61e00fb15ac Mon Sep 17 00:00:00 2001 From: emv Date: Wed, 16 Sep 2020 15:05:50 +0300 Subject: [PATCH] 0030559: BOP Fuse: result is inconsistent Implement alternative approach for making the edge seam (closed) on the face. This approach is useful for non-periodic surfaces (e.g. tore-like surface of revolution is periodic in U direction only). Avoid internal faces in the affected solids of the result of BOP Fuse. --- src/BOPAlgo/BOPAlgo.msg | 3 + src/BOPAlgo/BOPAlgo_Alerts.hxx | 3 + src/BOPAlgo/BOPAlgo_BOP.cxx | 24 ++----- src/BOPAlgo/BOPAlgo_BOPAlgo_msg.pxx | 5 +- src/BOPAlgo/BOPAlgo_Builder_2.cxx | 14 +++- src/BOPTools/BOPTools_AlgoTools3D.cxx | 100 ++++++++++++++++++++++++-- src/BOPTools/BOPTools_AlgoTools3D.hxx | 12 ++-- tests/bugs/modalg_2/bug22109_2 | 2 +- tests/bugs/modalg_7/bug30559 | 25 +++++++ 9 files changed, 158 insertions(+), 30 deletions(-) create mode 100644 tests/bugs/modalg_7/bug30559 diff --git a/src/BOPAlgo/BOPAlgo.msg b/src/BOPAlgo/BOPAlgo.msg index 7217079492..9843277bc1 100644 --- a/src/BOPAlgo/BOPAlgo.msg +++ b/src/BOPAlgo/BOPAlgo.msg @@ -124,3 +124,6 @@ Unable to glue the shapes .BOPAlgo_AlertShapeIsNotPeriodic The shape is not periodic + +.BOPAlgo_AlertUnableToMakeClosedEdgeOnFace +Unable to make closed edge on face. diff --git a/src/BOPAlgo/BOPAlgo_Alerts.hxx b/src/BOPAlgo/BOPAlgo_Alerts.hxx index 39fe789751..3871c58fcd 100644 --- a/src/BOPAlgo/BOPAlgo_Alerts.hxx +++ b/src/BOPAlgo/BOPAlgo_Alerts.hxx @@ -138,4 +138,7 @@ DEFINE_ALERT_WITH_SHAPE(BOPAlgo_AlertUnableToGlue) //! The shape is not periodic DEFINE_ALERT_WITH_SHAPE(BOPAlgo_AlertShapeIsNotPeriodic) +//! Unable to make closed edge on face (to make a seam) +DEFINE_ALERT_WITH_SHAPE(BOPAlgo_AlertUnableToMakeClosedEdgeOnFace) + #endif // _BOPAlgo_Alerts_HeaderFile diff --git a/src/BOPAlgo/BOPAlgo_BOP.cxx b/src/BOPAlgo/BOPAlgo_BOP.cxx index 5069c2132d..b516015c49 100644 --- a/src/BOPAlgo/BOPAlgo_BOP.cxx +++ b/src/BOPAlgo/BOPAlgo_BOP.cxx @@ -60,8 +60,7 @@ static // static void MapFacesToBuildSolids(const TopoDS_Shape& theSol, - TopTools_IndexedDataMapOfShapeListOfShape& theMFS, - TopTools_IndexedMapOfShape& theMFI); + TopTools_IndexedDataMapOfShapeListOfShape& theMFS); //======================================================================= //function : @@ -1002,8 +1001,6 @@ void BOPAlgo_BOP::BuildSolid() TopTools_IndexedMapOfShape aMUSols; // Use map to chose the most outer faces to build result solids aMFS.Clear(); - // Internal faces - TopTools_IndexedMapOfShape aMFI; // TopoDS_Iterator aIt(myRC); for (; aIt.More(); aIt.Next()) { @@ -1015,7 +1012,7 @@ void BOPAlgo_BOP::BuildSolid() } } // - MapFacesToBuildSolids(aSx, aMFS, aMFI); + MapFacesToBuildSolids(aSx, aMFS); } // for (; aIt.More(); aIt.Next()) { // // Process possibly untouched solids. @@ -1035,7 +1032,7 @@ void BOPAlgo_BOP::BuildSolid() } // if (aExp.More()) { - MapFacesToBuildSolids(aSx, aMFS, aMFI); + MapFacesToBuildSolids(aSx, aMFS); } else { BOPTools_Set aST; @@ -1058,13 +1055,6 @@ void BOPAlgo_BOP::BuildSolid() aSFS.Append(aFx); } } - // Internal faces - aNb = aMFI.Extent(); - for (i = 1; i <= aNb; ++i) { - TopoDS_Shape aFx = aMFI.FindKey(i); - aSFS.Append(aFx.Oriented(TopAbs_FORWARD)); - aSFS.Append(aFx.Oriented(TopAbs_REVERSED)); - } // TopoDS_Shape aRC; BOPTools_AlgoTools::MakeContainer(TopAbs_COMPOUND, aRC); @@ -1073,6 +1063,7 @@ void BOPAlgo_BOP::BuildSolid() BOPAlgo_BuilderSolid aBS; aBS.SetContext(myContext); aBS.SetShapes(aSFS); + aBS.SetAvoidInternalShapes (Standard_True); aBS.Perform(); if (aBS.HasErrors()) { AddError (new BOPAlgo_AlertSolidBuilderFailed); // SolidBuilder failed @@ -1518,19 +1509,16 @@ Standard_Integer NbCommonItemsInMap(const TopTools_MapOfShape& theM1, //======================================================================= //function : MapFacesToBuildSolids //purpose : Stores the faces of the given solid into outgoing maps: -// - not internal faces with reference to solid; -// - internal faces. +// - not internal faces with reference to solid. //======================================================================= void MapFacesToBuildSolids(const TopoDS_Shape& theSol, - TopTools_IndexedDataMapOfShapeListOfShape& theMFS, - TopTools_IndexedMapOfShape& theMFI) + TopTools_IndexedDataMapOfShapeListOfShape& theMFS) { TopExp_Explorer aExp(theSol, TopAbs_FACE); for (; aExp.More(); aExp.Next()) { const TopoDS_Shape& aF = aExp.Current(); // if (aF.Orientation() == TopAbs_INTERNAL) { - theMFI.Add(aF); continue; } // diff --git a/src/BOPAlgo/BOPAlgo_BOPAlgo_msg.pxx b/src/BOPAlgo/BOPAlgo_BOPAlgo_msg.pxx index d8fd52feae..6c971d1d4f 100644 --- a/src/BOPAlgo/BOPAlgo_BOPAlgo_msg.pxx +++ b/src/BOPAlgo/BOPAlgo_BOPAlgo_msg.pxx @@ -126,4 +126,7 @@ static const char BOPAlgo_BOPAlgo_msg[] = "Unable to glue the shapes\n" "\n" ".BOPAlgo_AlertShapeIsNotPeriodic\n" - "The shape is not periodic\n"; + "The shape is not periodic\n" + "\n" + ".BOPAlgo_AlertUnableToMakeClosedEdgeOnFace\n" + "Unable to make closed edge on face.\n"; diff --git a/src/BOPAlgo/BOPAlgo_Builder_2.cxx b/src/BOPAlgo/BOPAlgo_Builder_2.cxx index 17c29ada6a..09a57fa46a 100644 --- a/src/BOPAlgo/BOPAlgo_Builder_2.cxx +++ b/src/BOPAlgo/BOPAlgo_Builder_2.cxx @@ -17,6 +17,7 @@ #include +#include #include #include #include @@ -383,7 +384,18 @@ void BOPAlgo_Builder::BuildSplitFaces() if (bIsClosed) { if (aMFence.Add(aSp)) { if (!BRep_Tool::IsClosed(aSp, aF)){ - BOPTools_AlgoTools3D::DoSplitSEAMOnFace(aSp, aF); + if (!BOPTools_AlgoTools3D::DoSplitSEAMOnFace(aSp, aF)) + { + // try different approach + if (!BOPTools_AlgoTools3D::DoSplitSEAMOnFace(aE, aSp, aF)) + { + TopoDS_Compound aWS; + BRep_Builder().MakeCompound (aWS); + BRep_Builder().Add (aWS, aF); + BRep_Builder().Add (aWS, aSp); + AddWarning (new BOPAlgo_AlertUnableToMakeClosedEdgeOnFace (aWS)); + } + } } // aSp.Orientation(TopAbs_FORWARD); diff --git a/src/BOPTools/BOPTools_AlgoTools3D.cxx b/src/BOPTools/BOPTools_AlgoTools3D.cxx index 06653d1605..4de235a391 100644 --- a/src/BOPTools/BOPTools_AlgoTools3D.cxx +++ b/src/BOPTools/BOPTools_AlgoTools3D.cxx @@ -39,6 +39,7 @@ #include #include #include +#include #include #include #include @@ -72,8 +73,8 @@ static //function : DoSplitSEAMOnFace //purpose : //======================================================================= -void BOPTools_AlgoTools3D::DoSplitSEAMOnFace (const TopoDS_Edge& aSplit, - const TopoDS_Face& aF) +Standard_Boolean BOPTools_AlgoTools3D::DoSplitSEAMOnFace (const TopoDS_Edge& aSplit, + const TopoDS_Face& aF) { Standard_Boolean bIsUPeriodic, bIsVPeriodic, bIsLeft; Standard_Real aTol, a, b, anUPeriod, anVPeriod, aT, anU, dU, anU1; @@ -131,7 +132,7 @@ void BOPTools_AlgoTools3D::DoSplitSEAMOnFace (const TopoDS_Edge& aSplit, bIsVPeriodic=aSB->IsVPeriodic(); // if (!(bIsUPeriodic || bIsVPeriodic)) { - return; + return Standard_False; } anUPeriod = bIsUPeriodic ? aSB->UPeriod() : 0.; anVPeriod = bIsVPeriodic ? aSB->VPeriod() : 0.; @@ -139,7 +140,7 @@ void BOPTools_AlgoTools3D::DoSplitSEAMOnFace (const TopoDS_Edge& aSplit, // if (aRTS.IsNull()) { if (!bIsUClosed && !bIsVClosed) { - return; + return Standard_False; } // if (bIsUClosed) { @@ -191,7 +192,7 @@ void BOPTools_AlgoTools3D::DoSplitSEAMOnFace (const TopoDS_Edge& aSplit, } // if (anU1==anU && anV1==anV) { - return; + return Standard_False; } // aScPr = (anU1==anU) ? aDir2D1*aDOX : aDir2D1*aDOY; @@ -222,7 +223,96 @@ void BOPTools_AlgoTools3D::DoSplitSEAMOnFace (const TopoDS_Edge& aSplit, BB.UpdateEdge(aSp, aC2, aC1, aF, aTol); } } + return Standard_True; } + +//======================================================================= +//function : DoSplitSEAMOnFace +//purpose : +//======================================================================= +Standard_Boolean BOPTools_AlgoTools3D::DoSplitSEAMOnFace (const TopoDS_Edge& theEOrigin, + const TopoDS_Edge& theESplit, + const TopoDS_Face& theFace) +{ + if (!BRep_Tool::IsClosed (theEOrigin, theFace)) + return Standard_False; + + if (BRep_Tool::IsClosed (theESplit, theFace)) + return Standard_True; + + TopoDS_Edge aESplit = theESplit; + aESplit.Orientation (TopAbs_FORWARD); + + TopoDS_Face aFace = theFace; + aFace.Orientation (TopAbs_FORWARD); + + Standard_Real aTS1, aTS2; + Handle(Geom2d_Curve) aC2DSplit = BRep_Tool::CurveOnSurface (aESplit, aFace, aTS1, aTS2); + if (aC2DSplit.IsNull()) + return Standard_False; + + Standard_Real aT1, aT2; + Handle(Geom2d_Curve) aC2D1 = BRep_Tool::CurveOnSurface ( + TopoDS::Edge (theEOrigin.Oriented (TopAbs_FORWARD)), aFace, aT1, aT2); + Handle(Geom2d_Curve) aC2D2 = BRep_Tool::CurveOnSurface ( + TopoDS::Edge (theEOrigin.Oriented (TopAbs_REVERSED)), aFace, aT1, aT2); + + Standard_Real aT = BOPTools_AlgoTools2D::IntermediatePoint (aTS1, aTS2); + gp_Pnt2d aPMid; + gp_Vec2d aVTgt; + aC2DSplit->D1 (aT, aPMid, aVTgt); + + // project on original 2d curves + Geom2dAPI_ProjectPointOnCurve aProjPC1, aProjPC2; + aProjPC1.Init (aPMid, aC2D1, aT1, aT2); + aProjPC2.Init (aPMid, aC2D2, aT1, aT2); + + if (!aProjPC1.NbPoints() && !aProjPC2.NbPoints()) + return Standard_False; + + Standard_Real aDist1 = aProjPC1.NbPoints() ? aProjPC1.LowerDistance() : RealLast(); + Standard_Real aDist2 = aProjPC2.NbPoints() ? aProjPC2.LowerDistance() : RealLast(); + + if (aDist1 > Precision::PConfusion() && aDist2 > Precision::PConfusion()) + return Standard_False; + + // choose the closest and take corresponding point from the opposite + gp_Pnt2d aNewPnt = aDist1 < aDist2 ? aC2D2->Value (aProjPC1.LowerDistanceParameter()) : + aC2D1->Value (aProjPC2.LowerDistanceParameter()); + + Handle (Geom2d_Curve) aTmpC1 = Handle (Geom2d_Curve)::DownCast (aC2DSplit->Copy()); + Handle (Geom2d_Curve) aTmpC2 = Handle (Geom2d_Curve)::DownCast (aC2DSplit->Copy()); + + Handle (Geom2d_TrimmedCurve) aC1 = new Geom2d_TrimmedCurve (aTmpC1, aTS1, aTS2); + Handle (Geom2d_TrimmedCurve) aC2 = new Geom2d_TrimmedCurve (aTmpC2, aTS1, aTS2); + + gp_Vec2d aTrVec (aPMid, aNewPnt); + aC2->Translate (aTrVec); + + gp_Pnt2d aPProj; + gp_Vec2d aVTgtOrigin; + if (aDist1 < aDist2) + { + aC2D1->D1 (aProjPC1.LowerDistanceParameter(), aPProj, aVTgtOrigin); + } + else + { + aC2D2->D1 (aProjPC2.LowerDistanceParameter(), aPProj, aVTgtOrigin); + } + + Standard_Real aDot = aVTgt.Dot (aVTgtOrigin); + + if ((aDist1 < aDist2) == (aDot > 0)) + { + BRep_Builder().UpdateEdge (aESplit, aC1, aC2, aFace, BRep_Tool::Tolerance (aESplit)); + } + else + { + BRep_Builder().UpdateEdge (aESplit, aC2, aC1, aFace, BRep_Tool::Tolerance (aESplit)); + } + return Standard_True; +} + //======================================================================= //function : GetNormalToFaceOnEdge //purpose : diff --git a/src/BOPTools/BOPTools_AlgoTools3D.hxx b/src/BOPTools/BOPTools_AlgoTools3D.hxx index 9a3d5de623..10457b91e9 100644 --- a/src/BOPTools/BOPTools_AlgoTools3D.hxx +++ b/src/BOPTools/BOPTools_AlgoTools3D.hxx @@ -43,11 +43,15 @@ public: DEFINE_STANDARD_ALLOC - + //! Makes the edge seam edge for the face basing on the surface properties (U and V periods) + Standard_EXPORT static Standard_Boolean DoSplitSEAMOnFace (const TopoDS_Edge& theESplit, + const TopoDS_Face& theFace); - //! Make the edge seam edge for the face - Standard_EXPORT static void DoSplitSEAMOnFace (const TopoDS_Edge& aSp, - const TopoDS_Face& aF); + //! Makes the split edge seam edge for the face basing on the positions + //! of 2d curves of the original edge . + Standard_EXPORT static Standard_Boolean DoSplitSEAMOnFace (const TopoDS_Edge& theEOrigin, + const TopoDS_Edge& theESplit, + const TopoDS_Face& theFace); //! Computes normal to the face for the point on the edge //! at parameter .
diff --git a/tests/bugs/modalg_2/bug22109_2 b/tests/bugs/modalg_2/bug22109_2 index d53991b41d..b3ef5cb571 100755 --- a/tests/bugs/modalg_2/bug22109_2 +++ b/tests/bugs/modalg_2/bug22109_2 @@ -24,6 +24,6 @@ puts "Finish boolean operation ..." checkprops result -s 87449.7 checkshape result -checknbshapes result -vertex 17 -edge 25 -wire 14 -face 10 -shell 2 -solid 1 -compsolid 0 -compound 1 -shape 70 +checknbshapes result -vertex 17 -edge 24 -wire 13 -face 9 -shell 1 -solid 1 checkview -display result -2d -path ${imagedir}/${test_image}.png diff --git a/tests/bugs/modalg_7/bug30559 b/tests/bugs/modalg_7/bug30559 new file mode 100644 index 0000000000..b09929176a --- /dev/null +++ b/tests/bugs/modalg_7/bug30559 @@ -0,0 +1,25 @@ +puts "TODO CR29596 ALL: Warning: Intersection of pair of shapes has failed" + +puts "=============================================================================================" +puts "0030559: BOP Fuse: result is inconsistent" +puts "=============================================================================================" +puts "" + +restore [locate_data_file bug30559_VALVE_1_of_BRANCH_6028_B3.brep] a +set exp [explode a So] +bclearobjects +bcleartools +baddobjects a_1 +eval baddtools [lrange $exp 1 end] +bfillds +bbop result 1 + +checkshape result +checknbshapes result -wire 26 -face 20 -shell 1 -solid 1 +checkprops result -s 456495 -v 1.2165e+07 + +if {![regexp "This shape seems to be OK" [bopcheck result]]} { + puts "Error: result is self-interfered shape" +} + +checkview -display result -2d -path ${imagedir}/${test_image}.png