1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-03 17:56:21 +03:00
occt/src/IntPatch/IntPatch_Intersection.cxx

2289 lines
73 KiB
C++

// Created by: Modelization
// 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.
#include <stdio.h>
#include <IntPatch_Intersection.hxx>
#include <Adaptor3d_Surface.hxx>
#include <Adaptor3d_TopolTool.hxx>
#include <IntPatch_ALine.hxx>
#include <IntPatch_ALineToWLine.hxx>
#include <IntPatch_GLine.hxx>
#include <IntPatch_ImpImpIntersection.hxx>
#include <IntPatch_ImpPrmIntersection.hxx>
#include <IntPatch_PrmPrmIntersection.hxx>
#include <IntPatch_WLineTool.hxx>
#include <ProjLib_ProjectOnPlane.hxx>
#include <Geom_Plane.hxx>
#include <GeomAdaptor_Surface.hxx>
#include <ProjLib_ProjectedCurve.hxx>
#include <Geom2dInt_GInter.hxx>
#include <Geom2dAdaptor_Curve.hxx>
#include <ProjLib.hxx>
//======================================================================
// function: SequenceOfLine
//======================================================================
const IntPatch_SequenceOfLine& IntPatch_Intersection::SequenceOfLine() const
{
return (slin);
}
//======================================================================
// function: IntPatch_Intersection
//======================================================================
IntPatch_Intersection::IntPatch_Intersection()
: done(Standard_False),
empt(Standard_True),
tgte(Standard_False),
oppo(Standard_False),
myTolArc(0.0),
myTolTang(0.0),
myUVMaxStep(0.0),
myFleche(0.0),
myIsStartPnt(Standard_False),
myU1Start(0.0),
myV1Start(0.0),
myU2Start(0.0),
myV2Start(0.0)
{
}
//======================================================================
// function: IntPatch_Intersection
//======================================================================
IntPatch_Intersection::IntPatch_Intersection(const Handle(Adaptor3d_Surface)& S1,
const Handle(Adaptor3d_TopolTool)& D1,
const Handle(Adaptor3d_Surface)& S2,
const Handle(Adaptor3d_TopolTool)& D2,
const Standard_Real TolArc,
const Standard_Real TolTang)
: done(Standard_False),
empt(Standard_True),
tgte(Standard_False),
oppo(Standard_False),
myTolArc(TolArc),
myTolTang(TolTang),
myUVMaxStep(0.0),
myFleche(0.0),
myIsStartPnt(Standard_False),
myU1Start(0.0),
myV1Start(0.0),
myU2Start(0.0),
myV2Start(0.0)
{
if (myTolArc < 1e-8)
myTolArc = 1e-8;
if (myTolTang < 1e-8)
myTolTang = 1e-8;
if (myTolArc > 0.5)
myTolArc = 0.5;
if (myTolTang > 0.5)
myTolTang = 0.5;
Perform(S1, D1, S2, D2, TolArc, TolTang);
}
//======================================================================
// function: IntPatch_Intersection
//======================================================================
IntPatch_Intersection::IntPatch_Intersection(const Handle(Adaptor3d_Surface)& S1,
const Handle(Adaptor3d_TopolTool)& D1,
const Standard_Real TolArc,
const Standard_Real TolTang)
: done(Standard_False),
empt(Standard_True),
tgte(Standard_False),
oppo(Standard_False),
myTolArc(TolArc),
myTolTang(TolTang),
myUVMaxStep(0.0),
myFleche(0.0),
myIsStartPnt(Standard_False),
myU1Start(0.0),
myV1Start(0.0),
myU2Start(0.0),
myV2Start(0.0)
{
Perform(S1, D1, TolArc, TolTang);
}
//======================================================================
// function: SetTolerances
//======================================================================
void IntPatch_Intersection::SetTolerances(const Standard_Real TolArc,
const Standard_Real TolTang,
const Standard_Real UVMaxStep,
const Standard_Real Fleche)
{
myTolArc = TolArc;
myTolTang = TolTang;
myUVMaxStep = UVMaxStep;
myFleche = Fleche;
if (myTolArc < 1e-8)
myTolArc = 1e-8;
if (myTolTang < 1e-8)
myTolTang = 1e-8;
if (myTolArc > 0.5)
myTolArc = 0.5;
if (myTolTang > 0.5)
myTolTang = 0.5;
if (myFleche < 1.0e-3)
myFleche = 1e-3;
// if(myUVMaxStep<1.0e-3) myUVMaxStep=1e-3;
if (myFleche > 10)
myFleche = 10;
if (myUVMaxStep > 0.5)
myUVMaxStep = 0.5;
}
//======================================================================
// function: Perform
//======================================================================
void IntPatch_Intersection::Perform(const Handle(Adaptor3d_Surface)& S1,
const Handle(Adaptor3d_TopolTool)& D1,
const Standard_Real TolArc,
const Standard_Real TolTang)
{
myTolArc = TolArc;
myTolTang = TolTang;
if (myFleche == 0.0)
myFleche = 0.01;
if (myUVMaxStep == 0.0)
myUVMaxStep = 0.01;
done = Standard_True;
spnt.Clear();
slin.Clear();
empt = Standard_True;
tgte = Standard_False;
oppo = Standard_False;
switch (S1->GetType())
{
case GeomAbs_Plane:
case GeomAbs_Cylinder:
case GeomAbs_Sphere:
case GeomAbs_Cone:
case GeomAbs_Torus:
break;
case GeomAbs_SurfaceOfExtrusion: {
gp_Dir aDirection = S1->Direction();
gp_Ax3 anAxis(gp::Origin(), aDirection);
Handle(Adaptor3d_Curve) aBasisCurve = S1->BasisCurve();
ProjLib_ProjectOnPlane Projector(anAxis);
Projector.Load(aBasisCurve, Precision::Confusion());
Handle(GeomAdaptor_Curve) aProjCurve = Projector.GetResult();
Handle(Geom_Plane) aPlane = new Geom_Plane(anAxis);
Handle(GeomAdaptor_Surface) aGAHsurf = new GeomAdaptor_Surface(aPlane);
ProjLib_ProjectedCurve aProjectedCurve(aGAHsurf, aProjCurve);
Handle(Geom2d_Curve) aPCurve;
ProjLib::MakePCurveOfType(aProjectedCurve, aPCurve);
Geom2dAdaptor_Curve AC(aPCurve,
aProjectedCurve.FirstParameter(),
aProjectedCurve.LastParameter());
Geom2dInt_GInter Intersector(AC, Precision::Confusion(), Precision::Confusion());
if (Intersector.IsDone()
&& (Intersector.IsEmpty() || (AC.IsClosed() && Intersector.NbPoints() == 1)))
{
break;
}
}
Standard_FALLTHROUGH
default: {
IntPatch_PrmPrmIntersection interpp;
interpp.Perform(S1, D1, TolTang, TolArc, myFleche, myUVMaxStep);
if (interpp.IsDone())
{
done = Standard_True;
tgte = Standard_False;
empt = interpp.IsEmpty();
const Standard_Integer nblm = interpp.NbLines();
for (Standard_Integer i = 1; i <= nblm; i++)
slin.Append(interpp.Line(i));
}
}
break;
}
}
/////////////////////////////////////////////////////////////////////////////
// These several support functions provide methods which can help basic //
// algorithm to intersect infinite surfaces of the following types: //
// //
// a.) SurfaceOfExtrusion; //
// b.) SurfaceOfRevolution; //
// c.) OffsetSurface. //
// //
/////////////////////////////////////////////////////////////////////////////
#include <TColgp_Array1OfXYZ.hxx>
#include <TColgp_SequenceOfPnt.hxx>
#include <Extrema_ExtPS.hxx>
#include <Extrema_POnSurf.hxx>
#include <Geom2d_Curve.hxx>
#include <Geom2dAPI_InterCurveCurve.hxx>
#include <Geom_Plane.hxx>
#include <ProjLib_ProjectOnPlane.hxx>
#include <GeomProjLib.hxx>
#include <ElCLib.hxx>
#include <Geom_TrimmedCurve.hxx>
#include <Geom_Surface.hxx>
#include <Geom_OffsetSurface.hxx>
#include <Geom_RectangularTrimmedSurface.hxx>
//===============================================================
// function: FUN_GetMinMaxXYZPnt
//===============================================================
static void FUN_GetMinMaxXYZPnt(const Handle(Adaptor3d_Surface)& S, gp_Pnt& pMin, gp_Pnt& pMax)
{
const Standard_Real DU = 0.25 * Abs(S->LastUParameter() - S->FirstUParameter());
const Standard_Real DV = 0.25 * Abs(S->LastVParameter() - S->FirstVParameter());
Standard_Real tMinXYZ = RealLast();
Standard_Real tMaxXYZ = -tMinXYZ;
gp_Pnt PUV, ptMax, ptMin;
for (Standard_Real U = S->FirstUParameter(); U <= S->LastUParameter(); U += DU)
{
for (Standard_Real V = S->FirstVParameter(); V <= S->LastVParameter(); V += DV)
{
S->D0(U, V, PUV);
const Standard_Real cXYZ = PUV.XYZ().Modulus();
if (cXYZ > tMaxXYZ)
{
tMaxXYZ = cXYZ;
ptMax = PUV;
}
if (cXYZ < tMinXYZ)
{
tMinXYZ = cXYZ;
ptMin = PUV;
}
}
}
pMin = ptMin;
pMax = ptMax;
}
//==========================================================================
// function: FUN_TrimInfSurf
//==========================================================================
static void FUN_TrimInfSurf(const gp_Pnt& Pmin,
const gp_Pnt& Pmax,
const Handle(Adaptor3d_Surface)& InfSurf,
const Standard_Real& AlternativeTrimPrm,
Handle(Adaptor3d_Surface)& TrimS)
{
Standard_Real TP = AlternativeTrimPrm;
Extrema_ExtPS ext1(Pmin, *InfSurf, 1.e-7, 1.e-7);
Extrema_ExtPS ext2(Pmax, *InfSurf, 1.e-7, 1.e-7);
if (ext1.IsDone() || ext2.IsDone())
{
Standard_Real Umax = -1.e+100, Umin = 1.e+100, Vmax = -1.e+100, Vmin = 1.e+100, cU, cV;
if (ext1.IsDone())
{
for (Standard_Integer i = 1; i <= ext1.NbExt(); i++)
{
const Extrema_POnSurf& pons = ext1.Point(i);
pons.Parameter(cU, cV);
if (cU > Umax)
Umax = cU;
if (cU < Umin)
Umin = cU;
if (cV > Vmax)
Vmax = cV;
if (cV < Vmin)
Vmin = cV;
}
}
if (ext2.IsDone())
{
for (Standard_Integer i = 1; i <= ext2.NbExt(); i++)
{
const Extrema_POnSurf& pons = ext2.Point(i);
pons.Parameter(cU, cV);
if (cU > Umax)
Umax = cU;
if (cU < Umin)
Umin = cU;
if (cV > Vmax)
Vmax = cV;
if (cV < Vmin)
Vmin = cV;
}
}
TP = Max(Abs(Umin), Max(Abs(Umax), Max(Abs(Vmin), Abs(Vmax))));
}
if (TP == 0.)
{
TrimS = InfSurf;
return;
}
else
{
const Standard_Boolean Uinf = Precision::IsNegativeInfinite(InfSurf->FirstUParameter());
const Standard_Boolean Usup = Precision::IsPositiveInfinite(InfSurf->LastUParameter());
const Standard_Boolean Vinf = Precision::IsNegativeInfinite(InfSurf->FirstVParameter());
const Standard_Boolean Vsup = Precision::IsPositiveInfinite(InfSurf->LastVParameter());
Handle(Adaptor3d_Surface) TmpSS;
Standard_Integer IsTrimed = 0;
const Standard_Real tp = 1000.0 * TP;
if (Vinf && Vsup)
{
TrimS = InfSurf->VTrim(-tp, tp, 1.0e-7);
IsTrimed = 1;
}
if (Vinf && !Vsup)
{
TrimS = InfSurf->VTrim(-tp, InfSurf->LastVParameter(), 1.0e-7);
IsTrimed = 1;
}
if (!Vinf && Vsup)
{
TrimS = InfSurf->VTrim(InfSurf->FirstVParameter(), tp, 1.0e-7);
IsTrimed = 1;
}
if (IsTrimed)
{
TmpSS = TrimS;
if (Uinf && Usup)
TrimS = TmpSS->UTrim(-tp, tp, 1.0e-7);
if (Uinf && !Usup)
TrimS = TmpSS->UTrim(-tp, InfSurf->LastUParameter(), 1.0e-7);
if (!Uinf && Usup)
TrimS = TmpSS->UTrim(InfSurf->FirstUParameter(), tp, 1.0e-7);
}
else
{
if (Uinf && Usup)
TrimS = InfSurf->UTrim(-tp, tp, 1.0e-7);
if (Uinf && !Usup)
TrimS = InfSurf->UTrim(-tp, InfSurf->LastUParameter(), 1.0e-7);
if (!Uinf && Usup)
TrimS = InfSurf->UTrim(InfSurf->FirstUParameter(), tp, 1.0e-7);
}
}
}
//================================================================================
// function: FUN_GetUiso
//================================================================================
static void FUN_GetUiso(const Handle(Geom_Surface)& GS,
const GeomAbs_SurfaceType& T,
const Standard_Real& FirstV,
const Standard_Real& LastV,
const Standard_Boolean& IsVC,
const Standard_Boolean& IsVP,
const Standard_Real& U,
Handle(Geom_Curve)& I)
{
if (T != GeomAbs_OffsetSurface)
{
Handle(Geom_Curve) gc = GS->UIso(U);
if (IsVP && (FirstV == 0.0 && LastV == (2. * M_PI)))
I = gc;
else
{
Handle(Geom_TrimmedCurve) gtc = new Geom_TrimmedCurve(gc, FirstV, LastV);
// szv:I = Handle(Geom_Curve)::DownCast(gtc);
I = gtc;
}
}
else // OffsetSurface
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(GS);
const Handle(Geom_Surface) bs = gos->BasisSurface();
Handle(Geom_Curve) gcbs = bs->UIso(U);
GeomAdaptor_Curve gac(gcbs);
const GeomAbs_CurveType GACT = gac.GetType();
if (IsVP || IsVC || GACT == GeomAbs_BSplineCurve || GACT == GeomAbs_BezierCurve
|| Abs(LastV - FirstV) < 1.e+5)
{
Handle(Geom_Curve) gc = gos->UIso(U);
if (IsVP && (FirstV == 0.0 && LastV == (2 * M_PI)))
I = gc;
else
{
Handle(Geom_TrimmedCurve) gtc = new Geom_TrimmedCurve(gc, FirstV, LastV);
// szv:I = Handle(Geom_Curve)::DownCast(gtc);
I = gtc;
}
}
else // Offset Line, Parab, Hyperb
{
Standard_Real VmTr, VMTr;
if (GACT != GeomAbs_Hyperbola)
{
if (FirstV >= 0. && LastV >= 0.)
{
VmTr = FirstV;
VMTr = ((LastV - FirstV) > 1.e+4) ? (FirstV + 1.e+4) : LastV;
}
else if (FirstV < 0. && LastV < 0.)
{
VMTr = LastV;
VmTr = ((FirstV - LastV) < -1.e+4) ? (LastV - 1.e+4) : FirstV;
}
else
{
VmTr = (FirstV < -1.e+4) ? -1.e+4 : FirstV;
VMTr = (LastV > 1.e+4) ? 1.e+4 : LastV;
}
}
else // Hyperbola
{
if (FirstV >= 0. && LastV >= 0.)
{
if (FirstV > 4.)
return;
VmTr = FirstV;
VMTr = (LastV > 4.) ? 4. : LastV;
}
else if (FirstV < 0. && LastV < 0.)
{
if (LastV < -4.)
return;
VMTr = LastV;
VmTr = (FirstV < -4.) ? -4. : FirstV;
}
else
{
VmTr = (FirstV < -4.) ? -4. : FirstV;
VMTr = (LastV > 4.) ? 4. : LastV;
}
}
// Make trimmed surface
Handle(Geom_RectangularTrimmedSurface) rts =
new Geom_RectangularTrimmedSurface(gos, VmTr, VMTr, Standard_True);
I = rts->UIso(U);
}
}
}
//================================================================================
// function: FUN_GetViso
//================================================================================
static void FUN_GetViso(const Handle(Geom_Surface)& GS,
const GeomAbs_SurfaceType& T,
const Standard_Real& FirstU,
const Standard_Real& LastU,
const Standard_Boolean& IsUC,
const Standard_Boolean& IsUP,
const Standard_Real& V,
Handle(Geom_Curve)& I)
{
if (T != GeomAbs_OffsetSurface)
{
Handle(Geom_Curve) gc = GS->VIso(V);
if (IsUP && (FirstU == 0.0 && LastU == (2 * M_PI)))
I = gc;
else
{
Handle(Geom_TrimmedCurve) gtc = new Geom_TrimmedCurve(gc, FirstU, LastU);
// szv:I = Handle(Geom_Curve)::DownCast(gtc);
I = gtc;
}
}
else // OffsetSurface
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(GS);
const Handle(Geom_Surface) bs = gos->BasisSurface();
Handle(Geom_Curve) gcbs = bs->VIso(V);
GeomAdaptor_Curve gac(gcbs);
const GeomAbs_CurveType GACT = gac.GetType();
if (IsUP || IsUC || GACT == GeomAbs_BSplineCurve || GACT == GeomAbs_BezierCurve
|| Abs(LastU - FirstU) < 1.e+5)
{
Handle(Geom_Curve) gc = gos->VIso(V);
if (IsUP && (FirstU == 0.0 && LastU == (2 * M_PI)))
I = gc;
else
{
Handle(Geom_TrimmedCurve) gtc = new Geom_TrimmedCurve(gc, FirstU, LastU);
// szv:I = Handle(Geom_Curve)::DownCast(gtc);
I = gtc;
}
}
else // Offset Line, Parab, Hyperb
{
Standard_Real UmTr, UMTr;
if (GACT != GeomAbs_Hyperbola)
{
if (FirstU >= 0. && LastU >= 0.)
{
UmTr = FirstU;
UMTr = ((LastU - FirstU) > 1.e+4) ? (FirstU + 1.e+4) : LastU;
}
else if (FirstU < 0. && LastU < 0.)
{
UMTr = LastU;
UmTr = ((FirstU - LastU) < -1.e+4) ? (LastU - 1.e+4) : FirstU;
}
else
{
UmTr = (FirstU < -1.e+4) ? -1.e+4 : FirstU;
UMTr = (LastU > 1.e+4) ? 1.e+4 : LastU;
}
}
else // Hyperbola
{
if (FirstU >= 0. && LastU >= 0.)
{
if (FirstU > 4.)
return;
UmTr = FirstU;
UMTr = (LastU > 4.) ? 4. : LastU;
}
else if (FirstU < 0. && LastU < 0.)
{
if (LastU < -4.)
return;
UMTr = LastU;
UmTr = (FirstU < -4.) ? -4. : FirstU;
}
else
{
UmTr = (FirstU < -4.) ? -4. : FirstU;
UMTr = (LastU > 4.) ? 4. : LastU;
}
}
// Make trimmed surface
Handle(Geom_RectangularTrimmedSurface) rts =
new Geom_RectangularTrimmedSurface(gos, UmTr, UMTr, Standard_True);
I = rts->VIso(V);
}
}
}
//================================================================================
// function: FUN_PL_Intersection
//================================================================================
static void FUN_PL_Intersection(const Handle(Adaptor3d_Surface)& S1,
const GeomAbs_SurfaceType& T1,
const Handle(Adaptor3d_Surface)& S2,
const GeomAbs_SurfaceType& T2,
Standard_Boolean& IsOk,
TColgp_SequenceOfPnt& SP,
gp_Vec& DV)
{
IsOk = Standard_False;
// 1. Check: both surfaces have U(V)isos - lines.
DV = gp_Vec(0., 0., 1.);
Standard_Boolean isoS1isLine[2] = {0, 0};
Standard_Boolean isoS2isLine[2] = {0, 0};
Handle(Geom_Curve) C1, C2;
const GeomAdaptor_Surface& gas1 = *(GeomAdaptor_Surface*)(S1.get());
const GeomAdaptor_Surface& gas2 = *(GeomAdaptor_Surface*)(S2.get());
const Handle(Geom_Surface) gs1 = gas1.Surface();
const Handle(Geom_Surface) gs2 = gas2.Surface();
Standard_Real MS1[2], MS2[2];
MS1[0] = 0.5 * (S1->LastUParameter() + S1->FirstUParameter());
MS1[1] = 0.5 * (S1->LastVParameter() + S1->FirstVParameter());
MS2[0] = 0.5 * (S2->LastUParameter() + S2->FirstUParameter());
MS2[1] = 0.5 * (S2->LastVParameter() + S2->FirstVParameter());
if (T1 == GeomAbs_SurfaceOfExtrusion)
isoS1isLine[0] = Standard_True;
else if (!S1->IsVPeriodic() && !S1->IsVClosed())
{
if (T1 != GeomAbs_OffsetSurface)
C1 = gs1->UIso(MS1[0]);
else
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(gs1);
const Handle(Geom_Surface) bs = gos->BasisSurface();
C1 = bs->UIso(MS1[0]);
}
GeomAdaptor_Curve gac(C1);
if (gac.GetType() == GeomAbs_Line)
isoS1isLine[0] = Standard_True;
}
if (!S1->IsUPeriodic() && !S1->IsUClosed())
{
if (T1 != GeomAbs_OffsetSurface)
C1 = gs1->VIso(MS1[1]);
else
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(gs1);
const Handle(Geom_Surface) bs = gos->BasisSurface();
C1 = bs->VIso(MS1[1]);
}
GeomAdaptor_Curve gac(C1);
if (gac.GetType() == GeomAbs_Line)
isoS1isLine[1] = Standard_True;
}
if (T2 == GeomAbs_SurfaceOfExtrusion)
isoS2isLine[0] = Standard_True;
else if (!S2->IsVPeriodic() && !S2->IsVClosed())
{
if (T2 != GeomAbs_OffsetSurface)
C2 = gs2->UIso(MS2[0]);
else
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(gs2);
const Handle(Geom_Surface) bs = gos->BasisSurface();
C2 = bs->UIso(MS2[0]);
}
GeomAdaptor_Curve gac(C2);
if (gac.GetType() == GeomAbs_Line)
isoS2isLine[0] = Standard_True;
}
if (!S2->IsUPeriodic() && !S2->IsUClosed())
{
if (T2 != GeomAbs_OffsetSurface)
C2 = gs2->VIso(MS2[1]);
else
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(gs2);
const Handle(Geom_Surface) bs = gos->BasisSurface();
C2 = bs->VIso(MS2[1]);
}
GeomAdaptor_Curve gac(C2);
if (gac.GetType() == GeomAbs_Line)
isoS2isLine[1] = Standard_True;
}
Standard_Boolean IsBothLines =
((isoS1isLine[0] || isoS1isLine[1]) && (isoS2isLine[0] || isoS2isLine[1]));
if (!IsBothLines)
{
return;
}
// 2. Check: Uiso lines of both surfaces are collinear.
gp_Pnt puvS1, puvS2;
gp_Vec derS1[2], derS2[2];
S1->D1(MS1[0], MS1[1], puvS1, derS1[0], derS1[1]);
S2->D1(MS2[0], MS2[1], puvS2, derS2[0], derS2[1]);
C1.Nullify();
C2.Nullify();
Standard_Integer iso = 0;
if (isoS1isLine[0] && isoS2isLine[0] && derS1[1].IsParallel(derS2[1], Precision::Angular()))
{
iso = 1;
FUN_GetViso(gs1,
T1,
S1->FirstUParameter(),
S1->LastUParameter(),
S1->IsUClosed(),
S1->IsUPeriodic(),
MS1[1],
C1);
FUN_GetViso(gs2,
T2,
S2->FirstUParameter(),
S2->LastUParameter(),
S2->IsUClosed(),
S2->IsUPeriodic(),
MS2[1],
C2);
}
else if (isoS1isLine[0] && isoS2isLine[1] && derS1[1].IsParallel(derS2[0], Precision::Angular()))
{
iso = 1;
FUN_GetViso(gs1,
T1,
S1->FirstUParameter(),
S1->LastUParameter(),
S1->IsUClosed(),
S1->IsUPeriodic(),
MS1[1],
C1);
FUN_GetUiso(gs2,
T2,
S2->FirstVParameter(),
S2->LastVParameter(),
S2->IsVClosed(),
S2->IsVPeriodic(),
MS2[0],
C2);
}
else if (isoS1isLine[1] && isoS2isLine[0] && derS1[0].IsParallel(derS2[1], Precision::Angular()))
{
iso = 0;
FUN_GetUiso(gs1,
T1,
S1->FirstVParameter(),
S1->LastVParameter(),
S1->IsVClosed(),
S1->IsVPeriodic(),
MS1[0],
C1);
FUN_GetViso(gs2,
T2,
S2->FirstUParameter(),
S2->LastUParameter(),
S2->IsUClosed(),
S2->IsUPeriodic(),
MS2[1],
C2);
}
else if (isoS1isLine[1] && isoS2isLine[1] && derS1[0].IsParallel(derS2[0], Precision::Angular()))
{
iso = 0;
FUN_GetUiso(gs1,
T1,
S1->FirstVParameter(),
S1->LastVParameter(),
S1->IsVClosed(),
S1->IsVPeriodic(),
MS1[0],
C1);
FUN_GetUiso(gs2,
T2,
S2->FirstVParameter(),
S2->LastVParameter(),
S2->IsVClosed(),
S2->IsVPeriodic(),
MS2[0],
C2);
}
else
{
IsOk = Standard_False;
return;
}
IsOk = Standard_True;
// 3. Make intersections of V(U)isos
if (C1.IsNull() || C2.IsNull())
return;
DV = derS1[iso];
Handle(Geom_Plane) GPln = new Geom_Plane(gp_Pln(puvS1, gp_Dir(DV)));
Handle(Geom_Curve) C1Prj = GeomProjLib::ProjectOnPlane(C1, GPln, gp_Dir(DV), Standard_True);
Handle(Geom_Curve) C2Prj = GeomProjLib::ProjectOnPlane(C2, GPln, gp_Dir(DV), Standard_True);
if (C1Prj.IsNull() || C2Prj.IsNull())
return;
Handle(Geom2d_Curve) C1Prj2d = GeomProjLib::Curve2d(C1Prj, GPln);
Handle(Geom2d_Curve) C2Prj2d = GeomProjLib::Curve2d(C2Prj, GPln);
Geom2dAPI_InterCurveCurve ICC(C1Prj2d, C2Prj2d, 1.0e-7);
if (ICC.NbPoints() > 0)
{
for (Standard_Integer ip = 1; ip <= ICC.NbPoints(); ip++)
{
gp_Pnt2d P = ICC.Point(ip);
gp_Pnt P3d = ElCLib::To3d(gp_Ax2(puvS1, gp_Dir(DV)), P);
SP.Append(P3d);
}
}
}
//================================================================================
// function: FUN_NewFirstLast
//================================================================================
static void FUN_NewFirstLast(const GeomAbs_CurveType& ga_ct,
const Standard_Real& Fst,
const Standard_Real& Lst,
const Standard_Real& TrVal,
Standard_Real& NewFst,
Standard_Real& NewLst,
Standard_Boolean& NeedTr)
{
NewFst = Fst;
NewLst = Lst;
NeedTr = Standard_False;
switch (ga_ct)
{
case GeomAbs_Line:
case GeomAbs_Parabola: {
if (Abs(Lst - Fst) > TrVal)
{
if (Fst >= 0. && Lst >= 0.)
{
NewFst = Fst;
NewLst = ((Fst + TrVal) < Lst) ? (Fst + TrVal) : Lst;
}
if (Fst < 0. && Lst < 0.)
{
NewLst = Lst;
NewFst = ((Lst - TrVal) > Fst) ? (Lst - TrVal) : Fst;
}
else
{
NewFst = (Fst < -TrVal) ? -TrVal : Fst;
NewLst = (Lst > TrVal) ? TrVal : Lst;
}
NeedTr = Standard_True;
}
break;
}
case GeomAbs_Hyperbola: {
if (Abs(Lst - Fst) > 10.)
{
if (Fst >= 0. && Lst >= 0.)
{
if (Fst > 4.)
return;
NewFst = Fst;
NewLst = (Lst > 4.) ? 4. : Lst;
}
if (Fst < 0. && Lst < 0.)
{
if (Lst < -4.)
return;
NewLst = Lst;
NewFst = (Fst < -4.) ? -4. : Fst;
}
else
{
NewFst = (Fst < -4.) ? -4. : Fst;
NewLst = (Lst > 4.) ? 4. : Lst;
}
NeedTr = Standard_True;
}
break;
}
default:
break;
}
}
//================================================================================
// function: FUN_TrimBothSurf
//================================================================================
static void FUN_TrimBothSurf(const Handle(Adaptor3d_Surface)& S1,
const GeomAbs_SurfaceType& T1,
const Handle(Adaptor3d_Surface)& S2,
const GeomAbs_SurfaceType& T2,
const Standard_Real& TV,
Handle(Adaptor3d_Surface)& NS1,
Handle(Adaptor3d_Surface)& NS2)
{
const GeomAdaptor_Surface& gas1 = *(GeomAdaptor_Surface*)(S1.get());
const GeomAdaptor_Surface& gas2 = *(GeomAdaptor_Surface*)(S2.get());
const Handle(Geom_Surface) gs1 = gas1.Surface();
const Handle(Geom_Surface) gs2 = gas2.Surface();
const Standard_Real UM1 = 0.5 * (S1->LastUParameter() + S1->FirstUParameter());
const Standard_Real UM2 = 0.5 * (S2->LastUParameter() + S2->FirstUParameter());
const Standard_Real VM1 = 0.5 * (S1->LastVParameter() + S1->FirstVParameter());
const Standard_Real VM2 = 0.5 * (S2->LastVParameter() + S2->FirstVParameter());
Handle(Geom_Curve) visoS1, visoS2, uisoS1, uisoS2;
if (T1 != GeomAbs_OffsetSurface)
{
visoS1 = gs1->VIso(VM1);
uisoS1 = gs1->UIso(UM1);
}
else
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(gs1);
const Handle(Geom_Surface) bs = gos->BasisSurface();
visoS1 = bs->VIso(VM1);
uisoS1 = bs->UIso(UM1);
}
if (T2 != GeomAbs_OffsetSurface)
{
visoS2 = gs2->VIso(VM2);
uisoS2 = gs2->UIso(UM2);
}
else
{
const Handle(Geom_OffsetSurface) gos = Handle(Geom_OffsetSurface)::DownCast(gs2);
const Handle(Geom_Surface) bs = gos->BasisSurface();
visoS2 = bs->VIso(VM2);
uisoS2 = bs->UIso(UM2);
}
if (uisoS1.IsNull() || uisoS2.IsNull() || visoS1.IsNull() || visoS2.IsNull())
{
NS1 = S1;
NS2 = S2;
return;
}
GeomAdaptor_Curve gau1(uisoS1);
GeomAdaptor_Curve gav1(visoS1);
GeomAdaptor_Curve gau2(uisoS2);
GeomAdaptor_Curve gav2(visoS2);
GeomAbs_CurveType GA_U1 = gau1.GetType();
GeomAbs_CurveType GA_V1 = gav1.GetType();
GeomAbs_CurveType GA_U2 = gau2.GetType();
GeomAbs_CurveType GA_V2 = gav2.GetType();
Standard_Boolean TrmU1 = Standard_False;
Standard_Boolean TrmV1 = Standard_False;
Standard_Boolean TrmU2 = Standard_False;
Standard_Boolean TrmV2 = Standard_False;
Standard_Real V1S1, V2S1, U1S1, U2S1, V1S2, V2S2, U1S2, U2S2;
FUN_NewFirstLast(GA_U1, S1->FirstVParameter(), S1->LastVParameter(), TV, V1S1, V2S1, TrmV1);
FUN_NewFirstLast(GA_V1, S1->FirstUParameter(), S1->LastUParameter(), TV, U1S1, U2S1, TrmU1);
FUN_NewFirstLast(GA_U2, S2->FirstVParameter(), S2->LastVParameter(), TV, V1S2, V2S2, TrmV2);
FUN_NewFirstLast(GA_V2, S2->FirstUParameter(), S2->LastUParameter(), TV, U1S2, U2S2, TrmU2);
if (TrmV1)
NS1 = S1->VTrim(V1S1, V2S1, 1.0e-7);
if (TrmV2)
NS2 = S2->VTrim(V1S2, V2S2, 1.0e-7);
if (TrmU1)
{
if (TrmV1)
{
Handle(Adaptor3d_Surface) TS = NS1;
NS1 = TS->UTrim(U1S1, U2S1, 1.0e-7);
}
else
NS1 = S1->UTrim(U1S1, U2S1, 1.0e-7);
}
if (TrmU2)
{
if (TrmV2)
{
Handle(Adaptor3d_Surface) TS = NS2;
NS2 = TS->UTrim(U1S2, U2S2, 1.0e-7);
}
else
NS2 = S2->UTrim(U1S2, U2S2, 1.0e-7);
}
}
//=================================================================================================
void IntPatch_Intersection::Perform(const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
const Handle(Adaptor3d_TopolTool)& theD2,
const Standard_Real TolArc,
const Standard_Real TolTang,
const Standard_Boolean isGeomInt,
const Standard_Boolean theIsReqToKeepRLine,
const Standard_Boolean theIsReqToPostWLProc)
{
myTolArc = TolArc;
myTolTang = TolTang;
if (myFleche <= Precision::PConfusion())
myFleche = 0.01;
if (myUVMaxStep <= Precision::PConfusion())
myUVMaxStep = 0.01;
done = Standard_False;
spnt.Clear();
slin.Clear();
empt = Standard_True;
tgte = Standard_False;
oppo = Standard_False;
GeomAbs_SurfaceType typs1 = theS1->GetType();
GeomAbs_SurfaceType typs2 = theS2->GetType();
// treatment of the cases with cone or torus
Standard_Boolean TreatAsBiParametric = Standard_False;
Standard_Integer bGeomGeom = 0;
//
if (typs1 == GeomAbs_Cone || typs2 == GeomAbs_Cone || typs1 == GeomAbs_Torus
|| typs2 == GeomAbs_Torus)
{
gp_Ax1 aCTAx, aGeomAx;
GeomAbs_SurfaceType aCTType;
Standard_Boolean bToCheck;
//
const Handle(Adaptor3d_Surface)& aCTSurf =
(typs1 == GeomAbs_Cone || typs1 == GeomAbs_Torus) ? theS1 : theS2;
const Handle(Adaptor3d_Surface)& aGeomSurf =
(typs1 == GeomAbs_Cone || typs1 == GeomAbs_Torus) ? theS2 : theS1;
//
aCTType = aCTSurf->GetType();
bToCheck = Standard_False;
//
if (typs1 == GeomAbs_Cone || typs2 == GeomAbs_Cone)
{
const gp_Cone aCon1 = (aCTType == GeomAbs_Cone) ? aCTSurf->Cone() : aGeomSurf->Cone();
Standard_Real a1 = Abs(aCon1.SemiAngle());
bToCheck = (a1 < 0.02) || (a1 > 1.55);
//
if (typs1 == typs2)
{
const gp_Cone aCon2 = aGeomSurf->Cone();
Standard_Real a2 = Abs(aCon2.SemiAngle());
bToCheck = bToCheck || (a2 < 0.02) || (a2 > 1.55);
//
if (a1 > 1.55 && a2 > 1.55)
{ // quasi-planes: if same domain, treat as canonic
const gp_Ax1 A1 = aCon1.Axis(), A2 = aCon2.Axis();
if (A1.IsParallel(A2, Precision::Angular()))
{
const gp_Pnt Apex1 = aCon1.Apex(), Apex2 = aCon2.Apex();
const gp_Pln Plan1(Apex1, A1.Direction());
if (Plan1.Distance(Apex2) <= Precision::Confusion())
{
bToCheck = Standard_False;
}
}
}
}
//
TreatAsBiParametric = bToCheck;
if (aCTType == GeomAbs_Cone)
{
aCTAx = aCon1.Axis();
}
}
//
if (typs1 == GeomAbs_Torus || typs2 == GeomAbs_Torus)
{
const gp_Torus aTor1 = (aCTType == GeomAbs_Torus) ? aCTSurf->Torus() : aGeomSurf->Torus();
bToCheck = aTor1.MajorRadius() > aTor1.MinorRadius();
if (typs1 == typs2)
{
const gp_Torus aTor2 = aGeomSurf->Torus();
bToCheck = aTor2.MajorRadius() > aTor2.MinorRadius();
}
//
if (aCTType == GeomAbs_Torus)
{
aCTAx = aTor1.Axis();
}
}
//
if (bToCheck)
{
const gp_Lin aL1(aCTAx);
//
switch (aGeomSurf->GetType())
{
case GeomAbs_Plane: {
aGeomAx = aGeomSurf->Plane().Axis();
if (aCTType == GeomAbs_Cone)
{
bGeomGeom = 1;
if (Abs(aCTSurf->Cone().SemiAngle()) < 0.02)
{
Standard_Real ps = Abs(aCTAx.Direction().Dot(aGeomAx.Direction()));
if (ps < 0.015)
{
bGeomGeom = 0;
}
}
}
else
{
if (aCTAx.IsParallel(aGeomAx, Precision::Angular())
|| (aCTAx.IsNormal(aGeomAx, Precision::Angular())
&& (aGeomSurf->Plane().Distance(aCTAx.Location()) < Precision::Confusion())))
{
bGeomGeom = 1;
}
}
bToCheck = Standard_False;
break;
}
case GeomAbs_Sphere: {
if (aL1.Distance(aGeomSurf->Sphere().Location()) < Precision::Confusion())
{
bGeomGeom = 1;
}
bToCheck = Standard_False;
break;
}
case GeomAbs_Cylinder:
aGeomAx = aGeomSurf->Cylinder().Axis();
break;
case GeomAbs_Cone:
aGeomAx = aGeomSurf->Cone().Axis();
break;
case GeomAbs_Torus:
aGeomAx = aGeomSurf->Torus().Axis();
break;
default:
bToCheck = Standard_False;
break;
}
//
if (bToCheck)
{
if (aCTAx.IsParallel(aGeomAx, Precision::Angular())
&& (aL1.Distance(aGeomAx.Location()) <= Precision::Confusion()))
{
bGeomGeom = 1;
}
}
//
if (bGeomGeom == 1)
{
TreatAsBiParametric = Standard_False;
}
}
}
//
if (theD1->DomainIsInfinite() || theD2->DomainIsInfinite())
{
TreatAsBiParametric = Standard_False;
}
// Modified by skv - Mon Sep 26 14:58:30 2005 Begin
// if(TreatAsBiParametric) { typs1 = typs2 = GeomAbs_BezierSurface; }
if (TreatAsBiParametric)
{
if (typs1 == GeomAbs_Cone && typs2 == GeomAbs_Plane)
typs1 = GeomAbs_BezierSurface; // Using Imp-Prm Intersector
else if (typs1 == GeomAbs_Plane && typs2 == GeomAbs_Cone)
typs2 = GeomAbs_BezierSurface; // Using Imp-Prm Intersector
else
{
// Using Prm-Prm Intersector
typs1 = GeomAbs_BezierSurface;
typs2 = GeomAbs_BezierSurface;
}
}
// Modified by skv - Mon Sep 26 14:58:30 2005 End
// Surface type definition
Standard_Integer ts1 = 0;
switch (typs1)
{
case GeomAbs_Plane:
case GeomAbs_Cylinder:
case GeomAbs_Sphere:
case GeomAbs_Cone:
ts1 = 1;
break;
case GeomAbs_Torus:
ts1 = bGeomGeom;
break;
default:
break;
}
Standard_Integer ts2 = 0;
switch (typs2)
{
case GeomAbs_Plane:
case GeomAbs_Cylinder:
case GeomAbs_Sphere:
case GeomAbs_Cone:
ts2 = 1;
break;
case GeomAbs_Torus:
ts2 = bGeomGeom;
break;
default:
break;
}
//
// treatment of the cases with torus and any other geom surface
//
// Possible intersection types: 1. ts1 == ts2 == 1 <Geom-Geom>
// 2. ts1 != ts2 <Geom-Param>
// 3. ts1 == ts2 == 0 <Param-Param>
// Geom - Geom
if (ts1 == ts2 && ts1 == 1)
{
IntSurf_ListOfPntOn2S ListOfPnts;
ListOfPnts.Clear();
if (isGeomInt)
{
GeomGeomPerfom(theS1,
theD1,
theS2,
theD2,
TolArc,
TolTang,
ListOfPnts,
typs1,
typs2,
theIsReqToKeepRLine);
}
else
{
ParamParamPerfom(theS1, theD1, theS2, theD2, TolArc, TolTang, ListOfPnts, typs1, typs2);
}
}
// Geom - Param
if (ts1 != ts2)
{
GeomParamPerfom(theS1, theD1, theS2, theD2, ts1 == 0, typs1, typs2);
}
// Param - Param
if (ts1 == ts2 && ts1 == 0)
{
IntSurf_ListOfPntOn2S ListOfPnts;
ListOfPnts.Clear();
ParamParamPerfom(theS1, theD1, theS2, theD2, TolArc, TolTang, ListOfPnts, typs1, typs2);
}
if (!theIsReqToPostWLProc)
return;
for (Standard_Integer i = slin.Lower(); i <= slin.Upper(); i++)
{
Handle(IntPatch_WLine) aWL = Handle(IntPatch_WLine)::DownCast(slin.Value(i));
if (aWL.IsNull())
continue;
if (!aWL->IsPurgingAllowed())
continue;
Handle(IntPatch_WLine) aRW =
IntPatch_WLineTool::ComputePurgedWLine(aWL, theS1, theS2, theD1, theD2);
if (aRW.IsNull())
continue;
slin.InsertAfter(i, aRW);
slin.Remove(i);
}
}
//=================================================================================================
void IntPatch_Intersection::Perform(const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
const Handle(Adaptor3d_TopolTool)& theD2,
const Standard_Real TolArc,
const Standard_Real TolTang,
IntSurf_ListOfPntOn2S& ListOfPnts,
const Standard_Boolean isGeomInt,
const Standard_Boolean theIsReqToKeepRLine,
const Standard_Boolean theIsReqToPostWLProc)
{
myTolArc = TolArc;
myTolTang = TolTang;
if (myFleche <= Precision::PConfusion())
myFleche = 0.01;
if (myUVMaxStep <= Precision::PConfusion())
myUVMaxStep = 0.01;
done = Standard_False;
spnt.Clear();
slin.Clear();
empt = Standard_True;
tgte = Standard_False;
oppo = Standard_False;
GeomAbs_SurfaceType typs1 = theS1->GetType();
GeomAbs_SurfaceType typs2 = theS2->GetType();
//
// treatment of the cases with cone or torus
Standard_Boolean TreatAsBiParametric = Standard_False;
Standard_Integer bGeomGeom = 0;
//
if (typs1 == GeomAbs_Cone || typs2 == GeomAbs_Cone || typs1 == GeomAbs_Torus
|| typs2 == GeomAbs_Torus)
{
gp_Ax1 aCTAx, aGeomAx;
GeomAbs_SurfaceType aCTType;
Standard_Boolean bToCheck;
//
const Handle(Adaptor3d_Surface)& aCTSurf =
(typs1 == GeomAbs_Cone || typs1 == GeomAbs_Torus) ? theS1 : theS2;
const Handle(Adaptor3d_Surface)& aGeomSurf =
(typs1 == GeomAbs_Cone || typs1 == GeomAbs_Torus) ? theS2 : theS1;
//
aCTType = aCTSurf->GetType();
bToCheck = Standard_False;
//
if (typs1 == GeomAbs_Cone || typs2 == GeomAbs_Cone)
{
const gp_Cone aCon1 = (aCTType == GeomAbs_Cone) ? aCTSurf->Cone() : aGeomSurf->Cone();
Standard_Real a1 = Abs(aCon1.SemiAngle());
bToCheck = (a1 < 0.02) || (a1 > 1.55);
//
if (typs1 == typs2)
{
const gp_Cone aCon2 = aGeomSurf->Cone();
Standard_Real a2 = Abs(aCon2.SemiAngle());
bToCheck = bToCheck || (a2 < 0.02) || (a2 > 1.55);
//
if (a1 > 1.55 && a2 > 1.55)
{ // quasi-planes: if same domain, treat as canonic
const gp_Ax1 A1 = aCon1.Axis(), A2 = aCon2.Axis();
if (A1.IsParallel(A2, Precision::Angular()))
{
const gp_Pnt Apex1 = aCon1.Apex(), Apex2 = aCon2.Apex();
const gp_Pln Plan1(Apex1, A1.Direction());
if (Plan1.Distance(Apex2) <= Precision::Confusion())
{
bToCheck = Standard_False;
}
}
}
}
//
TreatAsBiParametric = bToCheck;
if (aCTType == GeomAbs_Cone)
{
aCTAx = aCon1.Axis();
}
}
//
if (typs1 == GeomAbs_Torus || typs2 == GeomAbs_Torus)
{
const gp_Torus aTor1 = (aCTType == GeomAbs_Torus) ? aCTSurf->Torus() : aGeomSurf->Torus();
bToCheck = aTor1.MajorRadius() > aTor1.MinorRadius();
if (typs1 == typs2)
{
const gp_Torus aTor2 = aGeomSurf->Torus();
bToCheck = (bToCheck && (aTor2.MajorRadius() > aTor2.MinorRadius()))
|| (Abs(aTor1.MajorRadius() - aTor2.MajorRadius()) < TolTang
&& Abs(aTor1.MinorRadius() - aTor2.MinorRadius()) < TolTang);
}
//
if (aCTType == GeomAbs_Torus)
{
aCTAx = aTor1.Axis();
}
}
//
if (bToCheck)
{
const gp_Lin aL1(aCTAx);
//
switch (aGeomSurf->GetType())
{
case GeomAbs_Plane: {
aGeomAx = aGeomSurf->Plane().Axis();
if (aCTType == GeomAbs_Cone)
{
bGeomGeom = 1;
if (Abs(aCTSurf->Cone().SemiAngle()) < 0.02)
{
Standard_Real ps = Abs(aCTAx.Direction().Dot(aGeomAx.Direction()));
if (ps < 0.015)
{
bGeomGeom = 0;
}
}
}
else
{
if (aCTAx.IsParallel(aGeomAx, Precision::Angular())
|| (aCTAx.IsNormal(aGeomAx, Precision::Angular())
&& (aGeomSurf->Plane().Distance(aCTAx.Location()) < Precision::Confusion())))
{
bGeomGeom = 1;
}
}
bToCheck = Standard_False;
break;
}
case GeomAbs_Sphere: {
if (aL1.Distance(aGeomSurf->Sphere().Location()) < Precision::Confusion())
{
bGeomGeom = 1;
}
bToCheck = Standard_False;
break;
}
case GeomAbs_Cylinder:
aGeomAx = aGeomSurf->Cylinder().Axis();
break;
case GeomAbs_Cone:
aGeomAx = aGeomSurf->Cone().Axis();
break;
case GeomAbs_Torus:
aGeomAx = aGeomSurf->Torus().Axis();
break;
default:
bToCheck = Standard_False;
break;
}
//
if (bToCheck)
{
if (aCTAx.IsParallel(aGeomAx, Precision::Angular())
&& (aL1.Distance(aGeomAx.Location()) <= Precision::Confusion()))
{
bGeomGeom = 1;
}
}
//
if (bGeomGeom == 1)
{
TreatAsBiParametric = Standard_False;
}
}
}
//
if (theD1->DomainIsInfinite() || theD2->DomainIsInfinite())
{
TreatAsBiParametric = Standard_False;
}
if (TreatAsBiParametric)
{
// Using Prm-Prm Intersector
typs1 = GeomAbs_BezierSurface;
typs2 = GeomAbs_BezierSurface;
}
// Surface type definition
Standard_Integer ts1 = 0;
switch (typs1)
{
case GeomAbs_Plane:
case GeomAbs_Cylinder:
case GeomAbs_Sphere:
case GeomAbs_Cone:
ts1 = 1;
break;
case GeomAbs_Torus:
ts1 = bGeomGeom;
break;
default:
break;
}
Standard_Integer ts2 = 0;
switch (typs2)
{
case GeomAbs_Plane:
case GeomAbs_Cylinder:
case GeomAbs_Sphere:
case GeomAbs_Cone:
ts2 = 1;
break;
case GeomAbs_Torus:
ts2 = bGeomGeom;
break;
default:
break;
}
//
// Possible intersection types: 1. ts1 == ts2 == 1 <Geom-Geom>
// 2. ts1 != ts2 <Geom-Param>
// 3. ts1 == ts2 == 0 <Param-Param>
if (!isGeomInt)
{
ParamParamPerfom(theS1, theD1, theS2, theD2, TolArc, TolTang, ListOfPnts, typs1, typs2);
}
else if (ts1 != ts2)
{
GeomParamPerfom(theS1, theD1, theS2, theD2, ts1 == 0, typs1, typs2);
}
else if (ts1 == 0)
{
ParamParamPerfom(theS1, theD1, theS2, theD2, TolArc, TolTang, ListOfPnts, typs1, typs2);
}
else if (ts1 == 1)
{
GeomGeomPerfom(theS1,
theD1,
theS2,
theD2,
TolArc,
TolTang,
ListOfPnts,
typs1,
typs2,
theIsReqToKeepRLine);
}
if (!theIsReqToPostWLProc)
return;
for (Standard_Integer i = slin.Lower(); i <= slin.Upper(); i++)
{
Handle(IntPatch_WLine) aWL = Handle(IntPatch_WLine)::DownCast(slin.Value(i));
if (aWL.IsNull())
continue;
if (!aWL->IsPurgingAllowed())
continue;
Handle(IntPatch_WLine) aRW =
IntPatch_WLineTool::ComputePurgedWLine(aWL, theS1, theS2, theD1, theD2);
if (aRW.IsNull())
continue;
slin.InsertAfter(i, aRW);
slin.Remove(i);
}
}
//=================================================================================================
void IntPatch_Intersection::ParamParamPerfom(const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
const Handle(Adaptor3d_TopolTool)& theD2,
const Standard_Real TolArc,
const Standard_Real TolTang,
IntSurf_ListOfPntOn2S& ListOfPnts,
const GeomAbs_SurfaceType typs1,
const GeomAbs_SurfaceType typs2)
{
IntPatch_PrmPrmIntersection interpp;
//
if (!theD1->DomainIsInfinite() && !theD2->DomainIsInfinite())
{
Standard_Boolean ClearFlag = Standard_True;
if (!ListOfPnts.IsEmpty())
{
interpp
.Perform(theS1, theD1, theS2, theD2, TolTang, TolArc, myFleche, myUVMaxStep, ListOfPnts);
ClearFlag = Standard_False;
}
interpp.Perform(theS1, theD1, theS2, theD2, TolTang, TolArc, myFleche, myUVMaxStep, ClearFlag);
}
else if ((theD1->DomainIsInfinite()) ^ (theD2->DomainIsInfinite()))
{
gp_Pnt pMaxXYZ, pMinXYZ;
if (theD1->DomainIsInfinite())
{
FUN_GetMinMaxXYZPnt(theS2, pMinXYZ, pMaxXYZ);
const Standard_Real MU = Max(Abs(theS2->FirstUParameter()), Abs(theS2->LastUParameter()));
const Standard_Real MV = Max(Abs(theS2->FirstVParameter()), Abs(theS2->LastVParameter()));
const Standard_Real AP = Max(MU, MV);
Handle(Adaptor3d_Surface) SS;
FUN_TrimInfSurf(pMinXYZ, pMaxXYZ, theS1, AP, SS);
interpp.Perform(SS, theD1, theS2, theD2, TolTang, TolArc, myFleche, myUVMaxStep);
}
else
{
FUN_GetMinMaxXYZPnt(theS1, pMinXYZ, pMaxXYZ);
const Standard_Real MU = Max(Abs(theS1->FirstUParameter()), Abs(theS1->LastUParameter()));
const Standard_Real MV = Max(Abs(theS1->FirstVParameter()), Abs(theS1->LastVParameter()));
const Standard_Real AP = Max(MU, MV);
Handle(Adaptor3d_Surface) SS;
FUN_TrimInfSurf(pMinXYZ, pMaxXYZ, theS2, AP, SS);
interpp.Perform(theS1, theD1, SS, theD2, TolTang, TolArc, myFleche, myUVMaxStep);
}
} //(theD1->DomainIsInfinite()) ^ (theD2->DomainIsInfinite())
else
{
if (typs1 == GeomAbs_OtherSurface || typs2 == GeomAbs_OtherSurface)
{
done = Standard_False;
return;
}
Standard_Boolean IsPLInt = Standard_False;
TColgp_SequenceOfPnt sop;
gp_Vec v;
FUN_PL_Intersection(theS1, typs1, theS2, typs2, IsPLInt, sop, v);
if (IsPLInt)
{
if (sop.Length() > 0)
{
for (Standard_Integer ip = 1; ip <= sop.Length(); ip++)
{
gp_Lin lin(sop.Value(ip), gp_Dir(v));
Handle(IntPatch_Line) gl = new IntPatch_GLine(lin, Standard_False);
slin.Append(gl);
}
done = Standard_True;
}
else
done = Standard_False;
return;
} // 'COLLINEAR LINES'
else
{
Handle(Adaptor3d_Surface) nS1 = theS1;
Handle(Adaptor3d_Surface) nS2 = theS2;
FUN_TrimBothSurf(theS1, typs1, theS2, typs2, 1.e+8, nS1, nS2);
interpp.Perform(nS1, theD1, nS2, theD2, TolTang, TolArc, myFleche, myUVMaxStep);
} // 'NON - COLLINEAR LINES'
} // both domains are infinite
if (interpp.IsDone())
{
done = Standard_True;
tgte = Standard_False;
empt = interpp.IsEmpty();
for (Standard_Integer i = 1; i <= interpp.NbLines(); i++)
{
if (interpp.Line(i)->ArcType() != IntPatch_Walking)
slin.Append(interpp.Line(i));
}
for (Standard_Integer i = 1; i <= interpp.NbLines(); i++)
{
if (interpp.Line(i)->ArcType() == IntPatch_Walking)
slin.Append(interpp.Line(i));
}
}
}
//=======================================================================
////function : GeomGeomPerfom
// purpose :
//=======================================================================
void IntPatch_Intersection::GeomGeomPerfom(const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
const Handle(Adaptor3d_TopolTool)& theD2,
const Standard_Real TolArc,
const Standard_Real TolTang,
IntSurf_ListOfPntOn2S& ListOfPnts,
const GeomAbs_SurfaceType theTyps1,
const GeomAbs_SurfaceType theTyps2,
const Standard_Boolean theIsReqToKeepRLine)
{
IntPatch_ImpImpIntersection
interii(theS1, theD1, theS2, theD2, myTolArc, myTolTang, theIsReqToKeepRLine);
if (!interii.IsDone())
{
done = Standard_False;
ParamParamPerfom(theS1, theD1, theS2, theD2, TolArc, TolTang, ListOfPnts, theTyps1, theTyps2);
return;
}
done = (interii.GetStatus() == IntPatch_ImpImpIntersection::IntStatus_OK);
empt = interii.IsEmpty();
if (empt)
{
return;
}
const Standard_Integer aNbPointsInALine = 200;
tgte = interii.TangentFaces();
if (tgte)
oppo = interii.OppositeFaces();
Standard_Boolean isWLExist = Standard_False;
IntPatch_ALineToWLine AToW(theS1, theS2, aNbPointsInALine);
for (Standard_Integer i = 1; i <= interii.NbLines(); i++)
{
const Handle(IntPatch_Line)& line = interii.Line(i);
if (line->ArcType() == IntPatch_Analytic)
{
isWLExist = Standard_True;
AToW.MakeWLine(Handle(IntPatch_ALine)::DownCast(line), slin);
}
else
{
if (line->ArcType() == IntPatch_Walking)
{
Handle(IntPatch_WLine)::DownCast(line)->EnablePurging(Standard_False);
}
if ((line->ArcType() != IntPatch_Restriction) || theIsReqToKeepRLine)
slin.Append(line);
}
}
for (Standard_Integer i = 1; i <= interii.NbPnts(); i++)
{
spnt.Append(interii.Point(i));
}
if ((theTyps1 == GeomAbs_Cylinder) && (theTyps2 == GeomAbs_Cylinder))
{
IntPatch_WLineTool::JoinWLines(slin, spnt, theS1, theS2, TolTang);
}
if (isWLExist)
{
Bnd_Box2d aBx1, aBx2;
const Standard_Real aU1F = theS1->FirstUParameter(), aU1L = theS1->LastUParameter(),
aV1F = theS1->FirstVParameter(), aV1L = theS1->LastVParameter(),
aU2F = theS2->FirstUParameter(), aU2L = theS2->LastUParameter(),
aV2F = theS2->FirstVParameter(), aV2L = theS2->LastVParameter();
aBx1.Add(gp_Pnt2d(aU1F, aV1F));
aBx1.Add(gp_Pnt2d(aU1L, aV1F));
aBx1.Add(gp_Pnt2d(aU1L, aV1L));
aBx1.Add(gp_Pnt2d(aU1F, aV1L));
aBx2.Add(gp_Pnt2d(aU2F, aV2F));
aBx2.Add(gp_Pnt2d(aU2L, aV2F));
aBx2.Add(gp_Pnt2d(aU2L, aV2L));
aBx2.Add(gp_Pnt2d(aU2F, aV2L));
aBx1.Enlarge(Precision::PConfusion());
aBx2.Enlarge(Precision::PConfusion());
const Standard_Real anArrOfPeriod[4] = {theS1->IsUPeriodic() ? theS1->UPeriod() : 0.0,
theS1->IsVPeriodic() ? theS1->VPeriod() : 0.0,
theS2->IsUPeriodic() ? theS2->UPeriod() : 0.0,
theS2->IsVPeriodic() ? theS2->VPeriod() : 0.0};
NCollection_List<gp_Pnt> aListOfCriticalPoints;
if (theS1->GetType() == GeomAbs_Cone)
{
aListOfCriticalPoints.Append(theS1->Cone().Apex());
}
else if (theS1->GetType() == GeomAbs_Sphere)
{
aListOfCriticalPoints.Append(theS1->Value(0.0, M_PI_2));
aListOfCriticalPoints.Append(theS1->Value(0.0, -M_PI_2));
}
if (theS2->GetType() == GeomAbs_Cone)
{
aListOfCriticalPoints.Append(theS2->Cone().Apex());
}
else if (theS2->GetType() == GeomAbs_Sphere)
{
aListOfCriticalPoints.Append(theS2->Value(0.0, M_PI_2));
aListOfCriticalPoints.Append(theS2->Value(0.0, -M_PI_2));
}
IntPatch_WLineTool::ExtendTwoWLines(slin,
theS1,
theS2,
TolTang,
anArrOfPeriod,
aBx1,
aBx2,
aListOfCriticalPoints);
}
}
//=================================================================================================
void IntPatch_Intersection::GeomParamPerfom(const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
const Handle(Adaptor3d_TopolTool)& theD2,
const Standard_Boolean isNotAnalitical,
const GeomAbs_SurfaceType typs1,
const GeomAbs_SurfaceType typs2)
{
IntPatch_ImpPrmIntersection interip;
if (myIsStartPnt)
{
if (isNotAnalitical /*ts1 == 0*/)
interip.SetStartPoint(myU1Start, myV1Start);
else
interip.SetStartPoint(myU2Start, myV2Start);
}
if (theD1->DomainIsInfinite() && theD2->DomainIsInfinite())
{
Standard_Boolean IsPLInt = Standard_False;
TColgp_SequenceOfPnt sop;
gp_Vec v;
FUN_PL_Intersection(theS1, typs1, theS2, typs2, IsPLInt, sop, v);
if (IsPLInt)
{
if (sop.Length() > 0)
{
for (Standard_Integer ip = 1; ip <= sop.Length(); ip++)
{
gp_Lin lin(sop.Value(ip), gp_Dir(v));
Handle(IntPatch_Line) gl = new IntPatch_GLine(lin, Standard_False);
slin.Append(gl);
}
done = Standard_True;
}
else
done = Standard_False;
return;
}
else
{
Handle(Adaptor3d_Surface) nS1 = theS1;
Handle(Adaptor3d_Surface) nS2 = theS2;
FUN_TrimBothSurf(theS1, typs1, theS2, typs2, 1.e+5, nS1, nS2);
interip.Perform(nS1, theD1, nS2, theD2, myTolArc, myTolTang, myFleche, myUVMaxStep);
}
}
else
interip.Perform(theS1, theD1, theS2, theD2, myTolArc, myTolTang, myFleche, myUVMaxStep);
if (interip.IsDone())
{
done = Standard_True;
empt = interip.IsEmpty();
if (!empt)
{
const Standard_Integer aNbLines = interip.NbLines();
for (Standard_Integer i = 1; i <= aNbLines; i++)
{
if (interip.Line(i)->ArcType() != IntPatch_Walking)
slin.Append(interip.Line(i));
}
for (Standard_Integer i = 1; i <= aNbLines; i++)
{
if (interip.Line(i)->ArcType() == IntPatch_Walking)
slin.Append(interip.Line(i));
}
for (Standard_Integer i = 1; i <= interip.NbPnts(); i++)
spnt.Append(interip.Point(i));
}
}
}
void IntPatch_Intersection::Perform(const Handle(Adaptor3d_Surface)& S1,
const Handle(Adaptor3d_TopolTool)& D1,
const Handle(Adaptor3d_Surface)& S2,
const Handle(Adaptor3d_TopolTool)& D2,
const Standard_Real U1,
const Standard_Real V1,
const Standard_Real U2,
const Standard_Real V2,
const Standard_Real TolArc,
const Standard_Real TolTang)
{
myTolArc = TolArc;
myTolTang = TolTang;
if (myFleche == 0.0)
{
#ifdef OCCT_DEBUG
// std::cout<<" -- IntPatch_Intersection::myFleche fixe par defaut a 0.01 --"<<std::endl;
// std::cout<<" -- Utiliser la Methode SetTolerances( ... ) "<<std::endl;
#endif
myFleche = 0.01;
}
if (myUVMaxStep == 0.0)
{
#ifdef OCCT_DEBUG
// std::cout<<" -- IntPatch_Intersection::myUVMaxStep fixe par defaut a 0.01 --"<<std::endl;
// std::cout<<" -- Utiliser la Methode SetTolerances( ... ) "<<std::endl;
#endif
myUVMaxStep = 0.01;
}
done = Standard_False;
spnt.Clear();
slin.Clear();
empt = Standard_True;
tgte = Standard_False;
oppo = Standard_False;
const GeomAbs_SurfaceType typs1 = S1->GetType();
const GeomAbs_SurfaceType typs2 = S2->GetType();
if (typs1 == GeomAbs_Plane || typs1 == GeomAbs_Cylinder || typs1 == GeomAbs_Sphere
|| typs1 == GeomAbs_Cone || typs2 == GeomAbs_Plane || typs2 == GeomAbs_Cylinder
|| typs2 == GeomAbs_Sphere || typs2 == GeomAbs_Cone)
{
myIsStartPnt = Standard_True;
myU1Start = U1;
myV1Start = V1;
myU2Start = U2;
myV2Start = V2;
Perform(S1, D1, S2, D2, TolArc, TolTang);
myIsStartPnt = Standard_False;
}
else
{
IntPatch_PrmPrmIntersection interpp;
interpp.Perform(S1, D1, S2, D2, U1, V1, U2, V2, TolTang, TolArc, myFleche, myUVMaxStep);
if (interpp.IsDone())
{
done = Standard_True;
tgte = Standard_False;
empt = interpp.IsEmpty();
const Standard_Integer nblm = interpp.NbLines();
Standard_Integer i = 1;
for (; i <= nblm; i++)
slin.Append(interpp.Line(i));
}
}
for (Standard_Integer i = slin.Lower(); i <= slin.Upper(); i++)
{
Handle(IntPatch_WLine) aWL = Handle(IntPatch_WLine)::DownCast(slin.Value(i));
if (aWL.IsNull())
continue;
if (!aWL->IsPurgingAllowed())
continue;
Handle(IntPatch_WLine) aRW = IntPatch_WLineTool::ComputePurgedWLine(aWL, S1, S2, D1, D2);
if (aRW.IsNull())
continue;
slin.InsertAfter(i, aRW);
slin.Remove(i);
}
}
#ifdef DUMPOFIntPatch_Intersection
void IntPatch_Intersection__MAJ_R(Handle(Adaptor2d_Curve2d)* R1,
Handle(Adaptor2d_Curve2d)*,
int* NR1,
int*,
Standard_Integer nbR1,
Standard_Integer,
const IntPatch_Point& VTX)
{
if (VTX.IsOnDomS1())
{
//-- long unsigned ptr= *((long unsigned *)(((Handle(Standard_Transient)
//*)(&(VTX.ArcOnS1())))));
for (Standard_Integer i = 0; i < nbR1; i++)
{
if (VTX.ArcOnS1() == R1[i])
{
NR1[i]++;
printf("\n ******************************");
return;
}
}
printf("\n R Pas trouvee (IntPatch)\n");
}
}
#endif
void IntPatch_Intersection::Dump(const Standard_Integer /*Mode*/,
const Handle(Adaptor3d_Surface)& /*S1*/,
const Handle(Adaptor3d_TopolTool)& /*D1*/,
const Handle(Adaptor3d_Surface)& /*S2*/,
const Handle(Adaptor3d_TopolTool)& /*D2*/) const
{
#ifdef DUMPOFIntPatch_Intersection
const int MAXR = 200;
//-- ----------------------------------------------------------------------
//-- construction de la liste des restrictions & vertex
//--
int NR1[MAXR], NR2[MAXR];
Handle(Adaptor2d_Curve2d) R1[MAXR], R2[MAXR];
Standard_Integer nbR1 = 0, nbR2 = 0;
for (D1->Init(); D1->More() && nbR1 < MAXR; D1->Next())
{
R1[nbR1] = D1->Value();
NR1[nbR1] = 0;
nbR1++;
}
for (D2->Init(); D2->More() && nbR2 < MAXR; D2->Next())
{
R2[nbR2] = D2->Value();
NR2[nbR2] = 0;
nbR2++;
}
printf("\nDUMP_INT: ----empt:%2ud tgte:%2ud oppo:%2ud ---------------------------------",
empt,
tgte,
empt);
Standard_Integer i, nbr1, nbr2, nbgl, nbgc, nbge, nbgp, nbgh, nbl, nbr, nbg, nbw, nba;
nbl = nbr = nbg = nbw = nba = nbgl = nbge = nbr1 = nbr2 = nbgc = nbgp = nbgh = 0;
nbl = NbLines();
for (i = 1; i <= nbl; i++)
{
const Handle(IntPatch_Line)& line = Line(i);
const IntPatch_IType IType = line->ArcType();
if (IType == IntPatch_Walking)
nbw++;
else if (IType == IntPatch_Restriction)
{
nbr++;
Handle(IntPatch_RLine) rlin(Handle(IntPatch_RLine)::DownCast(line));
if (rlin->IsArcOnS1())
nbr1++;
if (rlin->IsArcOnS2())
nbr2++;
}
else if (IType == IntPatch_Analytic)
nba++;
else
{
nbg++;
if (IType == IntPatch_Lin)
nbgl++;
else if (IType == IntPatch_Circle)
nbgc++;
else if (IType == IntPatch_Parabola)
nbgp++;
else if (IType == IntPatch_Hyperbola)
nbgh++;
else if (IType == IntPatch_Ellipse)
nbge++;
}
}
printf("\nDUMP_INT:Lines:%2d Wlin:%2d Restr:%2d(On1:%2d On2:%2d) Ana:%2d Geom:%2d(L:%2d C:%2d "
"E:%2d H:%2d P:%2d)",
nbl,
nbw,
nbr,
nbr1,
nbr2,
nba,
nbg,
nbgl,
nbgc,
nbge,
nbgh,
nbgp);
IntPatch_LineConstructor LineConstructor(2);
Standard_Integer nbllc = 0;
nbw = nbr = nbg = nba = 0;
Standard_Integer nbva, nbvw, nbvr, nbvg;
nbva = nbvr = nbvw = nbvg = 0;
for (j = 1; j <= nbl; j++)
{
Standard_Integer v, nbvtx;
const Handle(IntPatch_Line)& intersLinej = Line(j);
Standard_Integer NbLines;
LineConstructor.Perform(SequenceOfLine(), intersLinej, S1, D1, S2, D2, 1e-7);
NbLines = LineConstructor.NbLines();
for (Standard_Integer k = 1; k <= NbLines; k++)
{
nbllc++;
const Handle(IntPatch_Line)& LineK = LineConstructor.Line(k);
if (LineK->ArcType() == IntPatch_Analytic)
{
Handle(IntPatch_ALine) alin(Handle(IntPatch_ALine)::DownCast(LineK));
nbvtx = alin->NbVertex();
nbva += nbvtx;
nba++;
for (v = 1; v <= nbvtx; v++)
{
IntPatch_Intersection__MAJ_R(R1, R2, NR1, NR2, nbR1, nbR2, alin->Vertex(v));
}
}
else if (LineK->ArcType() == IntPatch_Restriction)
{
Handle(IntPatch_RLine) rlin(Handle(IntPatch_RLine)::DownCast(LineK));
nbvtx = rlin->NbVertex();
nbvr += nbvtx;
nbr++;
for (v = 1; v <= nbvtx; v++)
{
IntPatch_Intersection__MAJ_R(R1, R2, NR1, NR2, nbR1, nbR2, rlin->Vertex(v));
}
}
else if (LineK->ArcType() == IntPatch_Walking)
{
Handle(IntPatch_WLine) wlin(Handle(IntPatch_WLine)::DownCast(LineK));
nbvtx = wlin->NbVertex();
nbvw += nbvtx;
nbw++;
for (v = 1; v <= nbvtx; v++)
{
IntPatch_Intersection__MAJ_R(R1, R2, NR1, NR2, nbR1, nbR2, wlin->Vertex(v));
}
}
else
{
Handle(IntPatch_GLine) glin(Handle(IntPatch_GLine)::DownCast(LineK));
nbvtx = glin->NbVertex();
nbvg += nbvtx;
nbg++;
for (v = 1; v <= nbvtx; v++)
{
IntPatch_Intersection__MAJ_R(R1, R2, NR1, NR2, nbR1, nbR2, glin->Vertex(v));
}
}
}
}
printf("\nDUMP_LC :Lines:%2d WLin:%2d Restr:%2d Ana:%2d Geom:%2d", nbllc, nbw, nbr, nba, nbg);
printf("\nDUMP_LC :vtx :%2d r:%2d :%2d :%2d", nbvw, nbvr, nbva, nbvg);
printf("\n");
#endif
}
//=================================================================================================
Standard_Boolean IntPatch_Intersection::CheckSingularPoints(
const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
Standard_Real& theDist)
{
theDist = Precision::Infinite();
Standard_Boolean isSingular = Standard_False;
if (theS1 == theS2)
{
return isSingular;
}
//
const Standard_Integer aNbBndPnts = 5;
constexpr Standard_Real aTol = Precision::Confusion();
Standard_Integer i;
theD1->Init();
Standard_Boolean isU = Standard_True;
for (; theD1->More(); theD1->Next())
{
Handle(Adaptor2d_Curve2d) aBnd = theD1->Value();
Standard_Real pinf = aBnd->FirstParameter(), psup = aBnd->LastParameter();
if (Precision::IsNegativeInfinite(pinf) || Precision::IsPositiveInfinite(psup))
{
continue;
}
Standard_Real t, dt = (psup - pinf) / (aNbBndPnts - 1);
gp_Pnt2d aP1;
gp_Vec2d aDir;
aBnd->D1((pinf + psup) / 2., aP1, aDir);
if (Abs(aDir.X()) > Abs(aDir.Y()))
isU = Standard_True;
else
isU = Standard_False;
gp_Pnt aPP1;
gp_Vec aDU, aDV;
Standard_Real aD1NormMax = 0.;
gp_XYZ aPmid(0., 0., 0.);
Standard_Integer aNb = 0;
for (t = pinf; t <= psup; t += dt)
{
aP1 = aBnd->Value(t);
theS1->D1(aP1.X(), aP1.Y(), aPP1, aDU, aDV);
if (isU)
aD1NormMax = Max(aD1NormMax, aDU.Magnitude());
else
aD1NormMax = Max(aD1NormMax, aDV.Magnitude());
aPmid += aPP1.XYZ();
aNb++;
if (aD1NormMax > aTol)
break;
}
if (aD1NormMax <= aTol)
{
// Singular point aPP1;
aPmid /= aNb;
aPP1.SetXYZ(aPmid);
constexpr Standard_Real aTolU = Precision::PConfusion(), aTolV = Precision::PConfusion();
Extrema_ExtPS aProj(aPP1, *theS2.get(), aTolU, aTolV, Extrema_ExtFlag_MIN);
if (aProj.IsDone())
{
Standard_Integer aNbExt = aProj.NbExt();
for (i = 1; i <= aNbExt; ++i)
{
theDist = Min(theDist, aProj.SquareDistance(i));
}
}
}
}
if (!Precision::IsInfinite(theDist))
{
theDist = Sqrt(theDist);
isSingular = Standard_True;
}
return isSingular;
}
//=================================================================================================
Standard_Real IntPatch_Intersection::DefineUVMaxStep(const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
const Handle(Adaptor3d_TopolTool)& theD2)
{
Standard_Real anUVMaxStep = 0.001;
Standard_Real aDistToSing1 = Precision::Infinite();
Standard_Real aDistToSing2 = Precision::Infinite();
constexpr Standard_Real aTolMin = Precision::Confusion(), aTolMax = 1.e-5;
if (theS1 != theS2)
{
Standard_Boolean isSing1 = CheckSingularPoints(theS1, theD1, theS2, aDistToSing1);
if (isSing1)
{
if (aDistToSing1 > aTolMin && aDistToSing1 < aTolMax)
{
anUVMaxStep = 0.0001;
}
else
{
isSing1 = Standard_False;
}
}
if (!isSing1)
{
Standard_Boolean isSing2 = CheckSingularPoints(theS2, theD2, theS1, aDistToSing2);
if (isSing2)
{
if (aDistToSing2 > aTolMin && aDistToSing2 < aTolMax)
{
anUVMaxStep = 0.0001;
}
}
}
}
return anUVMaxStep;
}
//=======================================================================
// function : splitCone
// purpose : Splits cone by the apex
//=======================================================================
static void splitCone(const Handle(Adaptor3d_Surface)& theS,
const Handle(Adaptor3d_TopolTool)& theD,
const Standard_Real theTol,
NCollection_Vector<Handle(Adaptor3d_Surface)>& theVecHS)
{
if (theS->GetType() != GeomAbs_Cone)
{
throw Standard_NoSuchObject("IntPatch_Intersection : Surface is not Cone");
}
gp_Cone aCone = theS->Cone();
Standard_Real aU0, aV0;
Adaptor3d_TopolTool::GetConeApexParam(aCone, aU0, aV0);
TopAbs_State aState = theD->Classify(gp_Pnt2d(aU0, aV0), theTol);
if (aState == TopAbs_IN || aState == TopAbs_ON)
{
const Handle(Adaptor3d_Surface) aHSDn =
theS->VTrim(theS->FirstVParameter(), aV0, Precision::PConfusion());
const Handle(Adaptor3d_Surface) aHSUp =
theS->VTrim(aV0, theS->LastVParameter(), Precision::PConfusion());
theVecHS.Append(aHSDn);
theVecHS.Append(aHSUp);
}
else
{
theVecHS.Append(theS);
}
}
//=======================================================================
// function : PrepareSurfaces
// purpose : Prepares surfaces for intersection
//=======================================================================
void IntPatch_Intersection::PrepareSurfaces(
const Handle(Adaptor3d_Surface)& theS1,
const Handle(Adaptor3d_TopolTool)& theD1,
const Handle(Adaptor3d_Surface)& theS2,
const Handle(Adaptor3d_TopolTool)& theD2,
const Standard_Real theTol,
NCollection_Vector<Handle(Adaptor3d_Surface)>& theVecHS1,
NCollection_Vector<Handle(Adaptor3d_Surface)>& theVecHS2)
{
if ((theS1->GetType() == GeomAbs_Cone)
&& (Abs(M_PI / 2. - Abs(theS1->Cone().SemiAngle())) < theTol))
{
splitCone(theS1, theD1, theTol, theVecHS1);
}
else
{
theVecHS1.Append(theS1);
}
if ((theS2->GetType() == GeomAbs_Cone)
&& (Abs(M_PI / 2. - Abs(theS2->Cone().SemiAngle())) < theTol))
{
splitCone(theS2, theD2, theTol, theVecHS2);
}
else
{
theVecHS2.Append(theS2);
}
}