1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00
occt/src/TopOpeBRepBuild/TopOpeBRepBuild_Builder1.cxx
omy d3f26155b5 0024058: Eliminate compiler warning C4702 in MSVC++ with warning level 4
Got rid of most of warnings of C4702 type: unreachable code.
Returned some #ifdef DEB
Fixed tabs formatting
Fixed some mistakes in code
2013-07-12 12:54:01 +04:00

1959 lines
63 KiB
C++
Executable File

// Created on: 1999-09-29
// Created by: Maxim ZVEREV
// Copyright (c) 1999-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.
#include <TopOpeBRepBuild_Builder1.ixx>
#include <TopOpeBRepTool_ShapeExplorer.hxx>
#include <TopAbs_Orientation.hxx>
#include <TopoDS.hxx>
#include <TopOpeBRepDS_HDataStructure.hxx>
#include <TopOpeBRepDS_ShapeWithState.hxx>
#include <TopOpeBRepBuild_WireEdgeSet.hxx>
#include <TopAbs_ShapeEnum.hxx>
#include <TopExp.hxx>
#include <TopOpeBRepTool_2d.hxx>
#include <Geom2d_Curve.hxx>
#include <BRepAdaptor_Surface.hxx>
#include <BRep_Tool.hxx>
#include <TopOpeBRepDS_EXPORT.hxx>
#include <Geom2dAPI_ProjectPointOnCurve.hxx>
#include <TopOpeBRepBuild_Tools.hxx>
#include <BRepTopAdaptor_FClass2d.hxx>
#include <TopOpeBRepDS_Interference.hxx>
#include <TopOpeBRepDS_ShapeShapeInterference.hxx>
#include <TopOpeBRepDS_ListOfInterference.hxx>
#include <TopOpeBRepDS_ListIteratorOfListOfInterference.hxx>
#include <TopOpeBRepDS_Transition.hxx>
#include <TColStd_MapOfInteger.hxx>
//define parameter division number as 10*e^(-PI) = 0.43213918
const Standard_Real PAR_T = 0.43213918;
static TopTools_IndexedMapOfShape mySDEdgeMap;
static TopAbs_State ClassifyEdgeToFaceByOnePoint(const TopoDS_Edge& E,
const TopoDS_Face& F);
//modified by NIZHNY-MZV Thu Apr 20 09:58:59 2000
///////////////////
//this variable used to separate old algo from the new one
//because new algo can not be used in LocOpe and Mechanical Features (for the moment)
//that's why we use new algo only in BRepAlgoAPI_BooleanOperation
//in all other cases old algo is called (see the methods GFillSolidSFS, GFillShellSFS, etc.);
Standard_Boolean GLOBAL_USE_NEW_BUILDER = Standard_False;
//=======================================================================
//function : Constructor
//purpose :
//=======================================================================
TopOpeBRepBuild_Builder1::TopOpeBRepBuild_Builder1(const TopOpeBRepDS_BuildTool& BT)
: TopOpeBRepBuild_Builder(BT)
{
mySameDomMap.Clear();
myMapOfEdgeFaces.Clear();
mySplitsONtoKeep.Clear();
myProcessedPartsOut2d.Clear();
myProcessedPartsON2d.Clear();
}
//modified by NIZNHY-PKV Mon Dec 16 11:37:59 2002 f
/*
//=======================================================================
//function : Destroy
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::Destroy()
{
}
*/
//modified by NIZNHY-PKV Mon Dec 16 11:38:05 2002 t
//=======================================================================
//function : Clear
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::Clear()
{
TopOpeBRepBuild_Builder::Clear();
// mySameDomMap.Clear();
// myMapOfEdgeFaces.Clear();
// mySplitsONtoKeep.Clear();
// myProcessedPartsOut2d.Clear();
// myProcessedPartsON2d.Clear();
// myDataStructure -> ChangeDS().ChangeMapOfShapeWithStateObj().Clear();
// myDataStructure -> ChangeDS().ChangeMapOfShapeWithStateTool().Clear();
}
//=======================================================================
//function : Perform
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::Perform(const Handle(TopOpeBRepDS_HDataStructure)& HDS)
{
TopOpeBRepBuild_Builder::Perform(HDS);
}
//=======================================================================
//function : Perform
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::Perform(const Handle(TopOpeBRepDS_HDataStructure)& HDS,
const TopoDS_Shape& S1,
const TopoDS_Shape& S2)
{
//modified by NIZHNY-MZV Wed Apr 19 17:23:12 2000
//see the comments at the top of file about this global variable
if(!GLOBAL_USE_NEW_BUILDER) {
TopOpeBRepBuild_Builder::Perform(HDS, S1, S2);
return;
}
mySameDomMap.Clear();
myMapOfEdgeFaces.Clear();
mySplitsONtoKeep.Clear();
myProcessedPartsOut2d.Clear();
myProcessedPartsON2d.Clear();
myShape1 = S1; myShape2 = S2;
Perform(HDS);
myIsKPart = FindIsKPart();
if((myIsKPart == 1) || (myIsKPart == 5))
myIsKPart=4;
if (myIsKPart==4) {
// For the moment States will be calculated in case SOLID/SOLID only
PerformShapeWithStates();
}
}
//=======================================================================
//function : MergeKPart
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::MergeKPart(const TopAbs_State TB1,
const TopAbs_State TB2)
{
TopOpeBRepBuild_Builder::MergeKPart(TB1, TB2);
}
//=======================================================================
//function : MergeKPart
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::MergeKPart()
{
if ( myIsKPart == 1 ) { // iskole
MergeKPartiskole();
}
else if ( myIsKPart == 5 ) { // iskoletge
MergeKPartiskoletge();
}
else if (myIsKPart == 2) { // isdisj
MergeKPartisdisj();
}
else if ( myIsKPart == 3 ) { // isfafa
MergeKPartisfafa();
}
else if ( myIsKPart == 4 ) { // issoso
MergeKPartissoso();
TopTools_ListIteratorOfListOfShape its(Merged(myShape1,myState1));
for (; its.More(); its.Next()) {
CorrectResult2d(its.Value());
}
}
End();
}
//=======================================================================
//function : GFillSolidSFS
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillSolidSFS(const TopoDS_Shape& SO1,
const TopTools_ListOfShape& LSO2,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_ShellFaceSet& SFS)
{
//modified by NIZHNY-MZV Wed Apr 19 17:23:12 2000
//see the comments at the top of file about this global variable
if(!GLOBAL_USE_NEW_BUILDER) {
TopOpeBRepBuild_Builder::GFillSolidSFS(SO1, LSO2, G1, SFS);
return;
}
myMapOfEdgeFaces.Clear();
TopExp::MapShapesAndAncestors(myShape1, TopAbs_EDGE, TopAbs_FACE, myMapOfEdgeFaces);
TopExp::MapShapesAndAncestors(myShape2, TopAbs_EDGE, TopAbs_FACE, myMapOfEdgeFaces);
TopAbs_State TB1,TB2;
G1.StatesON(TB1,TB2);
// printf("TB1 =%d, TB2 = %d\n", TB1, TB2);
Standard_Boolean RevOri1 = G1.IsToReverse1();
TopoDS_Shape SOF = SO1;
mySolidToFill = TopoDS::Solid(SOF);
TopOpeBRepTool_ShapeExplorer exShell(SOF,TopAbs_SHELL);
for (; exShell.More(); exShell.Next()) {
TopoDS_Shape SH = exShell.Current();
Standard_Boolean hasshape = myDataStructure->HasShape(SH);
if ( ! hasshape ) {
// shell SH is not in DS : Get its state (to the LS02) from map and define to keep or not
TopAbs_State shSt = myDataStructure -> DS().GetShapeWithState(SH).State();
Standard_Boolean keep = (shSt == TB1) ? Standard_True : Standard_False;
if (keep) {
TopAbs_Orientation oriSH = SH.Orientation();
TopAbs_Orientation neworiSH = Orient(oriSH,RevOri1);
SH.Orientation(neworiSH);
SFS.AddShape(SH);
}
}
else { // shell SH has faces(s) with geometry : split SH faces
GFillShellSFS(SH,LSO2,G1,SFS);
}
}
}
//=======================================================================
//function : GFillShellSFS
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillShellSFS (const TopoDS_Shape& SH,
const TopTools_ListOfShape& LSO2,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_ShellFaceSet& SFS)
{
//modified by NIZHNY-MZV Wed Apr 19 17:23:12 2000
//see the comments at the top of file about this global variable
if(!GLOBAL_USE_NEW_BUILDER) {
TopOpeBRepBuild_Builder::GFillShellSFS(SH, LSO2, G1, SFS);
return;
}
TopAbs_State TB1,TB2;
G1.StatesON(TB1,TB2);
Standard_Boolean RevOri1 = G1.IsToReverse1();
TopOpeBRepTool_ShapeExplorer exFace;
TopoDS_Shape SH1 = SH;// SH1.Orientation(TopAbs_FORWARD);
//1) process firstly same domain faces and non-interference faces
for (exFace.Init(SH1,TopAbs_FACE); exFace.More(); exFace.Next()) {
TopoDS_Shape FOR = exFace.Current();
if(!myDataStructure -> HasShape(FOR)) {
//DS doesn't contain FACE , get its state and define to keep or not
TopAbs_State shSt = myDataStructure -> DS().GetShapeWithState(FOR).State();
Standard_Boolean keep = (shSt == TB1) ? Standard_True : Standard_False;
if (keep) {
TopAbs_Orientation oriF = FOR.Orientation();
TopAbs_Orientation neworiF = Orient(oriF,RevOri1);
FOR.Orientation(neworiF);
SFS.AddElement(FOR);
}
continue;
}
Standard_Boolean hsd = myDataStructure->HasSameDomain(FOR);
if ( hsd && !mySameDomMap.Contains(FOR))
GFillFaceSameDomSFS(FOR,LSO2,G1,SFS);
}
//2 Process all other faces
for (exFace.Init(SH1,TopAbs_FACE); exFace.More(); exFace.Next()) {
TopoDS_Shape FOR = exFace.Current();
if(!myDataStructure -> HasShape(FOR)
||
myDataStructure->HasSameDomain(FOR))
continue;
GFillFaceNotSameDomSFS(FOR, LSO2, G1, SFS);
}
} // GFillShellSFS
//=======================================================================
//function : GFillFaceNotSameDomSFS
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillFaceNotSameDomSFS(const TopoDS_Shape& FOR,
const TopTools_ListOfShape& LSO2,
const TopOpeBRepBuild_GTopo& Gin,
TopOpeBRepBuild_ShellFaceSet& SFS)
{
TopOpeBRepBuild_GTopo G1 = Gin;
Standard_Boolean RevOri = Standard_False;
G1.SetReverse(RevOri);
TopAbs_State TB1,TB2; G1.StatesON(TB1,TB2);
// work on a FORWARD face <FForward>
TopoDS_Shape FF = FOR; FF.Orientation(TopAbs_FORWARD);
// make a WireEdgeSet WES on face FF
TopOpeBRepBuild_WireEdgeSet WES(FF,this);
// Add ON parts (edges ON solid)
GFillONPartsWES(FOR,G1,LSO2,WES);
// save these edges
TopTools_ListOfShape anEdgesON;
TopTools_ListIteratorOfListOfShape it;
if (myProcessON) {
Standard_Boolean toRevOri = Opefus();
for (it.Initialize(WES.StartElements()); it.More(); it.Next())
anEdgesON.Append(toRevOri ? it.Value().Reversed() : it.Value());
myONElemMap.Clear();
}
// split the edges of FF : add split edges to WES
GFillFaceNotSameDomWES(FF,LSO2,G1,WES);
// add edges built on curves supported by FF
GFillCurveTopologyWES(FF,G1,WES);
myEdgeAvoid.Clear();
// mark FF as split TB1
MarkSplit(FF,TB1);
// build the new faces LOF on FF from the Wire/Edge set WES
TopTools_ListOfShape LOF;
GWESMakeFaces(FF,WES,LOF);
if (myProcessON && (!anEdgesON.IsEmpty() || !myONElemMap.IsEmpty())) {
// try to make patches with only ON parts.
// prepare the map of used edges to not take the same matter two times
TopTools_IndexedMapOfOrientedShape aMapOE;
for (it.Initialize(LOF); it.More(); it.Next())
for (TopExp_Explorer ex(it.Value(),TopAbs_EDGE); ex.More(); ex.Next())
aMapOE.Add(ex.Current());
FillOnPatches(anEdgesON,FOR,aMapOE);
myONElemMap.Clear();
}
// LOFS : LOF faces located TB1 / LSclass = split faces of state TB1 of FF
TopTools_ListOfShape& LOFS = ChangeSplit(FF,TB1);
LOFS.Clear();
GKeepShapes(FF,myEmptyShapeList,TB1,LOF,LOFS);
GSplitFaceSFS(FOR, LSO2, Gin, SFS);
} // GFillFaceSFS
//=======================================================================
//function : GFillFaceNotSameDomWES
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillFaceNotSameDomWES(const TopoDS_Shape& FOR1,
const TopTools_ListOfShape& LFclass,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_WireEdgeSet& WES)
{
TopAbs_State TB1,TB2; G1.StatesON(TB1,TB2);
Standard_Boolean RevOri1 = G1.IsToReverse1();
mySourceShapes.Clear();
// work on a FORWARD face FF
TopoDS_Shape FF = FOR1; FF.Orientation(TopAbs_FORWARD);
TopOpeBRepTool_ShapeExplorer exWire(FF,TopAbs_WIRE);
for (; exWire.More(); exWire.Next()) {
TopoDS_Shape W = exWire.Current();
Standard_Boolean hasshape = myDataStructure->HasShape(W);
if ( ! hasshape ) {
// wire W is not in DS : get its state and define to keep or not
TopAbs_State shSt = myDataStructure -> DS().GetShapeWithState(W).State();
Standard_Boolean keep = (shSt == TB1) ? Standard_True : Standard_False;
if (keep || (myProcessON && shSt == TopAbs_ON)) {
TopAbs_Orientation oriW = W.Orientation();
TopAbs_Orientation neworiW = Orient(oriW,RevOri1);
W.Orientation(neworiW);
if (keep) WES.AddShape(W);
else myONElemMap.Add(W);
mySourceShapes.Add(W);
}
}
else { // wire W has edges(s) with geometry : split W edges
GFillWireNotSameDomWES(W,LFclass,G1,WES);
}
}
return;
} // GFillFaceWES
//=======================================================================
//function : GFillWireNotSameDomWES
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillWireNotSameDomWES(const TopoDS_Shape& W,
const TopTools_ListOfShape& LSclass,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_WireEdgeSet& WES)
{
TopAbs_State TB1,TB2; G1.StatesON(TB1,TB2);
Standard_Boolean RevOri1 = G1.IsToReverse1();
TopoDS_Shape WW = W; //WW.Orientation(TopAbs_FORWARD);
TopOpeBRepTool_ShapeExplorer exEdge(WW,TopAbs_EDGE);
for (; exEdge.More(); exEdge.Next()) {
TopoDS_Shape EOR = exEdge.Current();
Standard_Boolean hasshape = myDataStructure->HasShape(EOR);
if ( ! hasshape ) {
// edge EOR is not in DS : get its state and define to keep or not
TopAbs_State shSt = myDataStructure -> DS().GetShapeWithState(EOR).State();
Standard_Boolean keep = (shSt == TB1) ? Standard_True : Standard_False;
if (keep || (myProcessON && shSt == TopAbs_ON)) {
TopAbs_Orientation oriE = EOR.Orientation();
TopAbs_Orientation neworiE = Orient(oriE,RevOri1);
EOR.Orientation(neworiE);
if (keep) WES.AddElement(EOR);
else myONElemMap.Add(EOR);
mySourceShapes.Add(EOR);
}
}
else { // wire W has edges(s) with geometry : split W edges
GFillEdgeNotSameDomWES(EOR,LSclass,G1,WES);
}
}
} // GFillWireWES
//=======================================================================
//function : GFillEdgeNotSameDomWES
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillEdgeNotSameDomWES(const TopoDS_Shape& EOR,
const TopTools_ListOfShape& /*LSclass*/,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_WireEdgeSet& WES)
{
TopAbs_State TB1,TB2; G1.StatesON(TB1,TB2);
Standard_Boolean RevOri1 = G1.IsToReverse1();
TopAbs_Orientation oriE = EOR.Orientation();
TopAbs_Orientation neworiE = Orient(oriE,RevOri1);
//1) Get split parts of edge with state TB1
const TopTools_ListOfShape& LSE = myDataStructure -> DS().GetShapeWithState(EOR).Part(TB1);
TopTools_ListIteratorOfListOfShape it (LSE);
for(; it.More(); it.Next()) {
TopoDS_Edge newE = TopoDS::Edge(it.Value());
newE.Orientation(neworiE);
WES.AddStartElement(newE);
mySourceShapes.Add(newE);
}
//2) Get ON parts of the edge and define to keep it or not
const TopTools_ListOfShape& LSEOn = myDataStructure -> DS().GetShapeWithState(EOR).Part(TopAbs_ON);
TopTools_ListIteratorOfListOfShape itON (LSEOn);
for(; itON.More(); itON.Next()) {
TopoDS_Edge newE = TopoDS::Edge(itON.Value());
newE.Orientation(neworiE);
if(mySplitsONtoKeep.Contains(newE)) {
WES.AddStartElement(newE);
continue;
}
// we keep all degenerated edges here because FillONPartsWES can not process them
if(BRep_Tool::Degenerated(newE)) {
WES.AddStartElement(newE);
mySourceShapes.Add(newE);
}
if (myProcessON) {
myONElemMap.Add(newE);
mySourceShapes.Add(newE);
}
}
} // GFillEdgeWES
/////////////////// ALL FUNCTIONS FOR SAME DOMAIN FACES
//=======================================================================
//function : GFillFaceSameDomSFS
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillFaceSameDomSFS(const TopoDS_Shape& FOR,
const TopTools_ListOfShape& LSO2,
const TopOpeBRepBuild_GTopo& Gin,
TopOpeBRepBuild_ShellFaceSet& SFS)
{
myProcessedPartsOut2d.Clear();
myProcessedPartsON2d.Clear();
myMapOfEdgeWithFaceState.Clear();
mySDEdgeMap.Clear();
mySourceShapes.Clear();
//we process all same domain faces during cycling throught the Shape1
if(myDataStructure -> DS().AncestorRank(FOR) != 1)
return;
TopOpeBRepBuild_GTopo G1 = Gin;
TopAbs_State TB1,TB2; G1.StatesON(TB1,TB2);
// work on a FORWARD face <FForward>
TopoDS_Shape FF = FOR; FF.Orientation(TopAbs_FORWARD);
// make a WireEdgeSet WES on face FF
TopOpeBRepBuild_WireEdgeSet WES(FF,this);
// split the edges of FF : add split edges to WES
GFillFaceSameDomWES(FOR,LSO2,G1,WES);
myEdgeAvoid.Clear();
// mark FF as split TB1
MarkSplit(FF,TB1);
// build the new faces LOF on FF from the Wire/Edge set WES
TopTools_ListOfShape LOF, oriLOF;
GWESMakeFaces(FF,WES,LOF);
// LOFS : LOF faces located TB1 / LSclass = split faces of state TB1 of FF
TopTools_ListOfShape& LOFS = ChangeSplit(FF,TB1);
//orientate new faces by the right way
Standard_Boolean OrigRev = (FOR.Orientation() == TopAbs_FORWARD ? Standard_False : Standard_True);
TopTools_ListIteratorOfListOfShape LOFit(LOF);
for(; LOFit.More(); LOFit.Next()) {
TopoDS_Shape aFace = LOFit.Value();
TopTools_IndexedMapOfShape aEM;
TopExp::MapShapes(aFace, TopAbs_EDGE, aEM);
Standard_Boolean rev = Standard_False;
for(Standard_Integer i = 1; i <= aEM.Extent(); i++) {
const TopoDS_Shape& anEdge = aEM(i);
if(myMapOfEdgeWithFaceState.IsBound(anEdge)) {
rev = (Standard_Boolean)myMapOfEdgeWithFaceState.Find(anEdge);
break;
}
}
if(OrigRev)
aFace.Reverse();
if(rev)
aFace.Reverse();
oriLOF.Append(aFace);
SFS.AddStartElement(aFace);
}
LOFS.Clear();
GKeepShapes(FF,myEmptyShapeList,TB1,oriLOF,LOFS);
} // GFillFaceSFS
//=======================================================================
//function : GFillFaceSameDomWES
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillFaceSameDomWES(const TopoDS_Shape& FOR1,
const TopTools_ListOfShape& /*LFclass*/,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_WireEdgeSet& WES)
{
TopAbs_State TB1,TB2; G1.StatesON(TB1,TB2);
myBaseFaceToFill = TopoDS::Face(FOR1);
TopTools_IndexedMapOfShape curSameDomMap;
curSameDomMap.Add(FOR1);
Standard_Integer i, nF;
for(i = 1; i <= curSameDomMap.Extent(); i++) {
TopTools_ListIteratorOfListOfShape it = myDataStructure -> SameDomain(curSameDomMap(i));
for(; it.More(); it.Next()) {
const TopoDS_Shape& SDF = it.Value();
curSameDomMap.Add(SDF);
mySameDomMap.Add(SDF);
TopExp::MapShapes(SDF, TopAbs_EDGE, mySDEdgeMap);
}
}
nF = curSameDomMap.Extent();
for(i = 1; i<= nF; i++) {
TopoDS_Shape curF = curSameDomMap(i);
TopoDS_Shape curFF = curF;
curFF.Orientation(TopAbs_FORWARD);
mySDFaceToFill = TopoDS::Face(curF);
Standard_Integer iref = myDataStructure -> DS().AncestorRank(curFF);
TopAbs_State TB;
Standard_Boolean RevOri = Standard_False;
TopOpeBRepBuild_GTopo GFTW = G1;
if(iref == 1) {//object
TB = TB1;
RevOri = G1.IsToReverse1();
}
else {//tool
RevOri = G1.IsToReverse2();
TB = TB2;
if(RevOri)
GFTW = G1.CopyPermuted();
}
//we need to pass GTopo according to ancestor rank
GFillCurveTopologyWES(curFF,GFTW,WES);
//process ON parts from not SD faces
PerformONParts(curFF, curSameDomMap, G1, WES);
const TopTools_ListOfShape& LSF = myDataStructure -> DS().ShapeSameDomain(curFF);
TopOpeBRepTool_ShapeExplorer exWire(curFF,TopAbs_WIRE);
for (; exWire.More(); exWire.Next()) {
TopoDS_Shape W = exWire.Current();
Standard_Boolean hasshape = myDataStructure->HasShape(W);
TopAbs_State shSt = myDataStructure -> DS().GetShapeWithState(W).State();
if ( ! hasshape && (shSt != TopAbs_ON)) {
// wire W is not in DS : get its state and define to keep or not
Standard_Boolean keep = (shSt == TB) ? Standard_True : Standard_False;
if (keep) {
TopAbs_Orientation oriW = W.Orientation();
TopAbs_Orientation neworiW = Orient(oriW,RevOri);
if(myBaseFaceToFill != mySDFaceToFill)
TopOpeBRepBuild_Tools::UpdatePCurves(TopoDS::Wire(W),
TopoDS::Face(mySDFaceToFill),
TopoDS::Face(myBaseFaceToFill));
else {
mySourceShapes.Add(W);
}
TopExp_Explorer we(W, TopAbs_EDGE);
Standard_Boolean stateOfFaceOri = Standard_False;
Standard_Boolean UseEdges = Standard_False;
for(; we.More(); we.Next()) {
TopoDS_Edge EOR = TopoDS::Edge(we.Current());
TopAbs_Orientation oldori = EOR.Orientation();
OrientateEdgeOnFace(EOR,TopoDS::Face(myBaseFaceToFill),
TopoDS::Face(mySDFaceToFill), G1, stateOfFaceOri);
// OrientateEdgeOnFace(TopoDS::Edge(EOR), TopoDS::Face(myBaseFaceToFill),
// TopoDS::Face(mySDFaceToFill), G1, stateOfFaceOri);
if(EOR.Orientation() != oldori) {
UseEdges = Standard_True;
WES.AddStartElement(EOR);
}
myMapOfEdgeWithFaceState.Bind(EOR, (Standard_Integer)stateOfFaceOri);
}
if(!UseEdges) {
W.Orientation(neworiW);
WES.AddShape(W);
}
}
}
else { // wire W has edges(s) with geometry : split W edges
GFillWireSameDomWES(W, LSF,G1,WES);
}
}
}
return;
} // GFillFaceWES
//=======================================================================
//function : GFillWireSameDomWES
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillWireSameDomWES(const TopoDS_Shape& W,
const TopTools_ListOfShape& LSclass,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_WireEdgeSet& WES)
{
TopAbs_State TB1,TB2; G1.StatesON(TB1,TB2);
TopoDS_Shape WW = W; //WW.Orientation(TopAbs_FORWARD);
Standard_Integer iref = myDataStructure -> DS().AncestorRank(W);
Standard_Boolean RevOri;
TopAbs_State TB;
if(iref == 1) {//object
TB = TB1;
RevOri = G1.IsToReverse1();
}
else {//tool
RevOri = G1.IsToReverse2();
TB = TB2;
}
TopOpeBRepTool_ShapeExplorer exEdge(WW,TopAbs_EDGE);
for (; exEdge.More(); exEdge.Next()) {
TopoDS_Shape EOR = exEdge.Current();
Standard_Boolean hasshape = myDataStructure->HasShape(EOR);
if ( ! hasshape ) {
// edge EOR is not in DS : get its state and define to keep or not
TopAbs_State shSt = myDataStructure -> DS().GetShapeWithState(EOR).State();
Standard_Boolean keep = (shSt == TB) ? Standard_True : Standard_False;
if (keep) {
TopAbs_Orientation oriE = EOR.Orientation();
Orient(oriE,RevOri);
if(mySDFaceToFill != myBaseFaceToFill) {
TopOpeBRepBuild_Tools::UpdateEdgeOnFace(TopoDS::Edge(EOR),
TopoDS::Face(mySDFaceToFill),
TopoDS::Face(myBaseFaceToFill));
}
else {
mySourceShapes.Add(EOR);
}
Standard_Boolean stateOfFaceOri = Standard_False;
OrientateEdgeOnFace(TopoDS::Edge(EOR), TopoDS::Face(myBaseFaceToFill),
TopoDS::Face(mySDFaceToFill), G1, stateOfFaceOri);
myMapOfEdgeWithFaceState.Bind(EOR, (Standard_Integer)stateOfFaceOri);
WES.AddElement(EOR);
}
}
else { // wire W has edges(s) with geometry : split W edges
GFillEdgeSameDomWES(EOR,LSclass,G1,WES);
}
}
} // GFillWireWES
//=======================================================================
//function : GFillEdgeSameDomWES
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GFillEdgeSameDomWES(const TopoDS_Shape& EOR,
const TopTools_ListOfShape& LSclass,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_WireEdgeSet& WES)
{
TopAbs_State TB1,TB2, TB; G1.StatesON(TB1,TB2);
Standard_Integer iref = myDataStructure -> DS().AncestorRank(EOR);
Standard_Boolean RevOri;
if(iref == 1) {//object
TB = TB1;
RevOri = G1.IsToReverse1();
}
else {//tool
RevOri = G1.IsToReverse2();
TB = TB2;
}
TopAbs_Orientation oriE = EOR.Orientation();
Orient(oriE,RevOri);
//1) Get split parts of edge with state TB
const TopTools_ListOfShape& LSE = myDataStructure -> DS().GetShapeWithState(EOR).Part(TB);
TopTools_ListIteratorOfListOfShape it (LSE);
Standard_Boolean first = Standard_True;
for(; it.More(); it.Next()) {
TopoDS_Edge newE = TopoDS::Edge(it.Value());
newE.Orientation(oriE);
if(mySDFaceToFill != myBaseFaceToFill) {
TopOpeBRepBuild_Tools::UpdateEdgeOnFace(newE,
TopoDS::Face(mySDFaceToFill),
TopoDS::Face(myBaseFaceToFill));
}
else {
mySourceShapes.Add(newE);
}
Standard_Boolean stateOfFaceOri = Standard_False;
OrientateEdgeOnFace(newE, TopoDS::Face(myBaseFaceToFill),
TopoDS::Face(mySDFaceToFill), G1, stateOfFaceOri);
myMapOfEdgeWithFaceState.Bind(newE, (Standard_Integer)stateOfFaceOri);
WES.AddStartElement(newE);
}
//2) Get ON parts of the edge and define to keep it or not
const TopTools_ListOfShape& LSEOn = myDataStructure -> DS().GetShapeWithState(EOR).Part(TopAbs_ON);
first = Standard_True;
it.Initialize(LSEOn);
for(; it.More(); it.Next()) {
TopoDS_Edge aSplitPart = TopoDS::Edge(it.Value());
aSplitPart.Orientation(EOR.Orientation());
//do ON 2D computation
if(myDataStructure -> HasSameDomain(aSplitPart) || myDataStructure -> HasSameDomain(EOR)) {
Standard_Integer flag = 0;
//First of All : if SDFaceToFill is REVERSED we need to reverse aSplitPart
TopoDS_Shape eON = aSplitPart;
TopoDS_Shape nEOR = EOR;
if(mySDFaceToFill.Orientation() == TopAbs_REVERSED) {
eON.Reverse();
nEOR.Reverse();
}
TopTools_ListOfShape aListOfPieces, aListOfFaces, aListOfPieceOut2d;
//Out 2d pieces we compute only one time : for the Object
if(myProcessedPartsOut2d.Contains(eON))
continue;
flag = PerformPieceOn2D (eON, mySDFaceToFill, nEOR, aListOfPieces, aListOfFaces, aListOfPieceOut2d);
TopTools_ListIteratorOfListOfShape aPIt2d(aListOfPieceOut2d);
for(; aPIt2d.More(); aPIt2d.Next()) {
TopoDS_Shape aFP = aPIt2d.Value();
TopoDS_Shape aRP = aPIt2d.Value();
aFP.Reverse();
WES.AddStartElement(aFP);
WES.AddStartElement(aRP);
myProcessedPartsOut2d.Add(aFP);
}
TopTools_ListIteratorOfListOfShape aPIt(aListOfPieces), aFIt(aListOfFaces);
for(; aPIt.More(); aPIt.Next()) {
TopoDS_Shape aPieceToKeep = aPIt.Value();
const TopoDS_Shape& aPieceFace = aFIt.Value();
if(aPieceFace == mySDFaceToFill) {
Standard_Boolean IsRev = (aPieceToKeep.Orientation() == nEOR.Orientation());
Standard_Boolean stateOfFaceOri = Standard_False;
aPieceToKeep.Orientation(oriE);
OrientateEdgeOnFace(TopoDS::Edge(aPieceToKeep), TopoDS::Face(myBaseFaceToFill),
TopoDS::Face(mySDFaceToFill), G1, stateOfFaceOri);
//if edge was in not the right orientation we need to reverse it
if(!IsRev)
aPieceToKeep.Reverse();
myMapOfEdgeWithFaceState.Bind(aPieceToKeep, (Standard_Integer)stateOfFaceOri);
WES.AddStartElement(aPieceToKeep);
}
aFIt.Next();
}
if(flag) //if flag == 0 we should go to the IN2D or OUT2D
continue;
}
//do IN 2D computation
TopoDS_Shape aSDShape;
TopAbs_State aState = TopAbs_UNKNOWN;
if(LSclass.Extent() == 1) {
aSDShape = LSclass.First();
aState = ClassifyEdgeToFaceByOnePoint(aSplitPart, TopoDS::Face(aSDShape));
}
else { //if face has more than one same domain faces we need to detect for each part complement same domain face
TopTools_ListIteratorOfListOfShape LSClassIt(LSclass);
for(; LSClassIt.More(); LSClassIt.Next()) {
TopoDS_Face curSD = TopoDS::Face(LSClassIt.Value());
aState = ClassifyEdgeToFaceByOnePoint(aSplitPart,curSD);
// aState = ClassifyEdgeToFaceByOnePoint(aSplitPart, TopoDS::Face(curSD));
if(aState == TopAbs_IN || aState == TopAbs_ON) {
aSDShape = curSD;
break;
}
}
}
//we should process all same domain edges (ON2D) in the code above
//and we can not proceess edges with UNKNOWN state
if(aState == TopAbs_ON || aState == TopAbs_UNKNOWN)
continue;
//OUT2D computation
if(aState == TopAbs_OUT || aSDShape.IsNull()) {
//it means that SplitPart is ON
//comparing with myShape2 but Out of all this SD faces
//so we need to find adjacent faces and they also MUST be SameDomain and compute all in reverse order
Standard_Boolean keep = Standard_False;
if(aSDShape.IsNull()) {
aSDShape = LSclass.First();
}
//compute adjacents
TopoDS_Shape aAdjSDFace;
const TopTools_ListOfShape& aFEL = myMapOfEdgeFaces.FindFromKey(EOR);
TopTools_ListIteratorOfListOfShape aEFIt(aFEL);
if(aFEL.Extent() <= 2) { //we don't compute adjacent if we have more than one adjacent face
for(; aEFIt.More(); aEFIt.Next()) {
if(mySDFaceToFill.IsSame(aEFIt.Value()))
continue;
else {
if(myDataStructure -> HasSameDomain(aEFIt.Value())) {
aAdjSDFace = aEFIt.Value();
break;
}
}
}
}
if(!aAdjSDFace.IsNull()) {
TopTools_IndexedMapOfShape aEAdjMap;
TopExp::MapShapes(aAdjSDFace, TopAbs_EDGE, aEAdjMap);
Standard_Integer index = aEAdjMap.FindIndex(EOR);
TopoDS_Shape AdjEOR = aEAdjMap.FindKey(index);
TopTools_ListIteratorOfListOfShape it1 = myDataStructure -> SameDomain(aAdjSDFace);
TopoDS_Shape aSDToAdjFace = it1.Value();
TopoDS_Edge aSplitP = aSplitPart;
aSplitP.Orientation(AdjEOR.Orientation());
gp_Vec aTg, aN1, aN2,aN3, aBiN;
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(TopoDS::Face(aSDToAdjFace), aSplitP, aN2);
if(aSDToAdjFace.Orientation() == TopAbs_REVERSED)
aN2.Reverse();
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(TopoDS::Face(aAdjSDFace), aSplitP, aN3);
if(aAdjSDFace.Orientation() == TopAbs_REVERSED)
aN3.Reverse();
TopOpeBRepBuild_Tools::GetTangentToEdge(aSplitP, aTg);
if (aSplitP.Orientation() == TopAbs_REVERSED) {
aTg.Reverse();
}
aBiN = aTg^aN2;
Standard_Real scalarPr = 0.;
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(TopoDS::Face(mySDFaceToFill), aSplitP, aN1);
if(mySDFaceToFill.Orientation() == TopAbs_REVERSED)
aN1.Reverse();
scalarPr = aBiN*aN1;
if(fabs(scalarPr) <= 1e-10) {//try to step inside
TopOpeBRepBuild_Tools::GetNormalInNearestPoint(TopoDS::Face(mySDFaceToFill), aSplitP, aN1);
if(mySDFaceToFill.Orientation() == TopAbs_REVERSED)
aN1.Reverse();
scalarPr = aBiN*aN1;
if(fabs(scalarPr) <= 1e-10) // this can not be
keep = (TB == TopAbs_IN); //just to do something
}
TopAbs_State aPartState = (scalarPr > 0) ? TopAbs_IN : TopAbs_OUT;
keep = (aPartState == TB) ? Standard_True : Standard_False;
}
else { //if aAdjFace.IsNull() - it must not happen
keep = (TB == TopAbs_IN);
}
if(keep) {
if(mySDFaceToFill != myBaseFaceToFill) {
TopOpeBRepBuild_Tools::UpdateEdgeOnFace(aSplitPart,
TopoDS::Face(mySDFaceToFill),
TopoDS::Face(myBaseFaceToFill));
}
else {
mySourceShapes.Add(aSplitPart);
}
Standard_Boolean stateOfFaceOri = Standard_False;
OrientateEdgeOnFace(aSplitPart, TopoDS::Face(myBaseFaceToFill),
TopoDS::Face(mySDFaceToFill), G1, stateOfFaceOri);
myMapOfEdgeWithFaceState.Bind(aSplitPart, (Standard_Integer)stateOfFaceOri);
WES.AddStartElement(aSplitPart);
}
continue;
}
//end case OUT2D
//IN2D computation
TopoDS_Edge aSplitP = aSplitPart;
aSplitP.Orientation(EOR.Orientation());
TopoDS_Face aSDFace = TopoDS::Face(aSDShape);
Standard_Boolean keep = Standard_False;
PerformPieceIn2D(aSplitPart, TopoDS::Edge(EOR),
TopoDS::Face(mySDFaceToFill), aSDFace, G1, keep);
if(keep) {
mySplitsONtoKeep.Add(aSplitPart);
//compute orientation of the future face
Standard_Boolean stateOfFaceOri = Standard_False;
OrientateEdgeOnFace(aSplitPart, TopoDS::Face(myBaseFaceToFill), aSDFace, G1, stateOfFaceOri);
myMapOfEdgeWithFaceState.Bind(aSplitPart, (Standard_Integer)stateOfFaceOri);
if(myBaseFaceToFill == mySDFaceToFill) {
mySourceShapes.Add(aSplitPart);
}
WES.AddStartElement(aSplitPart);
}
}
} // GFillEdgeWES
extern Standard_Boolean TopOpeBRepBuild_FUN_aresamegeom(const TopoDS_Shape& S1,const TopoDS_Shape& S2);
//=======================================================================
//function : PerformONParts
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::PerformONParts(const TopoDS_Shape& FOR1,
const TopTools_IndexedMapOfShape& /*SDFaces*/,
const TopOpeBRepBuild_GTopo& G1,
TopOpeBRepBuild_WireEdgeSet& WES)
{
TopAbs_State ETB1,ETB2, ETB; G1.StatesON(ETB1,ETB2);
TopAbs_State FTB1,FTB2, FTB; G1.StatesON(FTB1,FTB2);
Standard_Integer iref = myDataStructure -> DS().AncestorRank(FOR1);
Standard_Boolean RevOri;
if(iref == 1) {//object
FTB = FTB1;
RevOri = G1.IsToReverse1();
}
else {//tool
RevOri = G1.IsToReverse2();
FTB = FTB2;
}
//3 Process parts that can not be found on SD faces but must be included because they are ON the SD faces
const TopOpeBRepDS_ListOfInterference& LI = myDataStructure -> DS().ShapeInterferences(FOR1);
for (TopOpeBRepDS_ListIteratorOfListOfInterference ILI(LI);ILI.More();ILI.Next() ) {
const Handle(TopOpeBRepDS_Interference)& I=ILI.Value();
Handle(TopOpeBRepDS_ShapeShapeInterference) SSI
= Handle(TopOpeBRepDS_ShapeShapeInterference)::DownCast(I);
if (SSI.IsNull())
continue;
TopOpeBRepDS_Kind GT,ST;
Standard_Integer GI,SI;
FDS_data(SSI,GT,GI,ST,SI);
if (GT != TopOpeBRepDS_EDGE || ST != TopOpeBRepDS_FACE)
continue;
const TopoDS_Edge& EG=TopoDS::Edge(myDataStructure -> DS().Shape(GI, Standard_False));
//we process only the edges which are not from the current SD faces
if(mySDEdgeMap.Contains(EG))
continue;
//take ON splits of the edge
const TopTools_ListOfShape& splON = myDataStructure -> DS().GetShapeWithState(EG).Part(TopAbs_ON);
if(!splON.Extent())
continue;
const TopOpeBRepDS_Transition& aTr = SSI -> Transition();
Standard_Integer irefE = myDataStructure -> DS().AncestorRank(EG);
Standard_Boolean RevOriE;
if(irefE == 1) {//object
ETB = ETB1;
RevOriE = G1.IsToReverse1();
}
else {//tool
RevOriE = G1.IsToReverse2();
ETB = ETB2;
}
//take list of edge faces
const TopTools_ListOfShape& EdgeFaces = myMapOfEdgeFaces.FindFromKey(EG);
TopExp_Explorer Exp;
for(TopTools_ListIteratorOfListOfShape itON(splON); itON.More(); itON.Next()) {
TopoDS_Shape newE = itON.Value();
TopoDS_Shape aSDShape = FOR1;
TopAbs_State aState = TopAbs_UNKNOWN;
aState = ClassifyEdgeToFaceByOnePoint(TopoDS::Edge(newE), TopoDS::Face(FOR1));
if(!(aState == TopAbs_IN || aState == TopAbs_ON))
continue;
Standard_Boolean keep = Standard_False;
TopoDS_Face aSDFace = TopoDS::Face(aSDShape);
TopAbs_Orientation oriE;
TopAbs_Orientation neworiE;
for(TopTools_ListIteratorOfListOfShape it(EdgeFaces); it.More(); it.Next()) {
const TopoDS_Shape& FOR = it.Value();
Exp.Init(FOR, TopAbs_EDGE);
TopoDS_Shape EOR;
for(; Exp.More(); Exp.Next()) {
EOR = Exp.Current();
if(EG.IsSame(EOR))
break;
}
if(EOR.IsNull())
continue;
//else we have found a face , we process it
oriE = EOR.Orientation();
neworiE = Orient(oriE,RevOriE);
newE.Orientation(oriE);
gp_Vec aTg, aN2,aN3, aBiN;
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(TopoDS::Face(FOR), TopoDS::Edge(newE), aN2);
if(FOR.Orientation() == TopAbs_REVERSED)
aN2.Reverse();
TopOpeBRepBuild_Tools::GetTangentToEdge(TopoDS::Edge(newE), aTg);
if (newE.Orientation() == TopAbs_REVERSED) {
aTg.Reverse();
}
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(aSDFace, TopoDS::Edge(newE), aN3);
if(aSDFace.Orientation() == TopAbs_REVERSED)
aN3.Reverse();
keep = Standard_False;
aBiN = aTg^aN2;
Standard_Real scalarPr = 0.;
scalarPr = aBiN*aN3;
if(fabs(scalarPr) <= 1e-10) {//try to step inside
TopOpeBRepBuild_Tools::GetNormalInNearestPoint(TopoDS::Face(FOR), TopoDS::Edge(newE), aN2);
if(FOR.Orientation() == TopAbs_REVERSED)
aN2.Reverse();
aBiN = aTg^aN2;
scalarPr = aBiN*aN3;
if(fabs(scalarPr) <= 1e-10)
continue;
}
TopAbs_State aPartState = (scalarPr > 0) ? TopAbs_IN : TopAbs_OUT;
keep = (aPartState == ETB) ? Standard_True : Standard_False;
if(keep)
break;
}
if(keep) {
//compute orientation of the future face
Standard_Boolean stateOfFaceOri = Standard_False;
gp_Vec aNbf, aNsf , OrigNormalbf; //aTg, aBiN, aOut;
TopoDS_Edge aLocalEdge = TopoDS::Edge(newE);
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(aSDFace,aLocalEdge, aNsf);
// TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(TopoDS::Face(aSDFace), TopoDS::Edge(newE), aNsf);
if(aSDFace.Orientation() == TopAbs_REVERSED)
aNsf.Reverse();
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(TopoDS::Face(myBaseFaceToFill), TopoDS::Edge(newE), OrigNormalbf);
aNbf = OrigNormalbf;
if(myBaseFaceToFill.Orientation() == TopAbs_REVERSED)
aNbf.Reverse();
if(aNsf*aNbf < 0) {
stateOfFaceOri = Standard_True;
}
if(myDataStructure -> DS().AncestorRank(aSDFace) == 2) {//for tool we need to reverse face in cut
if(Opec12() || Opec21()) {
stateOfFaceOri = !stateOfFaceOri;
}
}
//adjust orientation of the edge
neworiE = aTr.Orientation(FTB);
Standard_Boolean samegeom = TopOpeBRepBuild_FUN_aresamegeom(FOR1,myBaseFaceToFill);
if (!samegeom) {
neworiE = TopAbs::Complement(neworiE);
}
newE.Orientation(neworiE);
myMapOfEdgeWithFaceState.Bind(newE, (Standard_Integer)stateOfFaceOri);
WES.AddStartElement(newE);
}
}//end iteration on splON
}//end iteration of interferences
}
//=======================================================================
//function : GWESMakeFaces
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::GWESMakeFaces(const TopoDS_Shape& FF,
TopOpeBRepBuild_WireEdgeSet& WES,
TopTools_ListOfShape& LOF)
{
TopOpeBRepBuild_Builder::GWESMakeFaces(FF, WES, LOF);
TopTools_ListIteratorOfListOfShape aLOFit(LOF);
TopTools_ListOfShape corrLOF;
if(myIsKPart == 4) {
for(; aLOFit.More(); aLOFit.Next()) {
const TopoDS_Shape& ff = aLOFit.Value();
TopoDS_Shape corrFF;
TopOpeBRepBuild_Tools::NormalizeFace(ff, corrFF);
corrLOF.Append(corrFF);
}
}
else
corrLOF.Assign(LOF);
LOF.Clear(); LOF.Assign(corrLOF);
//corect face2d
aLOFit.Initialize(corrLOF);
TopTools_ListOfShape corrLOF1;
for(; aLOFit.More(); aLOFit.Next()) {
const TopoDS_Shape& ff = aLOFit.Value();
TopoDS_Shape corrFF;
TopOpeBRepBuild_Tools::CorrectFace2d(ff, corrFF, mySourceShapes, myMapOfCorrect2dEdges);
corrLOF1.Append(corrFF);
}
LOF.Clear(); LOF.Assign(corrLOF1);
}
//=======================================================================
//function : PerformPieceIN2d
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::PerformPieceIn2D(const TopoDS_Edge& EdgeToPerform,
const TopoDS_Edge& EOR,
const TopoDS_Face& edgeFace,
const TopoDS_Face& toFace,
const TopOpeBRepBuild_GTopo& G1,
Standard_Boolean& keep)
{
keep = Standard_False;
TopAbs_State TB1,TB2, TB; G1.StatesON(TB1,TB2);
Standard_Integer iref = myDataStructure -> DS().AncestorRank(EOR);
TB = (iref == 1) ? TB1 : TB2;
gp_Vec aTg, aN1, aN2,aN3, aBiN;
TopAbs_Orientation O1 = edgeFace.Orientation();
TopAbs_Orientation O2 = toFace.Orientation();
TopAbs_Orientation oriE = EdgeToPerform.Orientation();
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(toFace, EdgeToPerform, aN2);
if(O2 == TopAbs_REVERSED)
aN2.Reverse();
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(edgeFace, EdgeToPerform, aN3);
if(O1 == TopAbs_REVERSED)
aN3.Reverse();
TopOpeBRepBuild_Tools::GetTangentToEdge(EdgeToPerform, aTg);
if (oriE == TopAbs_REVERSED)
aTg.Reverse();
if(O1 == TopAbs_REVERSED)
aTg.Reverse();
aBiN = aTg^aN2;
const TopTools_ListOfShape& aFEL = myMapOfEdgeFaces.FindFromKey(EOR);
TopTools_ListIteratorOfListOfShape aEFIt(aFEL);
Standard_Real scalarPr = 0.;
/// Why ????? Need to be checked
if(aFEL.Extent() <= 2) { //we don't compute adjacent if we have more than one adjacent face
for(; aEFIt.More(); aEFIt.Next()) {
if(edgeFace.IsSame(aEFIt.Value()))
continue;
else { //compute bi-normal state
TopoDS_Face aAdjF = TopoDS::Face(aEFIt.Value());
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(aAdjF, EdgeToPerform, aN1);
if(aAdjF.Orientation() == TopAbs_REVERSED)
aN1.Reverse();
scalarPr = aBiN*aN1;
if(fabs(scalarPr) <= 1e-10) { //try to step inside
TopOpeBRepBuild_Tools::GetNormalInNearestPoint(aAdjF, EdgeToPerform, aN1);
// TopOpeBRepBuild_Tools::GetNormalInNearestPoint(TopoDS::Face(aAdjF), EdgeToPerform, aN1);
if(aAdjF.Orientation() == TopAbs_REVERSED)
aN1.Reverse();
scalarPr = aBiN*aN1;
if(fabs(scalarPr) <= 1e-10)
continue;
}
TopAbs_State aPartState = (scalarPr > 0) ? TopAbs_IN : TopAbs_OUT;
keep = (aPartState == TB) ? Standard_True : Standard_False;
if(keep)
break;
}
}
}
//if scalar can not be found that means that adjacent face doesn't exist
//WARNING !!! May be this code is not good but for the moment it is only one solution
if(fabs(scalarPr) <= 1e-10) {
if(Opefus()) {
keep = aN3*aN2 > 0;
}
if(Opec12() || Opec21())
keep = aN3*aN2 < 0;
if(Opecom())
keep = aN3*aN2 > 0;
}
}
//=======================================================================
//function : PerformPieceOn2D
//purpose :
//=======================================================================
Standard_Integer TopOpeBRepBuild_Builder1::PerformPieceOn2D (const TopoDS_Shape& aPieceObj,
const TopoDS_Shape& aFaceObj,
const TopoDS_Shape& anEdgeObj,
TopTools_ListOfShape& aListOfPieces,
TopTools_ListOfShape& aListOfFaces,
TopTools_ListOfShape& aListOfPiecesOut2d)
{
// eap 30 May occ417, aCasesMap instead of aCase14 and aCase12
Standard_Integer i, j, k, flag=0, priz;//, aCase14=0, aCase12=0;
TColStd_MapOfInteger aCasesMap;
Standard_Integer iRef = myDataStructure -> DS().AncestorRank(aFaceObj);
if(!myDataStructure -> HasSameDomain(aFaceObj))
return -1;
// Main DataStructure
TopOpeBRepDS_DataStructure& aDS= myDataStructure-> ChangeDS();
// Main Map for Tool (relative Tool)
TopOpeBRepDS_IndexedDataMapOfShapeWithState& aMapOfShapeWithStateTool=
(iRef == 1) ? aDS.ChangeMapOfShapeWithStateTool() : aDS.ChangeMapOfShapeWithStateObj();
// Loop on faces same domain to aFaceObj
TopTools_ListIteratorOfListOfShape anIt(myDataStructure->SameDomain(aFaceObj));
for (i=1; anIt.More(); anIt.Next(), i++) {
const TopoDS_Shape& aFaceTool=anIt.Value();
TopTools_IndexedMapOfShape anEdgesToolMap;
TopExp::MapShapes(aFaceTool, TopAbs_EDGE, anEdgesToolMap);
if(myDataStructure -> HasSameDomain(anEdgeObj)) {
TopTools_ListIteratorOfListOfShape anItE=myDataStructure->SameDomain(anEdgeObj);
for (j=1; anItE.More(); anItE.Next(), j++) {
TopoDS_Shape anEdgeTool=anItE.Value();
if (anEdgesToolMap.Contains (anEdgeTool)) {
TopExp_Explorer anExpEdges;
for (anExpEdges.Init (aFaceTool, TopAbs_EDGE); anExpEdges.More(); anExpEdges.Next()) {
const TopoDS_Shape& anExpEdgeTool=anExpEdges.Current();
if (!anExpEdgeTool.IsSame(anEdgeTool)) continue;
anEdgeTool.Orientation(anExpEdgeTool.Orientation());
const TopOpeBRepDS_ShapeWithState& aSWSTool=
aMapOfShapeWithStateTool.FindFromKey(anEdgeTool);
const TopTools_ListOfShape& aPartOnTool=aSWSTool.Part(TopAbs_ON);
// we are looking for the same piece as aPieceObj among aPartOnTool
TopTools_ListIteratorOfListOfShape anItTool(aPartOnTool);
for (k=1; anItTool.More(); anItTool.Next(), k++) {
TopoDS_Shape& aPieceTool=anItTool.Value();
aPieceTool.Orientation(anEdgeTool.Orientation());
Standard_Boolean aIsSameCnd, IsDegFlag;
IsDegFlag=
BRep_Tool::Degenerated (TopoDS::Edge(aPieceObj)) &&
BRep_Tool::Degenerated (TopoDS::Edge(aPieceTool)) ;
aIsSameCnd=IsDegFlag ? TopOpeBRepBuild_Tools::IsDegEdgesTheSame(aPieceObj, aPieceTool) : aPieceObj.IsSame(aPieceTool);
if (aIsSameCnd) {
TopTools_SequenceOfShape aSeq;
aSeq.Append(aFaceObj) ; aSeq.Append(anEdgeObj) ; aSeq.Append(aPieceObj) ;
aSeq.Append(aFaceTool); aSeq.Append(anEdgeTool); aSeq.Append(aPieceTool);
flag++;
priz=TwoPiecesON (aSeq, aListOfPieces, aListOfFaces, aListOfPiecesOut2d);
//if (priz==14) aCase14=1;
//if (priz==12) aCase12=1;
aCasesMap.Add(priz);
break;
}
}
if (!flag) {
//printf("Warning : => aPieceTool is not found\n");
//modified by NIZHNY-MZV Thu Dec 23 17:30:20 1999
//return -2;
}
}
}
}
}
}
//this case dedicated for the computation then edge has sim (F and R at one time) SD edge
if ( flag>1 ) {
if ( aCasesMap.Contains(14) && aCasesMap.Contains(12) && Opefus() )
aListOfPieces.Clear();
// eap 30 May occ417, add :
if ( aCasesMap.Contains(11) && aCasesMap.Contains(13) && (Opec12() || Opec21()) )
aListOfPieces.Clear();
}
return flag; //Ok
}
//=======================================================================
//function : TwoPiecesON
//purpose :
//=======================================================================
Standard_Integer TopOpeBRepBuild_Builder1::TwoPiecesON (const TopTools_SequenceOfShape& aSeq,
TopTools_ListOfShape& aListOfPieces,
TopTools_ListOfShape& aListOfFaces,
TopTools_ListOfShape& aListOfPiecesOut2d)
{
// Restore Data
if (aSeq.Length() < 6)
return -2;
TopoDS_Shape aFaceObj =aSeq(1);
TopoDS_Shape anEObj =aSeq(2);
TopoDS_Shape aPieceObj =aSeq(3);
TopoDS_Shape aFaceTool =aSeq(4);
TopoDS_Shape anETool =aSeq(5);
TopoDS_Shape aPieceTool=aSeq(6);
// The two Maps for adjacent faces
Standard_Integer iRef = myDataStructure -> DS().AncestorRank(aFaceObj);
TopTools_IndexedDataMapOfShapeListOfShape anEdgeFaceMapObj, anEdgeFaceMapTool;
if(iRef == 1) {
TopExp::MapShapesAndAncestors(myShape1, TopAbs_EDGE, TopAbs_FACE, anEdgeFaceMapObj );
TopExp::MapShapesAndAncestors(myShape2, TopAbs_EDGE, TopAbs_FACE, anEdgeFaceMapTool);
}
else {
TopExp::MapShapesAndAncestors(myShape1, TopAbs_EDGE, TopAbs_FACE, anEdgeFaceMapObj );
TopExp::MapShapesAndAncestors(myShape2, TopAbs_EDGE, TopAbs_FACE, anEdgeFaceMapTool);
TopoDS_Shape tmpFace = aFaceObj, tmpPiece = aPieceObj, tmpEdge = anEObj;
aFaceObj = aFaceTool; aPieceObj = aPieceTool; anEObj = anETool;
aFaceTool = tmpFace; aPieceTool = tmpPiece; anETool = tmpEdge;
}
//
Standard_Boolean IsFacesDifOriented , IsEdgesRevSense,
anAd1=Standard_False , anAd2=Standard_False,
aScPrFlag1=Standard_False, aScPrFlag2=Standard_False,
Rejected1=Standard_True , Rejected2=Standard_True;
TopAbs_State aStateObj =TopAbs_UNKNOWN, aStateTool=TopAbs_UNKNOWN;
Standard_Real aScProductObj =0. , aScProductTool=0.,
aTol=1.e-5;
Standard_Real aScPrObj=0., aScPrTool=0.;
gp_Vec anyN;
TopoDS_Shape anAdjFaceObj, anAdjFaceTool;
// Faces
TopoDS_Face aFObj = TopoDS::Face(aFaceObj);
TopoDS_Face aFTool = TopoDS::Face(aFaceTool);
// Pieces
TopoDS_Edge anEdgeObj = TopoDS::Edge(aPieceObj);
TopoDS_Edge anEdgeTool= TopoDS::Edge(aPieceTool);
//OldEdges
TopoDS_Edge aOriEObj = TopoDS::Edge(anEObj);
TopoDS_Edge aOriETool = TopoDS::Edge(anETool);
// Normals to the Faces
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge (aFObj, anEdgeObj, anyN);
if(aFObj.Orientation() == TopAbs_REVERSED)
anyN.Reverse();
gp_Dir aDNObj(anyN);
//
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge (aFTool, anEdgeTool, anyN);
if(aFTool.Orientation() == TopAbs_REVERSED)
anyN.Reverse();
gp_Dir aDNTool (anyN);
// are aFaceObj & aFaceTool different oriented faces or not ?
IsFacesDifOriented=aDNObj*aDNTool < 0.;
// Sense of the Pieces
Standard_Boolean RevObj = TopOpeBRepBuild_Tools::GetTangentToEdgeEdge (aFObj, anEdgeObj, aOriEObj, anyN);
if(RevObj) {
aPieceObj.Reverse();
anEdgeObj.Reverse();
}
gp_Dir aDTObj(anyN);
Standard_Boolean RevTool = TopOpeBRepBuild_Tools::GetTangentToEdgeEdge (aFTool, anEdgeTool, aOriETool, anyN);
if(RevTool) {
aPieceTool.Reverse();
anEdgeTool.Reverse();
}
gp_Dir aDTTool(anyN);
IsEdgesRevSense= aDTObj*aDTTool < 0.;
// try to get adjacent faces for Obj and Tool. Ad1, Ad2 indicate that the face exists.
anAd1=TopOpeBRepBuild_Tools::GetAdjacentFace (aFaceObj, anEObj, anEdgeFaceMapObj, anAdjFaceObj);
anAd2=TopOpeBRepBuild_Tools::GetAdjacentFace (aFaceTool, anETool, anEdgeFaceMapTool, anAdjFaceTool);
if (anAd1 && anAd2) {
// both adjacents are found , so we can calculate the scalar products
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge (TopoDS::Face(anAdjFaceObj), anEdgeObj, anyN);
if(anAdjFaceObj.Orientation() == TopAbs_REVERSED)
anyN.Reverse();
gp_Dir aDNAObj (anyN);
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge (TopoDS::Face(anAdjFaceTool), anEdgeTool, anyN);
if(anAdjFaceTool.Orientation() == TopAbs_REVERSED)
anyN.Reverse();
gp_Dir aDNATool (anyN);
aScPrObj =(aDTObj^aDNTool)*aDNAObj;
aScPrTool=(aDTTool^aDNObj)*aDNATool;
if(fabs(aScPrObj) <= aTol) {//if scalar product is == 0 try to move a little from this point
TopOpeBRepBuild_Tools::GetNormalInNearestPoint (TopoDS::Face(anAdjFaceObj), anEdgeObj, anyN);
if(anAdjFaceObj.Orientation() == TopAbs_REVERSED)
anyN.Reverse();
aDNAObj.SetXYZ (anyN.XYZ());
aScPrObj =(aDTObj^aDNTool)*aDNAObj;
}
if(fabs(aScPrTool) <= aTol) {//if scalar product is == 0 try to move a little from this point
TopOpeBRepBuild_Tools::GetNormalInNearestPoint (TopoDS::Face(anAdjFaceTool), anEdgeTool, anyN);
if(anAdjFaceTool.Orientation() == TopAbs_REVERSED)
anyN.Reverse();
aDNATool.SetXYZ (anyN.XYZ());
aScPrTool =(aDTTool^aDNObj)*aDNATool;
}
// Scalar prouducts must not have too small value:
aScProductObj=aScPrObj;
aScProductTool=aScPrTool;
/*
// mine
gp_Dir aDBNObj(aDNObj^aDTObj);
aScProductObj=aDBNObj*aDNATool;
gp_Dir aDBNTool(aDNTool^aDTTool);
aScProductTool=aDBNTool*aDNAObj;
*/
// Scalar prouducts must not have too small value:
if (fabs(aScProductTool) > aTol) aScPrFlag1=Standard_True;
if (fabs(aScProductObj ) > aTol) aScPrFlag2=Standard_True;
}
// Management
if (!anAd1 || !anAd2 || !aScPrFlag1 || !aScPrFlag2) {
// manage without adjacents.
// case a. No==Nt , To!=Tt
if (IsEdgesRevSense && !IsFacesDifOriented ) {
if (Opec12()) {
Standard_Boolean poisc = BRep_Tool::IsClosed(anEdgeObj,aFObj);
if(!poisc)
{
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
}
return 11;
}
// case b. No!=Nt , To!=Tt
if (IsEdgesRevSense && IsFacesDifOriented) {
if (Opec12()) {
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
if(!anAd1 || !anAd2)
return 12;
else
return 10; //10 doesn't mean anything just to retutn something
}
// case c. No==Nt , To==Tt
if (!IsEdgesRevSense && !IsFacesDifOriented) {
//Begin modified by NIZHNY-MZV Mon Jan 24 10:03:58 2000
// WRNG!!
if(anAd1 && anAd2) {
if(!Opecom()) {
if(!aScPrFlag2) {
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
if(!aScPrFlag1) {
aListOfPieces.Append (aPieceTool);
aListOfFaces.Append (aFaceTool);
}
}
}
else {
if(Opefus()) {
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
//End modified by NIZHNY-MZV Mon Jan 24 11:21:17 2000
}
return 13;
}
// case d. No!=Nt , To==Tt
if (!IsEdgesRevSense && IsFacesDifOriented) {
//modified by NIZHNY-MZV Fri Jan 21 18:16:01 2000
// WRNG!!
if(anAd1 && anAd2) {
if(!Opecom()) {
if(!aScPrFlag2) {
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
if(!aScPrFlag1) {
aListOfPieces.Append (aPieceTool);
aListOfFaces.Append (aFaceTool);
}
}
}
else {
if(Opefus()) {
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
}
if(!anAd1 || !anAd2)
return 14;
else
return 10; //10 doesn't mean anything just to retutn something
}
return 10;
} // end of if (!anAd1 || !anAd2 || !aScPrFlag1 || !aScPrFlag2)
else {
// We can use adjacents .
// The States :
/*
aStateObj =aScProductObj < 0. ? TopAbs_IN: TopAbs_OUT ;
aStateTool=aScProductTool < 0. ? TopAbs_IN: TopAbs_OUT ;
*/
aStateObj =aScProductObj > 0. ? TopAbs_IN: TopAbs_OUT ;
aStateTool=aScProductTool > 0. ? TopAbs_IN: TopAbs_OUT ;
// case I RevSense && DifOriented
if (IsEdgesRevSense && IsFacesDifOriented) {
if (Opec12()) {
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
return 1;
}
// case III SameSense && !DifOriented
if (!IsEdgesRevSense && !IsFacesDifOriented) {
if (!Opec12()) {
aListOfPieces.Append (aPieceObj);
aListOfFaces.Append (aFaceObj);
}
return 3;
}
// case II RevSense && !DifOriented
if (IsEdgesRevSense && !IsFacesDifOriented) {
if (Opefus()) { // Fusion
if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_IN) {
Rejected1=Standard_False;
}
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_OUT) {
Rejected2=Standard_False;
}
//// ????
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_IN) {
if(!myProcessedPartsON2d.Contains(aPieceObj)) {//we proceed IsSame only if we didn't it before
myProcessedPartsON2d.Add(aPieceObj);
IsSame2d (aSeq, aListOfPiecesOut2d); //Perform IsSame 2d and keep periodic parts
}
}
else if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_OUT) {
if(!myProcessedPartsON2d.Contains(aPieceObj)) {//we proceed IsSame only if we didn't it before
myProcessedPartsON2d.Add(aPieceObj);
IsSame2d (aSeq, aListOfPiecesOut2d); //Perform IsSame 2d and keep periodic parts
}
}
}
if (Opecom()) {// Common
if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_IN) {
Rejected2=Standard_False;
}
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_OUT) {
Rejected1=Standard_False;
}
}
if (Opec12()) {// Cut
if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_OUT) {
Rejected2=Standard_False;
}
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_IN) {
Rejected1=Standard_False;
}
else if (aStateObj==TopAbs_OUT && aStateTool == TopAbs_IN) {
Rejected1=Standard_False;
Rejected2=Standard_False;
}
}
if (!Rejected1) {
aListOfPieces.Append(aPieceObj);
aListOfFaces.Append (aFaceObj);
}
if (!Rejected2) {
aListOfPieces.Append(aPieceTool);
aListOfFaces.Append (aFaceTool);
}
return 2;
}
// case IV !RevSense && DifOriented
if (!IsEdgesRevSense && IsFacesDifOriented) {
if (Opefus()) {// Fusion
if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_OUT) {
Rejected1=Standard_False;
Rejected2=Standard_False;
}
else if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_IN) {
Rejected2=Standard_False;
}
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_OUT) {
Rejected1=Standard_False;
}
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_IN) {
if(!myProcessedPartsON2d.Contains(aPieceObj)) {//we proceed IsSame only if we didn't it before
myProcessedPartsON2d.Add(aPieceObj);
IsSame2d (aSeq, aListOfPiecesOut2d); //Perform IsSame 2d and keep periodic parts
}
}
}
if (Opecom()) {// Common
if (aStateObj==TopAbs_IN && aStateTool==TopAbs_IN) {
Rejected1=Standard_False;
Rejected2=Standard_False;
}
else if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_IN) {
Rejected1=Standard_False;
}
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_OUT) {
Rejected2=Standard_False;
}
}
if (Opec12()) { //Cut
if (aStateObj==TopAbs_OUT && aStateTool==TopAbs_OUT) {
Rejected1=Standard_False;
}
else if (aStateObj==TopAbs_IN && aStateTool==TopAbs_IN) {
Rejected2=Standard_False;
}
}
if (!Rejected1) {
aListOfPieces.Append(aPieceObj);
aListOfFaces.Append (aFaceObj);
}
if (!Rejected2) {
aListOfPieces.Append(aPieceTool);
aListOfFaces.Append (aFaceTool);
}
return 4;
}
// Unknowm case for existing adjacents
return 0;
}
}
//=======================================================================
//function : IsSame2d
//purpose :
//=======================================================================
Standard_Integer TopOpeBRepBuild_Builder1::IsSame2d (const TopTools_SequenceOfShape& aSeq,
TopTools_ListOfShape& aListOfPiecesOut2d)
{
if (aSeq.Length() < 6) return 0;
TopoDS_Shape aFaceObj =aSeq(1); TopoDS_Shape anEdgeObj =aSeq(2);
TopoDS_Shape aPieceObj =aSeq(3); TopoDS_Shape aFaceTool =aSeq(4);
TopoDS_Shape anEdgeTool=aSeq(5); TopoDS_Shape aPieceTool=aSeq(6);
TopoDS_Face aFObj =TopoDS::Face(aFaceObj) ; TopoDS_Face aFTool =TopoDS::Face(aFaceTool) ;
TopoDS_Edge anEObj =TopoDS::Edge(anEdgeObj) ; TopoDS_Edge anETool=TopoDS::Edge(anEdgeTool);
TopoDS_Edge aPObj =TopoDS::Edge(aPieceObj) ; TopoDS_Edge aPTool =TopoDS::Edge(aPieceTool);
BRepAdaptor_Surface aBAS(aFObj);
if (!(aBAS.IsUPeriodic() || aBAS.IsVPeriodic())) return 1;
//we process here only fully closed edges (Vf == Vl)
if(!anEdgeObj.Closed() || !anEdgeTool.Closed())
return 1;
Standard_Real f = 0., l = 0., tolpc = 0. ,
par = 0., parOri = 0., f1 = 0., l1 = 0., parP = 0., gp_Resolution = 1.e-10;
gp_Pnt2d aUV1;
Handle(Geom2d_Curve) C2D;
// C2DPieceTool
Handle(Geom2d_Curve) C2DPieceTool = FC2D_CurveOnSurface (aPTool, aFObj, f1, l1, tolpc, Standard_True);
parP= f1*PAR_T + (1 - PAR_T)*l1;
gp_Pnt2d aPPiece;
C2DPieceTool -> D0(parP, aPPiece);
// Tool Edge
C2D=FC2D_CurveOnSurface (anETool, aFObj, f, l, tolpc, Standard_True);
Geom2dAPI_ProjectPointOnCurve aPP2d(aPPiece, C2D);
parOri = aPP2d.LowerDistanceParameter();
Standard_Boolean IsTrFirst = Standard_True;
if(parOri < f ) {
parOri = 2*M_PI + parOri;
}
if(parOri > l ) {
parOri = parOri - 2*M_PI;
}
gp_Pnt2d aUV2;
C2D -> D0(parOri, aUV2);
// C2DPieceObj
Handle(Geom2d_Curve) C2DPieceObj=FC2D_CurveOnSurface (aPObj, aFObj, f, l, tolpc, Standard_True);
par=f*PAR_T + (1 - PAR_T)*l;
C2DPieceObj->D0 (par, aUV1);
gp_Vec2d aTranslateV (aUV1, aUV2);
if(aTranslateV.Magnitude() >= gp_Resolution) {
Handle(Geom2d_Curve) aTrC2D = Handle(Geom2d_Curve)::DownCast(C2DPieceTool->Copy());
aTrC2D->Translate(aTranslateV);
gp_Pnt2d aTFuv, aTLuv;
aTrC2D -> D0(f1, aTFuv);
aTrC2D -> D0(l1, aTLuv);
gp_Vec2d aTrVec (aTFuv, aTLuv);
Standard_Real fo = 0., lo = 0.;
Handle(Geom2d_Curve) C2DEdgeObj = FC2D_CurveOnSurface(anEObj, aFObj, fo, lo, tolpc, Standard_True);
gp_Pnt2d aOFuv, aOLuv;
C2DEdgeObj -> D0(fo, aOFuv);
C2DEdgeObj -> D0(lo, aOLuv);
gp_Vec2d aOVec (aOFuv, aOLuv);
if(anEObj.Orientation() == TopAbs_REVERSED)
aOVec.Reverse();
IsTrFirst = (aTrVec*aOVec > 0) ? Standard_False : Standard_True;
BRep_Builder BB;
Standard_Real tolE = BRep_Tool::Tolerance(aPTool);
if(IsTrFirst)
BB.UpdateEdge(aPTool , aTrC2D, C2DPieceTool, aFObj , tolE);
else
BB.UpdateEdge(aPTool ,C2DPieceTool, aTrC2D, aFObj , tolE);
aListOfPiecesOut2d.Append (aPTool);
return 0;
}
return 1;
}
//=======================================================================
//function : OrientateEdgeOnFace
//purpose :
//=======================================================================
void TopOpeBRepBuild_Builder1::OrientateEdgeOnFace(TopoDS_Edge& EdgeToPerform,
const TopoDS_Face& baseFace,
const TopoDS_Face& edgeFace,
const TopOpeBRepBuild_GTopo& G1,
Standard_Boolean& stateOfFaceOri) const
{
gp_Vec aN1, aN2;
stateOfFaceOri = Standard_False;
Standard_Integer currRef = myDataStructure -> DS().AncestorRank(mySDFaceToFill);
Standard_Integer faceRef = myDataStructure -> DS().AncestorRank(edgeFace);
Standard_Boolean RevOri = Standard_False;
if(currRef == 1) {//object
RevOri = G1.IsToReverse1();
}
else {//tool
RevOri = G1.IsToReverse2();
}
TopAbs_Orientation oriE = EdgeToPerform.Orientation();
TopAbs_Orientation neworiE = Orient(oriE, RevOri);
TopAbs_Orientation faceOri = edgeFace.Orientation();
TopAbs_Orientation baseOri = baseFace.Orientation();
TopAbs_Orientation currOri = mySDFaceToFill.Orientation();
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(edgeFace, EdgeToPerform, aN1);
if(faceOri == TopAbs_REVERSED)
aN1.Reverse();
TopOpeBRepBuild_Tools::GetNormalToFaceOnEdge(baseFace, EdgeToPerform, aN2);
if(baseOri == TopAbs_REVERSED)
aN2.Reverse();
if(aN1*aN2 < 0)
stateOfFaceOri = Standard_True;
if(faceRef == 2) //for tool we need to reverse face in cut
if(Opec12() || Opec21())
stateOfFaceOri = !stateOfFaceOri;
//orientate edge with neworiE
EdgeToPerform.Orientation(neworiE);
if(currOri != baseOri)
EdgeToPerform.Reverse();
if(stateOfFaceOri)
EdgeToPerform.Reverse();
}
/////////////// STATIC FUNCTIONS
static TopAbs_State ClassifyEdgeToFaceByOnePoint(const TopoDS_Edge& E,
const TopoDS_Face& F)
{
Standard_Real f2 = 0., l2 = 0., tolpc = 0. , par = 0.;
Handle(Geom2d_Curve) C2D = FC2D_CurveOnSurface(E, F, f2, l2, tolpc, Standard_True);
par = f2*PAR_T + (1 - PAR_T)*l2;
gp_Pnt2d aP2d;
if(C2D.IsNull())
return TopAbs_UNKNOWN;
C2D -> D0(par, aP2d);
BRepTopAdaptor_FClass2d FC(F, 1e-7);
TopAbs_State aState = FC.Perform(aP2d);
return aState;
}