From 241a61330adaa27e2c2be65b29ea5075cbf57413 Mon Sep 17 00:00:00 2001 From: emv Date: Wed, 29 Aug 2018 11:51:43 +0300 Subject: [PATCH] 0030092: Modeling Algorithms - Invalid result of Section operation The following improvements have been made in Boolean operations algorithm in order to fix the problem: 1. Initialization of the pave blocks which vertices have acquired the SD ones. 2. Removing from Data Structure the small edges having the same vertices on both ends (either initially or acquired). 3. Avoid adding empty SD connections when one vertex points to itself. Test case for the issue. --- src/BOPAlgo/BOPAlgo_PaveFiller.cxx | 2 + src/BOPAlgo/BOPAlgo_PaveFiller.hxx | 2 + src/BOPAlgo/BOPAlgo_PaveFiller_2.cxx | 14 +- src/BOPAlgo/BOPAlgo_PaveFiller_3.cxx | 10 +- src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx | 217 +++++++++++++++------------ tests/bugs/modalg_2/bug472_1 | 1 - tests/bugs/modalg_2/bug472_3 | 7 +- tests/bugs/modalg_7/bug30092 | 32 ++++ 8 files changed, 181 insertions(+), 104 deletions(-) create mode 100644 tests/bugs/modalg_7/bug30092 diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller.cxx b/src/BOPAlgo/BOPAlgo_PaveFiller.cxx index 0257af3d70..421c9fa8d3 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller.cxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller.cxx @@ -328,6 +328,8 @@ void BOPAlgo_PaveFiller::PerformInternal() myDS->ReleasePaveBlocks(); myDS->RefineFaceInfoOn(); // + RemoveMicroEdges(); + // MakePCurves(); if (HasErrors()) { return; diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller.hxx b/src/BOPAlgo/BOPAlgo_PaveFiller.hxx index bc1a27743c..8baf0fac55 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller.hxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller.hxx @@ -548,6 +548,8 @@ protected: Standard_EXPORT void RemoveMicroSectionEdges(BOPDS_IndexedDataMapOfShapeCoupleOfPaveBlocks& theMSCPB, BOPDS_IndexedMapOfPaveBlock& theMicroPB); + //! Check all edges on the micro status and remove the positive ones + Standard_EXPORT void RemoveMicroEdges(); TopTools_ListOfShape myArguments; BOPDS_PDS myDS; diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller_2.cxx b/src/BOPAlgo/BOPAlgo_PaveFiller_2.cxx index 115667a872..331c97e67c 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller_2.cxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller_2.cxx @@ -368,6 +368,9 @@ void BOPAlgo_PaveFiller::SplitPaveBlocks(const TColStd_MapOfInteger& theMEdges, BOPDS_ListOfPaveBlock, TColStd_MapTransientHasher> aMCBNewPB; // + // Map of vertices to init the pave blocks for them + TColStd_MapOfInteger aMVerticesToInitPB; + TColStd_MapIteratorOfMapOfInteger aItM(theMEdges); for (; aItM.More(); aItM.Next()) { Standard_Integer nE = aItM.Value(); @@ -429,6 +432,10 @@ void BOPAlgo_PaveFiller::SplitPaveBlocks(const TColStd_MapOfInteger& theMEdges, aLV.Append(nV1); aLV.Append(nV2); MakeSDVertices(aLV, theAddInterfs); + + // Save vertices to init pave blocks + aMVerticesToInitPB.Add(nV1); + aMVerticesToInitPB.Add(nV2); } continue; } @@ -531,6 +538,11 @@ void BOPAlgo_PaveFiller::SplitPaveBlocks(const TColStd_MapOfInteger& theMEdges, } } } + + // Init pave blocks for vertices which have acquired SD vertex + aItM.Initialize(aMVerticesToInitPB); + for (; aItM.More(); aItM.Next()) + myDS->InitPaveBlocksForVertex(aItM.Value()); } //======================================================================= @@ -547,4 +559,4 @@ void BOPAlgo_PaveFiller::AddIntersectionFailedWarning(const TopoDS_Shape& theS1, BRep_Builder().Add(aWC, theS2); // Add the warning AddWarning(new BOPAlgo_AlertIntersectionOfPairOfShapesFailed(aWC)); -} \ No newline at end of file +} diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller_3.cxx b/src/BOPAlgo/BOPAlgo_PaveFiller_3.cxx index c9b4d013f0..43740b6cdc 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller_3.cxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller_3.cxx @@ -629,9 +629,13 @@ void BOPAlgo_PaveFiller::FillShrunkData(Handle(BOPDS_PaveBlock)& thePB) const TopoDS_Vertex& aV1=(*(TopoDS_Vertex *)(&myDS->Shape(nV1))); const TopoDS_Vertex& aV2=(*(TopoDS_Vertex *)(&myDS->Shape(nV2))); // Get the edge - Standard_Integer nE = thePB->OriginalEdge(); - if (nE < 0 && !thePB->HasEdge(nE)) - return; + Standard_Integer nE = -1; + if (!thePB->HasEdge(nE)) + { + nE = thePB->OriginalEdge(); + if (nE < 0) + return; + } const TopoDS_Edge& aE=(*(TopoDS_Edge *)(&myDS->Shape(nE))); // Range diff --git a/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx b/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx index aed0a2fe15..d56ec9503b 100644 --- a/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx +++ b/src/BOPAlgo/BOPAlgo_PaveFiller_6.cxx @@ -900,7 +900,7 @@ void BOPAlgo_PaveFiller::PostTreatFF // // 1 prepare arguments TopTools_MapOfShape anAddedSD; - for (k=1; k<=aNbS; ++k) { + for (k = aNbS; k > 0; --k) { const TopoDS_Shape& aS=theMSCPB.FindKey(k); aLS.Append(aS); // add vertices-candidates for SD from the map aDMNewSD, @@ -1025,8 +1025,11 @@ void BOPAlgo_PaveFiller::PostTreatFF if (!bIntersectionPoint) { // save SD connection nSx = myDS->Index(aSx); - aDMNewSD.Bind(nSx, iV); - myDS->AddShapeSD(nSx, iV); + if (nSx != iV) + { + aDMNewSD.Bind(nSx, iV); + myDS->AddShapeSD(nSx, iV); + } } else { // update FF interference @@ -1075,8 +1078,7 @@ void BOPAlgo_PaveFiller::PostTreatFF // note we check not the edge aSx itself, but its image in aPDS const BOPDS_ListOfPaveBlock& aLPBx = aPDS->PaveBlocks(nSx); aNbLPBx = aLPBx.Extent(); - if (aPDS->HasPaveBlocks(nSx) && - (aNbLPBx == 0 || (aNbLPBx == 1 && !aLPBx.First()->HasShrunkData()))) { + if (aNbLPBx == 0 || (aNbLPBx == 1 && !aLPBx.First()->HasShrunkData())) { BOPDS_ListIteratorOfListOfPaveBlock it(aLPBC); for (; it.More(); it.Next()) { if (it.Value() == aPB1) { @@ -1084,6 +1086,12 @@ void BOPAlgo_PaveFiller::PostTreatFF break; } } + + // The edge became micro edge, check vertices for SD + TopoDS_Iterator itV(aSx); + for (; itV.More(); itV.Next()) + aLS.Append(itV.Value()); + continue; } // @@ -2302,21 +2310,21 @@ void BOPAlgo_PaveFiller::GetFullShapeMap(const Standard_Integer nF, void BOPAlgo_PaveFiller::RemoveUsedVertices(const BOPDS_Curve& aNC, TColStd_MapOfInteger& aMV) { - if (!aMV.Extent()) { + if (aMV.IsEmpty()) return; - } const BOPDS_ListOfPaveBlock& aLPBC = aNC.PaveBlocks(); BOPDS_ListIteratorOfListOfPaveBlock itPB(aLPBC); for (; itPB.More(); itPB.Next()) { - const BOPDS_ListOfPave& aLP = itPB.Value()->ExtPaves(); - BOPDS_ListIteratorOfListOfPave aItLP(aLP); - for (;aItLP.More();aItLP.Next()) { - BOPDS_Pave aPave = aItLP.Value(); - Standard_Integer nV = aPave.Index(); - aMV.Remove(nV); - } + const Handle(BOPDS_PaveBlock)& aPB = itPB.Value(); + const BOPDS_ListOfPave& aLP = aPB->ExtPaves(); + BOPDS_ListIteratorOfListOfPave itLP(aLP); + for (; itLP.More(); itLP.Next()) + aMV.Remove(itLP.Value().Index()); + + aMV.Remove(aPB->Pave1().Index()); + aMV.Remove(aPB->Pave2().Index()); } } @@ -2919,79 +2927,46 @@ void BOPAlgo_PaveFiller::UpdatePaveBlocks bRebuild = Standard_False; aPB->Indices(nV[0], nV[1]); aPB->Range(aT[0], aT[1]); - - Standard_Integer nE = aPB->OriginalEdge(); - if (nE < 0) - // new edge - nE = aPB->Edge(); - // remember the fact if the edge had different vertices before substitution Standard_Boolean wasRegularEdge = (nV[0] != nV[1]); - - const TopoDS_Edge& aE = TopoDS::Edge(myDS->Shape(nE)); - TopoDS_Vertex aVE1, aVE2; - TopExp::Vertices(aE, aVE1, aVE2); - Standard_Boolean isClosedE = !aVE1.IsNull() && !aVE2.IsNull() && aVE1.IsSame(aVE2); - Standard_Boolean isDegEdge = myDS->ShapeInfo(nE).HasFlag(); // - BOPDS_Pave aPave[2] = {aPB->Pave1(), aPB->Pave2()}; - for (j = 0; j < 2; ++j) { if (aDMNewSD.IsBound(nV[j])) { + BOPDS_Pave aPave; // nV[j] = aDMNewSD.Find(nV[j]); - // recompute the parameter - if (!isDegEdge) - { - const TopoDS_Vertex& aV = TopoDS::Vertex(myDS->Shape(nV[j])); - if (!isClosedE || - BRep_Tool::Pnt(aV).Distance(BRep_Tool::Pnt(aVE1)) > BRep_Tool::Tolerance(aV) + myFuzzyValue) - { - Standard_Real aDummy, aTnew; - Standard_Integer iErr = myContext->ComputeVE(aV, aE, aTnew, aDummy, myFuzzyValue); - if (!iErr) - aT[j] = aTnew; - } - else - { - // choose the correct boundary parameter - Standard_Real f, l; - BRep_Tool::Range(aE, f, l); - aT[j] = Abs(aT[j] - f) < Abs(aT[j] - l) ? f : l; - } - } - aPave[j].SetIndex(nV[j]); + aPave.SetIndex(nV[j]); + aPave.SetParameter(aT[j]); // bRebuild = Standard_True; + if (!j) { + aPB->SetPave1(aPave); + } + else { + aPB->SetPave2(aPave); + } } } - + // if (bRebuild) { - if (aT[0] < aT[1]) - { - // It seems the parameters have been recomputed successfully - aPave[0].SetParameter(aT[0]); - aPave[1].SetParameter(aT[1]); + Standard_Integer nE = aPB->Edge(); + // Check if the Pave Block has the edge set + if (nE < 0) { + // untouched edge + nE = aPB->OriginalEdge(); } - - aPB->SetPave1(aPave[0]); - aPB->SetPave2(aPave[1]); - + Standard_Boolean isDegEdge = myDS->ShapeInfo(nE).HasFlag(); if (wasRegularEdge && !isDegEdge && nV[0] == nV[1]) { // now edge has the same vertex on both ends; // check if it is not a regular closed curve. FillShrunkData(aPB); if (!aPB->HasShrunkData()) { - Standard_Integer nEMicro = aPB->Edge(); - if (nEMicro < 0) - nEMicro = aPB->OriginalEdge(); // micro edge, so mark it for removal - aMicroEdges.Add(nEMicro); + aMicroEdges.Add(nE); continue; } } - aPB->Range(aT[0], aT[1]); nSp = SplitEdge(nE, nV[0], aT[0], nV[1], aT[1]); if (bCB) aCB->SetEdge(nSp); @@ -3029,41 +3004,12 @@ void BOPAlgo_PaveFiller::RemovePaveBlocks(const TColStd_MapOfInteger theEdges) } } - // 2. from Face Info and section curves + // 2. from section curves TColStd_MapOfInteger aMPassed; BOPDS_VectorOfInterfFF& aFFs = myDS->InterfFF(); Standard_Integer aNbFF = aFFs.Length(), j; for (i = 0; i < aNbFF; ++i) { BOPDS_InterfFF& aFF = aFFs(i); - Standard_Integer nF1, nF2; - aFF.Indices(nF1, nF2); - // - // rebuild pave block maps of face info - for (j = 0; j < 2; j++) { - Standard_Integer nF = (j == 0 ? nF1 : nF2); - if (!aMPassed.Add(nF)) - continue; - BOPDS_FaceInfo& aFI = myDS->ChangeFaceInfo(nF); - BOPDS_IndexedMapOfPaveBlock* aIMPB[] = { &aFI.ChangePaveBlocksIn(), - &aFI.ChangePaveBlocksOn(), &aFI.ChangePaveBlocksSc() }; - for (Standard_Integer k = 0; k < 3; k++) { - Standard_Integer aNbPB = aIMPB[k]->Extent(), m; - for (m = 1; m <= aNbPB; ++m) { - const Handle(BOPDS_PaveBlock)& aPB = aIMPB[k]->FindKey(m); - if (theEdges.Contains(aPB->Edge())) - break; - } - if (m <= aNbPB) { - BOPDS_IndexedMapOfPaveBlock aMPBCopy = *aIMPB[k]; - aIMPB[k]->Clear(); - for (m = 1; m <= aNbPB; ++m) { - const Handle(BOPDS_PaveBlock)& aPB = aMPBCopy(m); - if (!theEdges.Contains(aPB->Edge())) - aIMPB[k]->Add(aPB); - } - } - } - } // remove from Section pave blocks BOPDS_VectorOfCurve& aVNC = aFF.ChangeCurves(); Standard_Integer aNbC = aVNC.Length(); @@ -3080,7 +3026,44 @@ void BOPAlgo_PaveFiller::RemovePaveBlocks(const TColStd_MapOfInteger theEdges) } } } + + // 3. From Face Info + for (i = 0; i < myDS->NbSourceShapes(); ++i) + { + const BOPDS_ShapeInfo& aSI = myDS->ShapeInfo(i); + if (aSI.ShapeType() != TopAbs_FACE) + continue; + if (!aSI.HasReference()) + continue; + + BOPDS_FaceInfo& aFI = myDS->ChangeFaceInfo(i); + BOPDS_IndexedMapOfPaveBlock* aIMPB[] = { &aFI.ChangePaveBlocksIn(), + &aFI.ChangePaveBlocksOn(), + &aFI.ChangePaveBlocksSc() }; + for (Standard_Integer k = 0; k < 3; k++) + { + Standard_Integer aNbPB = aIMPB[k]->Extent(), m; + for (m = 1; m <= aNbPB; ++m) + { + const Handle(BOPDS_PaveBlock)& aPB = aIMPB[k]->FindKey(m); + if (theEdges.Contains(aPB->Edge())) + break; + } + if (m <= aNbPB) + { + BOPDS_IndexedMapOfPaveBlock aMPBCopy = *aIMPB[k]; + aIMPB[k]->Clear(); + for (m = 1; m <= aNbPB; ++m) + { + const Handle(BOPDS_PaveBlock)& aPB = aMPBCopy(m); + if (!theEdges.Contains(aPB->Edge())) + aIMPB[k]->Add(aPB); + } + } + } + } } + //======================================================================= //function : ToleranceFF //purpose : Computes the TolFF according to the tolerance value and @@ -3507,3 +3490,49 @@ void BOPAlgo_PaveFiller::RemoveMicroSectionEdges if (aSEPBMap.Extent() != theMSCPB.Extent()) theMSCPB = aSEPBMap; } + +//======================================================================= +//function : RemoveMicroEdges +//purpose : +//======================================================================= +void BOPAlgo_PaveFiller::RemoveMicroEdges() +{ + // Fence map + BOPDS_MapOfPaveBlock aMPBFence; + // Resulting map of micro edges + TColStd_MapOfInteger aMicroEdges; + // Check all pave blocks from the pool to find the micro edges + BOPDS_VectorOfListOfPaveBlock& aPBP = myDS->ChangePaveBlocksPool(); + Standard_Integer aNbPBP = aPBP.Length(); + for (Standard_Integer i = 0; i < aNbPBP; ++i) + { + BOPDS_ListOfPaveBlock& aLPB = aPBP(i); + if (aLPB.Extent() < 2) + // No splits + continue; + + if (myDS->ShapeInfo(aLPB.First()->OriginalEdge()).HasFlag()) + continue; + + BOPDS_ListOfPaveBlock::Iterator it(aLPB); + for (; it.More(); it.Next()) + { + const Handle(BOPDS_PaveBlock)& aPB = it.Value(); + Handle(BOPDS_PaveBlock) aPBR = myDS->RealPaveBlock(aPB); + + if (aMPBFence.Add(aPBR)) + { + Standard_Integer nV1, nV2; + aPBR->Indices(nV1, nV2); + if (nV1 == nV2) + { + // Check if it has the valid range + FillShrunkData(aPBR); + if (!aPBR->HasShrunkData()) + aMicroEdges.Add(aPBR->Edge()); + } + } + } + } + RemovePaveBlocks(aMicroEdges); +} diff --git a/tests/bugs/modalg_2/bug472_1 b/tests/bugs/modalg_2/bug472_1 index 2e542e85aa..ed20f9ad90 100755 --- a/tests/bugs/modalg_2/bug472_1 +++ b/tests/bugs/modalg_2/bug472_1 @@ -1,4 +1,3 @@ -puts "TODO OCC25917 ALL: Faulty shapes in variables faulty_1 to faulty_" puts "TODO OCC25917 ALL: Error : The area of result shape is" puts "========================" diff --git a/tests/bugs/modalg_2/bug472_3 b/tests/bugs/modalg_2/bug472_3 index 4e11dafc83..fc77452dd6 100755 --- a/tests/bugs/modalg_2/bug472_3 +++ b/tests/bugs/modalg_2/bug472_3 @@ -1,6 +1,3 @@ -puts "TODO OCC25917 ALL: Faulty shapes in variables faulty_1 to faulty_" -puts "TODO OCC25917 ALL: Error : is WRONG because number of " -puts "TODO OCC25917 ALL: Error : The area of result shape is " puts "========================" puts " OCC472 " puts "(case 3)" @@ -17,9 +14,9 @@ checkshape b2 bfuse result b1 b2 -checknbshapes result -vertex 66 -edge 107 +checknbshapes result -shell 1 -solid 1 -checkprops result -s 229.516 +checkprops result -v -111.22 -s 229.516 checkshape result checkview -display result -2d -path ${imagedir}/${test_image}.png diff --git a/tests/bugs/modalg_7/bug30092 b/tests/bugs/modalg_7/bug30092 new file mode 100644 index 0000000000..cfa5f88294 --- /dev/null +++ b/tests/bugs/modalg_7/bug30092 @@ -0,0 +1,32 @@ +puts "========" +puts "OCC30092: Modeling Algorithms - Invalid result of Section operation" +puts "========" +puts "" + + +restore [locate_data_file bug26795_c.brep] c +explode c f + +# modify c_2 face +line l 20004.7983876926 2164.22470816234 210.858020395251 -0.716336050762455 0.65536108806844 0.239508886314894 +mkedge e l -0.2 0.2 +settolerance e 0.04 + +bclearobjects +bcleartools +baddobjects c_2 +baddtools e +bfillds +bsplit c_2 + +# perform section operation +bclearobjects +bcleartools +baddobjects e c_1 +baddtools c_2 +bfillds +bbop result 4 + +checkshape result +checknbshapes result -vertex 5 -edge 4 +checkprops result -l 1250.44