mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-03 17:56:21 +03:00
0026796: The result of General Fuse operation is self-intersecting shape The fix forces creation of new sub-shapes (vertex, edge) when the tolerance of some sub-shape of an argument is to be increased. This new behavior is turned off by default. It can be turned on using two ways: 1) Setting 'locking' flag of the arguments. 2) Calling the method SetNonDestructive(Standard_True) of the API classes. Various bug fixes in the algorithm: - Compute correct tolerance values for intersections of type Line/Line, Line/Plane, Plane/Plane. - In case of Line/Plane intersection check if line's vertices lie on the plane. - Do not allow decreasing of the tolerance value of the Line/Line intersection vertex. - In IntTools_EdgeEdge, call the method FindParameters with proper 3D tolerance of the curve. - Force making copy of a degenerated edge if its vertex is touched but no 2D intersection with other curves is found. - Remove pave blocks both ends of which became referring to the same vertex after vertices substitution. - Avoid exception in IntTools_Context::IsVertexOnLine if Extrema is not done. - Reduce tolerance of vertex/edge using actual distances to interfered shapes if it was increased due to line/line, line/plane, or plane/plane small intersection angle. - Update tolerance of edges to reach all representations in a common block. - In V-E intersections, check if a vertex hits beyond shrunk range, in such case create V-V interference. - Do not put a section edge to the result if it becomes to be a micro edge after updating its vertex. - Correctly make vertices same-domain during the work of MakeBlocks. - Decrease shrunk range at least on a Precision::Confusion() in addition to vertex tolerance. - Add Confusion to bounding boxes of new shapes in DS - Add tolerance Precision::Confusion() to compare distances of touching cases to fix regressions. TODO marks have been removed from (or modified in) the following test cases (Improvements): boolean bsection M3 N2 R2 boolean gdml_private B6 C2 C6 G7 I6 F6 J1 J4 M7 N1 N8 N9 O3 O4 O6 O8 O9 P1 P2 P5 Q1 Q3 Q5 S9 T2 U4 U5 U9 ZB5 ZB6 ZC1 ZC5 ZD3 ZD6 ZD7 ZH2 ZH5 ZI2 ZI5 ZI7 ZI9 ZJ3 ZJ4 ZJ7 F8 I6 G1 boolean volumemaker A5 A6 B3 B4 B7 B9 D3 D4 D7 F1 boolean bopcut_2d D5 bugs modalg_5 bug25043 bugs modalg_2 bug472_1 bug472_2 bug472_3 Test cases updated because they are still bad but can be accepted as non-regression: boolean volumemaker C4 A3 A7 E6 bugs modalg_1 bug10232 boolean bsection N2 Put new TODO in the scripts: bugs modalg_5 bug25232_9 bugs modalg_6 bug26619 bugs modalg_1 buc60462_2 bugs modalg_4 bug772 For the following tests the result in fix became better, so take fix result as the reference: bugs modalg_5 bug24628 bugs modalg_6 bug26954_3 boolean volumemaker A4 B5 B6 C3 C8 D2 D5 F2 bugs modalg_2 bug472_2 bugs modalg_1 buc60776_1 - Add the method SetNonDestructive to API classes of user level
1034 lines
29 KiB
C++
1034 lines
29 KiB
C++
// Created by: Peter KURNEV
|
|
// 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 <Bnd_Box.hxx>
|
|
#include <BRep_Tool.hxx>
|
|
#include <BRepAdaptor_Surface.hxx>
|
|
#include <BRepBndLib.hxx>
|
|
#include <BRepClass3d_SolidClassifier.hxx>
|
|
#include <Extrema_LocateExtPC.hxx>
|
|
#include <Geom2d_Curve.hxx>
|
|
#include <Geom2d_TrimmedCurve.hxx>
|
|
#include <Geom2dHatch_Hatcher.hxx>
|
|
#include <Geom2dHatch_Intersector.hxx>
|
|
#include <Geom_BoundedCurve.hxx>
|
|
#include <Geom_Curve.hxx>
|
|
#include <GeomAdaptor_Curve.hxx>
|
|
#include <GeomAPI_ProjectPointOnCurve.hxx>
|
|
#include <GeomAPI_ProjectPointOnSurf.hxx>
|
|
#include <gp_Pnt.hxx>
|
|
#include <gp_Pnt2d.hxx>
|
|
#include <IntTools_Context.hxx>
|
|
#include <IntTools_Curve.hxx>
|
|
#include <IntTools_FClass2d.hxx>
|
|
#include <IntTools_SurfaceRangeLocalizeData.hxx>
|
|
#include <IntTools_Tools.hxx>
|
|
#include <Precision.hxx>
|
|
#include <Standard_Type.hxx>
|
|
#include <TopAbs_State.hxx>
|
|
#include <TopExp_Explorer.hxx>
|
|
#include <TopoDS.hxx>
|
|
#include <TopoDS_Edge.hxx>
|
|
#include <TopoDS_Face.hxx>
|
|
#include <TopoDS_Shape.hxx>
|
|
#include <TopoDS_Solid.hxx>
|
|
#include <TopoDS_Vertex.hxx>
|
|
|
|
IMPLEMENT_STANDARD_RTTIEXT(IntTools_Context,MMgt_TShared)
|
|
|
|
//
|
|
//=======================================================================
|
|
//function :
|
|
//purpose :
|
|
//=======================================================================
|
|
IntTools_Context::IntTools_Context()
|
|
:
|
|
myAllocator(NCollection_BaseAllocator::CommonBaseAllocator()),
|
|
myFClass2dMap(100, myAllocator),
|
|
myProjPSMap(100, myAllocator),
|
|
myProjPCMap(100, myAllocator),
|
|
mySClassMap(100, myAllocator),
|
|
myProjPTMap(100, myAllocator),
|
|
myHatcherMap(100, myAllocator),
|
|
myProjSDataMap(100, myAllocator),
|
|
myBndBoxDataMap(100, myAllocator),
|
|
myCreateFlag(0),
|
|
myPOnSTolerance(1.e-12)
|
|
{
|
|
}
|
|
//=======================================================================
|
|
//function :
|
|
//purpose :
|
|
//=======================================================================
|
|
IntTools_Context::IntTools_Context
|
|
(const Handle(NCollection_BaseAllocator)& theAllocator)
|
|
:
|
|
myAllocator(theAllocator),
|
|
myFClass2dMap(100, myAllocator),
|
|
myProjPSMap(100, myAllocator),
|
|
myProjPCMap(100, myAllocator),
|
|
mySClassMap(100, myAllocator),
|
|
myProjPTMap(100, myAllocator),
|
|
myHatcherMap(100, myAllocator),
|
|
myProjSDataMap(100, myAllocator),
|
|
myBndBoxDataMap(100, myAllocator),
|
|
myCreateFlag(1),
|
|
myPOnSTolerance(1.e-12)
|
|
{
|
|
}
|
|
//=======================================================================
|
|
//function : ~
|
|
//purpose :
|
|
//=======================================================================
|
|
IntTools_Context::~IntTools_Context()
|
|
{
|
|
Standard_Address anAdr;
|
|
BOPCol_DataMapIteratorOfDataMapOfShapeAddress aIt;
|
|
BOPCol_DataMapIteratorOfDataMapOfTransientAddress aIt1;
|
|
//
|
|
IntTools_FClass2d* pFClass2d;
|
|
//
|
|
aIt.Initialize(myFClass2dMap);
|
|
for (; aIt.More(); aIt.Next()) {
|
|
anAdr=aIt.Value();
|
|
pFClass2d=(IntTools_FClass2d*)anAdr;
|
|
(*pFClass2d).~IntTools_FClass2d();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
myFClass2dMap.Clear();
|
|
//
|
|
clearCachedPOnSProjectors();
|
|
//
|
|
GeomAPI_ProjectPointOnCurve* pProjPC;
|
|
aIt.Initialize(myProjPCMap);
|
|
for (; aIt.More(); aIt.Next()) {
|
|
anAdr=aIt.Value();
|
|
pProjPC=(GeomAPI_ProjectPointOnCurve*)anAdr;
|
|
(*pProjPC).~GeomAPI_ProjectPointOnCurve();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
myProjPCMap.Clear();
|
|
//
|
|
//
|
|
BRepClass3d_SolidClassifier* pSC;
|
|
aIt.Initialize(mySClassMap);
|
|
for (; aIt.More(); aIt.Next()) {
|
|
anAdr=aIt.Value();
|
|
pSC=(BRepClass3d_SolidClassifier*)anAdr;
|
|
(*pSC).~BRepClass3d_SolidClassifier();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
mySClassMap.Clear();
|
|
//
|
|
GeomAPI_ProjectPointOnCurve* pProjPT;
|
|
aIt1.Initialize(myProjPTMap);
|
|
for (; aIt1.More(); aIt1.Next()) {
|
|
anAdr=aIt1.Value();
|
|
pProjPT=(GeomAPI_ProjectPointOnCurve*)anAdr;
|
|
(*pProjPT).~GeomAPI_ProjectPointOnCurve();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
myProjPTMap.Clear();
|
|
//
|
|
Geom2dHatch_Hatcher* pHatcher;
|
|
aIt.Initialize(myHatcherMap);
|
|
for (; aIt.More(); aIt.Next()) {
|
|
anAdr=aIt.Value();
|
|
pHatcher=(Geom2dHatch_Hatcher*)anAdr;
|
|
(*pHatcher).~Geom2dHatch_Hatcher();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
myHatcherMap.Clear();
|
|
//
|
|
IntTools_SurfaceRangeLocalizeData* pSData = NULL;
|
|
aIt.Initialize(myProjSDataMap);
|
|
for (; aIt.More(); aIt.Next()) {
|
|
anAdr=aIt.Value();
|
|
pSData = (IntTools_SurfaceRangeLocalizeData*)anAdr;
|
|
(*pSData).~IntTools_SurfaceRangeLocalizeData();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
myProjSDataMap.Clear();
|
|
//
|
|
Bnd_Box* pBox;
|
|
aIt.Initialize(myBndBoxDataMap);
|
|
for (; aIt.More(); aIt.Next()) {
|
|
anAdr=aIt.Value();
|
|
pBox=(Bnd_Box*)anAdr;
|
|
(*pBox).~Bnd_Box();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
myBndBoxDataMap.Clear();
|
|
}
|
|
//=======================================================================
|
|
//function : BndBox
|
|
//purpose :
|
|
//=======================================================================
|
|
Bnd_Box& IntTools_Context::BndBox(const TopoDS_Shape& aS)
|
|
{
|
|
Standard_Address anAdr;
|
|
Bnd_Box* pBox;
|
|
//
|
|
if (!myBndBoxDataMap.IsBound(aS)) {
|
|
//
|
|
pBox=(Bnd_Box*)myAllocator->Allocate(sizeof(Bnd_Box));
|
|
new (pBox) Bnd_Box();
|
|
//
|
|
Bnd_Box &aBox=*pBox;
|
|
BRepBndLib::Add(aS, aBox);
|
|
//
|
|
anAdr=(Standard_Address)pBox;
|
|
myBndBoxDataMap.Bind(aS, anAdr);
|
|
}
|
|
else {
|
|
anAdr=myBndBoxDataMap.Find(aS);
|
|
pBox=(Bnd_Box*)anAdr;
|
|
}
|
|
return *pBox;
|
|
}
|
|
//=======================================================================
|
|
//function : IsInfiniteFace
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsInfiniteFace
|
|
(const TopoDS_Face& aFace)
|
|
{
|
|
Standard_Boolean bRet;
|
|
Standard_Integer i;
|
|
Standard_Real aX[6];
|
|
//
|
|
bRet=Standard_False;
|
|
//
|
|
if (!BRep_Tool::NaturalRestriction(aFace)) {
|
|
return bRet;
|
|
}
|
|
//
|
|
Bnd_Box& aBox=BndBox(aFace);
|
|
//
|
|
aBox.Get(aX[0], aX[1], aX[2], aX[3], aX[4], aX[5]);
|
|
//
|
|
for (i=0; (i<6) && (!bRet); ++i) {
|
|
bRet=Precision::IsInfinite(aX[i]);
|
|
}
|
|
//
|
|
return bRet;
|
|
}
|
|
//=======================================================================
|
|
//function : FClass2d
|
|
//purpose :
|
|
//=======================================================================
|
|
IntTools_FClass2d& IntTools_Context::FClass2d(const TopoDS_Face& aF)
|
|
{
|
|
Standard_Address anAdr;
|
|
IntTools_FClass2d* pFClass2d;
|
|
//
|
|
if (!myFClass2dMap.IsBound(aF)) {
|
|
Standard_Real aTolF;
|
|
TopoDS_Face aFF;
|
|
//
|
|
aFF=aF;
|
|
aFF.Orientation(TopAbs_FORWARD);
|
|
aTolF=BRep_Tool::Tolerance(aFF);
|
|
//
|
|
pFClass2d=(IntTools_FClass2d*)myAllocator->Allocate(sizeof(IntTools_FClass2d));
|
|
new (pFClass2d) IntTools_FClass2d(aFF, aTolF);
|
|
//
|
|
anAdr=(Standard_Address)pFClass2d;
|
|
myFClass2dMap.Bind(aFF, anAdr);
|
|
}
|
|
else {
|
|
anAdr=myFClass2dMap.Find(aF);
|
|
pFClass2d=(IntTools_FClass2d*)anAdr;
|
|
}
|
|
return *pFClass2d;
|
|
}
|
|
//=======================================================================
|
|
//function : ProjPS
|
|
//purpose :
|
|
//=======================================================================
|
|
GeomAPI_ProjectPointOnSurf& IntTools_Context::ProjPS(const TopoDS_Face& aF)
|
|
{
|
|
Standard_Address anAdr;
|
|
GeomAPI_ProjectPointOnSurf* pProjPS;
|
|
|
|
if (!myProjPSMap.IsBound(aF)) {
|
|
Standard_Real Umin, Usup, Vmin, Vsup;
|
|
BRepAdaptor_Surface aBAS;
|
|
//
|
|
const Handle(Geom_Surface)& aS=BRep_Tool::Surface(aF);
|
|
aBAS.Initialize (aF, Standard_True);
|
|
//
|
|
Umin=aBAS.FirstUParameter();
|
|
Usup=aBAS.LastUParameter ();
|
|
Vmin=aBAS.FirstVParameter();
|
|
Vsup=aBAS.LastVParameter ();
|
|
//
|
|
pProjPS=(GeomAPI_ProjectPointOnSurf*)myAllocator->Allocate(sizeof(GeomAPI_ProjectPointOnSurf));
|
|
new (pProjPS) GeomAPI_ProjectPointOnSurf();
|
|
pProjPS->Init(aS ,Umin, Usup, Vmin, Vsup, myPOnSTolerance/*, Extrema_ExtAlgo_Tree*/);
|
|
Extrema_ExtPS& anExtAlgo = const_cast<Extrema_ExtPS&>(pProjPS->Extrema());
|
|
anExtAlgo.SetFlag(Extrema_ExtFlag_MIN);
|
|
//
|
|
anAdr=(Standard_Address)pProjPS;
|
|
myProjPSMap.Bind(aF, anAdr);
|
|
}
|
|
|
|
else {
|
|
anAdr=myProjPSMap.Find(aF);
|
|
pProjPS=(GeomAPI_ProjectPointOnSurf*)anAdr;
|
|
}
|
|
return *pProjPS;
|
|
}
|
|
//=======================================================================
|
|
//function : ProjPC
|
|
//purpose :
|
|
//=======================================================================
|
|
GeomAPI_ProjectPointOnCurve& IntTools_Context::ProjPC(const TopoDS_Edge& aE)
|
|
{
|
|
Standard_Address anAdr;
|
|
GeomAPI_ProjectPointOnCurve* pProjPC;
|
|
|
|
if (!myProjPCMap.IsBound(aE)) {
|
|
Standard_Real f, l;
|
|
//
|
|
Handle(Geom_Curve)aC3D=BRep_Tool::Curve (aE, f, l);
|
|
//
|
|
pProjPC=(GeomAPI_ProjectPointOnCurve*)myAllocator->Allocate(sizeof(GeomAPI_ProjectPointOnCurve));
|
|
new (pProjPC) GeomAPI_ProjectPointOnCurve();
|
|
pProjPC->Init(aC3D, f, l);
|
|
//
|
|
anAdr=(Standard_Address)pProjPC;
|
|
myProjPCMap.Bind(aE, anAdr);
|
|
}
|
|
|
|
else {
|
|
anAdr=myProjPCMap.Find(aE);
|
|
pProjPC=(GeomAPI_ProjectPointOnCurve*)anAdr;
|
|
}
|
|
return *pProjPC;
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : ProjPT
|
|
//purpose :
|
|
//=======================================================================
|
|
GeomAPI_ProjectPointOnCurve& IntTools_Context::ProjPT
|
|
(const Handle(Geom_Curve)& aC3D)
|
|
|
|
{
|
|
Standard_Address anAdr;
|
|
GeomAPI_ProjectPointOnCurve* pProjPT;
|
|
|
|
if (!myProjPTMap.IsBound(aC3D)) {
|
|
Standard_Real f, l;
|
|
f=aC3D->FirstParameter();
|
|
l=aC3D->LastParameter();
|
|
//
|
|
pProjPT=(GeomAPI_ProjectPointOnCurve*)myAllocator->Allocate(sizeof(GeomAPI_ProjectPointOnCurve));
|
|
new (pProjPT) GeomAPI_ProjectPointOnCurve();
|
|
pProjPT->Init(aC3D, f, l);
|
|
//
|
|
anAdr=(Standard_Address)pProjPT;
|
|
myProjPTMap.Bind(aC3D, anAdr);
|
|
}
|
|
|
|
else {
|
|
anAdr=myProjPTMap.Find(aC3D);
|
|
pProjPT=(GeomAPI_ProjectPointOnCurve*)anAdr;
|
|
}
|
|
return *pProjPT;
|
|
}
|
|
//=======================================================================
|
|
//function : SolidClassifier
|
|
//purpose :
|
|
//=======================================================================
|
|
BRepClass3d_SolidClassifier& IntTools_Context::SolidClassifier
|
|
(const TopoDS_Solid& aSolid)
|
|
{
|
|
Standard_Address anAdr;
|
|
BRepClass3d_SolidClassifier* pSC;
|
|
|
|
if (!mySClassMap.IsBound(aSolid)) {
|
|
//
|
|
pSC=(BRepClass3d_SolidClassifier*)myAllocator->Allocate(sizeof(BRepClass3d_SolidClassifier));
|
|
new (pSC) BRepClass3d_SolidClassifier(aSolid);
|
|
//
|
|
anAdr=(Standard_Address)pSC;
|
|
mySClassMap.Bind(aSolid, anAdr);
|
|
}
|
|
|
|
else {
|
|
anAdr=mySClassMap.Find(aSolid);
|
|
pSC =(BRepClass3d_SolidClassifier*)anAdr;
|
|
}
|
|
return *pSC;
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : Hatcher
|
|
//purpose :
|
|
//=======================================================================
|
|
Geom2dHatch_Hatcher& IntTools_Context::Hatcher(const TopoDS_Face& aF)
|
|
{
|
|
Standard_Address anAdr;
|
|
Geom2dHatch_Hatcher* pHatcher;
|
|
//
|
|
if (!myHatcherMap.IsBound(aF)) {
|
|
Standard_Real aTolArcIntr, aTolTangfIntr, aTolHatch2D, aTolHatch3D;
|
|
Standard_Real aU1, aU2, aEpsT;
|
|
TopAbs_Orientation aOrE;
|
|
Handle(Geom_Surface) aS;
|
|
Handle(Geom2d_Curve) aC2D;
|
|
Handle(Geom2d_TrimmedCurve) aCT2D;
|
|
TopoDS_Face aFF;
|
|
TopExp_Explorer aExp;
|
|
//
|
|
aTolHatch2D=1.e-8;
|
|
aTolHatch3D=1.e-8;
|
|
aTolArcIntr=1.e-10;
|
|
aTolTangfIntr=1.e-10;
|
|
aEpsT=Precision::PConfusion();
|
|
//
|
|
Geom2dHatch_Intersector aIntr(aTolArcIntr, aTolTangfIntr);
|
|
pHatcher=(Geom2dHatch_Hatcher*)
|
|
myAllocator->Allocate(sizeof(Geom2dHatch_Hatcher));
|
|
new (pHatcher) Geom2dHatch_Hatcher(aIntr,
|
|
aTolHatch2D, aTolHatch3D,
|
|
Standard_True, Standard_False);
|
|
//
|
|
aFF=aF;
|
|
aFF.Orientation(TopAbs_FORWARD);
|
|
aS=BRep_Tool::Surface(aFF);
|
|
|
|
aExp.Init (aFF, TopAbs_EDGE);
|
|
for (; aExp.More() ; aExp.Next()) {
|
|
const TopoDS_Edge& aE=*((TopoDS_Edge*)&aExp.Current());
|
|
aOrE=aE.Orientation();
|
|
//
|
|
aC2D=BRep_Tool::CurveOnSurface (aE, aFF, aU1, aU2);
|
|
if (aC2D.IsNull() ) {
|
|
continue;
|
|
}
|
|
if (fabs(aU1-aU2) < aEpsT) {
|
|
continue;
|
|
}
|
|
//
|
|
aCT2D=new Geom2d_TrimmedCurve(aC2D, aU1, aU2);
|
|
Geom2dAdaptor_Curve aGAC (aCT2D);
|
|
pHatcher->AddElement(aGAC, aOrE);
|
|
}// for (; aExp.More() ; aExp.Next()) {
|
|
//
|
|
anAdr=(Standard_Address)pHatcher;
|
|
myHatcherMap.Bind(aFF, anAdr);
|
|
}//if (!myHatcherMap.IsBound(aF)) {
|
|
//
|
|
else {
|
|
anAdr=myHatcherMap.Find(aF);
|
|
pHatcher=(Geom2dHatch_Hatcher*)anAdr;
|
|
}
|
|
|
|
return *pHatcher;
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : SurfaceData
|
|
//purpose :
|
|
//=======================================================================
|
|
IntTools_SurfaceRangeLocalizeData& IntTools_Context::SurfaceData
|
|
(const TopoDS_Face& aF)
|
|
{
|
|
Standard_Address anAdr;
|
|
IntTools_SurfaceRangeLocalizeData* pSData;
|
|
//
|
|
if (!myProjSDataMap.IsBound(aF)) {
|
|
pSData=(IntTools_SurfaceRangeLocalizeData*)
|
|
myAllocator->Allocate(sizeof(IntTools_SurfaceRangeLocalizeData));
|
|
new (pSData) IntTools_SurfaceRangeLocalizeData
|
|
(3,
|
|
3,
|
|
10. * Precision::PConfusion(),
|
|
10. * Precision::PConfusion());
|
|
//
|
|
anAdr=(Standard_Address)pSData;
|
|
myProjSDataMap.Bind(aF, anAdr);
|
|
}
|
|
|
|
else {
|
|
anAdr=myProjSDataMap.Find(aF);
|
|
pSData=(IntTools_SurfaceRangeLocalizeData*)anAdr;
|
|
}
|
|
return *pSData;
|
|
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : ComputePE
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Integer IntTools_Context::ComputePE
|
|
(const gp_Pnt& aP1,
|
|
const Standard_Real aTolP1,
|
|
const TopoDS_Edge& aE2,
|
|
Standard_Real& aT)
|
|
{
|
|
if (!BRep_Tool::IsGeometric(aE2)) {
|
|
return -2;
|
|
}
|
|
Standard_Real aDist, aTolE2, aTolSum;
|
|
Standard_Integer aNbProj;
|
|
//
|
|
GeomAPI_ProjectPointOnCurve& aProjector=ProjPC(aE2);
|
|
aProjector.Perform(aP1);
|
|
|
|
aNbProj=aProjector.NbPoints();
|
|
if (!aNbProj) {
|
|
return -3;
|
|
}
|
|
//
|
|
aDist=aProjector.LowerDistance();
|
|
//
|
|
aTolE2=BRep_Tool::Tolerance(aE2);
|
|
aTolSum = aTolP1 + aTolE2 + Precision::Confusion();
|
|
//
|
|
aT=aProjector.LowerDistanceParameter();
|
|
if (aDist > aTolSum) {
|
|
return -4;
|
|
}
|
|
return 0;
|
|
}
|
|
//=======================================================================
|
|
//function : ComputeVE
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Integer IntTools_Context::ComputeVE
|
|
(const TopoDS_Vertex& aV1,
|
|
const TopoDS_Edge& aE2,
|
|
Standard_Real& aParam,
|
|
Standard_Real& aTolVnew)
|
|
{
|
|
if (BRep_Tool::Degenerated(aE2)) {
|
|
return -1;
|
|
}
|
|
if (!BRep_Tool::IsGeometric(aE2)) {
|
|
return -2;
|
|
}
|
|
Standard_Real aDist, aTolV1, aTolE2, aTolSum;
|
|
Standard_Integer aNbProj;
|
|
gp_Pnt aP;
|
|
//
|
|
aP=BRep_Tool::Pnt(aV1);
|
|
//
|
|
GeomAPI_ProjectPointOnCurve& aProjector=ProjPC(aE2);
|
|
aProjector.Perform(aP);
|
|
|
|
aNbProj=aProjector.NbPoints();
|
|
if (!aNbProj) {
|
|
return -3;
|
|
}
|
|
//
|
|
aDist=aProjector.LowerDistance();
|
|
//
|
|
aTolV1=BRep_Tool::Tolerance(aV1);
|
|
aTolE2=BRep_Tool::Tolerance(aE2);
|
|
aTolSum = aTolV1 + aTolE2 + Precision::Confusion();
|
|
//
|
|
aTolVnew=aDist+aTolE2;
|
|
//
|
|
aParam=aProjector.LowerDistanceParameter();
|
|
if (aDist > aTolSum) {
|
|
return -4;
|
|
}
|
|
return 0;
|
|
}
|
|
//=======================================================================
|
|
//function : ComputeVF
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Integer IntTools_Context::ComputeVF
|
|
(const TopoDS_Vertex& aV1,
|
|
const TopoDS_Face& aF2,
|
|
Standard_Real& U,
|
|
Standard_Real& V,
|
|
Standard_Real& aTolVnew)
|
|
{
|
|
Standard_Real aTolV1, aTolF2, aTolSum, aDist;
|
|
gp_Pnt aP;
|
|
|
|
aP=BRep_Tool::Pnt(aV1);
|
|
//
|
|
// 1. Check if the point is projectable on the surface
|
|
GeomAPI_ProjectPointOnSurf& aProjector=ProjPS(aF2);
|
|
aProjector.Perform(aP);
|
|
//
|
|
if (!aProjector.IsDone()) { // the point is not projectable on the surface
|
|
return -1;
|
|
}
|
|
//
|
|
// 2. Check the distance between the projection point and
|
|
// the original point
|
|
aDist=aProjector.LowerDistance();
|
|
//
|
|
aTolV1=BRep_Tool::Tolerance(aV1);
|
|
aTolF2=BRep_Tool::Tolerance(aF2);
|
|
//
|
|
aTolSum = aTolV1 + aTolF2 + Precision::Confusion();
|
|
aTolVnew = aDist + aTolF2;
|
|
//
|
|
if (aDist > aTolSum) {
|
|
// the distance is too large
|
|
return -2;
|
|
}
|
|
aProjector.LowerDistanceParameters(U, V);
|
|
//
|
|
gp_Pnt2d aP2d(U, V);
|
|
Standard_Boolean pri=IsPointInFace (aF2, aP2d);
|
|
if (!pri) {// the point lays on the surface but out of the face
|
|
return -3;
|
|
}
|
|
return 0;
|
|
}
|
|
//=======================================================================
|
|
//function : StatePointFace
|
|
//purpose :
|
|
//=======================================================================
|
|
TopAbs_State IntTools_Context::StatePointFace
|
|
(const TopoDS_Face& aF,
|
|
const gp_Pnt2d& aP2d)
|
|
{
|
|
TopAbs_State aState;
|
|
IntTools_FClass2d& aClass2d=FClass2d(aF);
|
|
aState=aClass2d.Perform(aP2d);
|
|
return aState;
|
|
}
|
|
//=======================================================================
|
|
//function : IsPointInFace
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsPointInFace
|
|
(const TopoDS_Face& aF,
|
|
const gp_Pnt2d& aP2d)
|
|
{
|
|
TopAbs_State aState=StatePointFace(aF, aP2d);
|
|
if (aState==TopAbs_OUT || aState==TopAbs_ON) {
|
|
return Standard_False;
|
|
}
|
|
return Standard_True;
|
|
}
|
|
//=======================================================================
|
|
//function : IsPointInFace
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsPointInFace
|
|
(const gp_Pnt& aP,
|
|
const TopoDS_Face& aF,
|
|
const Standard_Real aTol)
|
|
{
|
|
Standard_Boolean bIn;
|
|
Standard_Real aDist;
|
|
//
|
|
GeomAPI_ProjectPointOnSurf& aProjector=ProjPS(aF);
|
|
aProjector.Perform(aP);
|
|
//
|
|
bIn = aProjector.IsDone();
|
|
if (bIn) {
|
|
aDist = aProjector.LowerDistance();
|
|
if (aDist < aTol) {
|
|
Standard_Real U, V;
|
|
//
|
|
aProjector.LowerDistanceParameters(U, V);
|
|
gp_Pnt2d aP2D(U, V);
|
|
bIn = IsPointInFace(aF, aP2D);
|
|
}
|
|
}
|
|
//
|
|
return bIn;
|
|
}
|
|
//=======================================================================
|
|
//function : IsPointInOnFace
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsPointInOnFace(const TopoDS_Face& aF,
|
|
const gp_Pnt2d& aP2d)
|
|
{
|
|
TopAbs_State aState=StatePointFace(aF, aP2d);
|
|
if (aState==TopAbs_OUT) {
|
|
return Standard_False;
|
|
}
|
|
return Standard_True;
|
|
}
|
|
//=======================================================================
|
|
//function : IsValidPointForFace
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsValidPointForFace
|
|
(const gp_Pnt& aP,
|
|
const TopoDS_Face& aF,
|
|
const Standard_Real aTol)
|
|
{
|
|
Standard_Boolean bFlag;
|
|
Standard_Real Umin, U, V;
|
|
|
|
GeomAPI_ProjectPointOnSurf& aProjector=ProjPS(aF);
|
|
aProjector.Perform(aP);
|
|
|
|
bFlag=aProjector.IsDone();
|
|
if (bFlag) {
|
|
|
|
Umin=aProjector.LowerDistance();
|
|
//if (Umin > 1.e-3) { // it was
|
|
if (Umin > aTol) {
|
|
return !bFlag;
|
|
}
|
|
//
|
|
aProjector.LowerDistanceParameters(U, V);
|
|
gp_Pnt2d aP2D(U, V);
|
|
bFlag=IsPointInOnFace (aF, aP2D);
|
|
}
|
|
return bFlag;
|
|
}
|
|
//=======================================================================
|
|
//function : IsValidPointForFaces
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsValidPointForFaces
|
|
(const gp_Pnt& aP,
|
|
const TopoDS_Face& aF1,
|
|
const TopoDS_Face& aF2,
|
|
const Standard_Real aTol)
|
|
{
|
|
Standard_Boolean bFlag1, bFlag2;
|
|
|
|
bFlag1=IsValidPointForFace(aP, aF1, aTol);
|
|
if (!bFlag1) {
|
|
return bFlag1;
|
|
}
|
|
bFlag2=IsValidPointForFace(aP, aF2, aTol);
|
|
return bFlag2;
|
|
}
|
|
//=======================================================================
|
|
//function : IsValidBlockForFace
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsValidBlockForFace
|
|
(const Standard_Real aT1,
|
|
const Standard_Real aT2,
|
|
const IntTools_Curve& aC,
|
|
const TopoDS_Face& aF,
|
|
const Standard_Real aTol)
|
|
{
|
|
Standard_Boolean bFlag;
|
|
Standard_Real aTInterm;
|
|
gp_Pnt aPInterm;
|
|
|
|
aTInterm=IntTools_Tools::IntermediatePoint(aT1, aT2);
|
|
|
|
Handle(Geom_Curve) aC3D=aC.Curve();
|
|
// point 3D
|
|
aC3D->D0(aTInterm, aPInterm);
|
|
//
|
|
bFlag=IsValidPointForFace (aPInterm, aF, aTol);
|
|
return bFlag;
|
|
}
|
|
//=======================================================================
|
|
//function : IsValidBlockForFaces
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsValidBlockForFaces
|
|
(const Standard_Real aT1,
|
|
const Standard_Real aT2,
|
|
const IntTools_Curve& aC,
|
|
const TopoDS_Face& aF1,
|
|
const TopoDS_Face& aF2,
|
|
const Standard_Real aTol)
|
|
{
|
|
Standard_Boolean bFlag1, bFlag2;
|
|
//
|
|
Handle(Geom2d_Curve) aPC1 = aC.FirstCurve2d();
|
|
Handle(Geom2d_Curve) aPC2 = aC.SecondCurve2d();
|
|
if( !aPC1.IsNull() && !aPC2.IsNull() ) {
|
|
Standard_Real aMidPar = IntTools_Tools::IntermediatePoint(aT1, aT2);
|
|
gp_Pnt2d aPnt2D;
|
|
|
|
|
|
aPC1->D0(aMidPar, aPnt2D);
|
|
bFlag1 = IsPointInOnFace(aF1, aPnt2D);
|
|
|
|
if( !bFlag1 )
|
|
return bFlag1;
|
|
|
|
aPC2->D0(aMidPar, aPnt2D);
|
|
bFlag2 = IsPointInOnFace(aF2, aPnt2D);
|
|
return bFlag2;
|
|
}
|
|
//
|
|
|
|
bFlag1=IsValidBlockForFace (aT1, aT2, aC, aF1, aTol);
|
|
if (!bFlag1) {
|
|
return bFlag1;
|
|
}
|
|
bFlag2=IsValidBlockForFace (aT1, aT2, aC, aF2, aTol);
|
|
return bFlag2;
|
|
}
|
|
//=======================================================================
|
|
//function : IsVertexOnLine
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsVertexOnLine
|
|
(const TopoDS_Vertex& aV,
|
|
const IntTools_Curve& aC,
|
|
const Standard_Real aTolC,
|
|
Standard_Real& aT)
|
|
{
|
|
Standard_Boolean bRet;
|
|
Standard_Real aTolV;
|
|
//
|
|
aTolV=BRep_Tool::Tolerance(aV);
|
|
bRet=IntTools_Context::IsVertexOnLine(aV, aTolV, aC, aTolC , aT);
|
|
//
|
|
return bRet;
|
|
}
|
|
//=======================================================================
|
|
//function : IsVertexOnLine
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::IsVertexOnLine
|
|
(const TopoDS_Vertex& aV,
|
|
const Standard_Real aTolV,
|
|
const IntTools_Curve& aC,
|
|
const Standard_Real aTolC,
|
|
Standard_Real& aT)
|
|
{
|
|
Standard_Real aFirst, aLast, aDist, aTolSum;
|
|
Standard_Integer aNbProj;
|
|
gp_Pnt aPv;
|
|
|
|
aPv=BRep_Tool::Pnt(aV);
|
|
|
|
Handle(Geom_Curve) aC3D=aC.Curve();
|
|
|
|
|
|
aTolSum=aTolV+aTolC;
|
|
//
|
|
GeomAdaptor_Curve aGAC(aC3D);
|
|
GeomAbs_CurveType aType=aGAC.GetType();
|
|
if (aType==GeomAbs_BSplineCurve ||
|
|
aType==GeomAbs_BezierCurve) {
|
|
aTolSum=2.*aTolSum;
|
|
if (aTolSum<1.e-5) {
|
|
aTolSum=1.e-5;
|
|
}
|
|
}
|
|
else {
|
|
aTolSum=2.*aTolSum;//xft
|
|
if(aTolSum < 1.e-6)
|
|
aTolSum = 1.e-6;
|
|
}
|
|
//
|
|
aFirst=aC3D->FirstParameter();
|
|
aLast =aC3D->LastParameter();
|
|
//
|
|
//Checking extermities first
|
|
if (!Precision::IsInfinite(aFirst)) {
|
|
gp_Pnt aPCFirst=aC3D->Value(aFirst);
|
|
aDist=aPv.Distance(aPCFirst);
|
|
if (aDist < aTolSum) {
|
|
aT=aFirst;
|
|
//
|
|
if(aDist > aTolV) {
|
|
Extrema_LocateExtPC anExt(aPv, aGAC, aFirst, 1.e-10);
|
|
|
|
if(anExt.IsDone()) {
|
|
Extrema_POnCurv aPOncurve = anExt.Point();
|
|
aT = aPOncurve.Parameter();
|
|
|
|
if((aT > (aLast + aFirst) * 0.5) ||
|
|
(aPv.Distance(aPOncurve.Value()) > aTolSum) ||
|
|
(aPCFirst.Distance(aPOncurve.Value()) < Precision::Confusion()))
|
|
aT = aFirst;
|
|
}
|
|
else
|
|
{
|
|
// Local search may fail. Try to use more precise algo.
|
|
Extrema_ExtPC anExt2(aPv, aGAC, 1.e-10);
|
|
Standard_Real aMinDist = RealLast();
|
|
Standard_Integer aMinIdx = -1;
|
|
if (anExt2.IsDone()) {
|
|
for (Standard_Integer anIdx = 1; anIdx <= anExt2.NbExt(); anIdx++)
|
|
{
|
|
if ( anExt2.IsMin(anIdx) &&
|
|
anExt2.SquareDistance(anIdx) < aMinDist )
|
|
{
|
|
aMinDist = anExt2.SquareDistance(anIdx);
|
|
aMinIdx = anIdx;
|
|
}
|
|
}
|
|
}
|
|
if (aMinIdx != -1)
|
|
{
|
|
const Extrema_POnCurv& aPOncurve = anExt2.Point(aMinIdx);
|
|
aT = aPOncurve.Parameter();
|
|
|
|
if((aT > (aLast + aFirst) * 0.5) ||
|
|
(aPv.Distance(aPOncurve.Value()) > aTolSum) ||
|
|
(aPCFirst.Distance(aPOncurve.Value()) < Precision::Confusion()))
|
|
aT = aFirst;
|
|
}
|
|
}
|
|
|
|
}
|
|
//
|
|
return Standard_True;
|
|
}
|
|
}
|
|
//
|
|
//if (!Precision::IsInfinite(aFirst)) {
|
|
if (!Precision::IsInfinite(aLast)) {
|
|
gp_Pnt aPCLast=aC3D->Value(aLast);
|
|
aDist=aPv.Distance(aPCLast);
|
|
if (aDist < aTolSum) {
|
|
aT=aLast;
|
|
//
|
|
if(aDist > aTolV) {
|
|
Extrema_LocateExtPC anExt(aPv, aGAC, aLast, 1.e-10);
|
|
|
|
if(anExt.IsDone()) {
|
|
Extrema_POnCurv aPOncurve = anExt.Point();
|
|
aT = aPOncurve.Parameter();
|
|
|
|
if((aT < (aLast + aFirst) * 0.5) ||
|
|
(aPv.Distance(aPOncurve.Value()) > aTolSum) ||
|
|
(aPCLast.Distance(aPOncurve.Value()) < Precision::Confusion()))
|
|
aT = aLast;
|
|
}
|
|
else
|
|
{
|
|
// Local search may fail. Try to use more precise algo.
|
|
Extrema_ExtPC anExt2(aPv, aGAC, 1.e-10);
|
|
Standard_Real aMinDist = RealLast();
|
|
Standard_Integer aMinIdx = -1;
|
|
if (anExt2.IsDone()) {
|
|
for (Standard_Integer anIdx = 1; anIdx <= anExt2.NbExt(); anIdx++)
|
|
{
|
|
if ( anExt2.IsMin(anIdx) &&
|
|
anExt2.SquareDistance(anIdx) < aMinDist )
|
|
{
|
|
aMinDist = anExt2.SquareDistance(anIdx);
|
|
aMinIdx = anIdx;
|
|
}
|
|
}
|
|
}
|
|
if (aMinIdx != -1)
|
|
{
|
|
const Extrema_POnCurv& aPOncurve = anExt2.Point(aMinIdx);
|
|
aT = aPOncurve.Parameter();
|
|
|
|
if((aT < (aLast + aFirst) * 0.5) ||
|
|
(aPv.Distance(aPOncurve.Value()) > aTolSum) ||
|
|
(aPCLast.Distance(aPOncurve.Value()) < Precision::Confusion()))
|
|
aT = aLast;
|
|
}
|
|
}
|
|
}
|
|
//
|
|
return Standard_True;
|
|
}
|
|
}
|
|
//
|
|
GeomAPI_ProjectPointOnCurve& aProjector=ProjPT(aC3D);
|
|
aProjector.Perform(aPv);
|
|
|
|
aNbProj=aProjector.NbPoints();
|
|
if (!aNbProj) {
|
|
Handle(Geom_BoundedCurve) aBC=
|
|
Handle(Geom_BoundedCurve)::DownCast(aC3D);
|
|
if (!aBC.IsNull()) {
|
|
gp_Pnt aPStart=aBC->StartPoint();
|
|
gp_Pnt aPEnd =aBC->EndPoint();
|
|
|
|
aDist=aPv.Distance(aPStart);
|
|
if (aDist < aTolSum) {
|
|
aT=aFirst;
|
|
return Standard_True;
|
|
}
|
|
|
|
aDist=aPv.Distance(aPEnd);
|
|
if (aDist < aTolSum) {
|
|
aT=aLast;
|
|
return Standard_True;
|
|
}
|
|
}
|
|
|
|
return Standard_False;
|
|
}
|
|
|
|
aDist=aProjector.LowerDistance();
|
|
|
|
if (aDist > aTolSum) {
|
|
return Standard_False;
|
|
}
|
|
|
|
aT=aProjector.LowerDistanceParameter();
|
|
|
|
return Standard_True;
|
|
}
|
|
//=======================================================================
|
|
//function : ProjectPointOnEdge
|
|
//purpose :
|
|
//=======================================================================
|
|
Standard_Boolean IntTools_Context::ProjectPointOnEdge
|
|
(const gp_Pnt& aP,
|
|
const TopoDS_Edge& anEdge,
|
|
Standard_Real& aT)
|
|
{
|
|
Standard_Integer aNbPoints;
|
|
|
|
GeomAPI_ProjectPointOnCurve& aProjector=ProjPC(anEdge);
|
|
aProjector.Perform(aP);
|
|
|
|
aNbPoints=aProjector.NbPoints();
|
|
if (aNbPoints) {
|
|
aT=aProjector.LowerDistanceParameter();
|
|
return Standard_True;
|
|
}
|
|
return Standard_False;
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : SetPOnSProjectionTolerance
|
|
//purpose :
|
|
//=======================================================================
|
|
void IntTools_Context::SetPOnSProjectionTolerance(const Standard_Real theValue)
|
|
{
|
|
myPOnSTolerance = theValue;
|
|
clearCachedPOnSProjectors();
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : clearCachedPOnSProjectors
|
|
//purpose :
|
|
//=======================================================================
|
|
void IntTools_Context::clearCachedPOnSProjectors()
|
|
{
|
|
GeomAPI_ProjectPointOnSurf* pProjPS;
|
|
BOPCol_DataMapIteratorOfDataMapOfShapeAddress aIt(myProjPSMap);
|
|
for (; aIt.More(); aIt.Next()) {
|
|
Standard_Address anAdr=aIt.Value();
|
|
pProjPS=(GeomAPI_ProjectPointOnSurf*)anAdr;
|
|
(*pProjPS).~GeomAPI_ProjectPointOnSurf();
|
|
myAllocator->Free(anAdr);
|
|
}
|
|
myProjPSMap.Clear();
|
|
}
|