diff --git a/src/Approx/Approx_SameParameter.cxx b/src/Approx/Approx_SameParameter.cxx index a4e467e611..9bb7b22e24 100644 --- a/src/Approx/Approx_SameParameter.cxx +++ b/src/Approx/Approx_SameParameter.cxx @@ -526,6 +526,8 @@ void Approx_SameParameter::Build(const Standard_Real Tolerance) } myDone = Standard_True; } + + myCurveOnSurface = Handle(Adaptor3d_CurveOnSurface)::DownCast(aData.myCOnS.ShallowCopy()); } //======================================================================= diff --git a/src/Approx/Approx_SameParameter.hxx b/src/Approx/Approx_SameParameter.hxx index 617c2b2573..257a019336 100644 --- a/src/Approx/Approx_SameParameter.hxx +++ b/src/Approx/Approx_SameParameter.hxx @@ -81,6 +81,21 @@ public: return myCurve2d; } + //! Returns the 3D curve that has the same parameter as + //! the 3D curve once evaluated on the surface up to the + //! specified tolerance. + Handle(Adaptor3d_Curve) Curve3d() const + { + return myC3d; + } + + //! Returns the 3D curve on surface that has the same parameter as + //! the 3D curve up to the specified tolerance. + Handle(Adaptor3d_CurveOnSurface) CurveOnSurface() const + { + return myCurveOnSurface; + } + private: //! Internal data structure to unify access to the most actively used data. @@ -176,6 +191,7 @@ private: Handle(Adaptor2d_Curve2d) myHCurve2d; Handle(Adaptor3d_Curve) myC3d; Handle(Adaptor3d_Surface) mySurf; + Handle(Adaptor3d_CurveOnSurface) myCurveOnSurface; }; #endif // _Approx_SameParameter_HeaderFile diff --git a/src/BRepFill/BRepFill_Sweep.cxx b/src/BRepFill/BRepFill_Sweep.cxx index a74fdfc452..1be188dd79 100644 --- a/src/BRepFill/BRepFill_Sweep.cxx +++ b/src/BRepFill/BRepFill_Sweep.cxx @@ -85,6 +85,7 @@ #include #include #include +#include #include //#include @@ -289,6 +290,34 @@ static Standard_Boolean CheckSameParameter return Standard_True; } +//======================================================================= +//function : CheckSameParameterExact +//purpose : Check a posteriori that sameparameter has worked correctly +// with exact calculation method of edge tolerance +//======================================================================= +static Standard_Boolean CheckSameParameterExact +(const Handle(Adaptor3d_Curve)& C3d, + const Handle(Adaptor3d_CurveOnSurface)& curveOnSurface, + const Standard_Real tol3d, + Standard_Real& tolreached) +{ + GeomLib_CheckCurveOnSurface aCheckCurveOnSurface(C3d); + aCheckCurveOnSurface.SetParallel(Standard_False); + aCheckCurveOnSurface.Perform(curveOnSurface); + + tolreached = aCheckCurveOnSurface.MaxDistance(); + + if (tolreached > tol3d) { + return Standard_False; + } + else + { + tolreached = Max(tolreached, Precision::Confusion()); + tolreached *= 1.05; + } + return Standard_True; +} + //======================================================================= //function : SameParameter //purpose : Encapsulation of Sameparameter @@ -340,8 +369,10 @@ static Standard_Boolean SameParameter(TopoDS_Edge& E, return Standard_False; } - ResTol = sp.TolReached(); - if(ResTol > tolreached ){ + Handle(Adaptor3d_Curve) curve3d = sp.Curve3d(); + Handle(Adaptor3d_CurveOnSurface) curveOnSurface = sp.CurveOnSurface(); + + if (!CheckSameParameterExact(curve3d, curveOnSurface, tol3d, ResTol) && ResTol > tolreached) { #ifdef OCCT_DEBUG std::cout<<"SameParameter : Tolerance not reached!"<InterfFF(); Standard_Integer aNbFFs = aFFs.Length(); - if(!SplitUEdges(myUEdges, theDS, myHistMap)) { + if(!SplitUEdges(myUEdges, theDS, myIntPointCrossDir, myHistMap)) { return; } @@ -425,7 +429,7 @@ BRepFill_TrimShellCorner::MakeFacesNonSec(const Standard_Integer Standard_Real apar1 = 0., apar2 = 0.; Standard_Boolean bvertexfound = - FindCommonVertex(theDS, anIndex1, anIndex2, aCommonVertex, apar1, apar2); + FindCommonVertex(theDS, anIndex1, anIndex2, myIntPointCrossDir, aCommonVertex, apar1, apar2); // search common vertex between bounds. end Handle(BRepTools_ReShape) aSubstitutor = new BRepTools_ReShape(); @@ -697,9 +701,9 @@ BRepFill_TrimShellCorner::MakeFacesSec(const Standard_Integer TopoDS_Vertex FirstVertex, LastVertex; Standard_Real ParamOnLeftE1, ParamOnLeftE2, ParamOnRightE1, ParamOnRightE2; - FindCommonVertex(theDS, IndexOfLeftE1, IndexOfLeftE2, + FindCommonVertex(theDS, IndexOfLeftE1, IndexOfLeftE2, myIntPointCrossDir, FirstVertex, ParamOnLeftE1, ParamOnLeftE2); - FindCommonVertex(theDS, IndexOfRightE1, IndexOfRightE2, + FindCommonVertex(theDS, IndexOfRightE1, IndexOfRightE2, myIntPointCrossDir, LastVertex, ParamOnRightE1, ParamOnRightE2); TopoDS_Shape SecWire; @@ -1079,7 +1083,8 @@ Standard_Boolean BRepFill_TrimShellCorner::ChooseSection(const TopoDS_Shape& Com // purpose: // ------------------------------------------------------------------------------------------ Standard_Boolean SplitUEdges(const Handle(TopTools_HArray2OfShape)& theUEdges, - const BOPDS_PDS& theDS, + const BOPDS_PDS& theDS, + const gp_Vec& theCrossDirection, TopTools_DataMapOfShapeListOfShape& theHistMap) { const BOPDS_VectorOfInterfVV& aVVs = theDS->InterfVV(); @@ -1105,7 +1110,7 @@ Standard_Boolean SplitUEdges(const Handle(TopTools_HArray2OfShape)& theUEdge TopoDS_Vertex aCommonVertex; Standard_Real apar1 = 0., apar2 = 0.; Standard_Boolean bvertexfound = - FindCommonVertex(theDS, anEIndex1, anEIndex2, aCommonVertex, apar1, apar2); + FindCommonVertex(theDS, anEIndex1, anEIndex2, theCrossDirection, aCommonVertex, apar1, apar2); // if(!bvertexfound) { TopoDS_Vertex V1 = TopExp::LastVertex(TopoDS::Edge(aE1)); @@ -1217,6 +1222,7 @@ void FindFreeVertices(const TopoDS_Shape& theShape, Standard_Boolean FindCommonVertex(const BOPDS_PDS& theDS, const Standard_Integer theEIndex1, const Standard_Integer theEIndex2, + const gp_Vec& theCrossDirection, TopoDS_Vertex& theCommonVertex, Standard_Real& theParamOnE1, Standard_Real& theParamOnE2) { @@ -1227,6 +1233,10 @@ Standard_Boolean FindCommonVertex(const BOPDS_PDS& theDS, TopoDS_Vertex aCommonVertex; Standard_Integer eeit = 0; + TopoDS_Edge theE1 = TopoDS::Edge(theDS->Shape(theEIndex1)); + TopoDS_Edge theE2 = TopoDS::Edge(theDS->Shape(theEIndex2)); + BRepAdaptor_Curve aBC1(theE1), aBC2(theE2); + Standard_Integer aNbEEs; aNbEEs = aEEs.Length(); for(eeit = 0; eeit < aNbEEs; ++eeit) { @@ -1247,10 +1257,21 @@ Standard_Boolean FindCommonVertex(const BOPDS_PDS& theDS, IntTools_Tools::VertexParameters(aCP, theParamOnE1, theParamOnE2); else IntTools_Tools::VertexParameters(aCP, theParamOnE2, theParamOnE1); - - // - bvertexfound = Standard_True; - break; + + gp_Pnt aPt; + gp_Vec aDirOnE1, aDirOnE2; + gp_Dir aIntersectPointCrossDir; + + // intersect point aDirOnE1.cross(aDirOnE2) should same direction with path theCrossDirection + aBC1.D1(theParamOnE1, aPt, aDirOnE1); + aBC2.D1(theParamOnE2, aPt, aDirOnE2); + aIntersectPointCrossDir = aDirOnE1.Crossed(aDirOnE2); + + if (aIntersectPointCrossDir.Dot(theCrossDirection) > Precision::SquareConfusion()) + { + bvertexfound = Standard_True; + break; + } } } } diff --git a/src/BRepFill/BRepFill_TrimShellCorner.hxx b/src/BRepFill/BRepFill_TrimShellCorner.hxx index 283e546b91..f25e1171df 100644 --- a/src/BRepFill/BRepFill_TrimShellCorner.hxx +++ b/src/BRepFill/BRepFill_TrimShellCorner.hxx @@ -41,9 +41,12 @@ public: //! Constructor: takes faces to intersect, //! type of transition (it can be RightCorner or RoundCorner) //! and axis of bisector plane + //! theIntersectPointCrossDirection : prev path direction at the origin point of theAxeOfBisPlane + //! cross next path direction at the origin point of theAxeOfBisPlane. used when EE has more than one vertices Standard_EXPORT BRepFill_TrimShellCorner(const Handle(TopTools_HArray2OfShape)& theFaces, const BRepFill_TransitionStyle theTransition, - const gp_Ax2& theAxeOfBisPlane); + const gp_Ax2& theAxeOfBisPlane, + const gp_Vec& theIntPointCrossDir); Standard_EXPORT void AddBounds (const Handle(TopTools_HArray2OfShape)& Bounds); @@ -92,6 +95,7 @@ private: BRepFill_TransitionStyle myTransition; gp_Ax2 myAxeOfBisPlane; + gp_Vec myIntPointCrossDir; TopoDS_Shape myShape1; TopoDS_Shape myShape2; Handle(TopTools_HArray2OfShape) myBounds; diff --git a/tests/bugs/modalg_7/bug29204 b/tests/bugs/modalg_7/bug29204 index 8ce39b7bdc..80d4fde2d0 100644 --- a/tests/bugs/modalg_7/bug29204 +++ b/tests/bugs/modalg_7/bug29204 @@ -19,7 +19,7 @@ checknbshapes result -solid 1 -shell 1 -face 28 -wire 28 -edge 64 -vertex 36 -sh set tolres [checkmaxtol result] -if { ${tolres} > 0.001} { +if { ${tolres} > 0.003} { puts "Error: bad tolerance of result" } diff --git a/tests/bugs/modalg_7/bug29663 b/tests/bugs/modalg_7/bug29663 index af381765e9..eb1b2911af 100644 --- a/tests/bugs/modalg_7/bug29663 +++ b/tests/bugs/modalg_7/bug29663 @@ -21,7 +21,7 @@ fixshape result rr checkshape result checkprops result -s 2.14316e+011 -checkmaxtol result -ref 0.049038750060552701 +checkmaxtol result -ref 0.046703571498366049 checknbshapes result -shell 1 -face 201 -wire 201 diff --git a/tests/evolved/voluved/HMC008 b/tests/evolved/voluved/HMC008 index 63c57b5117..ce56438bf1 100644 --- a/tests/evolved/voluved/HMC008 +++ b/tests/evolved/voluved/HMC008 @@ -212,7 +212,7 @@ if {[regexp "Faulties" [bopargcheck result]]} { # the dimensions of the shape "result" are about 1.0e+5. # So, this tolerance seems to be OK. -checkmaxtol result -ref 1.7319951447770465 +checkmaxtol result -ref 1.6514213338573371 smallview don result sw tw diff --git a/tests/pipe/bugs/bug32807 b/tests/pipe/bugs/bug32807 new file mode 100644 index 0000000000..a86c2e8b0e --- /dev/null +++ b/tests/pipe/bugs/bug32807 @@ -0,0 +1,17 @@ +puts "========" +puts "bug32807: Sweep algorithm deal with EE more than one intersect points" +puts "========" +puts "" + +restore [locate_data_file bug32807_path.brep] p +restore [locate_data_file bug32807_profile.brep] pr + +mksweep p +setsweep -CF +addsweep pr + +buildsweep result -C -S + +checkshape result -exact +checknbshapes result -vertex 34 -edge 68 -wire 34 -face 34 -shell 1 +checkview -display result -2d -path ${imagedir}/${test_image}.png