0023411: [CEA 2023] Bug MakeFillet1D SALOME master

This commit is contained in:
jfa 2017-02-16 12:47:37 +03:00
parent 7e333dca6e
commit dd458d72fb
2 changed files with 60 additions and 19 deletions

View File

@ -65,7 +65,7 @@ static Standard_Boolean IsDivideEdge(const TopoDS_Edge &theEdge,
gp_Pnt aPStart = aCurve->Value(theStart); gp_Pnt aPStart = aCurve->Value(theStart);
gp_Pnt aPEnd = aCurve->Value(theEnd); gp_Pnt aPEnd = aCurve->Value(theEnd);
TopoDS_Vertex aVFirst = TopExp::FirstVertex(theEdge); TopoDS_Vertex aVFirst = TopExp::FirstVertex(theEdge);
TopoDS_Vertex aVLast = TopExp::FirstVertex(theEdge); TopoDS_Vertex aVLast = TopExp::LastVertex(theEdge);
Standard_Real aTolFirst = BRep_Tool::Tolerance(aVFirst); Standard_Real aTolFirst = BRep_Tool::Tolerance(aVFirst);
Standard_Real aTolLast = BRep_Tool::Tolerance(aVLast); Standard_Real aTolLast = BRep_Tool::Tolerance(aVLast);
Standard_Real aTolConf = Precision::Confusion(); Standard_Real aTolConf = Precision::Confusion();
@ -81,7 +81,11 @@ static Standard_Boolean IsDivideEdge(const TopoDS_Edge &theEdge,
aDistSL <= aTolLast + aTolConf) { aDistSL <= aTolLast + aTolConf) {
if (aDistEF <= aTolFirst + aTolConf || if (aDistEF <= aTolFirst + aTolConf ||
aDistEL <= aTolLast + aTolConf) { aDistEL <= aTolLast + aTolConf) {
isSplit = Standard_False;
isSplit = Standard_False;
// in this case the original edge is thrown, and distance (gap) from new arc end
// to a vertex of original wire can reach (aVertexTolerance + Precision::Confusion()).
// Resulting wire is fixed (Mantis issue 0023411) in GEOMImpl_Fillet1dDriver::MakeFillet()
} }
} }
@ -367,7 +371,7 @@ Standard_Boolean GEOMImpl_Fillet1d::Perform(const Standard_Real theRadius)
aParam2 > myStart1 + aTol && aParam2 < myEnd1 - aTol) { aParam2 > myStart1 + aTol && aParam2 < myEnd1 - aTol) {
// Add the point in the list in increasing order. // Add the point in the list in increasing order.
const Standard_Real aParam = 0.5*(aParam1 + aParam2); const Standard_Real aParam = 0.5*(aParam1 + aParam2);
for(anIter.Initialize(aParams); anIter.More(); anIter.Next()) { for(anIter.Initialize(aParams); anIter.More(); anIter.Next()) {
if (anIter.Value() > aParam) { if (anIter.Value() > aParam) {
aParams.InsertBefore(aParam, anIter); aParams.InsertBefore(aParam, anIter);

View File

@ -48,6 +48,7 @@
#include <TopTools_IndexedMapOfShape.hxx> #include <TopTools_IndexedMapOfShape.hxx>
#include <BRep_Tool.hxx> #include <BRep_Tool.hxx>
#include <BRep_Builder.hxx>
#include <BRepTools.hxx> #include <BRepTools.hxx>
#include <BRepBuilderAPI_MakeWire.hxx> #include <BRepBuilderAPI_MakeWire.hxx>
#include <BRepCheck_Analyzer.hxx> #include <BRepCheck_Analyzer.hxx>
@ -270,11 +271,11 @@ Standard_Integer GEOMImpl_Fillet1dDriver::Execute(LOGBOOK& log) const
//function : MakeFillet //function : MakeFillet
//purpose : //purpose :
//======================================================================= //=======================================================================
bool GEOMImpl_Fillet1dDriver::MakeFillet(const TopoDS_Wire& aWire, bool GEOMImpl_Fillet1dDriver::MakeFillet(const TopoDS_Wire& theWire,
const TopTools_ListOfShape& aVertexList, const TopTools_ListOfShape& theVertexList,
const Standard_Real rad, const Standard_Real theRadius,
bool isFinalPass, bool isFinalPass,
TopoDS_Wire& aResult) const TopoDS_Wire& theResult) const
{ {
// this variable is needed to break execution // this variable is needed to break execution
// in case of fillet failure and try to fuse edges // in case of fillet failure and try to fuse edges
@ -290,8 +291,8 @@ bool GEOMImpl_Fillet1dDriver::MakeFillet(const TopoDS_Wire& aWire,
TopTools_ListOfShape aListOfNewEdge; TopTools_ListOfShape aListOfNewEdge;
// remember relation between initial and modified map // remember relation between initial and modified map
TopTools_IndexedDataMapOfShapeListOfShape aMapVToEdges; TopTools_IndexedDataMapOfShapeListOfShape aMapVToEdges;
TopExp::MapShapesAndAncestors( aWire, TopAbs_VERTEX, TopAbs_EDGE, aMapVToEdges ); TopExp::MapShapesAndAncestors( theWire, TopAbs_VERTEX, TopAbs_EDGE, aMapVToEdges );
TopTools_ListIteratorOfListOfShape anIt( aVertexList ); TopTools_ListIteratorOfListOfShape anIt( theVertexList );
for ( ; anIt.More(); anIt.Next() ) { for ( ; anIt.More(); anIt.Next() ) {
TopoDS_Vertex aV = TopoDS::Vertex( anIt.Value() ); TopoDS_Vertex aV = TopoDS::Vertex( anIt.Value() );
if ( aV.IsNull() || !aMapVToEdges.Contains( aV ) ) if ( aV.IsNull() || !aMapVToEdges.Contains( aV ) )
@ -313,7 +314,7 @@ bool GEOMImpl_Fillet1dDriver::MakeFillet(const TopoDS_Wire& aWire,
continue; // seems edges does not belong to same plane or parallel (fillet can not be build) continue; // seems edges does not belong to same plane or parallel (fillet can not be build)
GEOMImpl_Fillet1d aFilletAlgo (anEdge1, anEdge2, aPlane); GEOMImpl_Fillet1d aFilletAlgo (anEdge1, anEdge2, aPlane);
if (!aFilletAlgo.Perform(rad)) { if (!aFilletAlgo.Perform(theRadius)) {
if (isFinalPass) if (isFinalPass)
continue; // can not create fillet with given radius continue; // can not create fillet with given radius
else { else {
@ -356,23 +357,59 @@ bool GEOMImpl_Fillet1dDriver::MakeFillet(const TopoDS_Wire& aWire,
return false; return false;
// create new wire instead of original // create new wire instead of original
for (TopExp_Explorer anExp (aWire, TopAbs_EDGE); anExp.More(); anExp.Next()) { Standard_Real aTol;
Standard_Real aVertMaxTol = -RealLast();
for (TopExp_Explorer anExp (theWire, TopAbs_EDGE); anExp.More(); anExp.Next()) {
TopoDS_Shape anEdge = anExp.Current(); TopoDS_Shape anEdge = anExp.Current();
if (!anEdgeToEdgeMap.IsBound(anEdge)) if (!anEdgeToEdgeMap.IsBound(anEdge))
aListOfNewEdge.Append(anEdge); aListOfNewEdge.Append(anEdge);
else if (!anEdgeToEdgeMap.Find(anEdge).IsNull()) else if (!anEdgeToEdgeMap.Find(anEdge).IsNull())
aListOfNewEdge.Append(anEdgeToEdgeMap.Find(anEdge)); aListOfNewEdge.Append(anEdgeToEdgeMap.Find(anEdge));
// calculate maximum vertex tolerance of the initial wire
// to be used for resulting wire fixing (some gaps are possible)
for (TopExp_Explorer anExV (anEdge, TopAbs_VERTEX); anExV.More(); anExV.Next()) {
TopoDS_Vertex aVert = TopoDS::Vertex(anExV.Current());
aTol = BRep_Tool::Tolerance(aVert);
if (aTol > aVertMaxTol)
aVertMaxTol = aTol;
}
} }
GEOMUtils::SortShapes(aListOfNewEdge); // Fix for Mantis issue 0023411: BEGIN
BRep_Builder B;
TopoDS_Wire aResWire;
B.MakeWire(aResWire);
TopTools_ListIteratorOfListOfShape anItNewEdge (aListOfNewEdge);
for (; anItNewEdge.More(); anItNewEdge.Next()) {
B.Add(aResWire, TopoDS::Edge(anItNewEdge.Value()));
}
Handle(ShapeFix_Wire) aFW = new ShapeFix_Wire;
aFW->Load(aResWire);
aFW->FixReorder();
aFW->ClosedWireMode() = theWire.Closed();
// Fix for Mantis issue 0023411
// We forced to do this because of fillet 1d algorithm feature
// (see function DivideEdge in file GEOMImpl_Fillet1d.cxx):
// a distance (gap) from new arc end to a vertex of original wire
// can reach (aVertexTolerance + Precision::Confusion()),
// so a distance between two adjacent arcs will be covered by value below
aFW->FixConnected(aVertMaxTol*2.0 + Precision::Confusion()*3.0);
theResult = aFW->WireAPIMake();
GEOMUtils::FixShapeTolerance(theResult, TopAbs_VERTEX, Precision::Confusion());
BRepBuilderAPI_MakeWire aWireTool; // In OCCT 7.0.0 and earlier this gap was successfully covered by
aWireTool.Add(aListOfNewEdge); // implementation of BRepBuilderAPI_MakeWire::Add(TopTools_ListOfShape)
aWireTool.Build(); // (in the case described in Mantis issue 0023411)
if (!aWireTool.IsDone())
return 0;
aResult = aWireTool.Wire(); //GEOMUtils::SortShapes(aListOfNewEdge);
//BRepBuilderAPI_MakeWire aWireTool;
//aWireTool.Add(aListOfNewEdge);
//aWireTool.Build();
//if (!aWireTool.IsDone())
// return 0;
//theResult = aWireTool.Wire();
// Fix for Mantis issue 0023411: END
return isAllStepsOk; return isAllStepsOk;
} }
@ -408,7 +445,7 @@ GetCreationInformation(std::string& theOperationName,
default: default:
return false; return false;
} }
return true; return true;
} }