1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00
occt/src/ShapeFix/ShapeFix_ComposeShell.cxx
abv 42cf5bc1ca 0024002: Overall code and build procedure refactoring -- automatic
Automatic upgrade of OCCT code by command "occt_upgrade . -nocdl":
- WOK-generated header files from inc and sources from drv are moved to src
- CDL files removed
- All packages are converted to nocdlpack
2015-07-12 07:42:38 +03:00

2799 lines
102 KiB
C++

// Created on: 1999-04-27
// Created by: Andrey BETENEV
// Copyright (c) 1999-1999 Matra Datavision
// Copyright (c) 1999-2014 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and/or modify it under
// the terms of the GNU Lesser General Public License version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
// pdn 01.06.99 S4205: handling not-SameRange edges
// abv 22.07.99 implementing patch indices
// svv 10.01.00 porting on DEC
#include <Bnd_Box2d.hxx>
#include <BndLib_Add2dCurve.hxx>
#include <BRep_Builder.hxx>
#include <BRep_Tool.hxx>
#include <BRepTools.hxx>
#include <BRepTopAdaptor_FClass2d.hxx>
#include <Extrema_ExtPC2d.hxx>
#include <Geom2d_Curve.hxx>
#include <Geom2d_Line.hxx>
#include <Geom2dAdaptor_Curve.hxx>
#include <Geom2dInt_GInter.hxx>
#include <Geom_Curve.hxx>
#include <Geom_Surface.hxx>
#include <GeomAdaptor_Surface.hxx>
#include <gp_Dir2d.hxx>
#include <gp_Lin2d.hxx>
#include <gp_Pnt.hxx>
#include <gp_Pnt2d.hxx>
#include <IntRes2d_Domain.hxx>
#include <IntRes2d_IntersectionPoint.hxx>
#include <IntRes2d_IntersectionSegment.hxx>
#include <Precision.hxx>
#include <ShapeAnalysis.hxx>
#include <ShapeAnalysis_Curve.hxx>
#include <ShapeAnalysis_Edge.hxx>
#include <ShapeAnalysis_Surface.hxx>
#include <ShapeAnalysis_TransferParameters.hxx>
#include <ShapeAnalysis_TransferParametersProj.hxx>
#include <ShapeAnalysis_WireOrder.hxx>
#include <ShapeBuild_Edge.hxx>
#include <ShapeBuild_ReShape.hxx>
#include <ShapeBuild_Vertex.hxx>
#include <ShapeExtend.hxx>
#include <ShapeExtend_CompositeSurface.hxx>
#include <ShapeExtend_WireData.hxx>
#include <ShapeFix_ComposeShell.hxx>
#include <ShapeFix_Edge.hxx>
#include <ShapeFix_Face.hxx>
#include <ShapeFix_Wire.hxx>
#include <ShapeFix_WireSegment.hxx>
#include <Standard_Type.hxx>
#include <TColgp_SequenceOfPnt2d.hxx>
#include <TColStd_Array1OfBoolean.hxx>
#include <TColStd_Array1OfInteger.hxx>
#include <TColStd_Array1OfReal.hxx>
#include <TColStd_HArray1OfReal.hxx>
#include <TColStd_SequenceOfReal.hxx>
#include <TopLoc_Location.hxx>
#include <TopoDS.hxx>
#include <TopoDS_Edge.hxx>
#include <TopoDS_Face.hxx>
#include <TopoDS_Iterator.hxx>
#include <TopoDS_Shape.hxx>
#include <TopoDS_Shell.hxx>
#include <TopoDS_Vertex.hxx>
#include <TopoDS_Wire.hxx>
#include <TopTools_DataMapOfShapeListOfShape.hxx>
#include <TopTools_MapOfShape.hxx>
//=======================================================================
//function : ShapeFix_ComposeShell
//purpose :
//=======================================================================
ShapeFix_ComposeShell::ShapeFix_ComposeShell () :
myStatus(0), myClosedMode(Standard_False)
{
myTransferParamTool = new ShapeAnalysis_TransferParametersProj;
}
//=======================================================================
//function : Init
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::Init (const Handle(ShapeExtend_CompositeSurface) &Grid,
const TopLoc_Location& L,
const TopoDS_Face &Face,
const Standard_Real Prec)
{
myGrid = Grid;
myUClosed = myGrid->IsUClosed();
myVClosed = myGrid->IsVClosed();
myUPeriod = myGrid->UJointValue(myGrid->NbUPatches()+1) - myGrid->UJointValue(1);
myVPeriod = myGrid->VJointValue(myGrid->NbVPatches()+1) - myGrid->VJointValue(1);
// DTK-CKY 100531 : protection against very thin face
// Test "isclosed" should be filtered on the overall (non trimmed) surface, must be closed
Handle(Geom_Surface) theSurface = BRep_Tool::Surface(Face,myLoc);
Standard_Real U0,U1,V0,V1;
theSurface->Bounds(U0,U1,V0,V1);
if (myUClosed) {
gp_Pnt P0 = theSurface->Value(U0,(V0+V1)/2.);
gp_Pnt P1 = theSurface->Value(U1,(V0+V1)/2.);
if (P0.Distance(P1) > Precision::Confusion()*10)
myUClosed = Standard_False;
}
if (myVClosed) {
gp_Pnt P0 = theSurface->Value((U0+U1)/2.,V0);
gp_Pnt P1 = theSurface->Value((U0+U1)/2.,V1);
if (P0.Distance(P1) > Precision::Confusion()*10)
myVClosed = Standard_False;
}
// DTK-CKY 100531 end
myLoc = L;
//smh#8
TopoDS_Shape tmpF = Face.Oriented ( TopAbs_FORWARD );
myFace = TopoDS::Face ( tmpF ); // for correct dealing with seams
myOrient = Face.Orientation();
SetPrecision(Prec);
myStatus = ShapeExtend::EncodeStatus ( ShapeExtend_OK );
// Compute resolution (checking in 2d is necessary for splitting
// degenerated edges and avoiding NotClosed)
myUResolution = myVResolution = RealLast();
for ( Standard_Integer i=1; i <= myGrid->NbUPatches(); i++ ) {
Standard_Real uRange = myGrid->UJointValue(i+1)-myGrid->UJointValue(i);
for ( Standard_Integer j=1; j <= myGrid->NbVPatches(); j++ ) {
Standard_Real vRange = myGrid->VJointValue(j+1)-myGrid->VJointValue(j);
Standard_Real u1,u2,v1,v2;
myGrid->Patch(i,j)->Bounds(u1,u2,v1,v2);
GeomAdaptor_Surface GAS ( myGrid->Patch(i,j) );
Standard_Real ures = GAS.UResolution ( 1. )*uRange/(u2-u1);
Standard_Real vres = GAS.VResolution ( 1. )*vRange/(v2-v1);
if ( ures >0. && myUResolution > ures ) myUResolution = ures;
if ( vres >0. && myVResolution > vres ) myVResolution = vres;
}
}
if ( myUResolution == RealLast() ) myUResolution = ::Precision::Parametric ( 1. );
if ( myVResolution == RealLast() ) myVResolution = ::Precision::Parametric ( 1. );
}
//=======================================================================
//function : Perform
//purpose :
//=======================================================================
Standard_Boolean ShapeFix_ComposeShell::Perform ()
{
myStatus = ShapeExtend::EncodeStatus ( ShapeExtend_OK );
myInvertEdgeStatus = Standard_False;
ShapeFix_SequenceOfWireSegment seqw; // working data: wire segments
// Init seqw by initial set of wires (with corresponding orientation)
LoadWires ( seqw );
if(seqw.Length() == 0) {
myStatus = ShapeExtend::EncodeStatus ( ShapeExtend_FAIL6 );
return Standard_False;
}
// Split edges in the wires by grid and add internal segments of grid (parts of cutting lines)
SplitByGrid ( seqw );
// Split all the wires into segments by common vertices (intersections)
BreakWires ( seqw );
// Then, collect resulting wires
ShapeFix_SequenceOfWireSegment wires; // resulting wires
CollectWires ( wires, seqw );
// And construct resulting faces
TopTools_SequenceOfShape faces;
DispatchWires ( faces, wires );
// Finally, construct resulting shell
if ( faces.Length() !=1 ) {
TopoDS_Shell S;
BRep_Builder B;
B.MakeShell ( S );
for ( Standard_Integer i=1; i <= faces.Length(); i++ )
B.Add ( S, faces(i) );
myResult = S;
}
else myResult = faces(1);
myResult.Orientation ( myOrient );
myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_DONE1 );
return Standard_True;
}
//=======================================================================
//function : SplitEdges
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::SplitEdges ()
{
myStatus = ShapeExtend::EncodeStatus ( ShapeExtend_OK );
ShapeFix_SequenceOfWireSegment seqw; // working data: wire segments
// Init seqw by initial set of wires (with corresponding orientation)
LoadWires ( seqw );
// Split edges in the wires by grid and add internal segments of grid (parts of cutting lines)
SplitByGrid ( seqw );
}
//=======================================================================
//function : Result
//purpose :
//=======================================================================
const TopoDS_Shape& ShapeFix_ComposeShell::Result () const
{
return myResult;
}
//=======================================================================
//function : Status
//purpose :
//=======================================================================
Standard_Boolean ShapeFix_ComposeShell::Status (const ShapeExtend_Status status) const
{
return ShapeExtend::DecodeStatus ( myStatus, status );
}
//=======================================================================
// PRIVATE (working) METHODS
//=======================================================================
#define TOLINT 1.e-10 // precision for intersection
// Local definitions: characteristics of intersection point
#define IOR_UNDEF 0 // undefined side
#define IOR_LEFT 1 // to left side of cutting line
#define IOR_RIGHT 2 // to right side of cutting line
#define IOR_BOTH 3 // crossing
#define IOR_POS 4 // in case of cycle on full period, whether first point is right
#define ITP_INTER 8 // crossing
#define ITP_BEGSEG 16 // start of tangential segment
#define ITP_ENDSEG 32 // stop of tangential segment
#define ITP_TANG 64 // tangential point
//=======================================================================
//function : PointLineDeviation
//purpose : auxilary
//=======================================================================
// Return (signed) deviation of point from line
static Standard_Real PointLineDeviation (const gp_Pnt2d &p, const gp_Lin2d &line)
{
gp_Dir2d dir = line.Direction();
gp_Dir2d n ( -dir.Y(), dir.X() );
return n.XY() * ( p.XY() - line.Location().XY() );
}
//=======================================================================
//function : PointLinePosition
//purpose : auxilary
//=======================================================================
// Define position of point relative to line
static Standard_Integer PointLinePosition (const gp_Pnt2d &p, const gp_Lin2d &line,
Standard_Real &dev)
{
dev = PointLineDeviation ( p, line );
return ( dev > TOLINT ? IOR_LEFT : ( dev < -TOLINT ? IOR_RIGHT : IOR_UNDEF ) );
}
//=======================================================================
//function : PointLinePosition
//purpose : auxilary
//=======================================================================
// Define position of point relative to line
static Standard_Integer PointLinePosition (const gp_Pnt2d &p, const gp_Lin2d &line)
{
Standard_Real dev;
return PointLinePosition ( p, line, dev );
}
//=======================================================================
//function : ParamPointsOnLine
//purpose : auxilary
//=======================================================================
// Compute parameter of point on line
static inline Standard_Real ParamPointOnLine (const gp_Pnt2d &p, const gp_Lin2d &line)
{
return line.Direction().XY() * ( p.XY() - line.Location().XY() );
}
//=======================================================================
//function : ParamPointsOnLine
//purpose : auxilary
//=======================================================================
// Compute parameter of two points on line (as intersection of segment)
static Standard_Real ParamPointsOnLine (const gp_Pnt2d &p1, const gp_Pnt2d &p2,
const gp_Lin2d &line)
{
Standard_Real dist1 = PointLineDeviation ( p1, line );
Standard_Real dist2 = PointLineDeviation ( p2, line );
// in most cases, one of points is on line
if ( Abs ( dist1 ) < ::Precision::PConfusion() ) {
if ( Abs ( dist2 ) < ::Precision::PConfusion() )
return 0.5 * ( ParamPointOnLine ( p1, line ) + ParamPointOnLine ( p2, line ) );
return ParamPointOnLine ( p1, line );
}
if ( Abs ( dist2 ) < ::Precision::PConfusion() )
return ParamPointOnLine ( p2, line );
// just protection
if ( dist2 * dist1 >0 )
return 0.5 * ( ParamPointOnLine ( p1, line ) + ParamPointOnLine ( p2, line ) );
// else compute intersection
return ( ParamPointOnLine ( p1, line ) * dist2 -
ParamPointOnLine ( p2, line ) * dist1 ) / ( dist2 - dist1 );
}
//=======================================================================
//function : ProjectPointOnLine
//purpose : auxilary
//=======================================================================
// Compute projection of point on line
static inline gp_Pnt2d ProjectPointOnLine (const gp_Pnt2d &p, const gp_Lin2d &line)
{
return line.Location().XY() + line.Direction().XY() * ParamPointOnLine ( p, line );
}
//=======================================================================
//function : ApplyContext
//purpose : auxilary
//=======================================================================
// Apply context to one edge in the wire and put result into this wire
static Standard_Integer ApplyContext (ShapeFix_WireSegment &wire,
const Standard_Integer iedge,
const Handle(ShapeBuild_ReShape) &context)
{
TopoDS_Edge edge = wire.Edge ( iedge );
TopoDS_Shape res = context->Apply ( edge );
if ( res.IsSame ( edge ) ) return 1;
if ( res.ShapeType() == TopAbs_EDGE ) {
wire.SetEdge ( iedge, TopoDS::Edge ( res ) );
return 1;
}
Standard_Integer index = iedge;
Handle(ShapeExtend_WireData) segw = new ShapeExtend_WireData;
segw->ManifoldMode() = Standard_False;
for ( TopoDS_Iterator it(res); it.More(); it.Next() ) {
TopoDS_Edge E = TopoDS::Edge ( it.Value() );
if ( ! E.IsNull() ) segw->Add ( E );
#ifdef OCCT_DEBUG
else cout << "Error: ShapeFix_ComposeShell, ApplyContext: wrong mapping of edge" << endl;
#endif
}
// add edges into the wire in correct order
if ( segw->NbEdges() >0 ) {
Standard_Integer ind, iumin, iumax, ivmin, ivmax;
wire.GetPatchIndex ( iedge, iumin, iumax, ivmin, ivmax );
Standard_Integer nbEdges = segw->NbEdges();
for ( Standard_Integer i=1; i <= nbEdges; i++, index++ ) {
ind = ( edge.Orientation() == TopAbs_FORWARD || edge.Orientation() == TopAbs_INTERNAL ? i : segw->NbEdges()-i+1 );
TopoDS_Edge aE = segw->Edge ( ind );
if ( i==1 ) wire.SetEdge ( index, aE );
else wire.AddEdge ( index, aE, iumin, iumax, ivmin, ivmax );
}
}
#ifdef OCCT_DEBUG
else cout << "Warning: ShapeFix_ComposeShell, ApplyContext: edge is to remove - not implemented" << endl;
#endif
return index - iedge;
}
//=======================================================================
//function : IsCoincided
//purpose : auxilary
//=======================================================================
// check points coincidence
static inline Standard_Integer IsCoincided (const gp_Pnt2d &p1, const gp_Pnt2d &p2,
const Standard_Real UResolution,
const Standard_Real VResolution,
const Standard_Real tol)
{
//pdn Maximal accuracy is working precision of intersector.
Standard_Real UTolerance = UResolution * tol;
Standard_Real VTolerance = VResolution * tol;
return Abs ( p1.X() - p2.X() ) <= Max(TOLINT,UTolerance) &&
Abs ( p1.Y() - p2.Y() ) <= Max(TOLINT,VTolerance);
}
//=======================================================================
//function : GetPatchIndex
//purpose : auxilary
//=======================================================================
// computes index for the patch by given parameter Param
static Standard_Integer GetPatchIndex (const Standard_Real Param,
const Handle(TColStd_HArray1OfReal) &Params,
const Standard_Boolean isClosed)
{
Standard_Integer NP = Params->Upper();
Standard_Real period = Params->Value(NP) - Params->Value(1);
Standard_Real shift = 0;
if ( isClosed )
shift = ShapeAnalysis::AdjustToPeriod ( Param, Params->Value(1), Params->Value(NP) );
Standard_Real p = Param + shift;
// locate patch: the same algo as in SE_CS::LocateParameter()
Standard_Integer i; // svv #1
for ( i = 2; i < NP; i++ ) {
// Standard_Real par = Params->Value(i);
if ( p < Params->Value(i) ) break;
}
i--;
Standard_Real ish = shift / period;
Standard_Integer ishift = (Standard_Integer)( ish <0 ? ish - 0.5 : ish + 0.5 );
return i - ishift * ( NP - 1 );
}
//=======================================================================
//function : LoadWires
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::LoadWires (ShapeFix_SequenceOfWireSegment &seqw) const
{
seqw.Clear();
// Init seqw by initial set of wires (with corresponding orientation)
for ( TopoDS_Iterator iw(myFace,Standard_False); iw.More(); iw.Next() )
{
TopoDS_Shape tmpW = Context()->Apply ( iw.Value() ) ;
if(tmpW.ShapeType() != TopAbs_WIRE)
{
if(tmpW.ShapeType() == TopAbs_VERTEX)
{
ShapeFix_WireSegment seg; //(( isOuter ? TopAbs_REVERSED : TopAbs_FORWARD ) );
seg.SetVertex(TopoDS::Vertex(tmpW));
seg.Orientation(tmpW.Orientation());
seqw.Append ( seg );
}
continue;
}
TopoDS_Wire wire = TopoDS::Wire ( tmpW );
Standard_Boolean isNonManifold = ( wire.Orientation() != TopAbs_REVERSED &&
wire.Orientation() != TopAbs_FORWARD );
// protect against INTERNAL/EXTERNAL wires
// if ( wire.Orientation() != TopAbs_REVERSED &&
// wire.Orientation() != TopAbs_FORWARD ) continue;
// determine orientation of the wire
// TopoDS_Face face = TopoDS::Face ( myFace.EmptyCopied() );
// B.Add ( face, wire );
// Standard_Boolean isOuter = ShapeAnalysis::IsOuterBound ( face );
if(isNonManifold)
{
Handle(ShapeExtend_WireData) sbwd = new ShapeExtend_WireData ( wire ,Standard_True,Standard_False);
//pdn protection againts of wires w/o edges
Standard_Integer nbEdges = sbwd->NbEdges();
if(nbEdges)
{
//wire segments for non-manifold topology should have INTERNAL orientation
ShapeFix_WireSegment seg ( sbwd, TopAbs_INTERNAL);
seqw.Append ( seg );
}
}
else
{
//splitting wires containing manifold and non-manifold parts on a separate
//wire segment
Handle(ShapeExtend_WireData) sbwdM = new ShapeExtend_WireData();
Handle(ShapeExtend_WireData) sbwdNM = new ShapeExtend_WireData();
sbwdNM->ManifoldMode() = Standard_False;
TopoDS_Iterator aIt(wire);
for( ; aIt.More(); aIt.Next())
{
TopoDS_Edge E = TopoDS::Edge ( aIt.Value() );
if(E.Orientation() == TopAbs_FORWARD || E.Orientation() == TopAbs_REVERSED)
sbwdM->Add(E);
else
sbwdNM->Add(E);
}
Standard_Integer nbMEdges = sbwdM->NbEdges();
Standard_Integer nbNMEdges = sbwdNM->NbEdges();
if(nbNMEdges)
{
ShapeFix_WireSegment seg ( sbwdNM, TopAbs_INTERNAL); //(( isOuter ? TopAbs_REVERSED : TopAbs_FORWARD ) );
seqw.Append ( seg );
}
if(nbMEdges) {
// Orientation is set so as to allow the segment to be traversed in only one direction
// skl 01.04.2002
Handle(ShapeFix_Wire) sfw = new ShapeFix_Wire;
sfw->Load ( sbwdM );
Standard_Integer stat=0;
Handle(Geom_Surface) gs = BRep_Tool::Surface(myFace);
if( gs->IsUPeriodic() && gs->IsVPeriodic() )
{
// For torus-like shapes, first reorder in 2d since reorder is indifferent in 3d
ShapeAnalysis_WireOrder sawo(Standard_False, 0);
ShapeAnalysis_Edge sae;
for(Standard_Integer i = 1; i <= nbMEdges; i++) {
Standard_Real f,l;
Handle(Geom2d_Curve) c2d;
//smh#8
TopoDS_Shape tmpF = myFace.Oriented(TopAbs_FORWARD);
if(!sae.PCurve(sbwdM->Edge(i),TopoDS::Face(tmpF),c2d,f,l))
continue;
sawo.Add(c2d->Value(f).XY(),c2d->Value(l).XY());
}
sawo.Perform();
stat = (sawo.Status() < 0 ? -1 : 1);
sfw->FixReorder(sawo);
}
sfw->FixReorder();
if (sfw->StatusReorder(ShapeExtend_DONE3))
stat=-1;
if(stat < 0)
{
BRep_Builder B;
TopoDS_Shape dummy = myFace.EmptyCopied();
TopoDS_Face face = TopoDS::Face ( dummy );
B.Add ( face, wire );
Standard_Boolean isOuter = ShapeAnalysis::IsOuterBound ( face );
TopoDS_Wire w = sbwdM->Wire();
dummy = myFace.EmptyCopied();
face = TopoDS::Face ( dummy );
B.Add ( face, w );
Standard_Boolean isOuterAfter = ShapeAnalysis::IsOuterBound ( face );
if(isOuter!=isOuterAfter)
sbwdM->Reverse(face);
}
ShapeFix_WireSegment seg ( sbwdM, TopAbs_REVERSED ); //(( isOuter ? TopAbs_REVERSED : TopAbs_FORWARD ) );
seqw.Append ( seg );
}
}
}
}
//=======================================================================
//function : ComputeCode
//purpose : compute code for wire segment between two intersections (by deviation)
//=======================================================================
Standard_Integer ShapeFix_ComposeShell::ComputeCode (const Handle(ShapeExtend_WireData) &wire,
const gp_Lin2d &line,
const Standard_Integer begInd,
const Standard_Integer endInd,
const Standard_Real begPar,
const Standard_Real endPar,
const Standard_Boolean isInternal)
{
Standard_Integer code = IOR_UNDEF;
ShapeAnalysis_Edge sae;
const Standard_Integer NPOINTS = 5; // number of points for measuring deviation
// track special closed case: segment starts at end of edge and ends at its beginning
Standard_Integer special = ( begInd == endInd &&
( wire->Edge(begInd).Orientation() == TopAbs_FORWARD ||
wire->Edge(begInd).Orientation() == TopAbs_INTERNAL) ==
( begPar > endPar ) ? 1 : 0);
if ( ! special && begInd == endInd && begPar == endPar &&
(myClosedMode || isInternal))
special = 1;
// for tracking cases in closed mode
Standard_Boolean begin=Standard_True;
Standard_Real shift=0;
gp_Pnt2d p2d0;
// check if segment is tangency
// Segment is considered as tangency if deviation of pcurve from line
// (in 2d) measured by NPOINTS points is less than tolerance of edge
// (recomputed to 2d using Resolution).
Standard_Integer nb = wire->NbEdges();
Standard_Integer i; // svv #1
for ( i=begInd; ; i++ ) {
if ( i > nb ) i = 1;
TopoDS_Edge edge = wire->Edge ( i );;
Handle(Geom2d_Curve) c2d;
Standard_Real f, l;
if ( ! sae.PCurve ( edge, myFace, c2d, f, l, Standard_False ) ) {
myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL3 );
continue;
}
Standard_Real tol = LimitTolerance(BRep_Tool::Tolerance ( edge ));
Standard_Boolean isreversed = ( edge.Orientation() == TopAbs_REVERSED );
Standard_Real par1 = ( i == begInd && special >=0 ? begPar : ( isreversed ? l : f ) );
Standard_Real par2 = ( i == endInd && special <=0 ? endPar : ( isreversed ? f : l ) );
Standard_Real dpar = ( par2 - par1 ) / ( NPOINTS - 1 );
Standard_Integer np = ( Abs ( dpar ) < ::Precision::PConfusion() ? 1 : NPOINTS );
Standard_Integer j; // svv #1
for ( j=0; j < np; j++ ) {
Standard_Real par = par1 + dpar * j;
gp_Pnt2d p2d = c2d->Value ( par );
if ( myClosedMode ) {
if ( myUClosed && Abs ( line.Direction().X() ) < ::Precision::PConfusion() ) {
if ( begin ) shift = ShapeAnalysis::AdjustByPeriod ( p2d.X(), line.Location().X(), myUPeriod );
else if ( ! j ) shift = ShapeAnalysis::AdjustByPeriod ( p2d.X()-p2d0.X(), 0., myUPeriod );
p2d.SetX ( p2d.X() + shift );
}
if ( myVClosed && Abs ( line.Direction().Y() ) < ::Precision::PConfusion() ) {
if ( begin ) shift = ShapeAnalysis::AdjustByPeriod ( p2d.Y(), line.Location().Y(), myVPeriod );
else if ( ! j ) shift = ShapeAnalysis::AdjustByPeriod ( p2d.Y()-p2d0.Y(), 0., myVPeriod );
p2d.SetY ( p2d.Y() + shift );
}
begin = Standard_False;
}
p2d0 = p2d;
Standard_Integer pos = PointLinePosition ( p2d, line );
if ( pos == IOR_UNDEF ) continue;
// analyse the deviation
gp_Pnt2d p2dl = ProjectPointOnLine ( p2d, line );
if(!IsCoincided ( p2d, p2dl, myUResolution, myVResolution, tol )) {
if(!myClosedMode) { code = pos; break; }
else {
code |= pos;
}
}
}
if ( j < np ) { i = 0; break; } // not tangency
if ( i == endInd ) {
if ( special <=0 ) break;
else special = -1;
}
}
if ( myClosedMode ) {
if ( code != IOR_UNDEF && ! begin ) {
// in closed mode, if segment is of 2*pi length, it is BOTH
Standard_Real dev = PointLineDeviation ( p2d0, line );
if ( myUClosed && Abs ( line.Direction().X() ) < ::Precision::PConfusion() ) {
if ( Abs ( Abs ( dev ) - myUPeriod ) < 0.1 * myUPeriod ) {
code = IOR_BOTH;
if ( dev >0 ) code |= IOR_POS;
}
else if(code==IOR_BOTH)
code=IOR_UNDEF;
}
if ( myVClosed && Abs ( line.Direction().Y() ) < ::Precision::PConfusion() ) {
if ( Abs ( Abs ( dev ) - myVPeriod ) < 0.1 * myVPeriod ) {
code = IOR_BOTH;
if ( dev >0 ) code |= IOR_POS;
}
else if(code==IOR_BOTH)
code=IOR_UNDEF;
}
}
return code;
}
if ( i ) code = IOR_UNDEF; // tangency
else if ( code == IOR_BOTH ) { // parity error in intersector
code = IOR_LEFT;
myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL2 );
#ifdef OCCT_DEBUG
cout << "Warning: ShapeFix_ComposeShell::ComputeCode: lost intersection point" << endl;
#endif
}
return code;
}
//=======================================================================
//function : DistributeSplitPoints
//purpose : auxilary
//=======================================================================
// After applying context to (seam) edge, distribute its indices on new edges,
// according to their parameters on that edge
static void DistributeSplitPoints (const Handle(ShapeExtend_WireData) &sbwd,
const TopoDS_Face myFace,
const Standard_Integer index,
const Standard_Integer nsplit,
TColStd_SequenceOfInteger& indexes,
const TColStd_SequenceOfReal& values)
{
Standard_Boolean isreversed = ( nsplit >0 && sbwd->Edge(index).Orientation() == TopAbs_REVERSED );
TColStd_Array1OfReal params(0,nsplit);
Standard_Integer i; // svv #1
for ( i=0; i < nsplit; i++ ) {
Standard_Real f, l;
BRep_Tool::Range ( sbwd->Edge(index+i), myFace, f, l );
params.SetValue ( i, ( isreversed ? l : f ) );
}
for ( i=1; i <= indexes.Length() && indexes(i) < index; i++ );
for ( Standard_Integer shift = 1; i <= indexes.Length() && indexes(i) == index; i++ ) {
while ( shift < nsplit && isreversed != (Standard_Boolean) ( values(i) > params(shift) ) ) shift++;
indexes.SetValue ( i, index + shift - 1 );
}
for ( ; i <= indexes.Length(); i++ )
indexes.SetValue ( i, indexes(i) + nsplit - 1 );
}
//=======================================================================
//function : CheckByCurve3d
//purpose : auxilary
//=======================================================================
static Standard_Integer CheckByCurve3d (const gp_Pnt &pos,
const Handle(Geom_Curve) &c3d,
const Standard_Real param,
const gp_Trsf &T,
const Standard_Real tol)
{
if ( c3d.IsNull() ) return Standard_True;
gp_Pnt p = c3d->Value(param);
if ( T.Form() != gp_Identity ) p.Transform ( T );
return pos.SquareDistance ( p ) <= tol * tol;
}
//=======================================================================
//function : DefinePatch
//purpose : auxilary
//=======================================================================
static void DefinePatch (ShapeFix_WireSegment &wire, const Standard_Integer code,
const Standard_Boolean isCutByU, const Standard_Integer cutIndex,
const Standard_Integer number = -1)
{
Standard_Integer nb = (number > 0 ? number : wire.NbEdges());
if ( isCutByU ) {
if ( ! ( code & IOR_LEFT ) ) wire.DefineIUMin ( nb, cutIndex );
if ( ! ( code & IOR_RIGHT ) ) wire.DefineIUMax ( nb, cutIndex );
}
else {
if ( ! ( code & IOR_RIGHT ) ) wire.DefineIVMin ( nb, cutIndex );
if ( ! ( code & IOR_LEFT ) ) wire.DefineIVMax ( nb, cutIndex );
}
}
//=======================================================================
//function : DefinePatchForWire
//purpose : auxilary
//=======================================================================
static void DefinePatchForWire(ShapeFix_WireSegment &wire, const Standard_Integer code,
const Standard_Boolean isCutByU, const Standard_Integer cutIndex)
{
for(Standard_Integer i = 1; i <= wire.NbEdges(); i++)
DefinePatch(wire,code,isCutByU,cutIndex,i);
}
//=======================================================================
//function : GetGridResolution
//purpose : auxilary
//=======================================================================
static Standard_Real GetGridResolution(const Handle(TColStd_HArray1OfReal) SplitValues,
const Standard_Integer cutIndex)
{
Standard_Integer nb = SplitValues->Length();
Standard_Real leftLen = (cutIndex > 1 ? SplitValues->Value(cutIndex) - SplitValues->Value(cutIndex-1) :
SplitValues->Value(nb) -SplitValues->Value(nb-1));
Standard_Real rigthLen =(cutIndex < nb ? SplitValues->Value(cutIndex+1)-SplitValues->Value(cutIndex) :
SplitValues->Value(2) - SplitValues->Value(1));
return Min(leftLen,rigthLen)/3.;
}
//=======================================================================
//function : SplitWire
//purpose :
//=======================================================================
ShapeFix_WireSegment ShapeFix_ComposeShell::SplitWire (ShapeFix_WireSegment &wire,
TColStd_SequenceOfInteger& indexes,
const TColStd_SequenceOfReal& values,
TopTools_SequenceOfShape& vertices,
const TColStd_SequenceOfInteger &SegmentCodes,
const Standard_Boolean isCutByU,
const Standard_Integer cutIndex)
{
BRep_Builder B;
ShapeFix_WireSegment result;
Handle(ShapeAnalysis_Surface) aSurfTool =
new ShapeAnalysis_Surface ( BRep_Tool::Surface (myFace) );
Standard_Integer nbSplits = indexes.Length();
ShapeAnalysis_Edge sae;
Standard_Integer start = 1;
TopAbs_Orientation anWireOrient = wire.Orientation();
gp_Trsf T;
if ( ! myLoc.IsIdentity() ) T = myLoc.Inverted().Transformation();
// Processing edge by edge (assuming that split points are sorted along the wire)
for ( Standard_Integer i = 1; i <= wire.NbEdges(); i++ ) {
// for already splitted seam edge, redistribute its splitting points
Standard_Integer nsplit = ApplyContext ( wire, i, Context() );
if ( nsplit !=1 ) {
DistributeSplitPoints ( wire.WireData(), myFace, i, nsplit, indexes, values );
if ( nsplit <=0 ) {
#ifdef OCCT_DEBUG
cout << "Error: ShapeFix_ComposeShell::SplitWire: edge dismissed" << endl;
#endif
i--;
continue;
}
}
TopoDS_Edge edge = wire.Edge(i);
Standard_Integer iumin, iumax, ivmin, ivmax;
wire.GetPatchIndex ( i, iumin, iumax, ivmin, ivmax );
// Position code for first segment of edge
Standard_Integer code = SegmentCodes ( start >1 ? start-1 : SegmentCodes.Length() );
// Defining split parameters on edge
Standard_Integer stop = start;
while ( stop <= nbSplits && indexes(stop) == i ) stop++;
if ( stop == start ) {
result.AddEdge ( 0, edge, iumin, iumax, ivmin, ivmax );
if(code!=0 || wire.Orientation()!=TopAbs_EXTERNAL) // pdn 0 code handling for extertnal wires
DefinePatch ( result, code, isCutByU, cutIndex );
continue;
}
//find non-manifold vertices on edge
TopTools_SequenceOfShape aNMVertices;
TopoDS_Iterator aIt(edge,Standard_False);
for( ; aIt.More(); aIt.Next()) {
if(aIt.Value().Orientation() != TopAbs_FORWARD &&
aIt.Value().Orientation() != TopAbs_REVERSED)
aNMVertices.Append(aIt.Value());
}
// Collect data on edge
Standard_Real tolEdge = BRep_Tool::Tolerance(edge);
Standard_Real tol = LimitTolerance( tolEdge );
TopoDS_Vertex prevV = sae.FirstVertex(edge);
TopoDS_Vertex lastV = sae.LastVertex(edge);
Standard_Real prevVTol = LimitTolerance( BRep_Tool::Tolerance(prevV) );
Standard_Real lastVTol = LimitTolerance( BRep_Tool::Tolerance(lastV) );
gp_Pnt prevVPnt = BRep_Tool::Pnt(prevV);
gp_Pnt lastVPnt = BRep_Tool::Pnt(lastV);
if ( T.Form() != gp_Identity ) {
prevVPnt.Transform ( T );
lastVPnt.Transform ( T );
}
Handle(Geom_Curve) c3d;
Standard_Real f3d, l3d;
if ( ! sae.Curve3d ( edge, c3d, f3d, l3d ) ) { // not a crime
c3d.Nullify();
f3d = l3d = 0;
}
Standard_Real firstPar, lastPar;
Handle(Geom2d_Curve) C2d;
if ( ! sae.PCurve ( edge, myFace, C2d, firstPar, lastPar ) ) {
myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL2 );
}
//finding sequence of non-manifold parameters
Standard_Integer nbNMVert = aNMVertices.Length();
TColStd_SequenceOfReal aNMVertParams;
if( nbNMVert) {
Geom2dAdaptor_Curve adc(C2d);
Standard_Integer n =1;
for( ; n<= nbNMVert; n++) {
gp_Pnt apV = BRep_Tool::Pnt(TopoDS::Vertex(aNMVertices.Value(n)));
Standard_Real apar =firstPar;
Standard_Real adist2 =RealLast();
gp_Pnt aPproj;
if(!c3d.IsNull()) {
ShapeAnalysis_Curve asae;
adist2 = asae.Project(c3d,apV,Precision::Confusion(),aPproj,apar);
adist2 *= adist2;
}
else {
gp_Pnt2d aP2d = aSurfTool->ValueOfUV(apV,Precision::Confusion());
Extrema_ExtPC2d aExtr(aP2d,adc);
if(aExtr.IsDone() && aExtr.NbExt()) {
adist2 = aExtr.SquareDistance(1);
Standard_Integer index =1;
Standard_Integer k =2;
for( ; k <= aExtr.NbExt();k++) {
Standard_Real ad2 = aExtr.SquareDistance(k);
if( ad2 <adist2) {
adist2 = ad2;
index =k;
}
}
apar = aExtr.Point(index).Parameter();
}
}
aNMVertParams.Append(apar);
}
}
//pdn Claculating parametric shift
Standard_Boolean sp = (f3d == firstPar && l3d == lastPar);
Standard_Real span2d = lastPar - firstPar;
// Standard_Real ln2d = lastPar-prevPar;
// Standard_Real ln3d = l3d - f3d;
// Standard_Real fact = ln2d/ln3d;
// Standard_Real shift = prevPar - f3d*fact;
Standard_Real prevPar = firstPar;
gp_Pnt2d prevPnt2d = C2d->Value(prevPar);
gp_Pnt2d lastPnt2d = C2d->Value(lastPar);
gp_Pnt prevPnt = myGrid->Value ( prevPnt2d );
gp_Pnt lastPnt = myGrid->Value ( lastPnt2d );
Standard_Boolean isPeriodic = C2d->IsPeriodic();
Standard_Real aPeriod = (isPeriodic ? C2d->Period() :0.);
// Splitting edge
Standard_Integer NbEdgesStart = result.NbEdges();
Standard_Boolean splitted = Standard_False;
Standard_Real currPar=lastPar; //SK
for ( Standard_Integer j = start; j <= stop; prevPar = currPar, j++ ) {
if ( ! splitted && j >= stop ) { // no splitting at all
// code = SegmentCodes ( j >1 ? j-1 : SegmentCodes.Length() ); // classification code
break;
}
currPar = ( j < stop ? values.Value(j) : lastPar );
//fix for case when pcurve is periodic and first parameter of edge is more than 2P
//method ShapeBuild_Edge::CopyRanges shift pcurve to range 0-2P and parameters of cutting
//should be shifted too. gka SAMTECH 28.07.06
if(isPeriodic ) {
if (currPar > (Max(lastPar,firstPar) +Precision::PConfusion()) ||
currPar < (Min(firstPar,lastPar)- Precision::PConfusion())) {
Standard_Real aShift = ShapeAnalysis::AdjustByPeriod(currPar, (firstPar+lastPar)*0.5,aPeriod);
currPar+=aShift;
}
}
gp_Pnt2d currPnt2d;
gp_Pnt currPnt;
// Try to adjust current splitting point to previous or end of edge
Standard_Boolean doCut = Standard_True;
TopoDS_Vertex V;
if ( Abs ( currPar - lastPar ) < ::Precision::PConfusion() ) {
V = lastV;
doCut = Standard_False;
}
else if ( Abs ( currPar - prevPar ) < ::Precision::PConfusion() ) {
vertices.Append ( prevV );
code = SegmentCodes ( j ); // classification code - update for next segment
continue; // no splitting at this point, go to next one
}
else {
currPnt2d = C2d->Value(currPar);
currPnt = myGrid->Value ( currPnt2d );
if ( currPnt.Distance ( lastVPnt ) <= lastVTol &&
lastPnt.Distance ( currPnt ) <= tol &&
CheckByCurve3d ( lastVPnt, c3d, f3d+(currPar-firstPar)*(l3d-f3d)/span2d,
T, lastVTol ) &&
lastPnt.Distance ( myGrid->Value ( C2d->Value(0.5*(currPar+lastPar)) ) ) <= tol ) {
V = lastV;
Standard_Real uRes = myUResolution;
Standard_Real vRes = myVResolution;
if(isCutByU) {
Standard_Real gridRes = GetGridResolution(myGrid->UJointValues(),cutIndex)/tol;
uRes = Min(myUResolution,gridRes);
}
else {
Standard_Real gridRes = GetGridResolution(myGrid->VJointValues(),cutIndex)/tol;
vRes = Min(myVResolution,gridRes);
}
if ( IsCoincided ( lastPnt2d, currPnt2d, uRes, vRes, tol ) &&
IsCoincided ( lastPnt2d, C2d->Value(0.5*(currPar+lastPar)),
uRes, vRes, tol ) ) doCut = Standard_False;
}
else if ( currPnt.Distance ( prevVPnt ) <= prevVTol &&
prevPnt.Distance ( currPnt ) <= tol &&
CheckByCurve3d ( prevVPnt, c3d, f3d+(currPar-firstPar)*(l3d-f3d)/span2d,
T, prevVTol ) &&
prevPnt.Distance ( myGrid->Value ( C2d->Value(0.5*(currPar+prevPar)) ) ) <= tol ) {
V = prevV;
Standard_Real uRes = myUResolution;
Standard_Real vRes = myVResolution;
if(isCutByU) {
Standard_Real gridRes = GetGridResolution(myGrid->UJointValues(),cutIndex)/tol;
uRes = Min(myUResolution,gridRes);
}
else {
Standard_Real gridRes = GetGridResolution(myGrid->VJointValues(),cutIndex)/tol;
vRes = Min(myVResolution,gridRes);
}
if ( IsCoincided ( prevPnt2d, currPnt2d, uRes, vRes, tol ) &&
IsCoincided ( prevPnt2d, C2d->Value(0.5*(currPar+prevPar)),
uRes, vRes, tol ) ) {
vertices.Append ( prevV );
code = SegmentCodes ( j ); // classification code - update for next segment
continue; // no splitting at this point, go to next one
}
}
//:abv 28.05.02: OCC320 Sample_2: if maxtol = 1e-7, the vertex tolerance
// is actually ignored - protect against new vertex on degenerated edge
else if ( BRep_Tool::Degenerated(edge) && prevV.IsSame(lastV) ) {
V = prevV;
}
}
// classification code for current segment
if ( j > start ) code = SegmentCodes ( j >1 ? j-1 : SegmentCodes.Length() );
// if not adjusted, make new vertex
if ( V.IsNull() ) {
B.MakeVertex ( V, currPnt.Transformed(myLoc.Transformation()), tolEdge );
vertices.Append ( V );
}
// else adjusted to end, fill all resting vertices
else if ( ! doCut ) {
for ( ; j < stop; j++ ) vertices.Append ( lastV );
if ( ! splitted ) break; // no splitting at all
currPar = lastPar;
}
else vertices.Append ( V );
// When edge is about to be splitted, copy end vertices to protect
// original shape from increasing tolerance after fixing SameParameter
if ( ! splitted ) {
//smh#8
TopoDS_Shape emptyCopiedfV = prevV.EmptyCopied();
TopoDS_Vertex fV = TopoDS::Vertex (emptyCopiedfV );
Context()->Replace ( prevV, fV );
TopoDS_Vertex lV;
if ( prevV.IsSame ( lastV ) ) {
//smh#8
TopoDS_Shape tmpV = fV.Oriented ( lastV.Orientation() ) ;
lV = TopoDS::Vertex (tmpV);
}
else {
//smh#8
TopoDS_Shape emptyCopied = lastV.EmptyCopied();
lV = TopoDS::Vertex (emptyCopied);
Context()->Replace ( lastV, lV );
}
if ( V.IsSame ( lastV ) ) V = lV;
else if ( V.IsSame ( prevV ) ) V = fV;
lastV = lV;
prevV = fV;
}
// Splitting of the edge
splitted = Standard_True;
prevV.Orientation ( TopAbs_FORWARD );
V.Orientation ( TopAbs_REVERSED );
ShapeBuild_Edge sbe;
TopoDS_Edge anInitEdge = edge;
Standard_Boolean ismanifold = (edge.Orientation() == TopAbs_FORWARD ||
edge.Orientation() == TopAbs_REVERSED);
if(!ismanifold)
anInitEdge.Orientation(TopAbs_FORWARD);
TopoDS_Edge newEdge = sbe.CopyReplaceVertices (anInitEdge, prevV, V );
//addition internal vertices if they exists on edge
Standard_Integer n =1;
for( ; n <= aNMVertParams.Length(); n++) {
Standard_Real apar = aNMVertParams.Value(n);
TopoDS_Vertex aNMVert =TopoDS::Vertex(aNMVertices.Value(n));
TopoDS_Vertex atmpV = TopoDS::Vertex(Context()->Apply(aNMVert));
if(fabs(apar - prevPar) <= Precision::PConfusion()) {
Context()->Replace(atmpV,prevV);
aNMVertParams.Remove(n);
aNMVertices.Remove(n);
n--;
}
else if(fabs(apar - currPar) <= Precision::PConfusion()) {
Context()->Replace(atmpV,V);
aNMVertParams.Remove(n);
aNMVertices.Remove(n);
n--;
}
if(apar > prevPar && apar < currPar) {
B.Add(newEdge,atmpV);
aNMVertParams.Remove(n);
aNMVertices.Remove(n);
n--;
}
}
sbe.CopyPCurves ( newEdge, anInitEdge );
Handle(ShapeAnalysis_TransferParameters) theTransferParamtool = GetTransferParamTool();
theTransferParamtool->SetMaxTolerance(MaxTolerance());
theTransferParamtool->Init(anInitEdge,myFace);
theTransferParamtool->TransferRange(newEdge,prevPar,currPar,Standard_True);
if(!ismanifold) {
if(code == IOR_UNDEF) //tangential segment
newEdge.Orientation(TopAbs_EXTERNAL);
else
newEdge.Orientation(edge.Orientation());
}
if(!sp && !BRep_Tool::Degenerated(newEdge))
B.SameRange(newEdge, Standard_False);
//pdn take into account 0 codes (if ext)
if(code == 0 && wire.Orientation()==TopAbs_EXTERNAL){
code = ( ( isCutByU == (Standard_Boolean)( j == 1 ) ) ? 1 : 2 );
}
result.AddEdge ( 0, newEdge, iumin, iumax, ivmin, ivmax );
DefinePatch ( result, code, isCutByU, cutIndex );
// Changing prev parameters
prevV = V;
prevVTol = LimitTolerance( BRep_Tool::Tolerance ( V ) );
prevVPnt = BRep_Tool::Pnt ( V );
prevPnt = currPnt;
prevPnt2d = currPnt2d;
}
start = stop;
if ( splitted ) {
// record replacement in context
// NOTE: order of edges in the replacing wire corresponds to FORWARD orientation of the edge
TopoDS_Wire resWire;
B.MakeWire ( resWire );
for ( Standard_Integer k=NbEdgesStart; k < result.NbEdges(); k++ ) {
if ( edge.Orientation() == TopAbs_FORWARD || edge.Orientation() == TopAbs_INTERNAL)
B.Add ( resWire, result.Edge(k+1) );
else B.Add ( resWire, result.Edge(result.NbEdges()-k+NbEdgesStart) );
}
Context()->Replace ( edge, resWire );
}
else {
if(anWireOrient == TopAbs_INTERNAL && code ==0) {
ShapeBuild_Edge sbe;
if(edge.Orientation() == TopAbs_INTERNAL)
edge.Orientation(TopAbs_FORWARD);
TopoDS_Edge e1 = sbe.Copy(edge,Standard_False);
Handle(Geom2d_Curve) C2d2 = Handle(Geom2d_Curve)::DownCast(C2d->Copy());
B.UpdateEdge(e1,C2d,C2d2,myFace,0.);
e1.Orientation(TopAbs_EXTERNAL);
Context()->Replace ( edge,e1);
result.AddEdge ( 0,e1 , iumin, iumax, ivmin, ivmax );
}
else
result.AddEdge ( 0, edge, iumin, iumax, ivmin, ivmax );
if(code == 0 && wire.Orientation()==TopAbs_EXTERNAL){
//pdn defining code for intersection of two isos
code = ( ( isCutByU == (Standard_Boolean)( Abs(firstPar-currPar) < Abs(lastPar-currPar) ) ) ? 2 : 1 );
}
DefinePatch ( result, code, isCutByU, cutIndex );
}
}
result.Orientation ( anWireOrient );
return result;
}
//=======================================================================
//function : SplitByLine
//purpose :
//=======================================================================
Standard_Boolean ShapeFix_ComposeShell::SplitByLine (ShapeFix_WireSegment &wire,
const gp_Lin2d &line,
const Standard_Boolean isCutByU,
const Standard_Integer cutIndex,
TColStd_SequenceOfReal &SplitLinePar,
TColStd_SequenceOfInteger &SplitLineCode,
TopTools_SequenceOfShape &SplitLineVertex)
{
ShapeAnalysis_Edge sae;
// prepare data on cutting line
Handle(Geom2d_Line) jC2d = new Geom2d_Line ( line );
Geom2dAdaptor_Curve jGAC(jC2d);
TColStd_SequenceOfInteger IntEdgeInd; // index of intersecting edge
TColStd_SequenceOfReal IntEdgePar; // parameter of intersection point on edge
TColStd_SequenceOfReal IntLinePar; // parameter of intersection point on line
Standard_Boolean isnonmanifold = (wire.Orientation() == TopAbs_INTERNAL);
//gka correction for non-manifold vertices SAMTECH
if(wire.IsVertex()) {
Handle(ShapeAnalysis_Surface) aSurfTool = new ShapeAnalysis_Surface ( BRep_Tool::Surface (myFace) );
TopoDS_Vertex aVert = wire.GetVertex();
gp_Pnt aP3d = BRep_Tool::Pnt(aVert);
gp_Pnt2d aP2d = aSurfTool->ValueOfUV(aP3d,Precision::Confusion());
Standard_Real dev =0.;
Standard_Integer code = PointLinePosition(aP2d,line,dev);
if(code != IOR_UNDEF)
return Standard_False;
Standard_Real par = ParamPointOnLine (aP2d,line);
SplitLinePar.Append ( par );
//splitting codes for non-manifold topology should be tangential
SplitLineCode.Append (ITP_TANG); //ITP_INTER);
TopoDS_Vertex aVertNew;
BRep_Builder aB;
aB.MakeVertex(aVertNew,aP3d,BRep_Tool::Tolerance(aVert));
aVertNew.Orientation(TopAbs_FORWARD);
Context()->Replace(aVert,aVertNew);
SplitLineVertex.Append (aVertNew);
wire.SetVertex(aVertNew);
return Standard_True;
}
const Handle(ShapeExtend_WireData) sewd = wire.WireData();
Standard_Integer nbe = sewd->NbEdges();
//:abv 31.10.01: for closed mode
Standard_Integer closedDir = 0;
if ( myClosedMode ) {
if ( myUClosed && Abs ( line.Direction().X() ) < ::Precision::PConfusion() )
closedDir = -1;
else if ( myVClosed && Abs ( line.Direction().Y() ) < ::Precision::PConfusion() )
closedDir = 1;
}
Standard_Real halfPeriod = 0.5 * ( closedDir ? closedDir <0 ? myUPeriod : myVPeriod : 0. );
//============================================
// make intersections and collect all data on intersection points
Standard_Integer firstCode=0, prevCode=0;
gp_Pnt2d firstPos, prevPos;
Standard_Real firstDev=0., prevDev=0.;
for (Standard_Integer iedge = 1; iedge <= nbe; iedge++) {
TopoDS_Edge E= sewd->Edge ( iedge );
Standard_Boolean isreversed = ( E.Orientation() == TopAbs_REVERSED );
Standard_Real f, l;
Handle(Geom2d_Curve) c2d;
if ( ! sae.PCurve ( E, myFace, c2d, f, l, Standard_False ) ) continue;
// get end points
gp_Pnt2d posf = c2d->Value(f), posl = c2d->Value(l);
gp_XY pppf = posf.XY(), pppl = posl.XY();
// In case of ClosedMode, adjust curve and end points to period on closed surface
//:abv 16.10.01: Ziegler_CADDY01.sat -18: if pcurve is longer than period,
// ensure processing of all intersections
Standard_Integer nbIter = 1;
gp_Vec2d shiftNext(0.,0.);
if ( myClosedMode ) {
// get bounding box of pcurve
ShapeAnalysis_Curve sac;
Bnd_Box2d box;
const Standard_Integer aNbPoints = 41;
sac.FillBndBox ( c2d, f, l, aNbPoints, Standard_True, box );
Standard_Real umin, vmin, umax, vmax;
box.Get ( umin, vmin, umax, vmax );
// compute shifts and adjust points adjust
if ( closedDir < 0 ) {
Standard_Real x = line.Location().X();
Standard_Real shift = ShapeAnalysis::AdjustToPeriod ( umin, x-myUPeriod, x );
if ( shift != 0. ) {
c2d = Handle(Geom2d_Curve)::DownCast ( c2d->Copy() );
gp_Vec2d V ( shift, 0. );
c2d->Translate ( V );
pppf.SetX ( pppf.X() + shift );
pppl.SetX ( pppl.X() + shift );
}
shiftNext.SetX ( -myUPeriod );
nbIter = (Standard_Integer)( 1 + Abs ( umax + shift - x ) / myUPeriod );
shift = ShapeAnalysis::AdjustByPeriod ( posf.X(), x, myUPeriod );
posf.SetX ( posf.X() + shift );
shift = ShapeAnalysis::AdjustByPeriod ( posl.X(), x, myUPeriod );
posl.SetX ( posl.X() + shift );
}
else if ( closedDir > 0 ) {
Standard_Real y = line.Location().Y();
Standard_Real shift = ShapeAnalysis::AdjustToPeriod ( vmin, y-myVPeriod, y );
if ( shift != 0. ) {
c2d = Handle(Geom2d_Curve)::DownCast ( c2d->Copy() );
gp_Vec2d V ( 0., shift );
c2d->Translate ( V );
pppf.SetY ( pppf.Y() + shift );
pppl.SetY ( pppl.Y() + shift );
}
shiftNext.SetY ( -myVPeriod );
nbIter = (Standard_Integer)( 1 + Abs ( umax + shift - y ) / myVPeriod );
shift = ShapeAnalysis::AdjustByPeriod ( posf.Y(), y, myVPeriod );
posf.SetY ( posf.Y() + shift );
shift = ShapeAnalysis::AdjustByPeriod ( posl.Y(), y, myVPeriod );
posl.SetY ( posl.Y() + shift );
}
}
// detect intersections at junction of two edges
gp_Pnt2d pos = ( isreversed ? posl : posf );
Standard_Real dev;
Standard_Integer code = PointLinePosition ( pos, line, dev );
if ( iedge ==1 ) { firstCode = code; firstPos = pos; firstDev = dev; }
else if ( code == IOR_UNDEF || code != prevCode ) {
if ( ! closedDir || Abs ( dev - prevDev ) < halfPeriod ) {
IntLinePar.Append ( ParamPointsOnLine ( pos, prevPos, line ) ); // !! - maybe compute exactly ?
IntEdgePar.Append ( isreversed ? l : f );
IntEdgeInd.Append ( iedge );
}
}
// fill data on end point (for next edge)
pos = ( isreversed ? posf : posl );
prevCode = PointLinePosition ( pos, line, prevDev );
prevPos = pos;
// cycle with shift in order to track all possible intersections
for ( Standard_Integer iter=1; iter <= nbIter; iter++ ) {
// data for intersection
IntRes2d_Domain iDom ( pppf, f, TOLINT, pppl, l, TOLINT );
Geom2dAdaptor_Curve iGAC(c2d);
// intersection
Geom2dInt_GInter Inter;
Inter.Perform ( jGAC, /*jDom,*/ iGAC, iDom, TOLINT, TOLINT );
// Fill arrays with new intersection points
if ( Inter.IsDone() ) {
Standard_Integer i;
for ( i = 1; i <= Inter.NbPoints(); i++ ) {
IntRes2d_IntersectionPoint IP = Inter.Point (i);
IntLinePar.Append ( IP.ParamOnFirst() );
IntEdgePar.Append ( IP.ParamOnSecond() );
}
for ( i = 1; i <= Inter.NbSegments(); i++ ) {
IntRes2d_IntersectionSegment IS = Inter.Segment (i);
if ( IS.HasFirstPoint() ) {
IntRes2d_IntersectionPoint IP = IS.FirstPoint();
IntLinePar.Append ( IP.ParamOnFirst() );
IntEdgePar.Append ( IP.ParamOnSecond() );
}
if ( IS.HasLastPoint() ) {
IntRes2d_IntersectionPoint IP = IS.LastPoint();
IntLinePar.Append ( IP.ParamOnFirst() );
IntEdgePar.Append ( IP.ParamOnSecond() );
}
}
}
if ( iter < nbIter ) {
if ( iter == 1 ) c2d = Handle(Geom2d_Curve)::DownCast ( c2d->Copy() );
pppf += shiftNext.XY();
pppl += shiftNext.XY();
c2d->Translate ( shiftNext );
}
}
Standard_Integer start = IntEdgeInd.Length() + 1; // first of the new points
// Move all points into range [f,l] (intersector sometimes gives params out of range)
Standard_Integer i;
for ( i = start; i <= IntEdgePar.Length(); i++ ) {
if ( IntEdgePar(i) < f ) IntEdgePar.SetValue ( i, f );
else if ( IntEdgePar(i) > l ) IntEdgePar.SetValue ( i, l );
}
// Sort by parameter on edge
for ( i = IntEdgePar.Length(); i > start; i-- )
for ( Standard_Integer j = start; j < i; j++ ) {
if ( isreversed == (Standard_Boolean) ( IntEdgePar(j+1) < IntEdgePar(j) ) ) continue;
IntLinePar.Exchange ( j, j+1 );
IntEdgePar.Exchange ( j, j+1 );
}
// and fill indices
for ( i = start; i <= IntEdgePar.Length(); i++ )
IntEdgeInd.Append ( iedge );
// Detect intersection at closing point
// Only wires which are not EXTERNAL are considered (as closed)
if ( iedge == nbe && wire.Orientation() != TopAbs_EXTERNAL &&
wire.Orientation() != TopAbs_INTERNAL &&
( prevCode == IOR_UNDEF || prevCode != firstCode ) ) {
if ( ! closedDir || Abs ( firstDev - prevDev ) < halfPeriod ) {
IntLinePar.Append ( ParamPointsOnLine ( pos, firstPos, line ) );
IntEdgePar.Append ( isreversed ? f : l );
IntEdgeInd.Append ( iedge );
}
}
}
if ( IntEdgePar.Length() <1 ) {
//pdn Defining position of wire. There is no intersection, so by any point.
DefinePatchForWire ( wire, firstCode, isCutByU, cutIndex );
return Standard_False; //pdn ??
}
//======================================
// Fill sequence of transition codes for intersection points
TColStd_SequenceOfInteger IntCode; // parameter of intersection point on line
TColStd_SequenceOfInteger SegmentCodes; // classification codes for segments of wire
// remove duplicated points to ensure correct results of ComputeCode
Standard_Integer i, j = IntEdgePar.Length();
if ( myClosedMode && j >1 ) {
for ( i = 1; i <= IntEdgePar.Length(); ) {
if ( i == j ) break;
if ( IntEdgeInd(i) == IntEdgeInd(j) &&
Abs ( IntEdgePar(i) - IntEdgePar(j) ) < ::Precision::PConfusion() ) {
IntLinePar.Remove(i);
IntEdgePar.Remove(i);
IntEdgeInd.Remove(i);
if ( j >i ) j--;
continue;
}
else if ( nbe ==1 || IntEdgeInd(i) == (IntEdgeInd(j)%nbe)+1 ) {
TopoDS_Edge E1 = sewd->Edge ( IntEdgeInd(j) );
TopoDS_Edge E2 = sewd->Edge ( IntEdgeInd(i) );
Standard_Real a1, b1, a2, b2;
BRep_Tool::Range ( E1, myFace, a1, b1 );
BRep_Tool::Range ( E2, myFace, a2, b2 );
if ( Abs ( IntEdgePar(j) - ( E1.Orientation() == TopAbs_FORWARD ? b1 : a1 ) ) < ::Precision::PConfusion() &&
Abs ( IntEdgePar(i) - ( E2.Orientation() == TopAbs_FORWARD ? a2 : b2 ) ) < ::Precision::PConfusion() ) {
IntLinePar.Remove(i);
IntEdgePar.Remove(i);
IntEdgeInd.Remove(i);
if ( j >i ) j--;
continue;
}
}
j=i++;
}
}
// Compute segment codes (left side of line, right or tangential)
for ( i=1; i <= IntEdgePar.Length(); i++ ) {
j = ( i < IntEdgePar.Length() ? i + 1 : 1 );
Standard_Integer code = ComputeCode ( sewd, line, IntEdgeInd(i), IntEdgeInd(j),
IntEdgePar(i), IntEdgePar(j),isnonmanifold );
SegmentCodes.Append ( code );
}
// for EXTERNAL wire, i.e. another joint line, every point is double intersection
if ( wire.Orientation() == TopAbs_EXTERNAL ) {
for ( i=1; i <= IntEdgePar.Length(); i++ )
IntCode.Append ( ITP_TANG | IOR_BOTH );
}
// For real (closed) wire, analyze tangencies
else {
if(wire.Orientation() != TopAbs_INTERNAL) {
// Two consecutive tangential segments are considered as one, merge them.
for ( i=1; i <= IntEdgePar.Length(); i++ ) {
j = ( i > 1 ? i-1 : IntEdgePar.Length() );
int k = ( i < IntEdgePar.Length() ? i + 1 : 1 ); // [ACIS22539]
if ( SegmentCodes(j) == IOR_UNDEF &&
SegmentCodes(i) == IOR_UNDEF ) {
// Very specific case when the constructed seam edge
// overlaps with spur edge [ACIS22539]
if (myClosedMode && (IntLinePar(i) - IntLinePar(j)) * (IntLinePar(k) - IntLinePar(i)) <= 0. )
continue;
IntEdgeInd.Remove(i);
IntEdgePar.Remove(i);
IntLinePar.Remove(i);
SegmentCodes.Remove(i);
i--;
}
}
}
//pdn exit if all split points removed
if ( IntEdgePar.Length() <1 ) {
//DefinePatchForWire ( wire, firstCode, isCutByU, cutIndex );
return Standard_False; //pdn ??
}
// Analyze type of intersection point and encode it
// Three kinds of points (ITP): clear intersection, tangency in-point,
// beginning and end of tangential segment.
// Orientation (IOR) tells on which side of line edge crosses it
j = IntEdgePar.Length();
for ( i=1; i <= IntEdgePar.Length(); j = i++ ) {
Standard_Integer codej = SegmentCodes(j);
Standard_Integer codei = SegmentCodes(i);
if ( myClosedMode ) {
if ( ( codej & IOR_BOTH ) == IOR_BOTH ) //IOR_LEFT : IOR_RIGHT
codej = ( codej & IOR_POS ? IOR_RIGHT : IOR_LEFT );
if ( ( codei & IOR_BOTH ) == IOR_BOTH ) //IOR_RIGHT : IOR_LEFT
codei = ( codei & IOR_POS ? IOR_LEFT : IOR_RIGHT );
}
Standard_Integer ipcode = ( codej | codei );
if ( codej == IOR_UNDEF ) { // previous segment was tangency
if ( IntLinePar(i) > IntLinePar (j) )
ipcode |= ITP_ENDSEG; // end of segment
else ipcode |= ITP_BEGSEG; // beginning of segment
}
else if ( codei == IOR_UNDEF ) { // current segment is tangency
if ( IntLinePar ( i < IntLinePar.Length() ? i+1 : 1 ) > IntLinePar(i) )
ipcode |= ITP_BEGSEG; // beginning of segment
else ipcode |= ITP_ENDSEG; // end of segment
}
//internal wire can be only tangent
else if ( i == j ) ipcode |= ( ( ipcode & IOR_BOTH ) == IOR_BOTH && !isnonmanifold ? ITP_INTER : ITP_TANG );
else if ( codei == codej || isnonmanifold) ipcode |= ITP_TANG; // tangency in-point
else ipcode |= ITP_INTER; // standard crossing
IntCode.Append ( ipcode );
}
}
//=======================================
// Split edges in the wire by intersection points and fill vertices array
TopTools_SequenceOfShape IntVertices;
wire = SplitWire ( wire, IntEdgeInd, IntEdgePar, IntVertices,
SegmentCodes, isCutByU, cutIndex );
// add all data to input arrays
for ( i=1; i <= IntLinePar.Length(); i++ ) {
SplitLinePar.Append ( IntLinePar(i) );
SplitLineCode.Append ( IntCode(i) );
SplitLineVertex.Append ( IntVertices(i) );
}
return Standard_True;
}
//=======================================================================
//function : SplitByLine
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::SplitByLine (ShapeFix_SequenceOfWireSegment &wires,
const gp_Lin2d &line,
const Standard_Boolean isCutByU,
const Standard_Integer cutIndex)
{
TColStd_SequenceOfReal SplitLinePar;
TColStd_SequenceOfInteger SplitLineCode;
TopTools_SequenceOfShape SplitLineVertex;
// split wires one by one, collecting data on intersection points
Standard_Integer i; // svv #1
for ( i=1; i <= wires.Length(); i++ ) {
SplitByLine ( wires(i), line, isCutByU, cutIndex,
SplitLinePar, SplitLineCode, SplitLineVertex );
}
// sort intersection points along parameter on cutting line
for ( i = SplitLinePar.Length(); i >1; i-- )
for ( Standard_Integer j=1; j < i; j++ ) {
if ( SplitLinePar(j) > SplitLinePar(j+1) ) {
SplitLinePar.Exchange ( j, j+1 );
SplitLineCode.Exchange ( j, j+1 );
SplitLineVertex.Exchange ( j, j+1 );
}
}
// merge null-length tangential segments into one-point tangencies or intersections
for ( i = 1; i < SplitLinePar.Length(); i++ ) {
if ( Abs ( SplitLinePar(i+1) - SplitLinePar(i) ) > ::Precision::PConfusion() ) continue;
if ( ( SplitLineCode(i) & ITP_ENDSEG &&
SplitLineCode(i+1) & ITP_BEGSEG ) ||
( SplitLineCode(i) & ITP_BEGSEG &&
SplitLineCode(i+1) & ITP_ENDSEG ) ) {
Standard_Integer code = ( SplitLineCode(i) | SplitLineCode(i+1) ) & IOR_BOTH;
SplitLineCode.SetValue ( i, code | ( code == IOR_BOTH ? ITP_INTER : ITP_TANG ) );
SplitLinePar.Remove(i+1);
SplitLineCode.Remove(i+1);
SplitLineVertex.Remove(i+1);
}
}
// go along line, split it by intersection points and create edges
// (only for internal parts, in particular not for tangential segments)
BRep_Builder B;
Standard_Integer parity = 0; // 0 - out, 1 - in
Standard_Integer halfparity = 0; // left/right for tangential segments
Standard_Integer tanglevel = 0; // tangency nesting level
for ( i = 1; i <= SplitLinePar.Length(); i++ ) {
Standard_Integer code = SplitLineCode(i);
Standard_Boolean interior = ( ! tanglevel && parity % 2 ); // create an edge
if ( code & ITP_INTER ) { // crossing
parity++;
}
else if ( code & ITP_BEGSEG ) { // beginning of tangential segment
tanglevel++;
if ( ! halfparity ) halfparity = ( code & IOR_BOTH );
else if ( halfparity != ( code & IOR_BOTH ) ) parity++;
}
else if ( code & ITP_ENDSEG ) { // end of tangential segment
tanglevel--;
if ( ! halfparity ) halfparity = ( code & IOR_BOTH );
else if ( halfparity != ( code & IOR_BOTH ) ) parity++;
}
if ( tanglevel <0 ) {
// myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL4 );
#ifdef OCCT_DEBUG
cout << "Warning: ShapeFix_ComposeShell::SplitByLine: tangency level <0 !" << endl;
#endif
}
if ( ! interior ) continue;
// apply context to vertices (to perform replacing/merging vertices)
//smh#8
TopoDS_Shape tmpV1 = Context()->Apply ( SplitLineVertex(i-1) );
TopoDS_Shape tmpV2 = Context()->Apply ( SplitLineVertex(i) );
TopoDS_Vertex V1 = TopoDS::Vertex ( tmpV1 );
TopoDS_Vertex V2 = TopoDS::Vertex ( tmpV2 );
// protection against creating null-length edges
if ( SplitLinePar(i) - SplitLinePar(i-1) < ::Precision::PConfusion() ) {
#ifdef OCCT_DEBUG
cout << "Info: ShapeFix_ComposeShell::SplitByLine: Short segment ignored" << endl;
#endif
if ( ! V1.IsSame ( V2 ) ) { // merge coincident vertices
ShapeBuild_Vertex sbv;
TopoDS_Vertex V = sbv.CombineVertex ( V1, V2 );
Context()->Replace ( V1, V.Oriented ( V1.Orientation() ) );
Context()->Replace ( V2, V.Oriented ( V2.Orientation() ) );
V1 = V2 = V;
#ifdef OCCT_DEBUG
cout << "Info: ShapeFix_ComposeShell::SplitByLine: Coincided vertices merged" << endl;
#endif
}
continue;
}
// create an edge (without 3d curve), put it in wire segment and add to sequence
// NOTE: i here is always >1
TopoDS_Edge edge;
B.MakeEdge ( edge );
V1.Orientation ( TopAbs_FORWARD );
V2.Orientation ( TopAbs_REVERSED );
B.Add ( edge, V1 );
B.Add ( edge, V2 );
Handle(Geom2d_Line) Lin1 = new Geom2d_Line ( line );
Handle(Geom2d_Line) Lin2 = new Geom2d_Line ( line );
B.UpdateEdge ( edge, Lin1, Lin2, myFace, ::Precision::Confusion() );
B.Range ( edge, myFace, SplitLinePar(i-1), SplitLinePar(i) );
Handle(ShapeExtend_WireData) sbwd = new ShapeExtend_WireData;
sbwd->Add ( edge );
ShapeFix_WireSegment seg ( sbwd, TopAbs_EXTERNAL );
// set patch indices
DefinePatch ( seg, IOR_UNDEF, isCutByU, cutIndex );
if ( ! isCutByU ) {
seg.DefineIUMin ( 1, GetPatchIndex ( SplitLinePar(i-1)+::Precision::PConfusion(),
myGrid->UJointValues(), myUClosed ) );
seg.DefineIUMax ( 1, GetPatchIndex ( SplitLinePar(i)-::Precision::PConfusion(),
myGrid->UJointValues(), myUClosed ) + 1 );
}
else {
seg.DefineIVMin ( 1, GetPatchIndex ( SplitLinePar(i-1)+::Precision::PConfusion(),
myGrid->VJointValues(), myVClosed ) );
seg.DefineIVMax ( 1, GetPatchIndex ( SplitLinePar(i)-::Precision::PConfusion(),
myGrid->VJointValues(), myVClosed ) + 1 );
}
wires.Append ( seg );
}
if ( parity % 2 ) {
myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL4 );
#ifdef OCCT_DEBUG
cout << "Error: ShapeFix_ComposeShell::SplitByLine: parity error" << endl;
#endif
}
// Apply context to all wires to perform all recorded replacements/merging
for ( i=1; i <= wires.Length(); i++ ) {
for ( Standard_Integer j=1; j <= wires(i).NbEdges(); )
j += ApplyContext ( wires(i), j, Context() );
}
}
//=======================================================================
//function : SplitByGrid
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::SplitByGrid (ShapeFix_SequenceOfWireSegment &seqw)
{
// process splitting by U- anv V-seams (i.e. U=const and V=const curves)
// closed composite surface is processed as periodic
Standard_Real Uf,Ul,Vf,Vl;
BRepTools::UVBounds(myFace,Uf,Ul,Vf,Vl);
Standard_Real Umin,Umax,Vmin,Vmax;
myGrid->Bounds(Umin,Umax,Vmin,Vmax);
Standard_Real pprec = ::Precision::PConfusion();
// split by u lines
Standard_Integer i; // svv #1
for ( i = ( myUClosed ? 1 : 2 ); i <= myGrid->NbUPatches(); i++ ) {
gp_Pnt2d pos ( myGrid->UJointValue(i), 0. ); // 0. - for infinite ranges: myGrid->VJointValue(1) ;
gp_Lin2d line ( pos, gp_Dir2d ( 0., 1. ) );
if ( ! myClosedMode && myUClosed ) {
Standard_Real period = Umax - Umin;
Standard_Real X = pos.X();
Standard_Real sh = ShapeAnalysis::AdjustToPeriod(X,Uf, Uf+period);
for( ; X+sh <= Ul+pprec; sh += period ) {
gp_Lin2d ln = line.Translated(gp_Vec2d(sh,0));
Standard_Integer cutIndex = GetPatchIndex ( X+sh+pprec, myGrid->UJointValues(), myUClosed );
SplitByLine ( seqw, ln, Standard_True, cutIndex );
}
}
else
SplitByLine ( seqw, line, Standard_True, i );
}
// split by v lines
for ( i = ( myVClosed ? 1 : 2 ); i <= myGrid->NbVPatches(); i++ ) {
gp_Pnt2d pos ( 0., myGrid->VJointValue(i) );
gp_Lin2d line ( pos, gp_Dir2d ( 1., 0. ) );
if ( ! myClosedMode && myVClosed ) {
Standard_Real period = Vmax - Vmin;
Standard_Real Y = pos.Y();
Standard_Real sh = ShapeAnalysis::AdjustToPeriod(Y,Vf, Vf+period);
for( ; Y+sh <= Vl+pprec; sh += period) {
gp_Lin2d ln = line.Translated(gp_Vec2d(0,sh));
Standard_Integer cutIndex = GetPatchIndex ( Y+sh+pprec, myGrid->VJointValues(), myVClosed );
SplitByLine ( seqw, ln, Standard_False, cutIndex );
}
}
else
SplitByLine ( seqw, line, Standard_False, i );
}
// limit patch indices to be in range of grid (extended for periodic)
Standard_Integer iumin = GetPatchIndex ( Uf+pprec, myGrid->UJointValues(), myUClosed );
Standard_Integer iumax = GetPatchIndex ( Ul-pprec, myGrid->UJointValues(), myUClosed ) + 1;
for ( i=1; i <= seqw.Length(); i++ ) {
ShapeFix_WireSegment &wire = seqw(i);
for ( Standard_Integer j=1; j <= wire.NbEdges(); j++ ) {
wire.DefineIUMin ( j, iumin );
wire.DefineIUMax ( j, iumax );
}
}
Standard_Integer ivmin = GetPatchIndex ( Vf+pprec, myGrid->VJointValues(), myVClosed );
Standard_Integer ivmax = GetPatchIndex ( Vl-pprec, myGrid->VJointValues(), myVClosed ) + 1;
for ( i=1; i <= seqw.Length(); i++ ) {
ShapeFix_WireSegment &wire = seqw(i);
for ( Standard_Integer j=1; j <= wire.NbEdges(); j++ ) {
wire.DefineIVMin ( j, ivmin );
wire.DefineIVMax ( j, ivmax );
}
}
}
//=======================================================================
//function : BreakWires
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::BreakWires (ShapeFix_SequenceOfWireSegment &seqw)
{
// split all the wires by vertices
TopTools_MapOfShape splitVertices;
ShapeAnalysis_Edge sae;
// first collect splitting vertices
Standard_Integer i; // svv #1
for ( i=1; i <= seqw.Length(); i++ ) {
TopAbs_Orientation ori_wire = seqw(i).Orientation();
if ( ori_wire != TopAbs_EXTERNAL &&
ori_wire != TopAbs_INTERNAL) continue;
Handle(ShapeExtend_WireData) sbwd = seqw(i).WireData();
for ( Standard_Integer j=1; j <= sbwd->NbEdges(); j++ ) {
TopoDS_Edge edge = sbwd->Edge ( j );
TopAbs_Orientation ori_edge = (ori_wire == TopAbs_EXTERNAL ? ori_wire : edge.Orientation());
if(ori_edge == TopAbs_EXTERNAL) {
splitVertices.Add ( sae.FirstVertex ( edge ) );
splitVertices.Add ( sae.LastVertex ( edge ) );
}
}
}
// and then split each vire
// Here each wire is supposed to be connected (while probably not closed)
for ( i=1; i <= seqw.Length(); i++ ) {
TopAbs_Orientation ori = seqw(i).Orientation();
ShapeFix_WireSegment wire = seqw(i);
if(wire.IsVertex())
continue;
Handle(ShapeExtend_WireData) sbwd = wire.WireData();
// find first vertex for split
Standard_Integer j; // svv #1
for ( j=1; j <= sbwd->NbEdges(); j++ ) {
TopoDS_Vertex V = sae.FirstVertex ( sbwd->Edge(j) );
if ( splitVertices.Contains ( V ) ) break;
}
if ( j > sbwd->NbEdges() ) continue; // splitting not needed
// if first split of closed edge is not its start, make permutation
Standard_Integer shift = 0;
if ( j >1 && ! myClosedMode && wire.IsClosed() ) {
TopoDS_Vertex V = sae.FirstVertex ( sbwd->Edge(1) );
if ( ! splitVertices.Contains ( V ) )
shift = j - 1;
// wire.SetLast ( j-1 );
}
// perform splitting
Standard_Integer nbnew = 0;
ShapeFix_WireSegment newwire;
TopAbs_Orientation curOri = ori;
for ( Standard_Integer ind=1; ind <= sbwd->NbEdges(); ind++ ) {
j = 1 + ( ind - 1 + shift ) % sbwd->NbEdges();
TopoDS_Edge edge = sbwd->Edge(j);
TopoDS_Vertex V = sae.FirstVertex ( edge );
if ( ind==1 || splitVertices.Contains ( V ) ) {
if ( newwire.NbEdges() ) {
newwire.Orientation ( curOri );
// ShapeFix_WireSegment seg ( newwire, ori );
seqw.InsertBefore ( i++, newwire );
nbnew++;
}
newwire.Clear();
curOri = ori;
}
Standard_Integer iumin, iumax, ivmin, ivmax;
wire.GetPatchIndex ( j, iumin, iumax, ivmin, ivmax );
if(ori == TopAbs_INTERNAL && edge.Orientation() == TopAbs_EXTERNAL ) {
curOri = TopAbs_EXTERNAL;
edge.Orientation(TopAbs_FORWARD);
nbnew++;
}
newwire.AddEdge ( 0, edge, iumin, iumax, ivmin, ivmax );
}
if ( nbnew ) {
newwire.Orientation ( curOri );
// ShapeFix_WireSegment seg ( newwire, ori );
seqw.SetValue ( i, newwire );
}
}
}
//=======================================================================
//function : IsShortSegment
//purpose : auxilary
//=======================================================================
// BUC60035 2053: check if wire segment is very short (in order not to skip it)
// 0 - long
// 1 - short even in 2d (to be taken always)
// -1 - short in 3d but not in 2d (to be checked after algo and atteching to
// another wire if alone)
static Standard_Integer IsShortSegment (const ShapeFix_WireSegment &seg,
const TopoDS_Face myFace,
const Handle(Geom_Surface)& myGrid,
const TopLoc_Location &myLoc,
const Standard_Real UResolution,
const Standard_Real VResolution)
{
TopoDS_Vertex Vf = seg.FirstVertex();
if ( ! Vf.IsSame ( seg.LastVertex() ) ) return Standard_False;
gp_Pnt pnt = BRep_Tool::Pnt(Vf);
Standard_Real tol = BRep_Tool::Tolerance(Vf);
Standard_Real tol2 = tol*tol;
Standard_Integer code = 1;
ShapeAnalysis_Edge sae;
Handle(ShapeExtend_WireData) sbwd = seg.WireData();
for ( Standard_Integer i=1; i <= sbwd->NbEdges(); i++ ) {
TopoDS_Edge edge = sbwd->Edge ( i );
if ( ! Vf.IsSame ( sae.LastVertex ( edge ) ) ) return Standard_False;
Handle(Geom2d_Curve) c2d;
Standard_Real f, l;
if ( ! sae.PCurve ( edge, myFace, c2d, f, l ) ) continue;
// check 2d
gp_Pnt2d endPnt = c2d->Value(l);
gp_Pnt2d midPnt = c2d->Value((f+l)/2);
if ( ! IsCoincided ( endPnt, midPnt, UResolution, VResolution, tol ) ) code = -1;
// check 3d
gp_Pnt midPnt3d = myGrid->Value(midPnt.X(),midPnt.Y());
if ( ! myLoc.IsIdentity() ) midPnt3d.Transform ( myLoc.Transformation() );
if ( midPnt3d.SquareDistance(pnt) > tol2) return 0;
}
return code;
}
//=======================================================================
//function : IsSamePatch
//purpose : auxilary
//=======================================================================
static Standard_Boolean IsSamePatch (const ShapeFix_WireSegment wire,
const Standard_Integer NU,
const Standard_Integer NV,
Standard_Integer &iumin,
Standard_Integer &iumax,
Standard_Integer &ivmin,
Standard_Integer &ivmax,
const Standard_Boolean extend=Standard_False)
{
// get patch indices for current segment
Standard_Integer jumin, jumax, jvmin, jvmax;
wire.GetPatchIndex ( 1, jumin, jumax, jvmin, jvmax );
// shift to the same period
Standard_Integer du=0, dv=0;
if ( jumin - iumin > NU ) du =-( jumin - iumin ) / NU;
else if ( iumin - jumin > NU ) du = ( iumin - jumin ) / NU;
if ( jvmin - ivmin > NV ) dv =-( jvmin - ivmin ) / NV;
else if ( ivmin - jvmin > NV ) dv = ( ivmin - jvmin ) / NV;
if ( du ) { jumin += du * NU; jumax += du * NU; }
if ( dv ) { jvmin += dv * NV; jvmax += dv * NV; }
// compute common (extended) indices
Standard_Integer iun = Min ( iumin, jumin );
Standard_Integer iux = Max ( iumax, jumax );
Standard_Integer ivn = Min ( ivmin, jvmin );
Standard_Integer ivx = Max ( ivmax, jvmax );
Standard_Boolean ok = ( iun == iux || iun+1 == iux ) &&
( ivn == ivx || ivn+1 == ivx );
if ( ok && extend ) { iumin = iun; iumax = iux; ivmin = ivn; ivmax = ivx; }
return ok;
}
//=======================================================================
//function : CollectWires
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::CollectWires (ShapeFix_SequenceOfWireSegment &wires,
ShapeFix_SequenceOfWireSegment &seqw)
{
ShapeAnalysis_Edge sae;
Standard_Integer i; // svv #1
// Collect information on short closed segments
TColStd_Array1OfInteger shorts(1,seqw.Length());
for ( i=1; i <= seqw.Length(); i++ ) {
if(seqw(i).IsVertex() || seqw(i).Orientation() == TopAbs_INTERNAL) {
wires.Append(seqw(i));
seqw.Remove(i);
i--;
continue;
}
#ifdef OCCT_DEBUG
for ( Standard_Integer k=1; ! myClosedMode && k <= seqw(i).NbEdges(); k++ )
if ( ! seqw(i).CheckPatchIndex ( k ) ) {
cout << "Warning: ShapeFix_ComposeShell::CollectWires: Wrong patch indices" << endl;
break;
}
#endif
Standard_Integer isshort = IsShortSegment ( seqw(i), myFace, myGrid, myLoc,
myUResolution, myVResolution );
shorts.SetValue ( i, isshort );
if ( isshort >0 &&
( seqw(i).Orientation() == TopAbs_EXTERNAL ||
( seqw(i).NbEdges() == 1 && //:abv 13.05.02: OCC320 - remove if degenerated
BRep_Tool::Degenerated ( seqw(i).Edge(1) ) ) ) ) {
#ifdef OCCT_DEBUG
cout << "Info: ShapeFix_ComposeShell::CollectWires: Short segment ignored" << endl;
#endif
seqw(i).Orientation ( TopAbs_INTERNAL );
}
}
Handle(ShapeExtend_WireData) sbwd;
gp_Pnt2d endPnt, firstPnt;
gp_Vec2d endTan, firstTan;
TopoDS_Vertex firstV, endV;
TopoDS_Edge firstEdge, lastEdge;
Standard_Real tol = 0;
Standard_Integer iumin = 0, iumax = 0, ivmin = 0, ivmax = 0;
Standard_Real dsu=0., dsv=0.;
Standard_Boolean canBeClosed = Standard_False;
for(;;) {
Standard_Integer index = 0;
Standard_Boolean misoriented = Standard_True, samepatch = Standard_False;
Standard_Boolean reverse = Standard_False, connected = Standard_False;
Standard_Real angle = -M_PI, mindist = RealLast();
Standard_Integer weigth = 0;
Standard_Real shiftu=0., shiftv=0.;
// find next segment to connect (or first if sbwd is NULL)
for ( i = 1; i <= seqw.Length(); i++ ) {
ShapeFix_WireSegment seg = seqw.Value(i);
if(seg.IsVertex())
continue;
TopAbs_Orientation anOr = seg.Orientation();
if ( anOr == TopAbs_INTERNAL ) continue;
// for first segment, take any
if ( sbwd.IsNull() ) {
if ( shorts(i) >0 ) continue;
if ( anOr == TopAbs_EXTERNAL ) continue;
if ( anOr == TopAbs_FORWARD ) reverse = Standard_True;
index = i;
seg.GetPatchIndex ( 1, iumin, iumax, ivmin, ivmax );
misoriented = Standard_False;
dsu = dsv = 0.;
break;
}
// check whether current segment is on the same patch with previous
Standard_Integer sp = ( myClosedMode || // no indexation in closed mode
IsSamePatch ( seg, myGrid->NbUPatches(), myGrid->NbVPatches(),
iumin, iumax, ivmin, ivmax ) );
// not same patch has lowest priority
if ( ! sp && ( canBeClosed || ( index && samepatch ) ) ) continue;
// try to connect, with the following priorities:
// The name of property Weigth:
// sharing vertex auto
// samepatch = 1 16
// ! sameedge auto
// misorientation = 0 8
// connected in 2d 4
// distance 2
// short auto
// angle ->> PI 1
Handle(ShapeExtend_WireData) wire = seg.WireData();
for ( Standard_Integer j=0; j <2; j++ ) {
if ( ! endV.IsSame ( j ? seg.LastVertex() : seg.FirstVertex() ) ) continue;
// check for misorientation only if nothing better is found
Standard_Integer misor = ( anOr == ( j ? TopAbs_REVERSED : TopAbs_FORWARD ) );
// if ( misor ) continue; // temporarily, to be improved
// returning back by the same edge is lowest priority
if ( lastEdge.IsSame ( wire->Edge ( j ? wire->NbEdges() : 1 ) ) ) {
if ( ! index && ! canBeClosed ) { // || ( sp && ! samepatch ) ) {
index = i;
reverse = j;
connected = Standard_True;
misoriented = misor;
samepatch = sp;
weigth = ( sp ? 16 : 0 ) + ( connected ? 8 : 0 ) + (misor==0 ? 4 : 0);
dsu = dsv = 0.;
}
continue;
}
// compute starting tangent
gp_Pnt2d lPnt;
gp_Vec2d lVec;
Standard_Integer k;
Standard_Real edgeTol = 0;
for ( k=1; k <= wire->NbEdges(); k++ ) {
TopoDS_Shape tmpE = wire->Edge(wire->NbEdges()-k+1).Reversed();
TopoDS_Edge edge = ( j ? TopoDS::Edge ( tmpE ) :
wire->Edge(k) );
edgeTol = BRep_Tool::Tolerance ( edge );
//if ( sae.GetEndTangent2d ( edge, myFace, Standard_False, lPnt, lVec ) ) break;
if ( sae.GetEndTangent2d ( edge, myFace, Standard_False, lPnt, lVec, 1.e-3 ) ) break;
}
if ( k > wire->NbEdges() ) myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL2 );
if ( myClosedMode ) {
if ( myUClosed ) {
shiftu = ShapeAnalysis::AdjustByPeriod ( lPnt.X(), endPnt.X(), myUPeriod );
lPnt.SetX ( lPnt.X() + shiftu );
}
if ( myVClosed ) {
shiftv = ShapeAnalysis::AdjustByPeriod ( lPnt.Y(), endPnt.Y(), myVPeriod );
lPnt.SetY ( lPnt.Y() + shiftv );
}
}
// short segment is to be taken with highest priority by angle
Standard_Real ang = ( shorts(i) >0 ? M_PI : endTan.Angle ( lVec ) );
if ( myClosedMode && shorts(i) <=0 && M_PI-ang < ::Precision::Angular() )
ang = 0.; // abv 21 Mar 00: trj3_s1-md-214.stp #2471: avoid going back
// abv 05 Feb 02: face from Parasolid: use tolerance of edges for check
// for coincidence (instead of vertex tolerance) in order
// this check to be in agreement with check for position of wire segments
// thus avoiding bad effects on overlapping edges
Standard_Real ctol = Max ( edgeTol, BRep_Tool::Tolerance(lastEdge) );
Standard_Boolean conn = IsCoincided ( endPnt, lPnt, myUResolution,
myVResolution, ctol );
Standard_Real dist = endPnt.SquareDistance ( lPnt );
// check if case is better than last found
Standard_Integer w1 = ( sp ? 16 : 0 ) + ( conn ? 4 : 0 ) + (misor==0 ? 8 : 0);
Standard_Integer tail1 = ( !conn && (dist < mindist) ? 2 : 0) + (ang > angle ? 1 : 0);
Standard_Integer tail2 = ( !connected &&(dist > mindist) ? 2 : 0) + (ang < angle ? 1 : 0);
if(w1+tail1 <= weigth+tail2)
continue;
index = i;
reverse = j;
angle = ang;
mindist = dist;
connected = conn;
misoriented = misor;
samepatch = sp;
weigth = w1;
dsu = shiftu;
dsv = shiftv;
}
}
// if next segment found, connect it
if ( index ) {
if(misoriented)
myInvertEdgeStatus = Standard_True;
ShapeFix_WireSegment seg = seqw.Value(index);
if ( sbwd.IsNull() ) sbwd = new ShapeExtend_WireData;
else if ( samepatch ) { // extend patch indices
IsSamePatch ( seg, myGrid->NbUPatches(), myGrid->NbVPatches(),
iumin, iumax, ivmin, ivmax, Standard_True );
}
// TopAbs_Orientation or = seg.Orientation();
if ( ! reverse ) sbwd->Add ( seg.WireData() );
else {
Handle(ShapeExtend_WireData) wire = new ShapeExtend_WireData;
wire->ManifoldMode() = Standard_False;
wire->Add ( seg.WireData() );
wire->Reverse ( myFace );
sbwd->Add ( wire );
}
if ( seg.Orientation() == TopAbs_EXTERNAL )
seg.Orientation ( reverse ? TopAbs_REVERSED : TopAbs_FORWARD );
else seg.Orientation ( TopAbs_INTERNAL );
seqw.SetValue ( index, seg );
}
else if ( sbwd.IsNull() ) break; // stop when no free segments available
// for first segment, remember start point
if ( endV.IsNull() ) {
firstEdge = sbwd->Edge(1);
firstV = sae.FirstVertex ( firstEdge );
//sae.GetEndTangent2d ( firstEdge, myFace, Standard_False, firstPnt, firstTan );
sae.GetEndTangent2d ( firstEdge, myFace, Standard_False, firstPnt, firstTan, 1.e-3 );
}
// update last edge and vertex (only for not short segments)
Standard_Boolean doupdate = ( index && ( shorts(index) <=0 || endV.IsNull() ) );
if ( doupdate ) {
lastEdge = sbwd->Edge ( sbwd->NbEdges() );
endV = sae.LastVertex ( lastEdge );
tol = BRep_Tool::Tolerance ( endV );
// BUC60035 2053: iteration on edges is required
Standard_Integer k; // svv #1
for ( k=sbwd->NbEdges(); k >=1; k-- )
//if ( sae.GetEndTangent2d ( sbwd->Edge ( k ), myFace, Standard_True, endPnt, endTan ) )
if ( sae.GetEndTangent2d ( sbwd->Edge ( k ), myFace, Standard_True, endPnt, endTan, 1.e-3 ) )
break;
if ( k <1 ) myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL2 );
if ( myUClosed ) endPnt.SetX ( endPnt.X() + dsu );
if ( myVClosed ) endPnt.SetY ( endPnt.Y() + dsv );
}
// if closed or no next segment found, add to wires
canBeClosed = endV.IsSame ( firstV );
if ( ! index || ( canBeClosed &&
! lastEdge.IsSame ( firstEdge ) && // cylinder (seam)
IsCoincided ( endPnt, firstPnt, myUResolution, myVResolution, 2.* tol ) ) ) {
if ( ! endV.IsSame ( sae.FirstVertex ( firstEdge ) ) ) {
myStatus |= ShapeExtend::EncodeStatus ( ShapeExtend_FAIL5 );
#ifdef OCCT_DEBUG
cout << "Warning: ShapeFix_ComposeShell::CollectWires: can't close wire" << endl;
#endif
}
ShapeFix_WireSegment s ( sbwd, TopAbs_FORWARD );
s.DefineIUMin(1,iumin);
s.DefineIUMax(1,iumax);
s.DefineIVMin(1,ivmin);
s.DefineIVMax(1,ivmax);
wires.Append ( s );
sbwd.Nullify();
endV.Nullify();
canBeClosed = Standard_False;
}
}
// Check if some wires are short in 3d (lie entirely inside one vertex),
// and if yes try to merge them with others
//pdn The short seqments are stil plased in "in" sequence.
for ( i=1; i <= seqw.Length(); i++ ) {
if ( shorts(i) != 1 || seqw(i).IsVertex() || seqw(i).Orientation() == TopAbs_INTERNAL ||
seqw(i).Orientation() == TopAbs_EXTERNAL ) continue;
// find any other wire containing the same vertex
Handle(ShapeExtend_WireData) wd = seqw(i).WireData();
TopoDS_Vertex V = seqw(i).FirstVertex();
Standard_Integer minj=0, mink=0;
gp_Pnt2d p2d;
gp_Vec2d vec;
Standard_Real mindist=0;
Standard_Boolean samepatch = Standard_False;
// Standard_Integer iumin, iumax, ivmin, ivmax;
seqw(i).GetPatchIndex ( 1, iumin, iumax, ivmin, ivmax );
sae.GetEndTangent2d ( wd->Edge(1), myFace, Standard_False, p2d, vec );
for ( Standard_Integer j=1; j <= wires.Length(); j++ ) {
// if ( j == i ) continue;
// Handle(ShapeExtend_WireData)
sbwd = wires(j).WireData();
for ( Standard_Integer k=1; k <= sbwd->NbEdges(); k++ ) {
if ( !V.IsSame ( sae.FirstVertex ( sbwd->Edge(k) ) ) ) continue; //pdn I suppose that short segment should be inserted into the SAME vertex.
Standard_Integer sp = IsSamePatch ( wires(j), myGrid->NbUPatches(), myGrid->NbVPatches(),
iumin, iumax, ivmin, ivmax );
if ( samepatch && !sp) continue;
gp_Pnt2d pp;
sae.GetEndTangent2d ( sbwd->Edge(k), myFace, Standard_False, pp, vec );
Standard_Real dist = pp.SquareDistance ( p2d );
if ( sp && ! samepatch ) { minj = j; mink = k; mindist = dist;samepatch=sp;}
else
if ( ! minj || mindist > dist ) { minj = j; mink = k; mindist = dist;samepatch=sp; }
}
}
if ( ! minj ) {
//pdn add into resulting sequence!
ShapeFix_WireSegment s ( wd, TopAbs_FORWARD );
wires.Append ( s );
#ifdef OCCT_DEBUG
cout <<"Warning: Short segment processed as separate wire"<<endl;
#endif
continue;
}
// and if found, merge
// Handle(ShapeExtend_WireData)
sbwd = wires(minj).WireData();
for ( Standard_Integer n=1; n <= wd->NbEdges(); n++ )
sbwd->Add ( wd->Edge(n), mink++ );
// wires.Remove ( i );
// i--;
}
}
//=======================================================================
//function : DispatchWires
//purpose :
//=======================================================================
static gp_Pnt2d GetMiddlePoint (const ShapeFix_WireSegment wire,
const TopoDS_Face face)
{
if(wire.IsVertex()) {
TopoDS_Vertex aV = wire.GetVertex();
gp_Pnt aP3D = BRep_Tool::Pnt(aV );
Handle(Geom_Surface) surf = BRep_Tool::Surface(face);
Handle(ShapeAnalysis_Surface) aSurfTool = new ShapeAnalysis_Surface(surf);
return aSurfTool->ValueOfUV(aP3D,Precision::Confusion());
}
Bnd_Box2d box;
ShapeAnalysis_Edge sae;
ShapeAnalysis_Curve sac;
Handle(ShapeExtend_WireData) wd = wire.WireData();
for(Standard_Integer i = 1; i <= wd->NbEdges(); i++) {
TopoDS_Edge E = wd->Edge (i);
Standard_Real cf,cl;
Handle(Geom2d_Curve) c2d;
if(sae.PCurve (E,face,c2d,cf,cl,Standard_False)) {
sac.FillBndBox ( c2d, cf, cl, 3, Standard_False, box );
// box.Add(c2d->Value(cf));
// box.Add(c2d->Value(cl));
// box.Add(c2d->Value((cl+cf)/2.));
}
}
if ( box.IsVoid() ) return gp_Pnt2d(0.,0.);
Standard_Real aXmin, aYmin, aXmax, aYmax;
box.Get(aXmin, aYmin, aXmax, aYmax);
return gp_Pnt2d ( 0.5 * ( aXmax + aXmin ), 0.5 * ( aYmax + aYmin ) );
}
//=======================================================================
//function : MakeFacesOnPatch
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::MakeFacesOnPatch (TopTools_SequenceOfShape &faces,
const Handle(Geom_Surface)& surf,
TopTools_SequenceOfShape &loops) const
{
BRep_Builder B;
// Case of single loop: just add it to face
if ( loops.Length() == 1 ) {
TopoDS_Face newFace;
B.MakeFace ( newFace, surf, myLoc, ::Precision::Confusion() );
TopoDS_Shape aSH = loops.Value(1);
if( aSH.ShapeType() != TopAbs_WIRE)
return;
TopoDS_Wire wire = TopoDS::Wire ( loops.Value(1) );
B.Add ( newFace, wire );
if(myInvertEdgeStatus) {
Handle(ShapeFix_Face) sff = new ShapeFix_Face(newFace);
sff->FixAddNaturalBoundMode() = Standard_False;
TopTools_DataMapOfShapeListOfShape MapWires;
MapWires.Clear();
sff->FixOrientation(MapWires);
newFace = sff->Face();
}
faces.Append ( newFace );
return;
}
// For several loops, first find roots
// make pseudo-face,
TopoDS_Face pf;
B.MakeFace ( pf, surf, myLoc, ::Precision::Confusion() );
Handle(Geom_Surface) atSurf = BRep_Tool::Surface(pf);
Handle(ShapeAnalysis_Surface) aSurfTool = new ShapeAnalysis_Surface(atSurf);
TopTools_SequenceOfShape roots;
Standard_Integer i; // svv #1
for ( i = 1; i <= loops.Length(); i++ ) {
gp_Pnt2d unp;
TopoDS_Wire wr;
TopoDS_Shape aShape = loops(i);
if(aShape.ShapeType() != TopAbs_WIRE ||
(aShape.Orientation() != TopAbs_FORWARD && aShape.Orientation() != TopAbs_REVERSED))
continue;
wr = TopoDS::Wire ( loops(i) );
TopoDS_Iterator ew (wr);
if ( ! ew.More() ) continue;
TopoDS_Edge ed = TopoDS::Edge ( ew.Value() );
while(ed.Orientation() != TopAbs_FORWARD &&
ed.Orientation() != TopAbs_REVERSED ) {
ew.Next();
if(ew.More())
ed = TopoDS::Edge ( ew.Value() );
else
break;
}
if ( ! ew.More() ) continue;
Standard_Real cf, cl;
Handle(Geom2d_Curve) cw = BRep_Tool::CurveOnSurface ( ed, pf, cf, cl );
if ( cw.IsNull() ) continue;
unp = cw->Value ( 0.5 * ( cf + cl ) );
Standard_Integer j; // svv #1
for ( j = 1; j <= loops.Length(); j++ ) {
if ( i == j ) continue;
TopoDS_Shape aShape2 = loops(j);
if(aShape2.ShapeType() != TopAbs_WIRE ||
(aShape2.Orientation() != TopAbs_FORWARD &&
aShape2.Orientation() != TopAbs_REVERSED))
continue;
TopoDS_Wire w1 = TopoDS::Wire (aShape2);
TopoDS_Wire awtmp;
B.MakeWire(awtmp);
awtmp.Orientation(TopAbs_FORWARD);
TopoDS_Iterator aIt(w1);
Standard_Integer nbe =0;
for( ; aIt.More() ; aIt.Next()) {
if(aIt.Value().Orientation() == TopAbs_FORWARD ||
aIt.Value().Orientation() == TopAbs_REVERSED) {
B.Add(awtmp,aIt.Value());
nbe++;
}
}
if(!nbe)
continue;
TopoDS_Face fc;
B.MakeFace ( fc, surf, myLoc, ::Precision::Confusion() );
B.Add ( fc, awtmp );
BRepTopAdaptor_FClass2d clas ( fc, ::Precision::PConfusion() );
TopAbs_State stPoint = clas.Perform (unp,Standard_False);
if(stPoint == TopAbs_ON || stPoint == TopAbs_UNKNOWN) {
TopoDS_Edge ed = TopoDS::Edge ( ew.Value() );
Standard_Real cf, cl;
Handle(Geom2d_Curve) cw = BRep_Tool::CurveOnSurface ( ed, pf, cf, cl );
// handle tangential case (ON)
while ( stPoint == TopAbs_ON || stPoint == TopAbs_UNKNOWN ) {
stPoint = clas.Perform ( cw->Value(cl), Standard_False );
if ( ! ew.More() ) break;
ew.Next();
if ( ! ew.More() ) break;
TopoDS_Edge edge = TopoDS::Edge ( ew.Value() );
if(edge.Orientation() !=TopAbs_FORWARD &&
edge.Orientation() !=TopAbs_REVERSED)
continue;
Handle(Geom2d_Curve) c2d = BRep_Tool::CurveOnSurface ( edge, pf, cf, cl );
if ( ! c2d.IsNull() ) cw = c2d;
}
}
TopAbs_State stInfin = clas.PerformInfinitePoint();
if ( stPoint != stInfin ) break;
}
if ( j > loops.Length()) {
roots.Append ( wr );
// loops.Remove ( i-- );
}
}
// And remove them from the list of loops
for ( i = 1; i <= loops.Length(); i++ )
for ( Standard_Integer j = 1; j <= roots.Length(); j++ )
if ( loops(i).IsSame ( roots(j) ) ) { loops.Remove(i--); break; }
// check for lost wires, and if they are, make them roots
if ( roots.Length() <=0 && loops.Length() >0 ) {
#ifdef OCCT_DEBUG
cout << "Error: ShapeFix_ComposeShell::MakeFacesOnPatch: can't dispatch wires" << endl;
#endif
for ( Standard_Integer j=1; j <= loops.Length(); j++ ) {
roots.Append ( loops(j) );
}
loops.Clear();
}
// Then iterate on loops
for ( i=1; i <= roots.Length(); i++ ) {
Standard_Boolean reverse = Standard_False;
TopoDS_Wire wire = TopoDS::Wire ( roots(i) );
TopoDS_Face fc;
B.MakeFace ( fc, surf, myLoc, ::Precision::Confusion() );
B.Add ( fc, wire );
BRepTopAdaptor_FClass2d clas ( fc, ::Precision::PConfusion() );
if ( clas.PerformInfinitePoint() == TopAbs_IN ) {
reverse = Standard_True;
#ifdef OCCT_DEBUG
cout << "Warning: ShapeFix_ComposeShell::MakeFacesOnPatch: badly oriented wire" << endl;
#endif
}
// find all holes for that loop
TopTools_SequenceOfShape holes; // holes in holes not supported
Standard_Integer j; // svv #1
for ( j=1; j <= loops.Length(); j++ ) {
gp_Pnt2d unp;
if(loops(j).ShapeType() == TopAbs_WIRE) {
TopoDS_Wire bw = TopoDS::Wire ( loops(j) );
TopoDS_Iterator ew ( bw );
if ( ! ew.More() ) continue;
TopoDS_Edge ed = TopoDS::Edge ( ew.Value() );
Standard_Real cf, cl;
Handle(Geom2d_Curve) cw = BRep_Tool::CurveOnSurface ( ed, pf, cf, cl );
if ( cw.IsNull() ) continue;
unp = cw->Value ( 0.5 * ( cf + cl ) );
}
else if(loops(j).ShapeType() == TopAbs_VERTEX) {
TopoDS_Vertex aV = TopoDS::Vertex(loops(j));
gp_Pnt aP = BRep_Tool::Pnt(aV);
unp = aSurfTool->ValueOfUV(aP,Precision::Confusion());
}
else
continue;
TopAbs_State state = clas.Perform (unp,Standard_False);
if ( (Standard_Boolean) ( state == TopAbs_OUT ) == reverse ) {
holes.Append ( loops(j) );
loops.Remove ( j-- );
}
}
// and add them to new face (no orienting is done)
TopoDS_Face newFace;
B.MakeFace ( newFace, surf, myLoc, ::Precision::Confusion() );
B.Add ( newFace, wire );
for ( j=1; j <= holes.Length(); j++ ) {
TopoDS_Shape aSh = holes(j);
if(aSh.ShapeType() == TopAbs_VERTEX) {
TopoDS_Vertex aNewV = ShapeAnalysis_TransferParametersProj::CopyNMVertex(TopoDS::Vertex(aSh), newFace,myFace);
Context()->Replace(aSh,aNewV);
B.Add ( newFace,aNewV);
}
else
B.Add ( newFace, holes(j) );
}
faces.Append ( newFace );
// check for lost wires, and if they are, make them roots
if ( i == roots.Length() && loops.Length() >0 ) {
#ifdef OCCT_DEBUG
cout << "Error: ShapeFix_ComposeShell::MakeFacesOnPatch: can't dispatch wires" << endl;
#endif
for ( j=1; j <= loops.Length(); j++ ) {
TopoDS_Shape aSh = loops(j);
if(aSh.ShapeType() == TopAbs_WIRE && (aSh.Orientation() == TopAbs_FORWARD ||
aSh.Orientation() == TopAbs_REVERSED))
roots.Append ( loops(j) );
}
loops.Clear();
}
}
}
//=======================================================================
//function : DispatchWires
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::DispatchWires (TopTools_SequenceOfShape &faces,
ShapeFix_SequenceOfWireSegment& wires) const
{
BRep_Builder B;
// in closed mode, apply FixShifted to all wires before dispatching them
if ( myClosedMode ) {
ShapeFix_Wire sfw;
sfw.SetFace ( myFace );
sfw.SetPrecision ( Precision() );
// pdn: shift pcurves in the seam to make OK shape w/o fixshifted
Standard_Integer i;
for ( i=1; i <= wires.Length(); i++ ) {
if(wires(i).IsVertex())
continue;
Handle(ShapeExtend_WireData) sbwd = wires(i).WireData();
for(Standard_Integer jL=1; jL <= sbwd->NbEdges(); jL++ ) {
TopoDS_Edge E = sbwd->Edge(jL);
if ( E.Orientation() == TopAbs_REVERSED && BRep_Tool::IsClosed(E,myFace) ) {
Standard_Real f1,l1, f2, l2;
Handle(Geom2d_Curve) c21 = BRep_Tool::CurveOnSurface(E,myFace,f1,l1);
TopoDS_Shape dummy = E.Reversed();
Handle(Geom2d_Curve) c22 = BRep_Tool::CurveOnSurface(TopoDS::Edge(dummy),myFace,f2,l2);
Standard_Real dPreci = ::Precision::PConfusion()*Precision::PConfusion();
gp_Pnt2d pf1 = c21->Value(f1);
gp_Pnt2d pl1 = c21->Value(l1);
gp_Pnt2d pf2 = c22->Value(f2);
gp_Pnt2d pl2 = c22->Value(l2);
if ( c21 == c22 || pf1.SquareDistance(pf2) < dPreci ||
pl1.SquareDistance(pl2) < dPreci ) {
gp_Vec2d shift(0.,0.);
if ( myUClosed && Abs ( pf2.X() - pl2.X() ) < ::Precision::PConfusion() )
shift.SetX(myUPeriod);
if ( myVClosed && Abs ( pf2.Y() - pl2.Y() ) < ::Precision::PConfusion() )
shift.SetY(myVPeriod);
c22->Translate(shift);
}
}
}
}
for ( i=1; i <= wires.Length(); i++ ) {
if(wires(i).IsVertex())
continue;
Handle(ShapeExtend_WireData) sbwd = wires(i).WireData();
//: abv 30.08.01: torHalf2.sat: if wire contains single degenerated
// edge, skip that wire
if ( sbwd->NbEdges() <=0 ||
( sbwd->NbEdges() ==1 && BRep_Tool::Degenerated(sbwd->Edge(1)) ) ) {
wires.Remove(i--);
continue;
}
sfw.Load ( sbwd );
sfw.FixShifted();
// force recomputation of degenerated edges (clear pcurves)
ShapeBuild_Edge sbe;
for (Standard_Integer jL=1; jL <= sbwd->NbEdges(); jL++ ) {
if ( BRep_Tool::Degenerated(sbwd->Edge(jL)) )
sbe.RemovePCurve(sbwd->Edge(jL),myFace);
// sfw.FixDegenerated(jL);
}
sfw.FixDegenerated();
}
}
// Compute center points for wires
TColgp_SequenceOfPnt2d mPnts;
Standard_Integer nb = wires.Length();
// pdn protection on empty sequence
if(nb == 0)
return;
Standard_Integer i; //svv #1
for ( i = 1; i <= nb; i++ )
mPnts.Append ( GetMiddlePoint ( wires(i), myFace ) );
// Put each wire on its own surface patch (by reassigning pcurves)
// and build 3d curve if necessary
ShapeBuild_ReShape rs;
ShapeBuild_Edge sbe;
ShapeAnalysis_Edge sae;
Handle(ShapeFix_Edge) sfe = new ShapeFix_Edge;
Standard_Real U1,U2,V1,V2;
myGrid->Bounds(U1,U2,V1,V2);
for ( i = 1; i <= nb; i++ ) {
gp_Pnt2d pnt = mPnts(i);
Standard_Real ush =0., vsh=0.;
if(myUClosed) {
ush = ShapeAnalysis::AdjustToPeriod(pnt.X(),U1,U2);
pnt.SetX(pnt.X()+ush);
}
if(myVClosed) {
vsh = ShapeAnalysis::AdjustToPeriod(pnt.Y(),V1,V2);
pnt.SetY(pnt.Y()+vsh);
}
mPnts(i) = pnt;
Standard_Integer indU = myGrid->LocateUParameter ( pnt.X() );
Standard_Integer indV = myGrid->LocateVParameter ( pnt.Y() );
// compute parametric transformation
gp_Trsf2d T;
Standard_Real uFact=1.;
Standard_Boolean needT = myGrid->GlobalToLocalTransformation ( indU, indV, uFact, T );
if ( ush != 0. || vsh != 0. ) {
gp_Trsf2d Sh;
Sh.SetTranslation ( gp_Vec2d ( ush, vsh ) );
T.Multiply ( Sh );
needT = Standard_True;
}
if(wires(i).IsVertex())
continue;
Handle(Geom_Surface) surf = myGrid->Patch ( indU, indV );
TopoDS_Face face;
B.MakeFace ( face, surf, myLoc, ::Precision::Confusion() );
Handle(ShapeExtend_WireData) sewd = wires(i).WireData();
for ( Standard_Integer j = 1; j <= sewd->NbEdges(); j++ ) {
// Standard_Integer nsplit = ApplyContext ( sewd, j, context );
// if ( nsplit <1 ) { j--; continue; }
TopoDS_Edge edge = sewd->Edge(j);
// !! Accurately copy pcurves for SEAMS and SEAM-like edges !!
// if edge is already copied, don`t copy any more
TopoDS_Edge newEdge;
TopoDS_Edge anInitEdge = edge;
Standard_Boolean ismanifold = (edge.Orientation() == TopAbs_FORWARD ||
edge.Orientation() == TopAbs_REVERSED);
if ( rs.IsRecorded ( edge ) ) {
//smh#8
TopoDS_Shape tmpNE = rs.Value(edge);
newEdge = TopoDS::Edge ( tmpNE );
}
else {
if(!ismanifold)
anInitEdge.Orientation(TopAbs_FORWARD);
newEdge = sbe.Copy ( anInitEdge, Standard_False );
if(!ismanifold)
newEdge.Orientation(edge.Orientation());
rs.Replace ( edge, newEdge );
Context()->Replace ( edge, newEdge );
}
sbe.ReassignPCurve ( newEdge, myFace, face );
// transform pcurve to parametric space of patch
if ( needT ) {
Standard_Real f, l;
Handle(Geom2d_Curve) c2d;
if ( sae.PCurve ( newEdge, face, c2d, f, l, Standard_False ) ) {
Standard_Real newf = f, newl = l;
Handle(Geom2d_Curve) c2dnew = sbe.TransformPCurve ( c2d, T, uFact, newf, newl );
if ( BRep_Tool::IsClosed ( newEdge, face ) ) {
Standard_Real cf, cl;
Handle(Geom2d_Curve) c2d2;
//smh#8
TopoDS_Shape tmpE = newEdge.Reversed();
TopoDS_Edge e2 = TopoDS::Edge (tmpE );
if ( sae.PCurve ( e2, face, c2d2, cf, cl, Standard_False ) ) {
if ( newEdge.Orientation() == TopAbs_FORWARD )
B.UpdateEdge ( newEdge, c2dnew, c2d2, face, 0. );
else B.UpdateEdge ( newEdge, c2d2, c2dnew, face, 0. );
}
else B.UpdateEdge ( newEdge, c2dnew, face, 0. );
}
else B.UpdateEdge ( newEdge, c2dnew, face, 0. );
B.Range ( newEdge, face, newf, newl );
if ( (newf != f || newl != l) && !BRep_Tool::Degenerated(newEdge) )
B.SameRange ( newEdge, Standard_False );
}
}
if(!BRep_Tool::SameRange(newEdge)) {
TopoDS_Edge etmp;
if(!ismanifold) {
TopoDS_Edge afe = TopoDS::Edge(newEdge.Oriented(TopAbs_FORWARD));
etmp = sbe.Copy (afe , Standard_False );
}
else
etmp = sbe.Copy ( newEdge, Standard_False );
sfe->FixAddCurve3d ( etmp );
Standard_Real cf, cl;
Handle(Geom_Curve) c3d;
if(sae.Curve3d(etmp,c3d,cf,cl,Standard_False)) {
B.UpdateEdge ( newEdge, c3d, 0. );
sbe.SetRange3d ( newEdge, cf, cl );
}
}
else
sfe->FixAddCurve3d ( newEdge );
sewd->Set ( newEdge, j );
}
}
// Collect wires in packets lying on same surface and dispatch them
TColStd_Array1OfBoolean used ( 1, nb );
used.Init ( Standard_False );
for(;;) {
TopTools_SequenceOfShape loops;
Handle(Geom_Surface) Surf;
for ( i = 1; i <= nb; i++ ) {
if ( used(i) ) continue;
Handle(Geom_Surface) S = myGrid->Patch ( mPnts(i) );
if ( Surf.IsNull() ) Surf = S;
else if ( S != Surf ) continue;
used(i) = Standard_True;
ShapeFix_WireSegment aSeg = wires(i);
if(aSeg.IsVertex()) {
TopoDS_Vertex aVert = aSeg.GetVertex();
if(aVert.Orientation() == TopAbs_INTERNAL)
loops.Append(wires(i).GetVertex());
}
else {
Handle(ShapeExtend_WireData) aWD = aSeg.WireData();
if(!aWD.IsNull())
loops.Append ( aWD->Wire() );
}
}
if ( Surf.IsNull() ) break;
MakeFacesOnPatch ( faces, Surf, loops );
}
}
//======================================================================
//function : SetTransferParamTool
//purpose :
//=======================================================================
void ShapeFix_ComposeShell::SetTransferParamTool(const Handle(ShapeAnalysis_TransferParameters)& TransferParam)
{
myTransferParamTool = TransferParam;
}
//=======================================================================
//function : GetTransferParamTool
//purpose :
//=======================================================================
Handle(ShapeAnalysis_TransferParameters) ShapeFix_ComposeShell::GetTransferParamTool() const
{
return myTransferParamTool;
}
//=======================================================================
//function : ClosedMode
//purpose :
//=======================================================================
Standard_Boolean &ShapeFix_ComposeShell::ClosedMode()
{
return myClosedMode;
}