1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00

0023982: Wire explorer raises exception

Add test case for this fix
This commit is contained in:
ifv 2013-05-31 17:10:17 +04:00
parent 80482e0178
commit a0f8845f56
2 changed files with 184 additions and 154 deletions

View File

@ -353,21 +353,21 @@ void BRepTools_WireExplorer::Next()
aNextPC = BRep_Tool::CurveOnSurface(aNextEdge, myFace, aPar21, aPar22); aNextPC = BRep_Tool::CurveOnSurface(aNextEdge, myFace, aPar21, aPar22);
if (aPrevPC.IsNull() || aNextPC.IsNull()) { if (aPrevPC.IsNull() || aNextPC.IsNull()) {
myEdge = TopoDS_Edge(); myEdge = TopoDS_Edge();
return; return;
} }
if (myEdge.Orientation() == TopAbs_FORWARD) if (myEdge.Orientation() == TopAbs_FORWARD)
aPrevPar = aPar12; aPrevPar = aPar12;
else else
aPrevPar = aPar11; aPrevPar = aPar11;
if (aNextEdge.Orientation() == TopAbs_FORWARD) { if (aNextEdge.Orientation() == TopAbs_FORWARD) {
aNextFPar = aPar21; aNextFPar = aPar21;
aNextLPar = aPar22; aNextLPar = aPar22;
} else { } else {
aNextFPar = aPar22; aNextFPar = aPar22;
aNextLPar = aPar21; aNextLPar = aPar21;
} }
gp_Pnt2d aPPrev = aPrevPC->Value(aPrevPar); gp_Pnt2d aPPrev = aPrevPC->Value(aPrevPar);
@ -375,8 +375,8 @@ void BRepTools_WireExplorer::Next()
gp_Pnt2d aPNextL = aNextPC->Value(aNextLPar); gp_Pnt2d aPNextL = aNextPC->Value(aNextLPar);
if (aPPrev.SquareDistance(aPNextF) > aPPrev.SquareDistance(aPNextL)) { if (aPPrev.SquareDistance(aPNextF) > aPPrev.SquareDistance(aPNextL)) {
myEdge = TopoDS_Edge(); myEdge = TopoDS_Edge();
return; return;
} }
} }
// Modified by Sergey KHROMOV - Fri Jun 21 11:08:16 2002 End // Modified by Sergey KHROMOV - Fri Jun 21 11:08:16 2002 End
@ -390,177 +390,194 @@ void BRepTools_WireExplorer::Next()
// At first degenerated edges. // At first degenerated edges.
TopoDS_Edge E = myEdge; TopoDS_Edge E = myEdge;
if (SelectDegenerated(l,E)) { if (SelectDegenerated(l,E)) {
myEdge = E; myEdge = E;
return; return;
} }
// At second double edges. // At second double edges.
E = myEdge; E = myEdge;
if (SelectDouble(myDoubles,l,E)) { if (SelectDouble(myDoubles,l,E)) {
myEdge = E; myEdge = E;
return; return;
} }
TopTools_ListIteratorOfListOfShape it(l); TopTools_ListIteratorOfListOfShape it(l);
Standard_Boolean notfound = Standard_True; Standard_Boolean notfound = Standard_True;
while (it.More()) { while (it.More()) {
if (!it.Value().IsSame(myEdge)) { if (!it.Value().IsSame(myEdge)) {
myEdge = TopoDS::Edge(it.Value()); myEdge = TopoDS::Edge(it.Value());
l.Remove(it); l.Remove(it);
notfound = Standard_False; notfound = Standard_False;
break; break;
} }
it.Next(); it.Next();
} }
if(notfound) { if(notfound) {
myEdge = TopoDS_Edge(); myEdge = TopoDS_Edge();
return; return;
} }
} }
else else
{
// If we have more than one edge attached to the list
// probably wire that we explore contains a loop or loops.
Standard_Real dfFPar = 0., dfLPar = 0.;
Handle(Geom2d_Curve) aPCurve = BRep_Tool::CurveOnSurface (myEdge, myFace, dfFPar, dfLPar);
if(aPCurve.IsNull())
{ {
// If we have more than one edge attached to the list myEdge = TopoDS_Edge();
// probably wire that we explore contains a loop or loops. return;
Standard_Real dfFPar = 0., dfLPar = 0.; }
Handle(Geom2d_Curve) aPCurve = BRep_Tool::CurveOnSurface (myEdge, myFace, dfFPar, dfLPar); // Note: current < myVertex > which is last on < myEdge >
if(aPCurve.IsNull()) // equals in 2D to following 2D points:
{ // edge is FORWARD - point with MAX parameter on PCurve;
myEdge = TopoDS_Edge(); // edge is REVERSED - point with MIN parameter on PCurve.
return;
}
// Note: current < myVertex > which is last on < myEdge >
// equals in 2D to following 2D points:
// edge is FORWARD - point with MAX parameter on PCurve;
// edge is REVERSED - point with MIN parameter on PCurve.
// Get 2D point equals to < myVertex > in 2D for current edge. // Get 2D point equals to < myVertex > in 2D for current edge.
gp_Pnt2d PRef; gp_Pnt2d PRef;
if( myEdge.Orientation() == TopAbs_REVERSED ) if( myEdge.Orientation() == TopAbs_REVERSED )
aPCurve->D0(dfFPar, PRef); aPCurve->D0(dfFPar, PRef);
else else
aPCurve->D0(dfLPar, PRef); aPCurve->D0(dfLPar, PRef);
// Get next 2D point from current edge's PCurve with parameter // Get next 2D point from current edge's PCurve with parameter
// F + dP (REV) or L - dP (FOR) // F + dP (REV) or L - dP (FOR)
Standard_Boolean isrevese = ( myEdge.Orientation() == TopAbs_REVERSED ); Standard_Boolean isrevese = ( myEdge.Orientation() == TopAbs_REVERSED );
Standard_Real dfMPar = GetNextParamOnPC(aPCurve,PRef,dfFPar,dfLPar,myTolU,myTolV,isrevese); Standard_Real dfMPar = GetNextParamOnPC(aPCurve,PRef,dfFPar,dfLPar,myTolU,myTolV,isrevese);
gp_Pnt2d PRefm; gp_Pnt2d PRefm;
aPCurve->D0(dfMPar, PRefm); aPCurve->D0(dfMPar, PRefm);
// Get vector from PRef to PRefm // Get vector from PRef to PRefm
gp_Vec2d anERefDir(PRef,PRefm); gp_Vec2d anERefDir(PRef,PRefm);
// Search the list of edges looking for the edge having hearest // Search the list of edges looking for the edge having hearest
// 2D point of connected vertex to current one and smallest angle. // 2D point of connected vertex to current one and smallest angle.
// First process all degenerated edges, then - all others. // First process all degenerated edges, then - all others.
TopTools_ListIteratorOfListOfShape it; TopTools_ListIteratorOfListOfShape it;
Standard_Integer k = 1, kMin = 0, iDone = 0; Standard_Integer k = 1, kMin = 0, iDone = 0;
Standard_Boolean isDegenerated = Standard_True; Standard_Boolean isDegenerated = Standard_True;
Standard_Real dmin = RealLast(); Standard_Real dmin = RealLast();
Standard_Real dfMinAngle = 3.0*M_PI, dfCurAngle = 3.0*M_PI; Standard_Real dfMinAngle = 3.0*M_PI, dfCurAngle = 3.0*M_PI;
for(iDone = 0; iDone < 2; iDone++) for(iDone = 0; iDone < 2; iDone++)
{ {
it.Initialize(l); it.Initialize(l);
while( it.More() ) while( it.More() )
{ {
const TopoDS_Edge& E = TopoDS::Edge(it.Value()); const TopoDS_Edge& E = TopoDS::Edge(it.Value());
if( E.IsSame(myEdge) ) if( E.IsSame(myEdge) )
{ {
it.Next(); it.Next();
k++; k++;
continue; continue;
} }
TopoDS_Vertex aVert1, aVert2; TopoDS_Vertex aVert1, aVert2;
TopExp::Vertices (E, aVert1, aVert2, Standard_True); TopExp::Vertices (E, aVert1, aVert2, Standard_True);
if( aVert1.IsNull() || aVert2.IsNull() ) if( aVert1.IsNull() || aVert2.IsNull() )
{ {
it.Next(); it.Next();
k++; k++;
continue; continue;
} }
aPCurve = BRep_Tool::CurveOnSurface (E, myFace, dfFPar, dfLPar); aPCurve = BRep_Tool::CurveOnSurface (E, myFace, dfFPar, dfLPar);
if( aPCurve.IsNull() ) if( aPCurve.IsNull() )
{ {
it.Next(); it.Next();
k++; k++;
continue; continue;
} }
gp_Pnt2d aPEb, aPEe; gp_Pnt2d aPEb, aPEe;
if( aVert1.IsSame(aVert2) == isDegenerated ) if( aVert1.IsSame(aVert2) == isDegenerated )
{ {
if( E.Orientation() == TopAbs_REVERSED ) if( E.Orientation() == TopAbs_REVERSED )
aPCurve->D0(dfLPar, aPEb); aPCurve->D0(dfLPar, aPEb);
else else
aPCurve->D0(dfFPar, aPEb); aPCurve->D0(dfFPar, aPEb);
if( Abs(dfLPar-dfFPar) > Precision::PConfusion() ) if( Abs(dfLPar-dfFPar) > Precision::PConfusion() )
{ {
isrevese = ( E.Orientation() == TopAbs_REVERSED ); isrevese = ( E.Orientation() == TopAbs_REVERSED );
isrevese = !isrevese; isrevese = !isrevese;
Standard_Real aEPm = GetNextParamOnPC(aPCurve,aPEb,dfFPar,dfLPar,myTolU,myTolV,isrevese); Standard_Real aEPm = GetNextParamOnPC(aPCurve,aPEb,dfFPar,dfLPar,myTolU,myTolV,isrevese);
aPCurve->D0 (aEPm, aPEe); aPCurve->D0 (aEPm, aPEe);
gp_Vec2d anEDir(aPEb, aPEe); if(aPEb.SquareDistance(aPEe) <= gp::Resolution())
dfCurAngle = Abs( anEDir.Angle(anERefDir) ); {
} //seems to be very short curve
gp_Vec2d aD;
aPCurve->D1(aEPm, aPEe, aD);
if( E.Orientation() == TopAbs_REVERSED )
aPEe.SetXY(aPEb.XY()-aD.XY());
else
aPEe.SetXY(aPEb.XY()+aD.XY());
if( dfCurAngle <= dfMinAngle ) if(aPEb.SquareDistance(aPEe) <= gp::Resolution())
{ {
Standard_Real d = PRef.SquareDistance(aPEb); it.Next();
if( d <= Precision::PConfusion() ) k++;
d = 0.; continue;
if( Abs(aPEb.X()-PRef.X()) < myTolU && Abs(aPEb.Y()-PRef.Y()) < myTolV ) }
{ }
if( d <= dmin ) gp_Vec2d anEDir(aPEb, aPEe);
{ dfCurAngle = Abs( anEDir.Angle(anERefDir) );
dfMinAngle = dfCurAngle; }
kMin = k;
dmin = d;
}
}
}
}
it.Next();
k++;
}// while it
if( kMin == 0 ) if( dfCurAngle <= dfMinAngle )
{ {
isDegenerated = Standard_False; Standard_Real d = PRef.SquareDistance(aPEb);
k = 1; if( d <= Precision::PConfusion() )
dmin = RealLast(); d = 0.;
} if( Abs(aPEb.X()-PRef.X()) < myTolU && Abs(aPEb.Y()-PRef.Y()) < myTolV )
else {
break; if( d <= dmin )
}// for iDone {
dfMinAngle = dfCurAngle;
kMin = k;
dmin = d;
}
}
}
}
it.Next();
k++;
}// while it
if(kMin == 0) if( kMin == 0 )
{ {
// probably unclosed in 2d space wire isDegenerated = Standard_False;
myEdge = TopoDS_Edge(); k = 1;
return; dmin = RealLast();
} }
else
break;
}// for iDone
// Selection the edge. if(kMin == 0)
it.Initialize(l); {
k = 1; // probably unclosed in 2d space wire
while( it.More() ) myEdge = TopoDS_Edge();
{ return;
if( k == kMin ) }
{
myEdge = TopoDS::Edge(it.Value()); // Selection the edge.
l.Remove(it); it.Initialize(l);
break; k = 1;
} while( it.More() )
it.Next(); {
k++; if( k == kMin )
} {
}//else face != NULL && l > 1 myEdge = TopoDS::Edge(it.Value());
l.Remove(it);
break;
}
it.Next();
k++;
}
}//else face != NULL && l > 1
}//else l > 1 }//else l > 1
} }

13
tests/bugs/moddata_3/bug23982 Executable file
View File

@ -0,0 +1,13 @@
puts "================"
puts "OCC23982"
puts "================"
puts ""
#######################################################
## Wire explorer raises exception
#######################################################
restore [locate_data_file bug23982_Cowl_965.brep] result
explode result w
wexplo result_1 result