mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-05 18:16:23 +03:00
1322 lines
42 KiB
C++
Executable File
1322 lines
42 KiB
C++
Executable File
// Created on: 1996-04-22
|
|
// Created by: Herve LOUESSARD
|
|
// Copyright (c) 1996-1999 Matra Datavision
|
|
// Copyright (c) 1999-2012 OPEN CASCADE SAS
|
|
//
|
|
// The content of this file is subject to the Open CASCADE Technology Public
|
|
// License Version 6.5 (the "License"). You may not use the content of this file
|
|
// except in compliance with the License. Please obtain a copy of the License
|
|
// at http://www.opencascade.org and read it completely before using this file.
|
|
//
|
|
// The Initial Developer of the Original Code is Open CASCADE S.A.S., having its
|
|
// main offices at: 1, place des Freres Montgolfier, 78280 Guyancourt, France.
|
|
//
|
|
// The Original Code and all software distributed under the License is
|
|
// distributed on an "AS IS" basis, without warranty of any kind, and the
|
|
// Initial Developer hereby disclaims all such warranties, including without
|
|
// limitation, any warranties of merchantability, fitness for a particular
|
|
// purpose or non-infringement. Please see the License for the specific terms
|
|
// and conditions governing the rights and limitations under the License.
|
|
|
|
|
|
// Modified: Sergey ZERCHANINOV
|
|
|
|
#include <BRepExtrema_DistanceSS.hxx>
|
|
|
|
#include <TopTools_IndexedMapOfShape.hxx>
|
|
#include <TopExp.hxx>
|
|
#include <gp_Pnt.hxx>
|
|
#include <gp_Pnt2d.hxx>
|
|
#include <BRep_Tool.hxx>
|
|
#include <BRepExtrema_SupportType.hxx>
|
|
#include <Standard_Real.hxx>
|
|
#include <BRepExtrema_SolutionElem.hxx>
|
|
#include <BRepExtrema_SeqOfSolution.hxx>
|
|
#include <Standard_Boolean.hxx>
|
|
#include <Standard_Integer.hxx>
|
|
#include <TopAbs_ShapeEnum.hxx>
|
|
#include <TopoDS.hxx>
|
|
#include <Bnd_Box.hxx>
|
|
#include <BRepExtrema_ExtPC.hxx>
|
|
#include <BRepExtrema_ExtPF.hxx>
|
|
#include <Extrema_ExtFlag.hxx>
|
|
#include <BRepExtrema_ExtCC.hxx>
|
|
#include <BRepExtrema_ExtCF.hxx>
|
|
#include <BRepExtrema_ExtFF.hxx>
|
|
#include <BRepClass_FaceClassifier.hxx>
|
|
#include <TopAbs.hxx>
|
|
#include <Geom_Curve.hxx>
|
|
#include <GeomAbs_Shape.hxx>
|
|
#include <GeomAdaptor_Curve.hxx>
|
|
#include <TColStd_Array1OfReal.hxx>
|
|
#include <BRepBuilderAPI_MakeVertex.hxx>
|
|
#include <BRepBuilderAPI_MakeEdge.hxx>
|
|
#include <BRepBuilderAPI_MakeFace.hxx>
|
|
#include <Geom_Surface.hxx>
|
|
#include <GeomAPI_ProjectPointOnSurf.hxx>
|
|
#include <GeomAPI_ProjectPointOnCurve.hxx>
|
|
#include <Geom_RectangularTrimmedSurface.hxx>
|
|
#include <Geom_TrimmedCurve.hxx>
|
|
#include <BRepBndLib.hxx>
|
|
#include <BRepTools.hxx>
|
|
#include <TColgp_Array1OfPnt.hxx>
|
|
|
|
|
|
/*********************************************************************************/
|
|
|
|
//------------------------------------------------------------------------------
|
|
// function: TRI_SOLUTION
|
|
//------------------------------------------------------------------------------
|
|
static Standard_Boolean TRI_SOLUTION (const BRepExtrema_SeqOfSolution& SeqSol, const gp_Pnt& Pt)
|
|
{
|
|
const Standard_Integer Nbsol = SeqSol.Length();
|
|
for (Standard_Integer i = 1; i <= Nbsol; i++)
|
|
{
|
|
const Standard_Real dst = SeqSol.Value(i).Point().Distance(Pt);
|
|
if (dst <= Precision::Confusion()) return Standard_False;
|
|
}
|
|
return Standard_True;
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// function: MIN_SOLUTION
|
|
//------------------------------------------------------------------------------
|
|
static void MIN_SOLUTION (const BRepExtrema_SeqOfSolution& SeqSol1,
|
|
const BRepExtrema_SeqOfSolution& SeqSol2,
|
|
const Standard_Real DstRef,
|
|
const Standard_Real Eps,
|
|
BRepExtrema_SeqOfSolution& seqSol1,
|
|
BRepExtrema_SeqOfSolution& seqSol2)
|
|
{
|
|
const Standard_Integer nbSol = SeqSol1.Length();
|
|
for (Standard_Integer i = 1; i <= nbSol; i++)
|
|
{
|
|
const Standard_Real dst1 = SeqSol1.Value(i).Dist();
|
|
if (fabs(dst1 - DstRef) < Eps)
|
|
{
|
|
seqSol1.Append(SeqSol1.Value(i));
|
|
seqSol2.Append(SeqSol2.Value(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// function: TRIM_INFINIT_EDGE
|
|
//------------------------------------------------------------------------------
|
|
static void TRIM_INFINIT_EDGE(const TopoDS_Edge& S1, const TopoDS_Edge& S2, TopoDS_Edge& aResEdge,
|
|
Standard_Boolean& bIsTrim1, Standard_Boolean& bIsTrim2)
|
|
{
|
|
if ( BRep_Tool::Degenerated(S1) || BRep_Tool::Degenerated(S2) )
|
|
return;
|
|
|
|
aResEdge = S2;
|
|
Standard_Real aFirst1, aLast1, aFirst2, aLast2;
|
|
Handle(Geom_Curve) pCurv1 = BRep_Tool::Curve(S1, aFirst1, aLast1);
|
|
Handle(Geom_Curve) pCurv2 = BRep_Tool::Curve(S2, aFirst2, aLast2);
|
|
|
|
if (Precision::IsInfinite(aFirst1) &&
|
|
Precision::IsInfinite(aLast1) &&
|
|
Precision::IsInfinite(aFirst2) &&
|
|
Precision::IsInfinite(aLast2))
|
|
return;
|
|
|
|
Standard_Real Umin, Umax;
|
|
Standard_Boolean bUmin, bUmax;
|
|
bUmin = bUmax = Standard_False;
|
|
|
|
Handle(Geom_Curve) pCurv;
|
|
if ( !pCurv1.IsNull() && (Precision::IsInfinite(aFirst1) || Precision::IsInfinite(aLast1)) )
|
|
{
|
|
pCurv = pCurv1;
|
|
bIsTrim1 = Standard_True;
|
|
if (!Precision::IsInfinite(aFirst1))
|
|
{
|
|
bUmin = Standard_True;
|
|
Umin = aFirst1;
|
|
}
|
|
else if (!Precision::IsInfinite(aLast1))
|
|
{
|
|
bUmax = Standard_True;
|
|
Umax = aLast1;
|
|
}
|
|
}
|
|
else if ( !pCurv2.IsNull() && (Precision::IsInfinite(aFirst2) || Precision::IsInfinite(aLast2)) )
|
|
{
|
|
pCurv = pCurv2;
|
|
bIsTrim2 = Standard_True;
|
|
if (!Precision::IsInfinite(aFirst2))
|
|
{
|
|
bUmin = Standard_True;
|
|
Umin = aFirst2;
|
|
}
|
|
else if (!Precision::IsInfinite(aLast2))
|
|
{
|
|
bUmax = Standard_True;
|
|
Umax = aLast2;
|
|
}
|
|
}
|
|
if (bIsTrim1 || bIsTrim2)
|
|
{
|
|
Bnd_Box aEdgeBox;
|
|
if (bIsTrim1)
|
|
BRepBndLib::Add(S2, aEdgeBox);
|
|
if (bIsTrim2)
|
|
BRepBndLib::Add(S1, aEdgeBox);
|
|
Standard_Real Xmin, Ymin, Zmin, Xmax, Ymax, Zmax;
|
|
aEdgeBox.Get(Xmin, Ymin, Zmin, Xmax, Ymax, Zmax);
|
|
|
|
const gp_Pnt aPnt0(Xmin, Ymin, Zmin);
|
|
const gp_Pnt aPnt1(Xmin, Ymax, Zmin);
|
|
const gp_Pnt aPnt2(Xmin, Ymax, Zmax);
|
|
const gp_Pnt aPnt3(Xmin, Ymin, Zmax);
|
|
const gp_Pnt aPnt4(Xmax, Ymax, Zmin);
|
|
const gp_Pnt aPnt5(Xmax, Ymax, Zmax);
|
|
const gp_Pnt aPnt6(Xmax, Ymin, Zmax);
|
|
const gp_Pnt aPnt7(Xmax, Ymin, Zmin);
|
|
|
|
Standard_Real arrU[8];
|
|
GeomAPI_ProjectPointOnCurve aProj(aPnt0, pCurv);
|
|
/*szv:aProj.Perform(aPnt0);*/ arrU[0] = aProj.LowerDistanceParameter();
|
|
aProj.Perform(aPnt1); arrU[1] = aProj.LowerDistanceParameter();
|
|
aProj.Perform(aPnt2); arrU[2] = aProj.LowerDistanceParameter();
|
|
aProj.Perform(aPnt3); arrU[3] = aProj.LowerDistanceParameter();
|
|
aProj.Perform(aPnt4); arrU[4] = aProj.LowerDistanceParameter();
|
|
aProj.Perform(aPnt5); arrU[5] = aProj.LowerDistanceParameter();
|
|
aProj.Perform(aPnt6); arrU[6] = aProj.LowerDistanceParameter();
|
|
aProj.Perform(aPnt7); arrU[7] = aProj.LowerDistanceParameter();
|
|
|
|
if (!bUmin)
|
|
Umin = arrU[0];
|
|
|
|
if (!bUmax)
|
|
Umax = arrU[0];
|
|
|
|
Standard_Integer i = 0;
|
|
while (i < 8)
|
|
{
|
|
const Standard_Real aU = arrU[i++];
|
|
if (aU < Umin)
|
|
Umin = aU;
|
|
else if (aU > Umax)
|
|
Umax = aU;
|
|
}
|
|
|
|
Standard_Real tol = Precision::Confusion();
|
|
if (bIsTrim1)
|
|
tol = BRep_Tool::Tolerance(S1);
|
|
else if (bIsTrim2)
|
|
tol = BRep_Tool::Tolerance(S2);
|
|
|
|
const Standard_Real EpsU = GeomAdaptor_Curve(pCurv).Resolution(3.*tol);
|
|
if (fabs(Umin - Umax) < EpsU)
|
|
{
|
|
Umin -= EpsU;
|
|
Umax += EpsU;
|
|
}
|
|
|
|
Handle(Geom_Curve) result = new Geom_TrimmedCurve(pCurv, Umin, Umax);
|
|
aResEdge = BRepBuilderAPI_MakeEdge(result);
|
|
}
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// function: TRIM_INFINIT_FACE
|
|
//------------------------------------------------------------------------------
|
|
static void TRIM_INFINIT_FACE(const TopoDS_Shape& S1, const TopoDS_Shape& S2,
|
|
TopoDS_Face& aResFace, Standard_Boolean& bIsInfinit)
|
|
{
|
|
bIsInfinit = Standard_False;
|
|
|
|
TopAbs_ShapeEnum Type1 = S1.ShapeType();
|
|
TopAbs_ShapeEnum Type2 = S2.ShapeType();
|
|
|
|
TopoDS_Edge aE;
|
|
TopoDS_Face aF;
|
|
|
|
if (Type1 == TopAbs_EDGE && Type2 == TopAbs_FACE)
|
|
{
|
|
aE = TopoDS::Edge(S1);
|
|
if ( BRep_Tool::Degenerated(aE) )
|
|
return;
|
|
aF = TopoDS::Face(S2);
|
|
}
|
|
else if (Type2 == TopAbs_EDGE && Type1 == TopAbs_FACE)
|
|
{
|
|
aE = TopoDS::Edge(S2);
|
|
if ( BRep_Tool::Degenerated(aE) )
|
|
return;
|
|
aF = TopoDS::Face(S1);
|
|
}
|
|
else
|
|
{
|
|
bIsInfinit = Standard_False;
|
|
return;
|
|
}
|
|
|
|
aResFace = aF;
|
|
Handle(Geom_Surface) pSurf = BRep_Tool::Surface(aF);
|
|
|
|
const Standard_Boolean bRestrict = BRep_Tool::NaturalRestriction(aF);
|
|
|
|
Standard_Real U1, V1, U2, V2;
|
|
Standard_Real Umin, Umax, Vmin, Vmax;
|
|
Standard_Boolean bUmin, bUmax, bVmin, bVmax;
|
|
bUmin = bUmax = bVmin = bVmax = Standard_False;
|
|
Standard_Boolean bIsTrim = Standard_False;
|
|
|
|
if (bRestrict)
|
|
{
|
|
pSurf->Bounds(U1, U2, V1, V2);
|
|
if (Precision::IsInfinite(U1))
|
|
bIsTrim = Standard_True;
|
|
else
|
|
{
|
|
Umin = U1;
|
|
bUmin = Standard_True;
|
|
}
|
|
|
|
if (Precision::IsInfinite(U2))
|
|
bIsTrim = Standard_True;
|
|
else
|
|
{
|
|
Umax = U2;
|
|
bUmax = Standard_True;
|
|
}
|
|
|
|
if (Precision::IsInfinite(V1))
|
|
bIsTrim = Standard_True;
|
|
else
|
|
{
|
|
Vmin = V1;
|
|
bVmin = Standard_True;
|
|
}
|
|
|
|
if (Precision::IsInfinite(V2))
|
|
bIsTrim = Standard_True;
|
|
else
|
|
{
|
|
Vmax = V2;
|
|
bVmax = Standard_True;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
BRepTools::UVBounds(aF, U1, U2, V1, V2);
|
|
if (Precision::IsInfinite(U1) &&
|
|
Precision::IsInfinite(U2) &&
|
|
Precision::IsInfinite(V1) &&
|
|
Precision::IsInfinite(V2))
|
|
bIsTrim = Standard_True;
|
|
}
|
|
|
|
if (bIsTrim)
|
|
{
|
|
Bnd_Box aEdgeBox;
|
|
BRepBndLib::Add(aE, aEdgeBox);
|
|
|
|
if(aEdgeBox.IsWhole())
|
|
return;
|
|
|
|
Standard_Real Xmin, Ymin, Zmin, Xmax, Ymax, Zmax;
|
|
aEdgeBox.Get(Xmin, Ymin, Zmin, Xmax, Ymax, Zmax);
|
|
|
|
const gp_Pnt aPnt0(Xmin, Ymin, Zmin);
|
|
const gp_Pnt aPnt1(Xmin, Ymax, Zmin);
|
|
const gp_Pnt aPnt2(Xmin, Ymax, Zmax);
|
|
const gp_Pnt aPnt3(Xmin, Ymin, Zmax);
|
|
const gp_Pnt aPnt4(Xmax, Ymax, Zmin);
|
|
const gp_Pnt aPnt5(Xmax, Ymax, Zmax);
|
|
const gp_Pnt aPnt6(Xmax, Ymin, Zmax);
|
|
const gp_Pnt aPnt7(Xmax, Ymin, Zmin);
|
|
|
|
Standard_Real arrU[8], arrV[8];
|
|
GeomAPI_ProjectPointOnSurf aProj(aPnt0, pSurf);
|
|
/*szv:aProj.Perform(aPnt0);*/ if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[0],arrV[0]);
|
|
aProj.Perform(aPnt1); if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[1],arrV[1]);
|
|
aProj.Perform(aPnt2); if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[2],arrV[2]);
|
|
aProj.Perform(aPnt3); if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[3],arrV[3]);
|
|
aProj.Perform(aPnt4); if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[4],arrV[4]);
|
|
aProj.Perform(aPnt5); if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[5],arrV[5]);
|
|
aProj.Perform(aPnt6); if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[6],arrV[6]);
|
|
aProj.Perform(aPnt7); if (aProj.IsDone()) aProj.LowerDistanceParameters(arrU[7],arrV[7]);
|
|
|
|
if (!bUmin)
|
|
Umin = arrU[0];
|
|
if (!bUmax)
|
|
Umax = arrU[0];
|
|
if (!bVmin)
|
|
Vmin = arrV[0];
|
|
if (!bVmax)
|
|
Vmax = arrV[0];
|
|
|
|
Standard_Integer i = 0;
|
|
while (i < 8)
|
|
{
|
|
const Standard_Real aU = arrU[i];
|
|
if (aU < Umin)
|
|
Umin = aU;
|
|
else if (aU > Umax)
|
|
Umax = aU;
|
|
|
|
const Standard_Real aV = arrV[i];
|
|
if (aV < Vmin)
|
|
Vmin = aV;
|
|
else if (aV > Vmax)
|
|
Vmax = aV;
|
|
|
|
i++;
|
|
}
|
|
|
|
GeomAdaptor_Surface aGAS(pSurf);
|
|
const Standard_Real tol = BRep_Tool::Tolerance(aF);
|
|
|
|
const Standard_Real EpsU = aGAS.UResolution(3.*tol);
|
|
if (fabs(Umin - Umax) < EpsU)
|
|
{
|
|
Umin -= EpsU;
|
|
Umax += EpsU;
|
|
}
|
|
|
|
const Standard_Real EpsV = aGAS.VResolution(3.*tol);
|
|
if (fabs(Vmin - Vmax) < EpsV)
|
|
{
|
|
Vmin -= EpsV;
|
|
Vmax += EpsV;
|
|
}
|
|
|
|
Handle(Geom_Surface) result = new Geom_RectangularTrimmedSurface(pSurf, Umin, Umax, Vmin, Vmax);
|
|
aResFace = BRepBuilderAPI_MakeFace(result, Precision::Confusion());
|
|
|
|
bIsInfinit = Standard_True;
|
|
}
|
|
else
|
|
bIsInfinit = Standard_False;
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// static function: PERFORM_C0
|
|
//------------------------------------------------------------------------------
|
|
static void PERFORM_C0(const TopoDS_Edge &S1, const TopoDS_Edge &S2,
|
|
BRepExtrema_SeqOfSolution& SeqSol1,
|
|
BRepExtrema_SeqOfSolution& SeqSol2,
|
|
const Standard_Real DstRef,
|
|
Standard_Real& mDstRef,
|
|
const Standard_Real Eps)
|
|
{
|
|
if ( BRep_Tool::Degenerated(S1) || BRep_Tool::Degenerated(S2) )
|
|
return;
|
|
|
|
Standard_Integer iE;
|
|
for (iE = 0; iE < 2; iE++)
|
|
{
|
|
TopoDS_Edge E, Eother;
|
|
if (iE == 0)
|
|
{
|
|
E = S1;
|
|
Eother = S2;
|
|
}
|
|
else
|
|
{
|
|
E = S2;
|
|
Eother = S1;
|
|
}
|
|
|
|
Standard_Real aFirst, aLast;
|
|
Handle(Geom_Curve) pCurv = BRep_Tool::Curve(E, aFirst, aLast);
|
|
|
|
Standard_Real aFOther, aLOther;
|
|
Handle(Geom_Curve) pCurvOther = BRep_Tool::Curve(Eother, aFOther, aLOther);
|
|
|
|
if (pCurv->Continuity() == GeomAbs_C0)
|
|
{
|
|
const Standard_Real epsP = Precision::PConfusion();
|
|
|
|
GeomAdaptor_Curve aAdaptorCurve(pCurv, aFirst, aLast);
|
|
const Standard_Integer nbIntervals = aAdaptorCurve.NbIntervals(GeomAbs_C1);
|
|
|
|
TColStd_Array1OfReal arrInter(1,1+nbIntervals);
|
|
aAdaptorCurve.Intervals(arrInter, GeomAbs_C1);
|
|
|
|
GeomAdaptor_Curve aAdaptorCurveOther(pCurvOther, aFOther, aLOther);
|
|
const Standard_Integer nbIntervalsOther = aAdaptorCurveOther.NbIntervals(GeomAbs_C1);
|
|
TColStd_Array1OfReal arrInterOther(1,1+nbIntervalsOther);
|
|
aAdaptorCurveOther.Intervals(arrInterOther, GeomAbs_C1);
|
|
|
|
Standard_Real Udeb,Ufin;
|
|
BRep_Tool::Range(Eother,Udeb,Ufin);
|
|
|
|
gp_Pnt P1,Pt;
|
|
Standard_Integer i, ii;
|
|
BRepClass_FaceClassifier classifier;
|
|
for (i = 1; i <= arrInter.Length(); i++)
|
|
{
|
|
const Standard_Real aParameter = arrInter(i);
|
|
const gp_Pnt aPnt = aAdaptorCurve.Value(aParameter);
|
|
const TopoDS_Vertex V1 = BRepBuilderAPI_MakeVertex(aPnt);
|
|
|
|
BRepExtrema_ExtPC Ext(V1,Eother);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? Ext.NbExt() : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance dstmin
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (ii = 2; ii <= NbExtrema; ii++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(ii);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
|
|
if ((Dstmin < DstRef - Eps) || (fabs(Dstmin-DstRef) < Eps))
|
|
for (ii = 1; ii <= NbExtrema; ii++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(ii)))<Eps)
|
|
{
|
|
Pt=Ext.Point(ii);
|
|
if (TRI_SOLUTION(SeqSol2,Pt))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
const Standard_Real t = Ext.Parameter(ii);
|
|
if ((fabs(t-Udeb)>=epsP)&&(fabs(t-Ufin)>epsP))
|
|
{
|
|
if (mDstRef > Dstmin)
|
|
mDstRef=Dstmin;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,aPnt,BRepExtrema_IsOnEdge,E,aParameter);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,Pt,BRepExtrema_IsOnEdge,Eother,t);
|
|
SeqSol1.Append(iE == 0 ? Sol1 : Sol2);
|
|
SeqSol2.Append(iE == 0 ? Sol2 : Sol1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for (Standard_Integer i2 = 1; i2<=arrInterOther.Length(); i2++)
|
|
{
|
|
const Standard_Real aParameterOther = arrInterOther(i2);
|
|
const gp_Pnt aPntOther = aAdaptorCurveOther.Value(aParameterOther);
|
|
const Standard_Real Dst = aPnt.Distance(aPntOther);
|
|
if ((Dst < DstRef - Eps) || (fabs(Dst-DstRef) < Eps))
|
|
{
|
|
if (mDstRef > Dst)
|
|
mDstRef=Dst;
|
|
const BRepExtrema_SolutionElem Sol1(Dst,aPnt,BRepExtrema_IsOnEdge,E,aParameter);
|
|
const BRepExtrema_SolutionElem Sol2(Dst,aPntOther,BRepExtrema_IsOnEdge,Eother,aParameterOther);
|
|
SeqSol1.Append(iE == 0 ? Sol1 : Sol2);
|
|
SeqSol2.Append(iE == 0 ? Sol2 : Sol1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Shape& S1, const TopoDS_Shape& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
SeqSolShape1.Clear();
|
|
SeqSolShape2.Clear();
|
|
myModif=Standard_False;
|
|
|
|
switch (S1.ShapeType())
|
|
{
|
|
case TopAbs_VERTEX :
|
|
{
|
|
TopoDS_Vertex V1 = TopoDS::Vertex(S1);
|
|
switch (S2.ShapeType())
|
|
{
|
|
case TopAbs_VERTEX :
|
|
{
|
|
TopoDS_Vertex V2 = TopoDS::Vertex(S2);
|
|
Perform( V1, V2 );
|
|
break;
|
|
}
|
|
case TopAbs_EDGE :
|
|
{
|
|
TopoDS_Edge E2 = TopoDS::Edge(S2);
|
|
Perform( V1, E2, B1, B2 );
|
|
break;
|
|
}
|
|
case TopAbs_FACE :
|
|
{
|
|
TopoDS_Face F2 = TopoDS::Face(S2);
|
|
Perform( V1, F2, B1, B2 );
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case TopAbs_EDGE :
|
|
{
|
|
TopoDS_Edge E1 = TopoDS::Edge(S1);
|
|
switch (S2.ShapeType())
|
|
{
|
|
case TopAbs_VERTEX :
|
|
{
|
|
TopoDS_Vertex V2 = TopoDS::Vertex(S2);
|
|
Perform( E1, V2, B1, B2 );
|
|
break;
|
|
}
|
|
case TopAbs_EDGE :
|
|
{
|
|
TopoDS_Edge E2 = TopoDS::Edge(S2);
|
|
TopoDS_Edge aTrimEdge;
|
|
Standard_Boolean bIsTrim1 = Standard_False;
|
|
Standard_Boolean bIsTrim2 = Standard_False;
|
|
TRIM_INFINIT_EDGE( E1, E2, aTrimEdge, bIsTrim1, bIsTrim2 );
|
|
if (bIsTrim1)
|
|
E1 = aTrimEdge;
|
|
if (bIsTrim2)
|
|
E2 = aTrimEdge;
|
|
Perform( E1, E2, B1, B2 );
|
|
break;
|
|
}
|
|
case TopAbs_FACE :
|
|
{
|
|
TopoDS_Face F2 = TopoDS::Face(S2);
|
|
TopoDS_Face aTrimFace;
|
|
Standard_Boolean bIsInfinit;
|
|
TRIM_INFINIT_FACE( E1, F2, aTrimFace, bIsInfinit );
|
|
if (bIsInfinit)
|
|
F2 = aTrimFace;
|
|
Perform( E1, F2, B1, B2 );
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case TopAbs_FACE :
|
|
{
|
|
TopoDS_Face F1 = TopoDS::Face(S1);
|
|
switch (S2.ShapeType())
|
|
{
|
|
case TopAbs_VERTEX :
|
|
{
|
|
TopoDS_Vertex V2 = TopoDS::Vertex(S2);
|
|
Perform( F1, V2, B1, B2 );
|
|
break;
|
|
}
|
|
case TopAbs_EDGE :
|
|
{
|
|
TopoDS_Edge E2 = TopoDS::Edge(S2);
|
|
TopoDS_Face aTrimFace;
|
|
Standard_Boolean bIsInfinit;
|
|
TRIM_INFINIT_FACE( F1, E2, aTrimFace, bIsInfinit );
|
|
if (bIsInfinit)
|
|
F1 = aTrimFace;
|
|
Perform( F1, E2, B1, B2 );
|
|
break;
|
|
}
|
|
case TopAbs_FACE :
|
|
{
|
|
TopoDS_Face F2 = TopoDS::Face(S2);
|
|
Perform( F1, F2, B1, B2 );
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Vertex& S1, const TopoDS_Vertex& S2)
|
|
{
|
|
const gp_Pnt P1 = BRep_Tool::Pnt(S1);
|
|
const gp_Pnt P2 = BRep_Tool::Pnt(S2);
|
|
|
|
const Standard_Real Dst = P1.Distance(P2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
if (myDstRef > Dst)
|
|
myDstRef=Dst;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dst,P1,BRepExtrema_IsVertex,S1);
|
|
const BRepExtrema_SolutionElem Sol2(Dst,P2,BRepExtrema_IsVertex,S2);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Vertex& S1, const TopoDS_Edge& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
if (BRep_Tool::Degenerated(S2))
|
|
return;
|
|
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
BRepExtrema_ExtPC Ext(S1,S2);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? Ext.NbExt() : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
Standard_Real Udeb,Ufin;
|
|
BRep_Tool::Range(S2,Udeb,Ufin);
|
|
|
|
gp_Pnt Pt,P1=BRep_Tool::Pnt(S1);
|
|
const Standard_Real epsP=Precision::PConfusion();
|
|
|
|
for (i = 1; i <= NbExtrema; i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt=Ext.Point(i);
|
|
if (TRI_SOLUTION(SeqSolShape2,Pt))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
const Standard_Real t = Ext.Parameter(i);
|
|
if ( (fabs(t-Udeb)>=epsP) && (fabs(t-Ufin)>epsP) )
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,P1,BRepExtrema_IsVertex,S1);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,Pt,BRepExtrema_IsOnEdge,S2,t);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Edge& S1,const TopoDS_Vertex& S2,
|
|
const Bnd_Box& B1,const Bnd_Box& B2)
|
|
{
|
|
if (BRep_Tool::Degenerated(S1))
|
|
return;
|
|
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
BRepExtrema_ExtPC Ext(S2,S1);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? Ext.NbExt() : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
Standard_Real Udeb,Ufin;
|
|
BRep_Tool::Range(S1,Udeb,Ufin);
|
|
|
|
gp_Pnt Pt, P2 = BRep_Tool::Pnt(S2);
|
|
const Standard_Real epsP=Precision::PConfusion();
|
|
|
|
for (i = 1; i <= NbExtrema; i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt=Ext.Point(i);
|
|
if (TRI_SOLUTION(SeqSolShape1,Pt))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
const Standard_Real t = Ext.Parameter(i);
|
|
if ( (fabs(t-Udeb)>=epsP) && (fabs(t-Ufin)>epsP) )
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,Pt,BRepExtrema_IsOnEdge,S1,t);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,P2,BRepExtrema_IsVertex,S2);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Vertex& S1, const TopoDS_Face& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
BRepExtrema_ExtPF Ext(S1,S2,myFlag,myAlgo);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? Ext.NbExt() : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
Standard_Real U,V;
|
|
gp_Pnt Pt,P1=BRep_Tool::Pnt(S1);
|
|
BRepClass_FaceClassifier classifier;
|
|
const Standard_Real tol = BRep_Tool::Tolerance(S2);
|
|
|
|
for (i = 1; i <= NbExtrema; i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt=Ext.Point(i);
|
|
if (TRI_SOLUTION(SeqSolShape2,Pt))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
Ext.Parameter(i,U,V);
|
|
const gp_Pnt2d PUV(U,V);
|
|
classifier.Perform(S2,PUV,tol);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,P1,BRepExtrema_IsVertex,S1);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,Pt,BRepExtrema_IsInFace,S2,U,V);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Face& S1, const TopoDS_Vertex& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
BRepExtrema_ExtPF Ext(S2,S1,myFlag,myAlgo);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? Ext.NbExt() : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
Standard_Real U,V;
|
|
gp_Pnt Pt,P2=BRep_Tool::Pnt(S2);
|
|
BRepClass_FaceClassifier classifier;
|
|
const Standard_Real tol = BRep_Tool::Tolerance(S1);
|
|
|
|
for (i = 1; i <= NbExtrema; i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt=Ext.Point(i);
|
|
if (TRI_SOLUTION(SeqSolShape1,Pt))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
Ext.Parameter(i,U,V);
|
|
const gp_Pnt2d PUV(U,V);
|
|
classifier.Perform(S1,PUV,tol);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,Pt,BRepExtrema_IsInFace,S1,U,V);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,P2,BRepExtrema_IsVertex,S2);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Edge& S1, const TopoDS_Edge& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
if (BRep_Tool::Degenerated(S1) || BRep_Tool::Degenerated(S2))
|
|
return;
|
|
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
const Standard_Real DstRef = myDstRef;
|
|
|
|
BRepExtrema_ExtCC Ext(S1,S2);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? (Ext.IsParallel()? 0 : Ext.NbExt()) : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
Standard_Real Udeb1,Ufin1,Udeb2,Ufin2;
|
|
BRep_Tool::Range(S1,Udeb1,Ufin1);
|
|
BRep_Tool::Range(S2,Udeb2,Ufin2);
|
|
|
|
gp_Pnt Pt1,Pt2;
|
|
const Standard_Real epsP=Precision::PConfusion();
|
|
|
|
for (i=1;i<=NbExtrema;i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt1=Ext.PointOnE1(i);
|
|
Pt2=Ext.PointOnE2(i);
|
|
if (TRI_SOLUTION(SeqSolShape1,Pt1) || TRI_SOLUTION(SeqSolShape2,Pt2))
|
|
{
|
|
// Check if the parameters do not correspond to a vertex
|
|
const Standard_Real t1 = Ext.ParameterOnE1(i);
|
|
const Standard_Real t2 = Ext.ParameterOnE2(i);
|
|
if ((fabs(t1-Udeb1)>=epsP)&&(fabs(t1-Ufin1)>epsP)&&(fabs(t2-Udeb2)>=epsP)&&(fabs(t2-Ufin2)>epsP))
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,Pt1,BRepExtrema_IsOnEdge,S1,t1);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,Pt2,BRepExtrema_IsOnEdge,S2,t2);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
BRepExtrema_SeqOfSolution SeqSolution1;
|
|
BRepExtrema_SeqOfSolution SeqSolution2;
|
|
|
|
PERFORM_C0(S1, S2, SeqSolution1, SeqSolution2, DstRef, myDstRef, myEps);
|
|
|
|
BRepExtrema_SeqOfSolution seqSol1;
|
|
BRepExtrema_SeqOfSolution seqSol2;
|
|
|
|
if (SeqSolution1.Length() > 0 && SeqSolution2.Length() > 0)
|
|
MIN_SOLUTION(SeqSolution1, SeqSolution2, myDstRef, myEps, seqSol1, seqSol2);
|
|
|
|
if (!seqSol1.IsEmpty() && !seqSol2.IsEmpty())
|
|
{
|
|
SeqSolShape1.Append(seqSol1);
|
|
SeqSolShape2.Append(seqSol2);
|
|
myModif = Standard_True;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Edge& S1, const TopoDS_Face& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
if (BRep_Tool::Degenerated(S1))
|
|
return;
|
|
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
BRepClass_FaceClassifier classifier;
|
|
|
|
BRepExtrema_ExtCF Ext(S1,S2);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? (Ext.IsParallel()? 0 : Ext.NbExt()) : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
Standard_Real Udeb,Ufin,U,V;
|
|
BRep_Tool::Range(S1,Udeb,Ufin);
|
|
const Standard_Real tol=BRep_Tool::Tolerance(S2);
|
|
|
|
gp_Pnt Pt1,Pt2;
|
|
const Standard_Real epsP=Precision::PConfusion();
|
|
|
|
for (i = 1; i <= NbExtrema; i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt1=Ext.PointOnEdge(i);
|
|
Pt2=Ext.PointOnFace(i);
|
|
if (TRI_SOLUTION(SeqSolShape1,Pt1) || TRI_SOLUTION(SeqSolShape2,Pt2))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
const Standard_Real t1 = Ext.ParameterOnEdge(i);
|
|
if ((fabs(t1-Udeb)>=epsP)&&(fabs(t1-Ufin)>epsP))
|
|
{
|
|
Ext.ParameterOnFace(i,U,V);
|
|
const gp_Pnt2d PUV(U,V);
|
|
classifier.Perform(S2,PUV,tol);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,Pt1,BRepExtrema_IsOnEdge,S1,t1);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,Pt2,BRepExtrema_IsInFace,S2,U,V);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Standard_Real aFirst, aLast;
|
|
Handle(Geom_Curve) pCurv = BRep_Tool::Curve(S1, aFirst, aLast);
|
|
if (pCurv->Continuity() == GeomAbs_C0)
|
|
{
|
|
BRepExtrema_SeqOfSolution SeqSolution1;
|
|
BRepExtrema_SeqOfSolution SeqSolution2;
|
|
|
|
GeomAdaptor_Curve aAdaptorCurve(pCurv, aFirst, aLast);
|
|
const Standard_Integer nbIntervals = aAdaptorCurve.NbIntervals(GeomAbs_C1);
|
|
|
|
TColStd_Array1OfReal arrInter(1,1+nbIntervals);
|
|
aAdaptorCurve.Intervals(arrInter, GeomAbs_C1);
|
|
|
|
gp_Pnt Pt;
|
|
Standard_Real U,V;
|
|
const Standard_Real tol = BRep_Tool::Tolerance(S2);
|
|
|
|
Standard_Integer i;
|
|
for (i = 1; i <= arrInter.Length(); i++)
|
|
{
|
|
const Standard_Real aParameter = arrInter(i);
|
|
gp_Pnt aPnt = aAdaptorCurve.Value(aParameter);
|
|
TopoDS_Vertex V1 = BRepBuilderAPI_MakeVertex(aPnt);
|
|
|
|
BRepExtrema_ExtPF ExtPF(V1,S2);
|
|
const Standard_Integer NbExtrema = ExtPF.IsDone()? ExtPF.NbExt() : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer ii;
|
|
Standard_Real Dstmin = ExtPF.SquareDistance(1);
|
|
for (ii = 2; ii <= NbExtrema; ii++)
|
|
{
|
|
const Standard_Real sDst = ExtPF.SquareDistance(ii);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
for (ii = 1; ii <= NbExtrema; ii++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(ExtPF.SquareDistance(ii)))<myEps)
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
ExtPF.Parameter(ii,U,V);
|
|
const gp_Pnt2d PUV(U,V);
|
|
classifier.Perform(S2,PUV,tol);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,aPnt,BRepExtrema_IsOnEdge,S1,aParameter);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,ExtPF.Point(ii),BRepExtrema_IsInFace,S2,U,V);
|
|
SeqSolution1.Append(Sol1);
|
|
SeqSolution2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
BRepExtrema_SeqOfSolution seqSol1;
|
|
BRepExtrema_SeqOfSolution seqSol2;
|
|
if (SeqSolution1.Length() > 0 && SeqSolution2.Length() > 0)
|
|
MIN_SOLUTION(SeqSolution1, SeqSolution2, myDstRef, myEps, seqSol1, seqSol2);
|
|
|
|
if (!seqSol1.IsEmpty() && !seqSol2.IsEmpty())
|
|
{
|
|
SeqSolShape1.Append(seqSol1);
|
|
SeqSolShape2.Append(seqSol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Face& S1, const TopoDS_Edge& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
if (BRep_Tool::Degenerated(S2))
|
|
return;
|
|
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
BRepClass_FaceClassifier classifier;
|
|
|
|
BRepExtrema_ExtCF Ext(S2,S1);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? (Ext.IsParallel()? 0 : Ext.NbExt()) : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
Standard_Real Udeb,Ufin,U,V;
|
|
BRep_Tool::Range(S2,Udeb,Ufin);
|
|
const Standard_Real tol=BRep_Tool::Tolerance(S1);
|
|
|
|
gp_Pnt Pt1,Pt2;
|
|
const Standard_Real epsP=Precision::PConfusion();
|
|
|
|
for (i = 1; i <= NbExtrema; i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt1=Ext.PointOnEdge(i);
|
|
Pt2=Ext.PointOnFace(i);
|
|
if (TRI_SOLUTION(SeqSolShape1,Pt1) || TRI_SOLUTION(SeqSolShape2,Pt2))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
const Standard_Real t1 = Ext.ParameterOnEdge(i);
|
|
if ((fabs(t1-Udeb)>=epsP)&&(fabs(t1-Ufin)>epsP))
|
|
{
|
|
Ext.ParameterOnFace(i,U,V);
|
|
const gp_Pnt2d PUV(U,V);
|
|
classifier.Perform(S1,PUV,tol);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,Pt1,BRepExtrema_IsOnEdge,S2,t1);
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,Pt2,BRepExtrema_IsInFace,S1,U,V);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Standard_Real aFirst, aLast;
|
|
Handle(Geom_Curve) pCurv = BRep_Tool::Curve(S2, aFirst, aLast);
|
|
if (pCurv->Continuity() == GeomAbs_C0)
|
|
{
|
|
BRepExtrema_SeqOfSolution SeqSolution1;
|
|
BRepExtrema_SeqOfSolution SeqSolution2;
|
|
|
|
GeomAdaptor_Curve aAdaptorCurve(pCurv, aFirst, aLast);
|
|
const Standard_Integer nbIntervals = aAdaptorCurve.NbIntervals(GeomAbs_C1);
|
|
|
|
TColStd_Array1OfReal arrInter(1,1+nbIntervals);
|
|
aAdaptorCurve.Intervals(arrInter, GeomAbs_C1);
|
|
|
|
gp_Pnt Pt;
|
|
Standard_Real U,V;
|
|
const Standard_Real tol = BRep_Tool::Tolerance(S1);
|
|
|
|
Standard_Integer i;
|
|
for (i = 1; i <= arrInter.Length(); i++)
|
|
{
|
|
const Standard_Real aParameter = arrInter(i);
|
|
gp_Pnt aPnt = aAdaptorCurve.Value(aParameter);
|
|
TopoDS_Vertex V1 = BRepBuilderAPI_MakeVertex(aPnt);
|
|
|
|
BRepExtrema_ExtPF ExtPF(V1,S1);
|
|
const Standard_Integer NbExtrema = ExtPF.IsDone()? ExtPF.NbExt() : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer ii;
|
|
Standard_Real Dstmin = ExtPF.SquareDistance(1);
|
|
for (ii = 2; ii <= NbExtrema; ii++)
|
|
{
|
|
const Standard_Real sDst = ExtPF.SquareDistance(ii);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
for (ii = 1; ii <= NbExtrema; ii++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(ExtPF.SquareDistance(ii)))<myEps)
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
ExtPF.Parameter(ii,U,V);
|
|
const gp_Pnt2d PUV(U,V);
|
|
classifier.Perform(S1,PUV,tol);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,aPnt,BRepExtrema_IsOnEdge,S2,aParameter);
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,ExtPF.Point(ii),BRepExtrema_IsInFace,S1,U,V);
|
|
SeqSolution1.Append(Sol1);
|
|
SeqSolution2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
BRepExtrema_SeqOfSolution seqSol1;
|
|
BRepExtrema_SeqOfSolution seqSol2;
|
|
if (SeqSolution1.Length() > 0 && SeqSolution2.Length() > 0)
|
|
MIN_SOLUTION(SeqSolution1, SeqSolution2, myDstRef, myEps, seqSol1, seqSol2);
|
|
|
|
if (!seqSol1.IsEmpty() && !seqSol2.IsEmpty())
|
|
{
|
|
SeqSolShape1.Append(seqSol1);
|
|
SeqSolShape2.Append(seqSol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|
|
|
|
void BRepExtrema_DistanceSS::Perform(const TopoDS_Face& S1, const TopoDS_Face& S2,
|
|
const Bnd_Box& B1, const Bnd_Box& B2)
|
|
{
|
|
const Standard_Real Dst=B1.Distance(B2);
|
|
if ((Dst < myDstRef - myEps) || (fabs(Dst-myDstRef) < myEps))
|
|
{
|
|
BRepExtrema_ExtFF Ext(S1,S2);
|
|
const Standard_Integer NbExtrema = Ext.IsDone()? (Ext.IsParallel()? 0 : Ext.NbExt()) : 0;
|
|
if ( NbExtrema > 0 )
|
|
{
|
|
// Search minimum distance Dstmin
|
|
Standard_Integer i;
|
|
Standard_Real Dstmin = Ext.SquareDistance(1);
|
|
for (i = 2; i <= NbExtrema; i++)
|
|
{
|
|
const Standard_Real sDst = Ext.SquareDistance(i);
|
|
if (sDst<Dstmin)
|
|
Dstmin=sDst;
|
|
}
|
|
Dstmin = sqrt(Dstmin);
|
|
if ((Dstmin < myDstRef - myEps) || (fabs(Dstmin-myDstRef) < myEps))
|
|
{
|
|
const Standard_Real tol1=BRep_Tool::Tolerance(S1);
|
|
const Standard_Real tol2=BRep_Tool::Tolerance(S2);
|
|
|
|
gp_Pnt Pt1,Pt2;
|
|
gp_Pnt2d PUV;
|
|
Standard_Real U1,V1,U2,V2;
|
|
BRepClass_FaceClassifier classifier;
|
|
|
|
for (i = 1; i <= NbExtrema; i++)
|
|
{
|
|
if (fabs(Dstmin-sqrt(Ext.SquareDistance(i)))<myEps)
|
|
{
|
|
Pt1=Ext.PointOnFace1(i);
|
|
Pt2=Ext.PointOnFace2(i);
|
|
if (TRI_SOLUTION(SeqSolShape1,Pt1) || TRI_SOLUTION(SeqSolShape2,Pt2))
|
|
{
|
|
// Check if the parameter does not correspond to a vertex
|
|
Ext.ParameterOnFace1(i,U1,V1);
|
|
PUV.SetCoord(U1,V1);
|
|
classifier.Perform(S1,PUV,tol1);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
Ext.ParameterOnFace2(i,U2,V2);
|
|
PUV.SetCoord(U2,V2);
|
|
classifier.Perform(S2,PUV,tol2);
|
|
if (classifier.State()==TopAbs_IN)
|
|
{
|
|
if (myDstRef > Dstmin)
|
|
myDstRef=Dstmin;
|
|
myModif=Standard_True;
|
|
const BRepExtrema_SolutionElem Sol1(Dstmin,Pt1,BRepExtrema_IsInFace,S1,U1,V1);
|
|
const BRepExtrema_SolutionElem Sol2(Dstmin,Pt2,BRepExtrema_IsInFace,S2,U2,V2);
|
|
SeqSolShape1.Append(Sol1);
|
|
SeqSolShape2.Append(Sol2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*********************************************************************************/
|