mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-04 18:06:22 +03:00
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
This commit is contained in:
parent
858aac0323
commit
253881cf19
@ -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;
|
||||
}
|
||||
|
@ -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);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -25,12 +25,14 @@
|
||||
#include <TopTools_ListOfShape.hxx>
|
||||
#include <TopTools_ListIteratorOfListOfShape.hxx>
|
||||
#include <TopTools_MapOfShape.hxx>
|
||||
#include <TopTools_MapOfOrientedShape.hxx>
|
||||
#include <TopTools_DataMapOfShapeListOfShape.hxx>
|
||||
#include <TopTools_DataMapIteratorOfDataMapOfShapeListOfShape.hxx>
|
||||
#include <TopTools_DataMapOfShapeInteger.hxx>
|
||||
#include <TopTools_DataMapOfShapeShape.hxx>
|
||||
#include <TopTools_DataMapIteratorOfDataMapOfShapeShape.hxx>
|
||||
#include <TopTools_MapIteratorOfMapOfShape.hxx>
|
||||
#include <TopTools_MapIteratorOfMapOfOrientedShape.hxx>
|
||||
#include <TopoDS_Iterator.hxx>
|
||||
#include <TopExp_Explorer.hxx>
|
||||
#include <BRep_Builder.hxx>
|
||||
@ -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"<<endl;
|
||||
Standard_ConstructionError::Raise();
|
||||
Standard_ConstructionError::Raise();
|
||||
}
|
||||
//fin MODIF.
|
||||
|
||||
tol1 = Max(BAS.UResolution(tol1), BAS.VResolution(tol1));
|
||||
|
||||
if(IsPeriodic) {
|
||||
cond = !(Vfirst.IsSame(Vlast) && SameUV(pfirst,plast,BAS));
|
||||
}
|
||||
else {
|
||||
cond = !(Vfirst.IsSame(Vlast));
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
for (exp.Init(wfirst.Oriented(TopAbs_FORWARD),TopAbs_EDGE);
|
||||
exp.More(); exp.Next()) {
|
||||
const TopoDS_Edge& edg = TopoDS::Edge(exp.Current());
|
||||
TopTools_ListIteratorOfListOfShape lexp(WiresFirst);
|
||||
for (; lexp.More(); lexp.Next()) {
|
||||
const TopoDS_Edge& edg = TopoDS::Edge(lexp.Value());
|
||||
if (!MapE.Contains(edg)) {
|
||||
B.Add(newW2,edg);
|
||||
MapE.Add(edg);
|
||||
}
|
||||
else if (myDblE.Contains(edg)) {
|
||||
for (itm.Initialize(MapE); itm.More(); itm.Next()) {
|
||||
const TopoDS_Edge& edg2 = TopoDS::Edge(itm.Key());
|
||||
if (edg.IsSame(edg2) && edg.Orientation() != edg2.Orientation()) {
|
||||
B.Add(newW2,edg);
|
||||
myDblE.Remove(edg);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
TopoDS_Face newF1,newF2;
|
||||
aLocalFace = FaceRef.EmptyCopied();
|
||||
newF1 = TopoDS::Face(aLocalFace);
|
||||
newF1.Orientation(TopAbs_FORWARD);
|
||||
aLocalFace = FaceRef.EmptyCopied();
|
||||
newF2 = TopoDS::Face(aLocalFace);
|
||||
// newF2 = TopoDS::Face(FaceRef.EmptyCopied());
|
||||
newF2.Orientation(TopAbs_FORWARD);
|
||||
|
||||
// modifs JAG 97.05.28
|
||||
@ -977,9 +943,7 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W,
|
||||
}
|
||||
}
|
||||
|
||||
B.Add(newF1,newW1.Oriented(orfila));
|
||||
B.Add(newF2,newW2.Oriented(orfila));
|
||||
// Standard_Boolean exch = Standard_False;
|
||||
newW1.Oriented(orfila);
|
||||
BRepTopAdaptor_FClass2d classif(newF1,Precision::PConfusion());
|
||||
if (classif.PerformInfinitePoint() == TopAbs_OUT) {
|
||||
BRepTopAdaptor_FClass2d classi(newF2,Precision::PConfusion());
|
||||
@ -987,13 +951,16 @@ void LocOpe_SplitShape::AddOpenWire(const TopoDS_Wire& W,
|
||||
TopoDS_Face tempF = newF2;
|
||||
newF2 = newF1;
|
||||
newF1 = tempF;
|
||||
newW2.Oriented(orfila);
|
||||
newW1.Oriented(TopAbs_FORWARD);
|
||||
}
|
||||
}
|
||||
B.Add(newF1,newW1);
|
||||
B.Add(newF2,newW2);
|
||||
|
||||
for (exp.ReInit(); exp.More(); exp.Next()) {
|
||||
const TopoDS_Wire& wir = TopoDS::Wire(exp.Current());
|
||||
if (!wir.IsSame(wfirst)) {
|
||||
// if (IsInside(F,wir,newW1) || IsInside(F,newW1,wir)) {
|
||||
if (IsInside(newF1, wir)) {
|
||||
B.Add(newF1,wir);
|
||||
}
|
||||
@ -1399,13 +1366,11 @@ static void ChoixUV(const TopoDS_Edge& Last,
|
||||
{
|
||||
|
||||
Standard_Real f,l;
|
||||
// gp_Pnt2d p2d,psav;
|
||||
gp_Pnt2d p2d;
|
||||
gp_Vec2d v2d;
|
||||
gp_Pnt aPCur, aPlst;
|
||||
|
||||
BRepAdaptor_Surface surf(F,Standard_False); // no restriction
|
||||
// Standard_Real tol = Precision::PConfusion() //BRep_Tool::Tolerance(Last));
|
||||
surf.D0 (plst.X(), plst.Y(), aPlst);
|
||||
|
||||
Standard_Real tol;
|
||||
|
Loading…
x
Reference in New Issue
Block a user