mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-06 18:26:22 +03:00
0025237: Wrong result of COMMON operation
Modifications in Edge/Edge intersection algorithm: 1. Condition to create common part of type TopAbs_EDGE is changed. 2. Correct treatment of closed edges. Small correction to eliminate warning. Test case for issue #25237
This commit is contained in:
parent
d33222c108
commit
3e594885aa
@ -125,16 +125,27 @@ is
|
|||||||
---Purpose:
|
---Purpose:
|
||||||
-- Computes Line/Line intersection.
|
-- Computes Line/Line intersection.
|
||||||
|
|
||||||
|
FindSolutions(me:out;
|
||||||
|
theRanges1 : out SequenceOfRanges from IntTools;
|
||||||
|
theRanges2 : out SequenceOfRanges from IntTools;
|
||||||
|
bSplit2 : out Boolean from Standard)
|
||||||
|
is protected;
|
||||||
|
---Purpose:
|
||||||
|
-- Intermediate function
|
||||||
|
|
||||||
FindSolutions(me:out;
|
FindSolutions(me:out;
|
||||||
theR1, theR2 : Range from IntTools;
|
theR1, theR2 : Range from IntTools;
|
||||||
|
theBox2 : Box from Bnd;
|
||||||
theRanges1 : out SequenceOfRanges from IntTools;
|
theRanges1 : out SequenceOfRanges from IntTools;
|
||||||
theRanges2 : out SequenceOfRanges from IntTools)
|
theRanges2 : out SequenceOfRanges from IntTools)
|
||||||
is protected;
|
is protected;
|
||||||
---Purpose:
|
---Purpose:
|
||||||
-- Looking for the exact intersection ranges
|
-- Looking for the exact intersection ranges
|
||||||
|
|
||||||
MergeSolutions(me:out;
|
MergeSolutions(me:out;
|
||||||
theRanges1, theRanges2 : SequenceOfRanges from IntTools)
|
theRanges1 : SequenceOfRanges from IntTools;
|
||||||
|
theRanges2 : SequenceOfRanges from IntTools;
|
||||||
|
bSplit2 : Boolean from Standard)
|
||||||
is protected;
|
is protected;
|
||||||
---Purpose:
|
---Purpose:
|
||||||
-- Merges found solutions
|
-- Merges found solutions
|
||||||
|
@ -34,9 +34,9 @@
|
|||||||
#include <BRep_Tool.hxx>
|
#include <BRep_Tool.hxx>
|
||||||
#include <BRepAdaptor_Curve.hxx>
|
#include <BRepAdaptor_Curve.hxx>
|
||||||
|
|
||||||
#include <IntTools_Tools.hxx>
|
|
||||||
#include <IntTools_CommonPrt.hxx>
|
#include <IntTools_CommonPrt.hxx>
|
||||||
|
|
||||||
|
#include <BOPCol_MapOfInteger.hxx>
|
||||||
|
|
||||||
static
|
static
|
||||||
void BndBuildBox(const BRepAdaptor_Curve& theBAC,
|
void BndBuildBox(const BRepAdaptor_Curve& theBAC,
|
||||||
@ -94,6 +94,12 @@ static
|
|||||||
static
|
static
|
||||||
Standard_Real CurveDeflection(const BRepAdaptor_Curve& theBAC,
|
Standard_Real CurveDeflection(const BRepAdaptor_Curve& theBAC,
|
||||||
const IntTools_Range& theRange);
|
const IntTools_Range& theRange);
|
||||||
|
static
|
||||||
|
Standard_Integer IsClosed(const Handle(Geom_Curve)& theCurve,
|
||||||
|
const Standard_Real aT1,
|
||||||
|
const Standard_Real aT2,
|
||||||
|
const Standard_Real theTol,
|
||||||
|
const Standard_Real theRes);
|
||||||
static
|
static
|
||||||
Standard_Integer TypeToInteger(const GeomAbs_CurveType theCType);
|
Standard_Integer TypeToInteger(const GeomAbs_CurveType theCType);
|
||||||
|
|
||||||
@ -211,10 +217,66 @@ void IntTools_EdgeEdge::Perform()
|
|||||||
IntTools_SequenceOfRanges aRanges1, aRanges2;
|
IntTools_SequenceOfRanges aRanges1, aRanges2;
|
||||||
//
|
//
|
||||||
//3.2. Find ranges containig solutions
|
//3.2. Find ranges containig solutions
|
||||||
FindSolutions(myRange1, myRange2, aRanges1, aRanges2);
|
Standard_Boolean bSplit2;
|
||||||
|
FindSolutions(aRanges1, aRanges2, bSplit2);
|
||||||
//
|
//
|
||||||
//4. Merge solutions and save common parts
|
//4. Merge solutions and save common parts
|
||||||
MergeSolutions(aRanges1, aRanges2);
|
MergeSolutions(aRanges1, aRanges2, bSplit2);
|
||||||
|
}
|
||||||
|
|
||||||
|
//=======================================================================
|
||||||
|
//function : FindSolutions
|
||||||
|
//purpose :
|
||||||
|
//=======================================================================
|
||||||
|
void IntTools_EdgeEdge::FindSolutions(IntTools_SequenceOfRanges& theRanges1,
|
||||||
|
IntTools_SequenceOfRanges& theRanges2,
|
||||||
|
Standard_Boolean& bSplit2)
|
||||||
|
{
|
||||||
|
Standard_Boolean bIsClosed2;
|
||||||
|
Standard_Real aT11, aT12, aT21, aT22;
|
||||||
|
Bnd_Box aB2;
|
||||||
|
//
|
||||||
|
bSplit2 = Standard_False;
|
||||||
|
myRange1.Range(aT11, aT12);
|
||||||
|
myRange2.Range(aT21, aT22);
|
||||||
|
//
|
||||||
|
bIsClosed2 = IsClosed(myGeom2, aT21, aT22, myTol2, myRes2);
|
||||||
|
//
|
||||||
|
if (bIsClosed2) {
|
||||||
|
Bnd_Box aB1;
|
||||||
|
BndBuildBox(myCurve1, aT11, aT12, myTol1, aB1);
|
||||||
|
//
|
||||||
|
gp_Pnt aP = myGeom2->Value(aT21);
|
||||||
|
bIsClosed2 = !aB1.IsOut(aP);
|
||||||
|
}
|
||||||
|
//
|
||||||
|
if (!bIsClosed2) {
|
||||||
|
BndBuildBox(myCurve2, aT21, aT22, myTol2, aB2);
|
||||||
|
FindSolutions(myRange1, myRange2, aB2, theRanges1, theRanges2);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
Standard_Integer i, j, aNb1, aNb2;
|
||||||
|
IntTools_SequenceOfRanges aSegments1, aSegments2;
|
||||||
|
//
|
||||||
|
aNb1 = IsClosed(myGeom1, aT11, aT12, myTol1, myRes1) ? 2 : 1;
|
||||||
|
aNb2 = 2;
|
||||||
|
//
|
||||||
|
SplitRangeOnSegments(aT11, aT12, myRes1, aNb1, aSegments1);
|
||||||
|
SplitRangeOnSegments(aT21, aT22, myRes2, aNb2, aSegments2);
|
||||||
|
//
|
||||||
|
aNb1 = aSegments1.Length();
|
||||||
|
aNb2 = aSegments2.Length();
|
||||||
|
for (i = 1; i <= aNb1; ++i) {
|
||||||
|
const IntTools_Range& aR1 = aSegments1(i);
|
||||||
|
for (j = 1; j <= aNb2; ++j) {
|
||||||
|
const IntTools_Range& aR2 = aSegments2(j);
|
||||||
|
BndBuildBox(myCurve2, aR2.First(), aR2.Last(), myTol2, aB2);
|
||||||
|
FindSolutions(aR1, aR2, aB2, theRanges1, theRanges2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//
|
||||||
|
bSplit2 = aNb2 > 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
@ -223,7 +285,8 @@ void IntTools_EdgeEdge::Perform()
|
|||||||
//=======================================================================
|
//=======================================================================
|
||||||
void IntTools_EdgeEdge::FindSolutions(const IntTools_Range& theR1,
|
void IntTools_EdgeEdge::FindSolutions(const IntTools_Range& theR1,
|
||||||
const IntTools_Range& theR2,
|
const IntTools_Range& theR2,
|
||||||
IntTools_SequenceOfRanges& theRanges1,
|
const Bnd_Box& theBox2,
|
||||||
|
IntTools_SequenceOfRanges& theRanges1,
|
||||||
IntTools_SequenceOfRanges& theRanges2)
|
IntTools_SequenceOfRanges& theRanges2)
|
||||||
{
|
{
|
||||||
Standard_Boolean bOut, bStop, bThin;
|
Standard_Boolean bOut, bStop, bThin;
|
||||||
@ -236,7 +299,7 @@ void IntTools_EdgeEdge::FindSolutions(const IntTools_Range& theR1,
|
|||||||
theR1.Range(aT11, aT12);
|
theR1.Range(aT11, aT12);
|
||||||
theR2.Range(aT21, aT22);
|
theR2.Range(aT21, aT22);
|
||||||
//
|
//
|
||||||
BndBuildBox(myCurve2, aT21, aT22, myTol2, aB2);
|
aB2 = theBox2;
|
||||||
//
|
//
|
||||||
bThin = Standard_False;
|
bThin = Standard_False;
|
||||||
bStop = Standard_False;
|
bStop = Standard_False;
|
||||||
@ -321,8 +384,8 @@ void IntTools_EdgeEdge::FindSolutions(const IntTools_Range& theR1,
|
|||||||
Standard_Real aT1, aT2;
|
Standard_Real aT1, aT2;
|
||||||
gp_Pnt aP1, aP2;
|
gp_Pnt aP1, aP2;
|
||||||
//
|
//
|
||||||
aT1 = IntTools_Tools::IntermediatePoint(aT11, aT12);
|
aT1 = (aT11 + aT12) * .5;
|
||||||
aT2 = IntTools_Tools::IntermediatePoint(aT21, aT22);
|
aT2 = (aT21 + aT22) * .5;
|
||||||
//
|
//
|
||||||
myGeom1->D0(aT1, aP1);
|
myGeom1->D0(aT1, aP1);
|
||||||
myGeom2->D0(aT2, aP2);
|
myGeom2->D0(aT2, aP2);
|
||||||
@ -348,12 +411,13 @@ void IntTools_EdgeEdge::FindSolutions(const IntTools_Range& theR1,
|
|||||||
IntTools_SequenceOfRanges aSegments1;
|
IntTools_SequenceOfRanges aSegments1;
|
||||||
//
|
//
|
||||||
IntTools_Range aR2(aT21, aT22);
|
IntTools_Range aR2(aT21, aT22);
|
||||||
|
BndBuildBox(myCurve2, aT21, aT22, myTol2, aB2);
|
||||||
//
|
//
|
||||||
SplitRangeOnSegments(aT11, aT12, myRes1, 3, aSegments1);
|
SplitRangeOnSegments(aT11, aT12, myRes1, 3, aSegments1);
|
||||||
aNb1 = aSegments1.Length();
|
aNb1 = aSegments1.Length();
|
||||||
for (i = 1; i <= aNb1; ++i) {
|
for (i = 1; i <= aNb1; ++i) {
|
||||||
const IntTools_Range& aR1 = aSegments1(i);
|
const IntTools_Range& aR1 = aSegments1(i);
|
||||||
FindSolutions(aR1, aR2, theRanges1, theRanges2);
|
FindSolutions(aR1, aR2, aB2, theRanges1, theRanges2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -457,48 +521,78 @@ Standard_Boolean IntTools_EdgeEdge::FindParameters(const BRepAdaptor_Curve& theB
|
|||||||
//purpose :
|
//purpose :
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
void IntTools_EdgeEdge::MergeSolutions(const IntTools_SequenceOfRanges& theRanges1,
|
void IntTools_EdgeEdge::MergeSolutions(const IntTools_SequenceOfRanges& theRanges1,
|
||||||
const IntTools_SequenceOfRanges& theRanges2)
|
const IntTools_SequenceOfRanges& theRanges2,
|
||||||
|
const Standard_Boolean bSplit2)
|
||||||
{
|
{
|
||||||
|
Standard_Integer aNbCP = theRanges1.Length();
|
||||||
|
if (aNbCP == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
//
|
||||||
IntTools_Range aRi1, aRi2, aRj1, aRj2;
|
IntTools_Range aRi1, aRi2, aRj1, aRj2;
|
||||||
Standard_Integer aNbCP, i, j;
|
Standard_Integer i, j;
|
||||||
TopAbs_ShapeEnum aType;
|
TopAbs_ShapeEnum aType;
|
||||||
Standard_Real aTi11, aTi12, aTi21, aTi22,
|
Standard_Real aTi11, aTi12, aTi21, aTi22,
|
||||||
aTj11, aTj12, aTj21, aTj22;
|
aTj11, aTj12, aTj21, aTj22,
|
||||||
|
dTR1, dTR2;
|
||||||
|
BOPCol_MapOfInteger aMI;
|
||||||
//
|
//
|
||||||
aNbCP = theRanges1.Length();
|
dTR1 = 20*myRes1;
|
||||||
|
dTR2 = (myRange2.Last() - myRange2.First()) / 2.;
|
||||||
aType = TopAbs_VERTEX;
|
aType = TopAbs_VERTEX;
|
||||||
//
|
//
|
||||||
for (i = 1; i <= aNbCP; ) {
|
for (i = 1; i <= aNbCP;) {
|
||||||
|
if (aMI.Contains(i)) {
|
||||||
|
++i;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
//
|
||||||
aRi1 = theRanges1(i);
|
aRi1 = theRanges1(i);
|
||||||
aRi2 = theRanges2(i);
|
aRi2 = theRanges2(i);
|
||||||
//
|
//
|
||||||
aRi1.Range(aTi11, aTi12);
|
aRi1.Range(aTi11, aTi12);
|
||||||
aRi2.Range(aTi21, aTi22);
|
aRi2.Range(aTi21, aTi22);
|
||||||
//
|
//
|
||||||
|
aMI.Add(i);
|
||||||
|
//
|
||||||
for (j = i+1; j <= aNbCP; ++j) {
|
for (j = i+1; j <= aNbCP; ++j) {
|
||||||
|
if (aMI.Contains(j)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
//
|
||||||
aRj1 = theRanges1(j);
|
aRj1 = theRanges1(j);
|
||||||
aRj2 = theRanges2(j);
|
aRj2 = theRanges2(j);
|
||||||
//
|
//
|
||||||
aRj1.Range(aTj11, aTj12);
|
aRj1.Range(aTj11, aTj12);
|
||||||
aRj2.Range(aTj21, aTj22);
|
aRj2.Range(aTj21, aTj22);
|
||||||
if (fabs(aTi12 - aTj11) < 10*myRes1 ||
|
if ((fabs(aTi12 - aTj11) < dTR1) &&
|
||||||
fabs(aTi22 - aTj21) < 10*myRes2) {
|
(fabs(aTi22 - aTj21) < dTR2)) {
|
||||||
aTi11 = Min(aTi11, aTj11);
|
aTi11 = Min(aTi11, aTj11);
|
||||||
aTi12 = Max(aTi12, aTj12);
|
aTi12 = Max(aTi12, aTj12);
|
||||||
aTi21 = Min(aTi21, aTj21);
|
aTi21 = Min(aTi21, aTj21);
|
||||||
aTi22 = Max(aTi22, aTj22);
|
aTi22 = Max(aTi22, aTj22);
|
||||||
} else {
|
aMI.Add(j);
|
||||||
|
}
|
||||||
|
else if (!bSplit2) {
|
||||||
|
i = j;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
i = j;
|
|
||||||
//
|
//
|
||||||
if (aTi11 == myRange1.First() && aTi12 == myRange1.Last() &&
|
if ((aTi11 == myRange1.First() && aTi12 == myRange1.Last()) ||
|
||||||
aTi21 == myRange2.First() && aTi22 == myRange2.Last()) {
|
(aTi21 == myRange2.First() && aTi22 == myRange2.Last())) {
|
||||||
aType = TopAbs_EDGE;
|
aType = TopAbs_EDGE;
|
||||||
|
myCommonParts.Clear();
|
||||||
}
|
}
|
||||||
//
|
//
|
||||||
AddSolution(aTi11, aTi12, aTi21, aTi22, aType);
|
AddSolution(aTi11, aTi12, aTi21, aTi22, aType);
|
||||||
|
if (aType == TopAbs_EDGE) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
if (bSplit2) {
|
||||||
|
++i;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -560,8 +654,8 @@ void IntTools_EdgeEdge::FindBestSolution(const Standard_Real aT11,
|
|||||||
GeomAPI_ProjectPointOnCurve aProj;
|
GeomAPI_ProjectPointOnCurve aProj;
|
||||||
IntTools_SequenceOfRanges aSeg1;
|
IntTools_SequenceOfRanges aSeg1;
|
||||||
//
|
//
|
||||||
aT1 = IntTools_Tools::IntermediatePoint(aT11, aT12);
|
aT1 = (aT11 + aT12) * .5;
|
||||||
aT2 = IntTools_Tools::IntermediatePoint(aT21, aT22);
|
aT2 = (aT21 + aT22) * .5;
|
||||||
//
|
//
|
||||||
aDMin = 100.;
|
aDMin = 100.;
|
||||||
aD = 100.;
|
aD = 100.;
|
||||||
@ -1103,12 +1197,12 @@ Standard_Integer TypeToInteger(const GeomAbs_CurveType theCType)
|
|||||||
case GeomAbs_Line:
|
case GeomAbs_Line:
|
||||||
iRet=0;
|
iRet=0;
|
||||||
break;
|
break;
|
||||||
case GeomAbs_Circle:
|
|
||||||
case GeomAbs_Ellipse:
|
|
||||||
iRet=1;
|
|
||||||
break;
|
|
||||||
case GeomAbs_Hyperbola:
|
case GeomAbs_Hyperbola:
|
||||||
case GeomAbs_Parabola:
|
case GeomAbs_Parabola:
|
||||||
|
iRet=1;
|
||||||
|
break;
|
||||||
|
case GeomAbs_Circle:
|
||||||
|
case GeomAbs_Ellipse:
|
||||||
iRet=2;
|
iRet=2;
|
||||||
break;
|
break;
|
||||||
case GeomAbs_BezierCurve:
|
case GeomAbs_BezierCurve:
|
||||||
@ -1244,3 +1338,30 @@ Standard_Real CurveDeflection(const BRepAdaptor_Curve& theBAC,
|
|||||||
return aDefl;
|
return aDefl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//=======================================================================
|
||||||
|
//function : IsClosed
|
||||||
|
//purpose :
|
||||||
|
//=======================================================================
|
||||||
|
Standard_Integer IsClosed(const Handle(Geom_Curve)& theCurve,
|
||||||
|
const Standard_Real aT1,
|
||||||
|
const Standard_Real aT2,
|
||||||
|
const Standard_Real theTol,
|
||||||
|
const Standard_Real theRes)
|
||||||
|
{
|
||||||
|
Standard_Boolean bClosed;
|
||||||
|
Standard_Real aD;
|
||||||
|
gp_Pnt aP1, aP2;
|
||||||
|
//
|
||||||
|
bClosed = Standard_False;
|
||||||
|
if (Abs(aT1 - aT2) < theRes) {
|
||||||
|
return bClosed;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
theCurve->D0(aT1, aP1);
|
||||||
|
theCurve->D0(aT2, aP2);
|
||||||
|
//
|
||||||
|
aD = aP1.Distance(aP2);
|
||||||
|
bClosed = aD < theTol;
|
||||||
|
//
|
||||||
|
return bClosed;
|
||||||
|
}
|
||||||
|
23
tests/bugs/modalg_5/bug25237
Normal file
23
tests/bugs/modalg_5/bug25237
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
puts "================"
|
||||||
|
puts "OCC25237"
|
||||||
|
puts "================"
|
||||||
|
puts ""
|
||||||
|
####################################
|
||||||
|
# Wrong result of COMMON operation
|
||||||
|
####################################
|
||||||
|
|
||||||
|
restore [locate_data_file bug25237_b26.brep] e1
|
||||||
|
restore [locate_data_file bug25237_b5.brep] e2
|
||||||
|
|
||||||
|
bop e1 e2
|
||||||
|
bopcommon result
|
||||||
|
|
||||||
|
set nb_v_good 2
|
||||||
|
set nb_e_good 1
|
||||||
|
set nb_w_good 1
|
||||||
|
set nb_f_good 0
|
||||||
|
set nb_sh_good 0
|
||||||
|
set nb_sol_good 0
|
||||||
|
set nb_compsol_good 0
|
||||||
|
set nb_compound_good 1
|
||||||
|
set nb_shape_good 5
|
Loading…
x
Reference in New Issue
Block a user