mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-03 17:56:21 +03:00
0031736: ShapeUpgrade_UnifySameDomain algorithm does not unify all possible faces
1. Correct processing faces that can be converted into planar ones. 2. Correct processing faces based on V-closed surfaces.
This commit is contained in:
parent
ed753e10df
commit
c3e0e1de43
@ -495,8 +495,9 @@ static void ReconstructMissedSeam(const TopTools_SequenceOfShape& theEdges,
|
||||
const TopoDS_Face& theFrefFace,
|
||||
const TopoDS_Vertex& theCurVertex,
|
||||
const gp_Pnt2d& theCurPoint,
|
||||
const Standard_Real theUperiod,
|
||||
const Standard_Real theFaceUmin,
|
||||
const Standard_Real thePeriod,
|
||||
const Standard_Real theFaceCoordMin,
|
||||
const Standard_Boolean theIsU,
|
||||
const Standard_Real theCoordTol,
|
||||
TopoDS_Edge& theNextEdge,
|
||||
TopoDS_Wire& theNewWire,
|
||||
@ -507,16 +508,24 @@ static void ReconstructMissedSeam(const TopTools_SequenceOfShape& theEdges,
|
||||
|
||||
{
|
||||
Handle(Geom_Surface) RefSurf = BRep_Tool::Surface(theFrefFace);
|
||||
GeomAbs_Shape aContinuity = (RefSurf->IsUPeriodic())? GeomAbs_CN : GeomAbs_C0;
|
||||
GeomAbs_Shape aContinuity;
|
||||
if (theIsU)
|
||||
aContinuity = (RefSurf->IsUPeriodic())? GeomAbs_CN : GeomAbs_C0;
|
||||
else
|
||||
aContinuity = (RefSurf->IsVPeriodic())? GeomAbs_CN : GeomAbs_C0;
|
||||
|
||||
Standard_Integer IndCoord = theIsU? 1 : 2;
|
||||
|
||||
Standard_Real Ydir = 1.; //up
|
||||
if (Abs(theCurPoint.X() - theFaceUmin) <= theCoordTol)
|
||||
Ydir = -1.; //down
|
||||
Standard_Real SeamDir = 1.; //up or right
|
||||
if (theIsU && Abs(theCurPoint.Coord(IndCoord) - theFaceCoordMin) <= theCoordTol)
|
||||
SeamDir = -1.; //down
|
||||
else if (!theIsU && Abs(theCurPoint.Coord(IndCoord) - theFaceCoordMin) > theCoordTol)
|
||||
SeamDir = -1.; //left
|
||||
|
||||
//Consider as the candidate to be next edge:
|
||||
//only the edge that has first point with X-coordinate close to X-coordinate of theCurPoint
|
||||
//Choose from candidates the edge that is closest to theCurPoint in the defined direction Ydir
|
||||
Standard_Real MinDeltaY = RealLast();
|
||||
//only the edge that has first point with X(or Y)-coordinate close to X(or Y)-coordinate of theCurPoint
|
||||
//Choose from candidates the edge that is closest to theCurPoint in the defined direction SeamDir
|
||||
Standard_Real MinDeltaSeamCoord = RealLast();
|
||||
for (Standard_Integer ind = 1; ind <= theEdges.Length(); ind++)
|
||||
{
|
||||
const TopoDS_Edge& aCandidate = TopoDS::Edge(theEdges(ind));
|
||||
@ -526,17 +535,17 @@ static void ReconstructMissedSeam(const TopTools_SequenceOfShape& theEdges,
|
||||
Standard_Real CandParam = (aCandidate.Orientation() == TopAbs_FORWARD)?
|
||||
BAcurve2d.FirstParameter() : BAcurve2d.LastParameter();
|
||||
gp_Pnt2d CandPoint = BAcurve2d.Value(CandParam);
|
||||
Standard_Real DeltaX = Abs(CandPoint.X() - theCurPoint.X());
|
||||
if (DeltaX > theCoordTol)
|
||||
Standard_Real DeltaCoord = Abs(CandPoint.Coord(IndCoord) - theCurPoint.Coord(IndCoord));
|
||||
if (DeltaCoord > theCoordTol)
|
||||
continue;
|
||||
|
||||
Standard_Real DeltaY = CandPoint.Y() - theCurPoint.Y();
|
||||
DeltaY *= Ydir;
|
||||
if (DeltaY < 0.) //on the other side from CurPoint
|
||||
Standard_Real DeltaSeamCoord = CandPoint.Coord(3-IndCoord) - theCurPoint.Coord(3-IndCoord);
|
||||
DeltaSeamCoord *= SeamDir;
|
||||
if (DeltaSeamCoord < 0.) //on the other side from CurPoint
|
||||
continue;
|
||||
if (DeltaY < MinDeltaY)
|
||||
if (DeltaSeamCoord < MinDeltaSeamCoord)
|
||||
{
|
||||
MinDeltaY = DeltaY;
|
||||
MinDeltaSeamCoord = DeltaSeamCoord;
|
||||
theNextEdge = aCandidate;
|
||||
theStartOfNextEdge = CandPoint;
|
||||
}
|
||||
@ -545,58 +554,68 @@ static void ReconstructMissedSeam(const TopTools_SequenceOfShape& theEdges,
|
||||
//Build missed seam edge
|
||||
theLastVertexOfSeam = TopExp::FirstVertex(theNextEdge, Standard_True); //with orientation
|
||||
TopoDS_Vertex V1, V2;
|
||||
Standard_Real Param1, Param2, anU = 0.;
|
||||
Handle(Geom_Curve) Uiso;
|
||||
Standard_Real Param1, Param2, aCoord = 0.;
|
||||
Handle(Geom_Curve) Iso;
|
||||
|
||||
TopoDS_Edge aRemovedEdge; //try to find it in <RemovedEdges>
|
||||
for (Standard_Integer i = 1; i <= theRemovedEdges.Length(); i++)
|
||||
{
|
||||
const TopoDS_Edge& anEdge = TopoDS::Edge(theRemovedEdges(i));
|
||||
Handle(Geom2d_Curve) aPC = BRep_Tool::CurveOnSurface(anEdge, theFrefFace, Param1, Param2);
|
||||
if (aPC.IsNull())
|
||||
continue;
|
||||
|
||||
GeomAbs_Shape aContOnRefFace = BRep_Tool::Continuity(anEdge, theFrefFace, theFrefFace);
|
||||
if (aContOnRefFace > aContinuity)
|
||||
aContinuity = aContOnRefFace;
|
||||
|
||||
TopoDS_Vertex aV1, aV2;
|
||||
TopExp::Vertices(anEdge, aV1, aV2);
|
||||
if ((aV1.IsSame(theCurVertex) && aV2.IsSame(theLastVertexOfSeam)) ||
|
||||
(aV1.IsSame(theLastVertexOfSeam) && aV2.IsSame(theCurVertex)))
|
||||
{
|
||||
Handle(Geom2d_Curve) aPC = BRep_Tool::CurveOnSurface(anEdge, theFrefFace, Param1, Param2);
|
||||
if (!aPC.IsNull())
|
||||
{
|
||||
aRemovedEdge = anEdge;
|
||||
break;
|
||||
}
|
||||
aRemovedEdge = anEdge;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (aRemovedEdge.IsNull())
|
||||
{
|
||||
Standard_Real CurTol = BRep_Tool::Tolerance(theCurVertex);
|
||||
Standard_Real LastTol = BRep_Tool::Tolerance(theLastVertexOfSeam);
|
||||
anU = (CurTol < LastTol)? theCurPoint.X() : theStartOfNextEdge.X();
|
||||
Uiso = RefSurf->UIso(anU);
|
||||
if (Ydir > 0)
|
||||
aCoord = (CurTol < LastTol)? theCurPoint.Coord(IndCoord) : theStartOfNextEdge.Coord(IndCoord);
|
||||
Iso = (theIsU)? RefSurf->UIso(aCoord) : RefSurf->VIso(aCoord);
|
||||
if (SeamDir > 0)
|
||||
{
|
||||
V1 = theCurVertex; V2 = theLastVertexOfSeam;
|
||||
Param1 = theCurPoint.Y(); Param2 = theStartOfNextEdge.Y();
|
||||
Param1 = theCurPoint.Coord(3-IndCoord); Param2 = theStartOfNextEdge.Coord(3-IndCoord);
|
||||
}
|
||||
else
|
||||
{
|
||||
V1 = theLastVertexOfSeam; V2 = theCurVertex;
|
||||
Param1 = theStartOfNextEdge.Y(); Param2 = theCurPoint.Y();
|
||||
Param1 = theStartOfNextEdge.Coord(3-IndCoord); Param2 = theCurPoint.Coord(3-IndCoord);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
TopExp::Vertices(aRemovedEdge, V1, V2);
|
||||
Uiso = BRep_Tool::Curve(aRemovedEdge, Param1, Param2);
|
||||
Iso = BRep_Tool::Curve(aRemovedEdge, Param1, Param2);
|
||||
}
|
||||
|
||||
TopoDS_Edge MissedSeam = BRepLib_MakeEdge(Uiso, V1, V2, Param1, Param2);
|
||||
TopoDS_Edge MissedSeam = BRepLib_MakeEdge(Iso, V1, V2, Param1, Param2);
|
||||
BRep_Builder BB;
|
||||
|
||||
gp_Vec2d Offset(theUperiod, 0.);
|
||||
//gp_Vec2d Offset(theUperiod, 0.);
|
||||
gp_Vec2d Offset;
|
||||
if (theIsU)
|
||||
Offset.SetCoord(thePeriod, 0.);
|
||||
else
|
||||
Offset.SetCoord(0., thePeriod);
|
||||
if (aRemovedEdge.IsNull())
|
||||
{
|
||||
Standard_Real Vorigin = 0.;
|
||||
Standard_Real SeamCoordOrigin = 0.;
|
||||
//Correct Param1 and Param2 if needed:
|
||||
//when Uiso-curve is periodic and Param1 and Param2 do not fit into V-range of surface,
|
||||
//when iso-curve is periodic and Param1 and Param2 do not fit into SeamCoord-range of surface,
|
||||
//(for example, V-range of sphere)
|
||||
//BRepLib_MakeEdge may shift Param1 and Param2
|
||||
Standard_Real InitialParam1 = Param1, InitialParam2 = Param2;
|
||||
Handle(Geom_Curve) MissedCurve = BRep_Tool::Curve(MissedSeam, Param1, Param2);
|
||||
@ -604,21 +623,29 @@ static void ReconstructMissedSeam(const TopTools_SequenceOfShape& theEdges,
|
||||
MissedCurve->IsPeriodic())
|
||||
{
|
||||
//Vorigin = -(MissedCurve->Period());
|
||||
Vorigin = -(Param1 - InitialParam1);
|
||||
SeamCoordOrigin = -(Param1 - InitialParam1);
|
||||
}
|
||||
/////////////////////////////////////
|
||||
Handle(Geom2d_Line) PC1 = new Geom2d_Line(gp_Pnt2d(anU, Vorigin), gp_Dir2d(0., 1.));
|
||||
//Handle(Geom2d_Line) PC1 = new Geom2d_Line(gp_Pnt2d(anU, Vorigin), gp_Dir2d(0., 1.));
|
||||
Handle(Geom2d_Line) PC1;
|
||||
if (theIsU)
|
||||
PC1 = new Geom2d_Line(gp_Pnt2d(aCoord, SeamCoordOrigin), gp_Dir2d(0., 1.));
|
||||
else
|
||||
PC1 = new Geom2d_Line(gp_Pnt2d(SeamCoordOrigin, aCoord), gp_Dir2d(1., 0.));
|
||||
|
||||
Handle(Geom2d_Curve) PC2 = Handle(Geom2d_Curve)::DownCast(PC1->Copy());
|
||||
if (Ydir > 0)
|
||||
if (theIsU && SeamDir > 0)
|
||||
Offset *= -1;
|
||||
else if (!theIsU && SeamDir < 0)
|
||||
Offset *= -1;
|
||||
PC2->Translate(Offset);
|
||||
|
||||
if (Ydir > 0)
|
||||
if (SeamDir > 0)
|
||||
BB.UpdateEdge(MissedSeam, PC1, PC2, theFrefFace, 0.);
|
||||
else
|
||||
BB.UpdateEdge(MissedSeam, PC2, PC1, theFrefFace, 0.);
|
||||
|
||||
if (Ydir < 0)
|
||||
if (SeamDir < 0)
|
||||
MissedSeam.Reverse();
|
||||
}
|
||||
else
|
||||
@ -631,19 +658,19 @@ static void ReconstructMissedSeam(const TopTools_SequenceOfShape& theEdges,
|
||||
Standard_Boolean IsSeam = (PC1 != PC2);
|
||||
if (!IsSeam) //it was not a seam
|
||||
{
|
||||
anU = theCurPoint.X();
|
||||
aCoord = theCurPoint.Coord(IndCoord);
|
||||
gp_Pnt2d PointOnRemovedEdge = PC1->Value(Param1);
|
||||
Standard_Real Uremovededge = PointOnRemovedEdge.X();
|
||||
if (Abs(anU - Uremovededge) > theUperiod/2)
|
||||
Standard_Real CoordOfRemovededge = PointOnRemovedEdge.Coord(IndCoord);
|
||||
if (Abs(aCoord - CoordOfRemovededge) > thePeriod/2)
|
||||
{
|
||||
Standard_Real Sign = (anU > Uremovededge)? 1 : -1;
|
||||
Standard_Real Sign = (aCoord > CoordOfRemovededge)? 1 : -1;
|
||||
Offset *= Sign;
|
||||
PC1 = Handle(Geom2d_Curve)::DownCast(PC2->Copy());
|
||||
PC1->Translate(Offset);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (Ydir > 0)
|
||||
if (SeamDir > 0)
|
||||
Offset *= -1;
|
||||
PC2 = Handle(Geom2d_Curve)::DownCast(PC1->Copy());
|
||||
PC2->Translate(Offset);
|
||||
@ -673,6 +700,7 @@ static void ReconstructMissedSeam(const TopTools_SequenceOfShape& theEdges,
|
||||
BAcurve2d.LastParameter() : BAcurve2d.FirstParameter();
|
||||
theNextPoint = BAcurve2d.Value(ParamOfNextPoint);
|
||||
}
|
||||
|
||||
//=======================================================================
|
||||
//function : SameSurf
|
||||
//purpose : auxilary
|
||||
@ -1109,90 +1137,6 @@ static Standard_Boolean GetNormalToSurface(const TopoDS_Face& theFace,
|
||||
return Standard_True;
|
||||
}
|
||||
|
||||
//=======================================================================
|
||||
//function : IsInSide
|
||||
//purpose :
|
||||
//=======================================================================
|
||||
static Standard_Boolean IsInSide(const gp_Pln& theRefPln,
|
||||
const Handle(Geom_Surface)& theRefSurf,
|
||||
const Handle(Geom_Surface)& theCheckedSurf, const Standard_Real theLinTol)
|
||||
{
|
||||
//
|
||||
Standard_Real uf, ul, vf, vl;
|
||||
theRefSurf->Bounds(uf, ul, vf, vl);
|
||||
|
||||
if (Precision::IsInfinite(uf) && Precision::IsInfinite(ul) &&
|
||||
Precision::IsInfinite(vf) && Precision::IsInfinite(vl))
|
||||
{
|
||||
return Standard_True;
|
||||
}
|
||||
else if (Precision::IsInfinite(uf) || Precision::IsInfinite(ul) ||
|
||||
Precision::IsInfinite(vf) || Precision::IsInfinite(vl))
|
||||
{
|
||||
//To avoid calculation of "infinite" isoline
|
||||
return Standard_False;
|
||||
}
|
||||
//
|
||||
Handle(Geom_Curve) anIsolines[4];
|
||||
anIsolines[0] = theRefSurf->VIso(vf);
|
||||
anIsolines[1] = theRefSurf->UIso(ul);
|
||||
anIsolines[2] = theRefSurf->VIso(vl)->Reversed();
|
||||
anIsolines[3] = theRefSurf->UIso(uf)->Reversed();
|
||||
|
||||
BRepBuilderAPI_MakeWire aMkWire;
|
||||
TopoDS_Edge E;
|
||||
Standard_Integer i;
|
||||
for (i = 0; i < 4; ++i)
|
||||
{
|
||||
GeomAdaptor_Curve aGAC(anIsolines[i]);
|
||||
Standard_Real length = GCPnts_AbscissaPoint::Length(aGAC);
|
||||
if (length > Precision::Confusion())
|
||||
{
|
||||
E = BRepBuilderAPI_MakeEdge(anIsolines[i]);
|
||||
aMkWire.Add(E);
|
||||
}
|
||||
}
|
||||
TopoDS_Face aRefFace;
|
||||
TopoDS_Wire aW;
|
||||
if (aMkWire.IsDone())
|
||||
{
|
||||
aW = aMkWire.Wire();
|
||||
}
|
||||
else
|
||||
{
|
||||
return Standard_False;
|
||||
}
|
||||
//
|
||||
BRepBuilderAPI_MakeFace aMkFace(theRefPln, aW);
|
||||
if (aMkFace.IsDone())
|
||||
{
|
||||
aRefFace = aMkFace.Face();
|
||||
}
|
||||
else
|
||||
{
|
||||
return Standard_False;
|
||||
}
|
||||
//
|
||||
Standard_Real up, vp;
|
||||
BRepTopAdaptor_FClass2d aFClass(aRefFace, theLinTol);
|
||||
theCheckedSurf->Bounds(uf, ul, vf, vl);
|
||||
const gp_Ax3& aPos = theRefPln.Position();
|
||||
//
|
||||
gp_Pnt2d aP2d[4] = { gp_Pnt2d(uf, vf), gp_Pnt2d(ul, vf), gp_Pnt2d(ul, vl), gp_Pnt2d(uf, vl) };
|
||||
for (i = 0; i < 4; ++i)
|
||||
{
|
||||
gp_Pnt aP = theCheckedSurf->Value(aP2d[i].X(), aP2d[i].Y());
|
||||
ElSLib::PlaneParameters(aPos, aP, up, vp);
|
||||
gp_Pnt2d aP2(up, vp);
|
||||
TopAbs_State aSt = aFClass.Perform(aP2);
|
||||
if (aSt == TopAbs_OUT || aSt == TopAbs_UNKNOWN)
|
||||
{
|
||||
return Standard_False;
|
||||
}
|
||||
}
|
||||
//
|
||||
return Standard_True;
|
||||
}
|
||||
//=======================================================================
|
||||
//function : IsSameDomain
|
||||
//purpose :
|
||||
@ -1200,7 +1144,8 @@ static Standard_Boolean IsInSide(const gp_Pln& theRefPln,
|
||||
static Standard_Boolean IsSameDomain(const TopoDS_Face& aFace,
|
||||
const TopoDS_Face& aCheckedFace,
|
||||
const Standard_Real theLinTol,
|
||||
const Standard_Real theAngTol)
|
||||
const Standard_Real theAngTol,
|
||||
ShapeUpgrade_UnifySameDomain::DataMapOfFacePlane& theFacePlaneMap)
|
||||
{
|
||||
//checking the same handles
|
||||
TopLoc_Location L1, L2;
|
||||
@ -1234,11 +1179,20 @@ static Standard_Boolean IsSameDomain(const TopoDS_Face& aFace,
|
||||
gp_Pln aPln2 = aPlanarityChecker2.Plan();
|
||||
|
||||
if (aPln1.Position().Direction().IsParallel(aPln2.Position().Direction(), theAngTol) &&
|
||||
aPln1.Distance(aPln2) < theLinTol) {
|
||||
Standard_Real uf, ul, vf, vl;
|
||||
BRepTools::UVBounds(aCheckedFace, uf, ul, vf, vl);
|
||||
S2 = new Geom_RectangularTrimmedSurface(S2, uf, ul, vf, vl);
|
||||
return IsInSide(aPln1, S1, S2, theLinTol);
|
||||
aPln1.Distance(aPln2) < theLinTol)
|
||||
{
|
||||
Handle(Geom_Plane) aPlaneOfFaces;
|
||||
if (theFacePlaneMap.IsBound(aFace))
|
||||
aPlaneOfFaces = theFacePlaneMap(aFace);
|
||||
else if (theFacePlaneMap.IsBound(aCheckedFace))
|
||||
aPlaneOfFaces = theFacePlaneMap(aCheckedFace);
|
||||
else
|
||||
aPlaneOfFaces = new Geom_Plane(aPln1);
|
||||
|
||||
theFacePlaneMap.Bind(aFace, aPlaneOfFaces);
|
||||
theFacePlaneMap.Bind(aCheckedFace, aPlaneOfFaces);
|
||||
|
||||
return Standard_True;
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -2167,6 +2121,7 @@ void ShapeUpgrade_UnifySameDomain::Initialize(const TopoDS_Shape& aShape,
|
||||
|
||||
myContext->Clear();
|
||||
myKeepShapes.Clear();
|
||||
myFacePlaneMap.Clear();
|
||||
myHistory->Clear();
|
||||
}
|
||||
|
||||
@ -2314,6 +2269,7 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
//these edges may be updated by temporary pcurves
|
||||
|
||||
Standard_Real Uperiod = (aBaseSurface->IsUPeriodic())? aBaseSurface->UPeriod() : 0.;
|
||||
Standard_Real Vperiod = (aBaseSurface->IsVPeriodic())? aBaseSurface->VPeriod() : 0.;
|
||||
|
||||
// find adjacent faces to union
|
||||
Standard_Integer i;
|
||||
@ -2373,7 +2329,7 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
}
|
||||
}
|
||||
//
|
||||
if (IsSameDomain(aFace,aCheckedFace, myLinTol, myAngTol)) {
|
||||
if (IsSameDomain(aFace,aCheckedFace, myLinTol, myAngTol, myFacePlaneMap)) {
|
||||
|
||||
if (AddOrdinaryEdges(edges, aCheckedFace, dummy, RemovedEdges)) {
|
||||
// sequence edges is modified
|
||||
@ -2388,6 +2344,12 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
}
|
||||
|
||||
if (faces.Length() > 1) {
|
||||
if (myFacePlaneMap.IsBound(faces(1)))
|
||||
{
|
||||
const Handle(Geom_Plane)& aPlane = myFacePlaneMap(faces(1));
|
||||
TopLoc_Location aLoc;
|
||||
BB.UpdateFace(RefFace, aPlane, aLoc, Precision::Confusion());
|
||||
}
|
||||
//Add correct pcurves for the reference surface to the edges of other faces
|
||||
TopoDS_Face F_RefFace = RefFace;
|
||||
F_RefFace.Orientation(TopAbs_FORWARD);
|
||||
@ -2528,6 +2490,10 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
if (faces.Length() > 1)
|
||||
{
|
||||
Standard_Real CoordTol = Precision::Confusion();
|
||||
TopTools_MapOfShape edgesMap;
|
||||
CoordTol = ComputeMinEdgeSize(edges, F_RefFace, edgesMap);
|
||||
CoordTol /= 10.;
|
||||
CoordTol = Max(CoordTol, Precision::Confusion());
|
||||
|
||||
TopTools_IndexedDataMapOfShapeListOfShape VEmap;
|
||||
for (Standard_Integer ind = 1; ind <= edges.Length(); ind++)
|
||||
@ -2537,11 +2503,6 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
//Define boundaries in 2d space of RefFace
|
||||
if (Uperiod != 0.)
|
||||
{
|
||||
TopTools_MapOfShape edgesMap;
|
||||
CoordTol = ComputeMinEdgeSize(edges, F_RefFace, edgesMap);
|
||||
CoordTol /= 10.;
|
||||
CoordTol = Max(CoordTol, Precision::Confusion());
|
||||
|
||||
//try to find a real seam edge - if it exists, do nothing
|
||||
Standard_Boolean SeamFound = Standard_False;
|
||||
for (Standard_Integer ii = 1; ii <= faces.Length(); ii++)
|
||||
@ -2643,32 +2604,41 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
|
||||
TopTools_SequenceOfShape NewFaces, NewWires;
|
||||
|
||||
if (Uperiod == 0)
|
||||
if (Uperiod == 0 || Vperiod == 0)
|
||||
{
|
||||
//Set the "period" for closed non-periodic surface
|
||||
//Set the "periods" for closed non-periodic surface
|
||||
TopLoc_Location aLoc;
|
||||
Handle(Geom_Surface) aSurf = BRep_Tool::Surface(RefFace, aLoc);
|
||||
if (aSurf->IsKind(STANDARD_TYPE(Geom_RectangularTrimmedSurface)))
|
||||
aSurf = (Handle(Geom_RectangularTrimmedSurface)::DownCast(aSurf))->BasisSurface();
|
||||
Standard_Real Ufirst, Ulast, Vfirst, Vlast;
|
||||
aSurf->Bounds(Ufirst, Ulast, Vfirst, Vlast);
|
||||
if (aSurf->IsUClosed())
|
||||
if (Uperiod == 0 && aSurf->IsUClosed())
|
||||
Uperiod = Ulast - Ufirst;
|
||||
if (Vperiod == 0 && aSurf->IsVClosed())
|
||||
Vperiod = Vlast - Vfirst;
|
||||
}
|
||||
|
||||
TopTools_MapOfShape UsedEdges;
|
||||
|
||||
Standard_Real FaceUmin = RealLast();
|
||||
Standard_Real FaceVmin = RealLast();
|
||||
for (Standard_Integer ii = 1; ii <= edges.Length(); ii++)
|
||||
{
|
||||
const TopoDS_Edge& anEdge = TopoDS::Edge(edges(ii));
|
||||
BRepAdaptor_Curve2d aBAcurve(anEdge, F_RefFace);
|
||||
gp_Pnt2d aFirstPoint = aBAcurve.Value(aBAcurve.FirstParameter());
|
||||
gp_Pnt2d aLastPoint = aBAcurve.Value(aBAcurve.LastParameter());
|
||||
|
||||
if (aFirstPoint.X() < FaceUmin)
|
||||
FaceUmin = aFirstPoint.X();
|
||||
if (aLastPoint.X() < FaceUmin)
|
||||
FaceUmin = aLastPoint.X();
|
||||
|
||||
if (aFirstPoint.Y() < FaceVmin)
|
||||
FaceVmin = aFirstPoint.Y();
|
||||
if (aLastPoint.Y() < FaceVmin)
|
||||
FaceVmin = aLastPoint.Y();
|
||||
}
|
||||
|
||||
//Building new wires from <edges>
|
||||
@ -2726,7 +2696,18 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
gp_Pnt2d StartOfNextEdge;
|
||||
TopoDS_Vertex LastVertexOfSeam;
|
||||
ReconstructMissedSeam(edges, RemovedEdges, UsedEdges, F_RefFace, CurVertex,
|
||||
CurPoint, Uperiod, FaceUmin, CoordTol,
|
||||
CurPoint, Uperiod, FaceUmin, Standard_True, CoordTol,
|
||||
NextEdge, aNewWire, NextPoint,
|
||||
StartOfNextEdge, LastVertexOfSeam, VEmap);
|
||||
}
|
||||
else if (Vperiod != 0. &&
|
||||
Abs(StartPoint.Y() - CurPoint.Y()) > Vperiod/2) //end of parametric space
|
||||
{
|
||||
//<edges> do not contain seams => we must reconstruct the seam up to <NextEdge>
|
||||
gp_Pnt2d StartOfNextEdge;
|
||||
TopoDS_Vertex LastVertexOfSeam;
|
||||
ReconstructMissedSeam(edges, RemovedEdges, UsedEdges, F_RefFace, CurVertex,
|
||||
CurPoint, Vperiod, FaceVmin, Standard_False, CoordTol,
|
||||
NextEdge, aNewWire, NextPoint,
|
||||
StartOfNextEdge, LastVertexOfSeam, VEmap);
|
||||
}
|
||||
@ -2754,7 +2735,7 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
TmpElist.Append(anEdge);
|
||||
}
|
||||
if (TmpElist.Extent() <= 1 ||
|
||||
Uperiod != 0.)
|
||||
(Uperiod != 0. || Vperiod != 0))
|
||||
TrueElist.Assign(TmpElist);
|
||||
else
|
||||
{
|
||||
@ -2797,20 +2778,25 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
Handle(Geom2d_Curve) aPCurve = BRep_Tool::CurveOnSurface(anEdge, F_RefFace, fpar, lpar);
|
||||
Standard_Real aParam = (anEdge.Orientation() == TopAbs_FORWARD)? fpar : lpar;
|
||||
gp_Pnt2d aPoint = aPCurve->Value(aParam);
|
||||
Standard_Real Diff = Abs(aPoint.X() - CurPoint.X());
|
||||
Standard_Real DiffU = Abs(aPoint.X() - CurPoint.X());
|
||||
Standard_Real DiffV = Abs(aPoint.Y() - CurPoint.Y());
|
||||
if (Uperiod != 0. &&
|
||||
Diff > CoordTol &&
|
||||
Abs(Diff - Uperiod) > CoordTol) //may be is is a deg.vertex
|
||||
DiffU > CoordTol &&
|
||||
Abs(DiffU - Uperiod) > CoordTol) //may be is is a deg.vertex
|
||||
continue;
|
||||
if (Vperiod != 0. &&
|
||||
DiffV > CoordTol &&
|
||||
Abs(DiffV - Vperiod) > CoordTol) //may be is is a deg.vertex
|
||||
continue;
|
||||
|
||||
//Check: may be <CurPoint> and <aPoint> are on Period from each other
|
||||
if (Uperiod != 0. && Diff > Uperiod/2) //end of parametric space
|
||||
if (Uperiod != 0. && DiffU > Uperiod/2) //end of parametric space
|
||||
{
|
||||
//<edges> do not contain seams => we must reconstruct the seam up to <NextEdge>
|
||||
gp_Pnt2d StartOfNextEdge;
|
||||
TopoDS_Vertex LastVertexOfSeam;
|
||||
ReconstructMissedSeam(edges, RemovedEdges, UsedEdges, F_RefFace, CurVertex,
|
||||
CurPoint, Uperiod, FaceUmin, CoordTol,
|
||||
CurPoint, Uperiod, FaceUmin, Standard_True, CoordTol,
|
||||
NextEdge, aNewWire, NextPoint,
|
||||
StartOfNextEdge, LastVertexOfSeam, VEmap);
|
||||
|
||||
@ -2821,6 +2807,23 @@ void ShapeUpgrade_UnifySameDomain::IntUnifyFaces(const TopoDS_Shape& theInpShape
|
||||
|
||||
break;
|
||||
} //if (Uperiod != 0. && Abs(aPoint.X() - CurPoint.X()) > Uperiod/2)
|
||||
else if (Vperiod != 0. && DiffV > Vperiod/2) //end of parametric space
|
||||
{
|
||||
//<edges> do not contain seams => we must reconstruct the seam up to <NextEdge>
|
||||
gp_Pnt2d StartOfNextEdge;
|
||||
TopoDS_Vertex LastVertexOfSeam;
|
||||
ReconstructMissedSeam(edges, RemovedEdges, UsedEdges, F_RefFace, CurVertex,
|
||||
CurPoint, Vperiod, FaceVmin, Standard_False, CoordTol,
|
||||
NextEdge, aNewWire, NextPoint,
|
||||
StartOfNextEdge, LastVertexOfSeam, VEmap);
|
||||
|
||||
//Check: may be it is the end
|
||||
if (LastVertexOfSeam.IsSame(StartVertex) &&
|
||||
Abs(StartPoint.Y() - StartOfNextEdge.Y()) < Vperiod/2)
|
||||
EndOfWire = Standard_True;
|
||||
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
NextEdge = anEdge;
|
||||
|
@ -27,6 +27,8 @@
|
||||
#include <TopTools_DataMapOfShapeShape.hxx>
|
||||
#include <TopTools_IndexedDataMapOfShapeListOfShape.hxx>
|
||||
#include <TopTools_MapOfShape.hxx>
|
||||
#include <Geom_Plane.hxx>
|
||||
#include <Precision.hxx>
|
||||
class ShapeBuild_ReShape;
|
||||
class TopoDS_Shape;
|
||||
|
||||
@ -66,6 +68,9 @@ class ShapeUpgrade_UnifySameDomain : public Standard_Transient
|
||||
{
|
||||
|
||||
public:
|
||||
|
||||
typedef NCollection_DataMap<TopoDS_Shape, Handle(Geom_Plane), TopTools_ShapeMapHasher> DataMapOfFacePlane;
|
||||
|
||||
//! Empty constructor
|
||||
Standard_EXPORT ShapeUpgrade_UnifySameDomain();
|
||||
|
||||
@ -177,6 +182,7 @@ private:
|
||||
TopoDS_Shape myShape;
|
||||
Handle(ShapeBuild_ReShape) myContext;
|
||||
TopTools_MapOfShape myKeepShapes;
|
||||
DataMapOfFacePlane myFacePlaneMap;
|
||||
|
||||
Handle(BRepTools_History) myHistory; //!< The history.
|
||||
};
|
||||
|
@ -7,7 +7,7 @@ restore [locate_data_file bug30185.brep] s
|
||||
|
||||
unifysamedom r s +b
|
||||
checkshape r
|
||||
checknbshapes r -vertex 54 -edge 78 -face 24
|
||||
checknbshapes r -vertex 50 -edge 71 -face 21
|
||||
checkmaxtol r -min_tol 2.49130e-03
|
||||
checkmaxtol s -min_tol 2.49130e-03
|
||||
|
||||
|
17
tests/bugs/heal/bug31736_1
Normal file
17
tests/bugs/heal/bug31736_1
Normal file
@ -0,0 +1,17 @@
|
||||
puts "=================================================================================="
|
||||
puts "OCC31736: ShapeUpgrade_UnifySameDomain algorithm does not unify all possible faces"
|
||||
puts "=================================================================================="
|
||||
puts ""
|
||||
|
||||
restore [locate_data_file bug31736_1.brep] a
|
||||
unifysamedom result a
|
||||
|
||||
checkshape result
|
||||
|
||||
checknbshapes result -vertex 18 -edge 26 -wire 9 -face 9 -shell 1 -solid 1
|
||||
|
||||
set tolres [checkmaxtol result]
|
||||
|
||||
if { ${tolres} > 0.5} {
|
||||
puts "Error: bad tolerance of result"
|
||||
}
|
17
tests/bugs/heal/bug31736_2
Normal file
17
tests/bugs/heal/bug31736_2
Normal file
@ -0,0 +1,17 @@
|
||||
puts "=================================================================================="
|
||||
puts "OCC31736: ShapeUpgrade_UnifySameDomain algorithm does not unify all possible faces"
|
||||
puts "=================================================================================="
|
||||
puts ""
|
||||
|
||||
restore [locate_data_file bug31736_2.brep] a
|
||||
unifysamedom result a
|
||||
|
||||
checkshape result
|
||||
|
||||
checknbshapes result -vertex 17 -edge 24 -wire 9 -face 9 -shell 1 -solid 1
|
||||
|
||||
set tolres [checkmaxtol result]
|
||||
|
||||
if { ${tolres} > 0.5} {
|
||||
puts "Error: bad tolerance of result"
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user