diff --git a/src/BRepClass3d/BRepClass3d_SClassifier.cxx b/src/BRepClass3d/BRepClass3d_SClassifier.cxx index 5d2e594fdf..ed27b3f82f 100644 --- a/src/BRepClass3d/BRepClass3d_SClassifier.cxx +++ b/src/BRepClass3d/BRepClass3d_SClassifier.cxx @@ -30,12 +30,14 @@ #include #include #include +#include +#include static - void FaceNormal (const TopoDS_Face& aF, - const Standard_Real U, - const Standard_Real V, - gp_Dir& aDN); + Standard_Boolean FaceNormal (const TopoDS_Face& aF, + const Standard_Real U, + const Standard_Real V, + gp_Dir& aDN); static Standard_Real GetAddToParam(const gp_Lin& L,const Standard_Real P,const Bnd_Box& B); @@ -73,13 +75,17 @@ BRepClass3d_SClassifier::BRepClass3d_SClassifier(BRepClass3d_SolidExplorer& S, //======================================================================= void BRepClass3d_SClassifier::PerformInfinitePoint(BRepClass3d_SolidExplorer& aSE, const Standard_Real /*Tol*/) { - //-- Idea : Take point A in face1 and point B in face B - //-- (if there is only one face, take 2 points in the same face.) - //-- - //-- Intersect straight line AB with the solid and produce transition of the - //-- first point. If the solid has only one face and the straight line AB does not cut it - //-- it is not possible to decide. + //Take a normal to the first extracted face in its random inner point + //and intersect this reversed normal with the faces of the solid. + //If the min.par.-intersection point is + // a) inner point of a face + // b) transition is not TANGENT + // (the line does not touch the face but pierces it) + //then set to IN or OUT according to transition + //else take the next random point inside the min.par.-intersected face + //and continue + if(aSE.Reject(gp_Pnt(0,0,0))) { myState=3; //-- in ds solid case without face return; @@ -87,150 +93,83 @@ void BRepClass3d_SClassifier::PerformInfinitePoint(BRepClass3d_SolidExplorer& aS // //------------------------------------------------------------ // 1 - Standard_Boolean bFound, bFlag; - Standard_Integer nump; - Standard_Real aParam, aU1 = 0., aV1 = 0., aU2 = 0., aV2 = 0.; - gp_Pnt A,B; - gp_Dir aDN1, aDN2; - TopoDS_Face aF1, aF2; - // - nump = 0; - aParam = 0.5; - myFace.Nullify(); - myState=2; - for(aSE.InitShell(); aSE.MoreShell() && nump<2; aSE.NextShell()) { - for(aSE.InitFace(); aSE.MoreFace() && nump<2; ) { - TopoDS_Face aF = aSE.CurrentFace(); - aSE.NextFace(); - if(!nump) { - nump++; - bFound=aSE.FindAPointInTheFace(aF, A, aU1, aV1, aParam); - if (!bFound) { - return; - } - aF1=aF; - if(!aSE.MoreFace()) { - nump++; - bFound=aSE.FindAPointInTheFace(aF, B, aU2, aV2, aParam); - if (!bFound) { - return; - } - aF2=aF; - } - }// if(nump==0) { - else if(nump==1) { - bFound=aSE.FindAPointInTheFace(aF, B, aU2, aV2, aParam); - if(!bFound) { - return; - } - aF2=aF; - nump++; - } - }// for(aSE.InitFace(); aSE.MoreFace() && nump<2; ) { - }// for(aSE.InitShell(); aSE.MoreShell() && nump<2; aSE.NextShell()) { - // - //------------------------------------------------------------ - // 2 - Standard_Integer cpasbon; - Standard_Real parmin, aD2, aSP; - IntCurveSurface_TransitionOnCurve aTC; - TopAbs_State aState; - // - parmin = RealLast(); - // - bFlag=Standard_False; - if (aF1!=aF2) { - FaceNormal(aF1, aU1, aV1, aDN1); - FaceNormal(aF2, aU2, aV2, aDN2); - aSP=1.-aDN1*aDN2; - if (aSP < 1.e-5) { - bFlag=!bFlag; - } - } - // - aD2=A.SquareDistance(B); - if(aD2<0.000001 || bFlag) { - B.SetCoord(A.X()+1,A.Y()+1,A.Z()+1); - } - // - cpasbon = 0; - gp_Vec AB(A,B); - // - do { - switch (cpasbon) - { - case 1 : AB.SetX(-AB.X());break; - case 2 : AB.SetY(-AB.Y());break; - case 3 : AB.SetZ(-AB.Z());break; - case 4 : AB.SetY(-AB.Y());break; - case 5 : AB.SetX(-AB.X());break; - } - gp_Lin L(A,gp_Dir(AB)); - //-- cout<<"\npoint A "<D1 (U, V, aPnt, aD1U, aD1V); aN=aD1U.Crossed(aD1V); + if (aN.Magnitude() <= gp::Resolution()) + return Standard_False; + aN.Normalize(); aDN.SetXYZ(aN.XYZ()); if (aF.Orientation() == TopAbs_REVERSED){ aDN.Reverse(); } - return; + return Standard_True; } diff --git a/src/BRepClass3d/BRepClass3d_SolidExplorer.cxx b/src/BRepClass3d/BRepClass3d_SolidExplorer.cxx index 9173aa113b..92085f82c0 100644 --- a/src/BRepClass3d/BRepClass3d_SolidExplorer.cxx +++ b/src/BRepClass3d/BRepClass3d_SolidExplorer.cxx @@ -54,6 +54,8 @@ #include #include //<-OCC454(apo) +#include + //======================================================================= //function : FindAPointInTheFace @@ -177,6 +179,14 @@ Standard_Boolean BRepClass3d_SolidExplorer::FindAPointInTheFace ParamInit *= 0.41234; u_ = P.X() + ParamInit* T.X(); v_ = P.Y() + ParamInit* T.Y(); + + //Additional check + BRepTopAdaptor_FClass2d Classifier(face, Precision::Confusion()); + gp_Pnt2d aPnt2d(u_, v_); + TopAbs_State StateOfResultingPoint = Classifier.Perform(aPnt2d); + if (StateOfResultingPoint != TopAbs_IN) + return Standard_False; + BRepAdaptor_Surface s; s.Initialize (face, Standard_False); s.D1 (u_, v_, APoint_, theVecD1U, theVecD1V); diff --git a/tests/bugs/modalg_5/bug24481_1 b/tests/bugs/modalg_5/bug24481_1 new file mode 100755 index 0000000000..283af0abb1 --- /dev/null +++ b/tests/bugs/modalg_5/bug24481_1 @@ -0,0 +1,18 @@ +puts "================" +puts "OCC24481" +puts "================" +puts "" +####################################################################### +# Test "Perform Infinite Point" provides wrong result for a solid +####################################################################### + +set BugNumber OCC24481 + +restore [locate_data_file bug24481_b1_64.brep] b + +set res [xclassify b] +if { [regexp "OUT" $res] == 1 } { + puts "OK ${BugNumber}" +} else { + puts "Faulty ${BugNumber}" +} diff --git a/tests/bugs/modalg_5/bug24481_2 b/tests/bugs/modalg_5/bug24481_2 new file mode 100755 index 0000000000..4e9d67ecfb --- /dev/null +++ b/tests/bugs/modalg_5/bug24481_2 @@ -0,0 +1,19 @@ +puts "================" +puts "OCC24481" +puts "================" +puts "" +####################################################################### +# Test "Perform Infinite Point" provides wrong result for a solid +####################################################################### + +set BugNumber OCC24481 + +restore [locate_data_file bug24481_sdp.brep] b + +set res [xclassify b] + +if { [regexp "OUT" $res] == 1 } { + puts "OK ${BugNumber}" +} else { + puts "Faulty ${BugNumber}" +}