1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-03 17:56:21 +03:00

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.
This commit is contained in:
emv 2018-08-29 11:51:43 +03:00 committed by bugmaster
parent 798a95ed9b
commit 241a61330a
8 changed files with 181 additions and 104 deletions

View File

@ -328,6 +328,8 @@ void BOPAlgo_PaveFiller::PerformInternal()
myDS->ReleasePaveBlocks();
myDS->RefineFaceInfoOn();
//
RemoveMicroEdges();
//
MakePCurves();
if (HasErrors()) {
return;

View File

@ -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;

View File

@ -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));
}
}

View File

@ -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

View File

@ -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);
}

View File

@ -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 "========================"

View File

@ -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

View File

@ -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