From 253881cf190b2f0700e4d58790402f978b358cc7 Mon Sep 17 00:00:00 2001 From: epa Date: Sat, 5 May 2012 16:32:03 +0400 Subject: [PATCH] 0023029: split shape algorithm works incorrectly in some cases Correct handling of periodic bsplines in extrema. If range of edge is out of the period, interval boundaries should be recomputed Face orientation is changed to forward for simplification of wires collection regressions were fixed Bounding box for periodic splines is corrected --- src/BndLib/BndLib_Add3dCurve.cxx | 15 ++-- src/Extrema/Extrema_GExtPC.gxx | 81 +++++++++++--------- src/LocOpe/LocOpe_SplitShape.cxx | 125 +++++++++++-------------------- 3 files changed, 98 insertions(+), 123 deletions(-) diff --git a/src/BndLib/BndLib_Add3dCurve.cxx b/src/BndLib/BndLib_Add3dCurve.cxx index 312bc9241b..8bfd50e270 100755 --- a/src/BndLib/BndLib_Add3dCurve.cxx +++ b/src/BndLib/BndLib_Add3dCurve.cxx @@ -146,12 +146,15 @@ void BndLib_Add3dCurve::Add( const Adaptor3d_Curve& C, //// modified by jgv, 24.10.01 for BUC61031 //// if (Bsaux->IsPeriodic()) ElCLib::AdjustPeriodic( Bsaux->FirstParameter(), Bsaux->LastParameter(), Precision::PConfusion(), u1, u2 ); - //////////////////////////////////////////////// - // modified by NIZHNY-EAP Fri Dec 3 14:29:14 1999 ___BEGIN___ - // To avoid exeption in Segment - if(Bsaux->FirstParameter() > U1) u1 = Bsaux->FirstParameter(); - if(Bsaux->LastParameter() < U2 ) u2 = Bsaux->LastParameter(); - // modified by NIZHNY-EAP Fri Dec 3 14:29:18 1999 ___END___ + else + { + //////////////////////////////////////////////// + // modified by NIZHNY-EAP Fri Dec 3 14:29:14 1999 ___BEGIN___ + // To avoid exeption in Segment + if(Bsaux->FirstParameter() > U1) u1 = Bsaux->FirstParameter(); + if(Bsaux->LastParameter() < U2 ) u2 = Bsaux->LastParameter(); + // modified by NIZHNY-EAP Fri Dec 3 14:29:18 1999 ___END___ + } Bsaux->Segment(u1, u2); Bs = Bsaux; } diff --git a/src/Extrema/Extrema_GExtPC.gxx b/src/Extrema/Extrema_GExtPC.gxx index 6661c088e0..ddaf85e83a 100755 --- a/src/Extrema/Extrema_GExtPC.gxx +++ b/src/Extrema/Extrema_GExtPC.gxx @@ -104,6 +104,8 @@ void Extrema_GExtPC::Perform(const ThePoint& P) Standard_Boolean IntIsNotValid; n = TheCurveTool::NbIntervals(*((TheCurve*)myC), GeomAbs_C2); TColStd_Array1OfReal theInter(1, n+1); + Standard_Boolean isPeriodic = TheCurveTool::IsPeriodic(*((TheCurve*)myC)); + Standard_Real aPeriodicShift = 0.; TheCurveTool::Intervals(*((TheCurve*)myC), theInter, GeomAbs_C2); mysample = Max(mysample/n, 17); TheVector V1; @@ -111,34 +113,40 @@ void Extrema_GExtPC::Perform(const ThePoint& P) Standard_Real s1 = 0.0 ; Standard_Real s2 = 0.0; for (i = 1; i <= n; i++) { - myintuinf = theInter(i); - myintusup = theInter(i+1); + myintuinf = theInter(i); + myintusup = theInter(i+1); + + Standard_Real anInfToCheck = myintuinf; + Standard_Real aSupToCheck = myintusup; + + if (isPeriodic) { + Standard_Real aPeriod = TheCurveTool::Period(*((TheCurve*)myC)); + anInfToCheck = ElCLib::InPeriod(myintuinf, myuinf, myuinf+aPeriod); + aSupToCheck = myintusup+(anInfToCheck-myintuinf); + } + IntIsNotValid = (myuinf > aSupToCheck) || (myusup < anInfToCheck); - IntIsNotValid = (myuinf > myintusup) || - (myusup < myintuinf); + if(IntIsNotValid) continue; - if(IntIsNotValid) continue; + if (myuinf >= anInfToCheck) anInfToCheck = myuinf; + if (myusup <= aSupToCheck) aSupToCheck = myusup; + if((aSupToCheck - anInfToCheck) <= mytolu) continue; - if (myuinf >= myintuinf) myintuinf = myuinf; - if (myusup <= myintusup) myintusup = myusup; - - if((myintusup - myintuinf) <= mytolu) continue; - - if (i != 1) { - TheCurveTool::D1(*((TheCurve*)myC), myintuinf, PP, V1); - s1 = (TheVector(P, PP))*V1; - if (s1*s2 < 0.0) { - mySqDist.Append(PP.SquareDistance(P)); - myismin.Append((s1 < 0.0)); - mypoint.Append(ThePOnC(myintuinf, PP)); - } - } - if (i != n) { - TheCurveTool::D1(*((TheCurve*)myC), myintusup, PP, V1); - s2 = (TheVector(P, PP))*V1; - } - IntervalPerform(P); - IntExtIsDone = IntExtIsDone || mydone; + if (i != 1) { + TheCurveTool::D1(*((TheCurve*)myC), myintuinf, PP, V1); + s1 = (TheVector(P, PP))*V1; + if (s1*s2 < 0.0) { + mySqDist.Append(PP.SquareDistance(P)); + myismin.Append((s1 < 0.0)); + mypoint.Append(ThePOnC(myintuinf, PP)); + } + } + if (i != n) { + TheCurveTool::D1(*((TheCurve*)myC), myintusup, PP, V1); + s2 = (TheVector(P, PP))*V1; + } + IntervalPerform(P); + IntExtIsDone = IntExtIsDone || mydone; } mydone = IntExtIsDone; return; @@ -154,17 +162,16 @@ void Extrema_GExtPC::Perform(const ThePoint& P) ThePOnC PC = myExtPElC.Point(i); U = PC.Parameter(); if (TheCurveTool::IsPeriodic(*((TheCurve*)myC))) { - U = ElCLib::InPeriod(U, myuinf, myuinf+TheCurveTool::Period(*((TheCurve*)myC))); + U = ElCLib::InPeriod(U, myuinf, myuinf+TheCurveTool::Period(*((TheCurve*)myC))); } if ((U >= myuinf-mytolu) && (U <= myusup+mytolu)){ - PC.SetValues(U, myExtPElC.Point(i).Value()); - mySqDist.Append(myExtPElC.SquareDistance(i)); - myismin.Append(myExtPElC.IsMin(i)); - mypoint.Append(PC); + PC.SetValues(U, myExtPElC.Point(i).Value()); + mySqDist.Append(myExtPElC.SquareDistance(i)); + myismin.Append(myExtPElC.IsMin(i)); + mypoint.Append(PC); } } - } - + } } @@ -211,13 +218,13 @@ void Extrema_GExtPC::IntervalPerform(const ThePoint& P) ThePOnC PC = myExtPC.Point(i); U = PC.Parameter(); if (TheCurveTool::IsPeriodic(*((TheCurve*)myC))) { - U = ElCLib::InPeriod(U, myuinf, myuinf+TheCurveTool::Period(*((TheCurve*)myC))); + U = ElCLib::InPeriod(U, myuinf, myuinf+TheCurveTool::Period(*((TheCurve*)myC))); } if ((U >= myuinf - mytolu) && (U <= myusup + mytolu)) { - PC.SetValues(U, PC.Value()); - mySqDist.Append(myExtPC.SquareDistance(i)); - myismin.Append(myExtPC.IsMin(i)); - mypoint.Append(PC); + PC.SetValues(U, PC.Value()); + mySqDist.Append(myExtPC.SquareDistance(i)); + myismin.Append(myExtPC.IsMin(i)); + mypoint.Append(PC); } } } diff --git a/src/LocOpe/LocOpe_SplitShape.cxx b/src/LocOpe/LocOpe_SplitShape.cxx index 907c755c4b..738cb1b915 100755 --- a/src/LocOpe/LocOpe_SplitShape.cxx +++ b/src/LocOpe/LocOpe_SplitShape.cxx @@ -25,12 +25,14 @@ #include #include #include +#include #include #include #include #include #include #include +#include #include #include #include @@ -671,28 +673,26 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, BRepTools::Update(F); - // TopExp::Vertices(W,Vfirst,Vlast); - Standard_Real tolf, toll, tol1; TopoDS_Shape aLocalShape = W.Oriented(TopAbs_FORWARD); TopExp::Vertices(TopoDS::Wire(aLocalShape),Vfirst,Vlast); - // TopExp::Vertices(TopoDS::Wire(W.Oriented(TopAbs_FORWARD)),Vfirst,Vlast); tolf = BRep_Tool::Tolerance(Vfirst); toll = BRep_Tool::Tolerance(Vlast); tol1 = Max(tolf, toll); - TopExp_Explorer exp,exp2; + TopExp_Explorer exp,exp2; TopoDS_Wire wfirst,wlast; for (; itl.More(); itl.Next()) { - const TopoDS_Face& fac = TopoDS::Face(itl.Value()); + TopoDS_Face fac = TopoDS::Face(itl.Value()); if (!IsInside(fac,W)) { continue; } - + + fac.Orientation(TopAbs_FORWARD); Standard_Boolean ffound = Standard_False; Standard_Boolean lfound = Standard_False; for (exp.Init(fac,TopAbs_WIRE); exp.More(); exp.Next()) { @@ -736,11 +736,12 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, if (wfirst.IsSame(wlast)) { // on cree 2 faces en remplacement de itl.Value() // Essai JAG - + TopTools_ListOfShape WiresFirst; for (exp.Init(wfirst,TopAbs_EDGE); exp.More(); exp.Next()) { if (BRep_Tool::IsClosed(TopoDS::Edge(exp.Current()),FaceRef)) { myDblE.Add(exp.Current()); } + WiresFirst.Append(exp.Current()); } TopAbs_Orientation orient; @@ -750,17 +751,19 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, B.MakeWire(newW2); newW2.Orientation(TopAbs_FORWARD); - Standard_Integer nbE = 0; + Standard_Integer nbE = 0; for (exp.Init(W.Oriented(TopAbs_FORWARD),TopAbs_EDGE); exp.More(); exp.Next()) { nbE++; const TopoDS_Edge& edg = TopoDS::Edge(exp.Current()); - orient = edg.Orientation(); - B.Add(newW1,edg); - B.Add(newW2,edg.Oriented(TopAbs::Reverse(orient))); - } + orient = edg.Orientation(); + WiresFirst.Append(edg); + WiresFirst.Append(edg.Oriented(TopAbs::Reverse(orient))); + myDblE.Add(edg); + } - TopTools_MapOfShape MapE, PossE; + TopTools_MapOfShape PossE; + TopTools_MapOfOrientedShape MapE; TopTools_MapIteratorOfMapOfShape itm; TopoDS_Vertex vdeb,vfin; Standard_Integer nbPoss; @@ -789,10 +792,7 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, TopoDS_Shape aLocalFace = FaceRef.Oriented(wfirst.Orientation()); C2d = BRep_Tool::CurveOnSurface(LastEdge, TopoDS::Face(aLocalFace), f, l); - // C2d = BRep_Tool::CurveOnSurface - // (LastEdge, - // TopoDS::Face(FaceRef.Oriented(wfirst.Orientation())), - // f,l); + if (LastEdge.Orientation() == TopAbs_FORWARD) { pfirst = C2d->Value(f); } @@ -818,16 +818,11 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, } aLocalFace = FaceRef.Oriented(wfirst.Orientation()); C2d = BRep_Tool::CurveOnSurface(LastEdge, TopoDS::Face(aLocalFace), f, l); - // C2d = BRep_Tool::CurveOnSurface - // (LastEdge, - // TopoDS::Face(FaceRef.Oriented(wfirst.Orientation())), - // f,l); + if (LastEdge.Orientation() == TopAbs_FORWARD) { C2d->D1(l,plast,dlast); - // plast = C2d->Value(l); } else { - // plast = C2d->Value(f); C2d->D1(f,plast,dlast); dlast.Reverse(); } @@ -841,17 +836,14 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, else { cond = !(Vfirst.IsSame(Vlast)); } - - while (cond) { + + while (cond) { PossE.Clear(); - // On enchaine par la fin - for (exp.Init(wfirst.Oriented(TopAbs_FORWARD),TopAbs_EDGE); - exp.More(); exp.Next()) { - const TopoDS_Edge& edg = TopoDS::Edge(exp.Current()); - if (MapE.Contains(edg) && !myDblE.Contains(edg)) { - continue; - } + // On enchaine par la fin + TopTools_ListIteratorOfListOfShape lexp(WiresFirst); + for (; lexp.More(); lexp.Next()) { + const TopoDS_Edge& edg = TopoDS::Edge(lexp.Value()); orient = edg.Orientation(); TopExp::Vertices(edg,vdeb,vfin); @@ -863,21 +855,21 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, } } nbPoss = PossE.Extent(); + if (nbPoss == 0) + { + break; + } + if (nbPoss == 1) { itm.Initialize(PossE); TopoDS_Shape aLocalFace = FaceRef.Oriented(wfirst.Orientation()); C2d = BRep_Tool::CurveOnSurface(TopoDS::Edge(itm.Key()), TopoDS::Face(aLocalFace), f, l); - // C2d = BRep_Tool::CurveOnSurface - // (TopoDS::Edge(itm.Key()), - // TopoDS::Face(FaceRef.Oriented(wfirst.Orientation())), - // f,l); + if (itm.Key().Orientation() == TopAbs_FORWARD) { - // plast = C2d->Value(l); C2d->D1(l,plast,dlast); } else { - // plast = C2d->Value(f); C2d->D1(f,plast,dlast); dlast.Reverse(); } @@ -888,24 +880,15 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, ChoixUV(LastEdge, TopoDS::Face(aLocalFace), PossE, itm, plast, dlast, toll); - // ChoixUV(LastEdge, - // TopoDS::Face(FaceRef.Oriented(wfirst.Orientation())), - // PossE, - // itm, - // plast, - // dlast, toll); - } if (nbPoss >= 1) { + if (MapE.Contains(itm.Key())) + break; B.Add(newW1,itm.Key()); - if (MapE.Contains(itm.Key())) { - myDblE.Remove(itm.Key()); - } - else { - MapE.Add(itm.Key()); - } + MapE.Add(itm.Key()); LastEdge = TopoDS::Edge(itm.Key()); + if (LastEdge.Orientation() == TopAbs_FORWARD) { Vlast = TopExp::LastVertex(LastEdge); } @@ -920,46 +903,29 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W, //MODIFICATION PIERRE SMEYERS : si pas de possibilite, on sort avec erreur else{ cout<<"erreur Spliter : pas de chainage du wire"<