1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-03 17:56:21 +03:00
occt/src/BRepOffset/BRepOffset_MakeOffset_1.cxx
akaftasev ae1683705e 0033420: Modeling Algorithm - Missing inner part after offset operation
Added condition to not remove face image, consisting only of invalid edges for artifactically invalid face.
This face image should be connected with other image faces minimum through the edge.
2023-08-07 11:32:08 +01:00

8811 lines
282 KiB
C++

// Created on: 2016
// Created by: Eugeny MALTCHIKOV
// Copyright (c) 2016 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.
// This is the implementation of the extension of the 3D offset algorithm
// to work in mode Complete and Join Type Intersection.
// Currently only the Planar cases are supported.
#include <BRepOffset_MakeOffset.hxx>
#include <Precision.hxx>
#include <BRepAlgo_AsDes.hxx>
#include <BRepAlgo_Image.hxx>
#include <BRep_Builder.hxx>
#include <BRep_Tool.hxx>
#include <BRepLib.hxx>
#include <BRepTools.hxx>
#include <BRepAdaptor_Curve.hxx>
#include <BRepOffset_Tool.hxx>
#include <BRepClass3d_SolidClassifier.hxx>
#include <BOPDS_DS.hxx>
#include <BOPAlgo_BuilderFace.hxx>
#include <BOPAlgo_MakerVolume.hxx>
#include <BOPAlgo_PaveFiller.hxx>
#include <BOPAlgo_Section.hxx>
#include <BOPAlgo_Splitter.hxx>
#include <BOPAlgo_BOP.hxx>
#include <TopTools_ListOfShape.hxx>
#include <TopTools_DataMapOfShapeShape.hxx>
#include <TopTools_IndexedDataMapOfShapeListOfShape.hxx>
#include <TopTools_MapOfOrientedShape.hxx>
#include <BOPTools_AlgoTools3D.hxx>
#include <BOPTools_AlgoTools.hxx>
#include <BOPTools_Set.hxx>
#include <IntTools_Context.hxx>
#include <IntTools_ShrunkRange.hxx>
#ifdef OFFSET_DEBUG
#include <BRepAlgoAPI_Check.hxx>
#endif
typedef NCollection_DataMap<TopoDS_Shape,
TopTools_MapOfShape,
TopTools_ShapeMapHasher> BRepOffset_DataMapOfShapeMapOfShape;
typedef NCollection_DataMap<TopoDS_Shape,
TopTools_IndexedMapOfShape,
TopTools_ShapeMapHasher> BRepOffset_DataMapOfShapeIndexedMapOfShape;
namespace {
//=======================================================================
//function : AddToContainer
//purpose : Set of methods to add a shape into container
//=======================================================================
static void AddToContainer (const TopoDS_Shape& theS,
TopTools_ListOfShape& theList)
{
theList.Append (theS);
}
static Standard_Boolean AddToContainer (const TopoDS_Shape& theS,
TopTools_MapOfShape& theMap)
{
return theMap.Add (theS);
}
static Standard_Boolean AddToContainer (const TopoDS_Shape& theS,
TopTools_IndexedMapOfShape& theMap)
{
const Standard_Integer aNb = theMap.Extent();
const Standard_Integer anInd = theMap.Add (theS);
return anInd > aNb;
}
static void AddToContainer (const TopoDS_Shape& theS,
TopoDS_Shape& theSOut)
{
BRep_Builder().Add (theSOut, theS);
}
static void AddToContainer (const TopoDS_Shape& theKey,
const TopoDS_Shape& theValue,
TopTools_DataMapOfShapeListOfShape& theMap)
{
if (TopTools_ListOfShape* pList = theMap.ChangeSeek (theKey))
{
pList->Append (theValue);
}
else
{
theMap.Bound (theKey, TopTools_ListOfShape())->Append (theValue);
}
}
//=======================================================================
//function : TakeModified
//purpose : Check if the shape has images in the given images map.
// Puts in the output map either the images or the shape itself.
//=======================================================================
template <class ContainerType, class FenceMapType>
static Standard_Boolean TakeModified (const TopoDS_Shape& theS,
const TopTools_DataMapOfShapeListOfShape& theImages,
ContainerType& theContainer,
FenceMapType* theMFence)
{
const TopTools_ListOfShape* pLSIm = theImages.Seek (theS);
if (pLSIm)
{
TopTools_ListIteratorOfListOfShape itLSIm (*pLSIm);
for (; itLSIm.More(); itLSIm.Next())
{
const TopoDS_Shape& aSIm = itLSIm.Value();
if (!theMFence || AddToContainer (aSIm, *theMFence))
AddToContainer (aSIm, theContainer);
}
return Standard_True;
}
else
{
if (!theMFence || AddToContainer (theS, *theMFence))
AddToContainer (theS, theContainer);
return Standard_False;
}
}
template <class ContainerType>
static Standard_Boolean TakeModified (const TopoDS_Shape& theS,
const TopTools_DataMapOfShapeListOfShape& theImages,
ContainerType& theMapOut)
{
TopTools_MapOfShape* aDummy = NULL;
return TakeModified (theS, theImages, theMapOut, aDummy);
}
//=======================================================================
//function : hasData
//purpose : Checks if container has any data in it
//=======================================================================
template <class Container>
static Standard_Boolean hasData (const Container* theData)
{
return (theData && !theData->IsEmpty());
}
//=======================================================================
//function : AppendToList
//purpose : Add to a list only unique elements
//=======================================================================
static void AppendToList (TopTools_ListOfShape& theList,
const TopoDS_Shape& theShape)
{
for (TopTools_ListOfShape::Iterator it (theList); it.More(); it.Next())
{
const TopoDS_Shape& aS = it.Value();
if (aS.IsSame (theShape))
{
return;
}
}
theList.Append (theShape);
}
//=======================================================================
//function : ProcessMicroEdge
//purpose : Checking if the edge is micro edge
//=======================================================================
static Standard_Boolean ProcessMicroEdge (const TopoDS_Edge& theEdge,
const Handle(IntTools_Context)& theCtx)
{
TopoDS_Vertex aV1, aV2;
TopExp::Vertices (theEdge, aV1, aV2);
if (aV1.IsNull() || aV2.IsNull())
{
return Standard_False;
}
Standard_Boolean bMicro = BOPTools_AlgoTools::IsMicroEdge (theEdge, theCtx);
if (bMicro && BRepAdaptor_Curve (theEdge).GetType() == GeomAbs_Line)
{
Standard_Real aLen = BRep_Tool::Pnt (aV1).Distance (BRep_Tool::Pnt (aV2));
BRep_Builder().UpdateVertex (aV1, aLen / 2.);
BRep_Builder().UpdateVertex (aV2, aLen / 2.);
}
return bMicro;
}
//=======================================================================
//function : UpdateOrigins
//purpose : Updating origins
//=======================================================================
static void UpdateOrigins (const TopTools_ListOfShape& theLA,
TopTools_DataMapOfShapeListOfShape& theOrigins,
BOPAlgo_Builder& theGF)
{
for (TopTools_ListOfShape::Iterator aItA (theLA); aItA.More(); aItA.Next())
{
const TopoDS_Shape& aS = aItA.Value();
const TopTools_ListOfShape& aLSIm = theGF.Modified (aS);
if (aLSIm.IsEmpty())
{
continue;
}
TopTools_ListOfShape aLSEmpt;
TopTools_ListOfShape* pLS = theOrigins.ChangeSeek (aS);
if (!pLS)
{
pLS = &aLSEmpt;
pLS->Append (aS);
}
for (TopTools_ListOfShape::Iterator aIt (aLSIm); aIt.More(); aIt.Next())
{
const TopoDS_Shape& aSIm = aIt.Value();
if (TopTools_ListOfShape* pLSOr = theOrigins.ChangeSeek (aSIm))
{
// merge two lists
for (TopTools_ListOfShape::Iterator aIt1 (*pLS); aIt1.More(); aIt1.Next())
{
AppendToList (*pLSOr, aIt1.Value());
}
}
else
{
theOrigins.Bind (aSIm, *pLS);
}
}
}
}
//=======================================================================
//function : UpdateImages
//purpose : Updating images of the shapes
//=======================================================================
static void UpdateImages (const TopTools_ListOfShape& theLA,
TopTools_DataMapOfShapeListOfShape& theImages,
BOPAlgo_Builder& theGF,
TopTools_MapOfShape& theModified)
{
for (TopTools_ListOfShape::Iterator aIt (theLA); aIt.More(); aIt.Next())
{
const TopoDS_Shape& aS = aIt.Value();
TopTools_ListOfShape* pLSIm = theImages.ChangeSeek (aS);
if (!pLSIm)
{
const TopTools_ListOfShape& aLSIm = theGF.Modified (aS);
if (aLSIm.Extent())
{
theImages.Bind (aS, aLSIm);
theModified.Add (aS);
}
continue;
}
TopTools_MapOfShape aMFence;
TopTools_ListOfShape aLSImNew;
Standard_Boolean bModified = Standard_False;
// check modifications of the images
for (TopTools_ListOfShape::Iterator aIt1 (*pLSIm); aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aSIm = aIt1.Value();
bModified |= TakeModified (aSIm, theGF.Images(), aLSImNew, &aMFence);
}
if (bModified)
{
*pLSIm = aLSImNew;
theModified.Add (aS);
}
}
}
//=======================================================================
//function : FindCommonParts
//purpose : Looking for the parts of type <theType> contained in both lists
//=======================================================================
static void FindCommonParts (const TopTools_ListOfShape& theLS1,
const TopTools_ListOfShape& theLS2,
TopTools_ListOfShape& theLSC,
const TopAbs_ShapeEnum theType = TopAbs_EDGE)
{
// map shapes in the first list
TopTools_IndexedMapOfShape aMS1;
for (TopTools_ListOfShape::Iterator aIt (theLS1); aIt.More(); aIt.Next())
{
const TopoDS_Shape& aS = aIt.Value();
TopExp::MapShapes (aS, theType, aMS1);
}
if (aMS1.IsEmpty())
{
return;
}
// check for such shapes in the other list
TopTools_MapOfShape aMFence;
for (TopTools_ListOfShape::Iterator aIt (theLS2); aIt.More(); aIt.Next())
{
const TopoDS_Shape& aS = aIt.Value();
TopExp_Explorer aExp (aS, theType);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aST = aExp.Current();
if (aMS1.Contains (aST) && aMFence.Add (aST))
{
theLSC.Append (aST);
}
}
}
}
//=======================================================================
//function : NbPoints
//purpose : Defines number of sample points to get average direction of the edge
//=======================================================================
static Standard_Integer NbPoints (const TopoDS_Edge& theEdge)
{
BRepAdaptor_Curve aBAC (theEdge);
switch (aBAC.GetType())
{
case GeomAbs_Line:
return 1;
default:
return 11;
}
}
//=======================================================================
//function : FindShape
//purpose : Looking for the same sub-shape in the shape
//=======================================================================
static Standard_Boolean FindShape (const TopoDS_Shape& theSWhat,
const TopoDS_Shape& theSWhere,
const BRepOffset_Analyse* theAnalyse,
TopoDS_Shape& theRes)
{
Standard_Boolean bFound = Standard_False;
TopAbs_ShapeEnum aType = theSWhat.ShapeType();
TopExp_Explorer aExp (theSWhere, aType);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aS = aExp.Current();
if (aS.IsSame (theSWhat))
{
theRes = aS;
bFound = Standard_True;
break;
}
}
if (!bFound && theAnalyse)
{
const TopTools_ListOfShape* pLD = theAnalyse->Descendants (theSWhere);
if (pLD)
{
for (TopTools_ListOfShape::Iterator it (*pLD); it.More(); it.Next())
{
const TopoDS_Shape& aS = it.Value();
if (aS.IsSame (theSWhat))
{
theRes = aS;
bFound = Standard_True;
break;
}
}
}
}
return bFound;
}
//=======================================================================
//function : BuildSplitsOfTrimmedFace
//purpose : Building the splits of offset face
//=======================================================================
static void BuildSplitsOfTrimmedFace (const TopoDS_Face& theFace,
const TopoDS_Shape& theEdges,
TopTools_ListOfShape& theLFImages,
const Message_ProgressRange& theRange)
{
BOPAlgo_Splitter aSplitter;
//
aSplitter.AddArgument (theFace);
aSplitter.AddArgument (theEdges);
aSplitter.SetToFillHistory (Standard_False);
aSplitter.Perform (theRange);
if (aSplitter.HasErrors())
{
return;
}
//
// splits of the offset shape
for (TopExp_Explorer anExp (aSplitter.Shape(), TopAbs_FACE); anExp.More(); anExp.Next())
{
theLFImages.Append (anExp.Current());
}
}
//=======================================================================
//function : BuildSplitsOfFace
//purpose : Building the splits of offset face
//=======================================================================
static void BuildSplitsOfFace (const TopoDS_Face& theFace,
const TopoDS_Shape& theEdges,
TopTools_DataMapOfShapeShape& theFacesOrigins,
TopTools_ListOfShape& theLFImages)
{
theLFImages.Clear();
//
// take edges to split the face
TopTools_ListOfShape aLE;
TopExp_Explorer aExp (theEdges, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
TopoDS_Edge aE = TopoDS::Edge (aExp.Current());
aE.Orientation (TopAbs_FORWARD);
aLE.Append (aE);
aE.Orientation (TopAbs_REVERSED);
aLE.Append (aE);
}
//
TopoDS_Face aFF = theFace;
TopAbs_Orientation anOr = theFace.Orientation();
aFF.Orientation (TopAbs_FORWARD);
//
// build pcurves for edges on the face
BRepLib::BuildPCurveForEdgesOnPlane (aLE, aFF);
//
// build splits of faces
BOPAlgo_BuilderFace aBF;
aBF.SetFace (aFF);
aBF.SetShapes (aLE);
aBF.Perform();
if (aBF.HasErrors())
{
return;
}
//
const TopTools_ListOfShape& aLFSp = aBF.Areas();
TopTools_ListIteratorOfListOfShape aItLF (aLFSp);
for (; aItLF.More(); aItLF.Next())
{
TopoDS_Shape& aFSp = aItLF.ChangeValue();
aFSp.Orientation (anOr);
theLFImages.Append (aFSp);
//
theFacesOrigins.Bind (aFSp, theFace);
}
}
//=======================================================================
//function : GetAverageTangent
//purpose : Computes average tangent vector along the curve
//=======================================================================
static gp_Vec GetAverageTangent (const TopoDS_Shape& theS,
const Standard_Integer theNbP)
{
gp_Vec aVA;
TopExp_Explorer aExp (theS, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Edge& aE = *(TopoDS_Edge*)&aExp.Current();
//
Standard_Real aT1, aT2;
const Handle(Geom_Curve)& aC = BRep_Tool::Curve (aE, aT1, aT2);
//
gp_Pnt aP;
gp_Vec aV, aVSum;
Standard_Real aT = aT1;
Standard_Real aDt = (aT2 - aT1) / theNbP;
while (aT <= aT2)
{
aC->D1 (aT, aP, aV);
aVSum += aV.Normalized();
aT += aDt;
}
//
if (aE.Orientation() == TopAbs_REVERSED)
{
aVSum.Reverse();
}
//
aVA += aVSum;
}
return aVA;
}
}
//=======================================================================
//function : BRepOffset_BuildOffsetFaces
//purpose : Auxiliary local class that is used here for building splits
// of offset faces, that are further used for building volumes.
//=======================================================================
class BRepOffset_BuildOffsetFaces
{
public: //! @name Constructor
//! Constructor, taking the history tool to be filled
BRepOffset_BuildOffsetFaces (BRepAlgo_Image& theImage) :
myFaces (NULL),
myAnalyzer (NULL),
myEdgesOrigins (NULL),
myFacesOrigins (NULL),
myETrimEInf (NULL),
myImage (&theImage)
{
myContext = new IntTools_Context();
}
public: //! @name Setting data
//! Sets faces to build splits
void SetFaces (const TopTools_ListOfShape& theFaces) { myFaces = &theFaces; }
//! Sets ascendants/descendants information
void SetAsDesInfo (const Handle(BRepAlgo_AsDes)& theAsDes) { myAsDes = theAsDes; }
//! Sets the analysis info of the input shape
void SetAnalysis (const BRepOffset_Analyse& theAnalyse) { myAnalyzer = &theAnalyse; }
//! Sets origins of the offset edges (from original shape)
void SetEdgesOrigins (TopTools_DataMapOfShapeListOfShape& theEdgesOrigins) { myEdgesOrigins = &theEdgesOrigins; }
//! Sets origins of the offset faces (from original shape)
void SetFacesOrigins (TopTools_DataMapOfShapeShape& theFacesOrigins) { myFacesOrigins = &theFacesOrigins; }
//! Sets infinite (extended) edges for the trimmed ones
void SetInfEdges (TopTools_DataMapOfShapeShape& theETrimEInf) { myETrimEInf = &theETrimEInf; }
public: //! @name Public methods to build the splits
//! Build splits of already trimmed faces
void BuildSplitsOfTrimmedFaces (const Message_ProgressRange& theRange);
//! Building splits of not-trimmed offset faces.
//! For the cases in which invalidities will be found, these invalidities will be rebuilt.
void BuildSplitsOfExtendedFaces (const Message_ProgressRange& theRange);
private: //! @name private methods performing the job
private: //! @name Intersection and post-treatment of edges
//! Intersection of the trimmed edges among themselves
void IntersectTrimmedEdges (const Message_ProgressRange& theRange);
//! Saving connection from trimmed edges to not trimmed ones
void UpdateIntersectedEdges (const TopTools_ListOfShape& theLA,
BOPAlgo_Builder& theGF);
//! Getting edges from AsDes map to build the splits of faces
Standard_Boolean GetEdges (const TopoDS_Face& theFace,
TopoDS_Shape& theEdges,
TopTools_IndexedMapOfShape* theInv = nullptr);
//! Looks for the invalid edges (edges with changed orientation)
//! in the splits of offset faces
void FindInvalidEdges (const TopoDS_Face& theF,
const TopTools_ListOfShape& theLFImages,
BRepOffset_DataMapOfShapeMapOfShape& theDMFMVE,
BRepOffset_DataMapOfShapeMapOfShape& theDMFMNE,
BRepOffset_DataMapOfShapeIndexedMapOfShape& theDMFMIE,
BRepOffset_DataMapOfShapeMapOfShape& theDMFMVIE,
TopTools_DataMapOfShapeListOfShape& theDMEOrLEIm,
TopTools_MapOfShape& theEdgesInvalidByVertex,
TopTools_MapOfShape& theEdgesValidByVertex,
const Message_ProgressRange& theRange);
//! Additional method to look for invalid edges
void FindInvalidEdges (const TopTools_ListOfShape& theLFOffset,
BRepOffset_DataMapOfShapeIndexedMapOfShape& theLocInvEdges,
BRepOffset_DataMapOfShapeMapOfShape& theLocValidEdges,
BRepOffset_DataMapOfShapeMapOfShape& theNeutralEdges);
//! Checks if the edge has been inverted
Standard_Boolean CheckInverted (const TopoDS_Edge& theEIm,
const TopoDS_Face& theFOr,
const TopTools_IndexedDataMapOfShapeListOfShape& theDMVE,
const TopTools_IndexedMapOfShape& theMEdges);
//! Looking for the invalid faces containing inverted edges that can be safely removed
void RemoveInvalidSplitsByInvertedEdges (TopTools_IndexedMapOfShape& theMERemoved);
//! Makes inverted edges located inside loop of invalid edges, invalid as well
void MakeInvertedEdgesInvalid (const TopTools_ListOfShape& theLFOffset);
//! Checks if it is possible to remove the block containing inverted edges
Standard_Boolean CheckInvertedBlock (const TopoDS_Shape& theCB,
const TopTools_ListOfShape& theLCBF,
BRepOffset_DataMapOfShapeMapOfShape& theDMCBVInverted,
BRepOffset_DataMapOfShapeMapOfShape& theDMCBVAll);
//! Updating the maps of images and origins of the offset edges
void FilterEdgesImages (const TopoDS_Shape& theS);
//! Checks additionally the unchecked edges originated from vertices
void CheckEdgesCreatedByVertex();
//! Filtering the invalid edges according to currently invalid faces
void FilterInvalidEdges (const BRepOffset_DataMapOfShapeIndexedMapOfShape& theDMFMIE,
const TopTools_IndexedMapOfShape& theMERemoved,
const TopTools_IndexedMapOfShape& theMEInside,
TopTools_MapOfShape& theMEUseInRebuild);
private: //! @name Checking faces
//! Build splits of faces
void BuildSplitsOfFaces (const Message_ProgressRange& theRange);
//! Looking for the invalid faces by analyzing their invalid edges
void FindInvalidFaces (TopTools_ListOfShape& theLFImages,
const BRepOffset_DataMapOfShapeMapOfShape& theDMFMVE,
const BRepOffset_DataMapOfShapeIndexedMapOfShape& theDMFMIE,
const TopTools_MapOfShape& theMENeutral,
const TopTools_MapOfShape& theEdgesInvalidByVertex,
const TopTools_MapOfShape& theEdgesValidByVertex,
const TopTools_MapOfShape& theMFHoles,
TopTools_IndexedMapOfShape& theMFInvInHole,
TopTools_ListOfShape& theInvFaces,
TopTools_ListOfShape& theInvertedFaces);
//! Find faces inside holes wires from the original face
void FindFacesInsideHoleWires (const TopoDS_Face& theFOrigin,
const TopoDS_Face& theFOffset,
const TopTools_ListOfShape& theLFImages,
const TopTools_DataMapOfShapeListOfShape& theDMEOrLEIm,
const TopTools_IndexedDataMapOfShapeListOfShape& theEFMap,
TopTools_MapOfShape& theMFHoles);
//! Removing invalid splits of faces from valid
void RemoveInvalidSplitsFromValid (const BRepOffset_DataMapOfShapeMapOfShape& theDMFMVIE);
//! Looking for the inside faces that can be safely removed
void RemoveInsideFaces (const TopTools_ListOfShape& theInvertedFaces,
const TopTools_IndexedMapOfShape& theMFToCheckInt,
const TopTools_IndexedMapOfShape& theMFInvInHole,
const TopoDS_Shape& theFHoles,
TopTools_IndexedMapOfShape& theMERemoved,
TopTools_IndexedMapOfShape& theMEInside,
const Message_ProgressRange& theRange);
//! Looking for the connections between faces not to miss some necessary intersection
void ShapesConnections (const TopTools_DataMapOfShapeShape& theDMFOr,
BOPAlgo_Builder& theBuilder);
//! Remove isolated invalid hanging parts
void RemoveHangingParts (const BOPAlgo_MakerVolume& theMV,
const TopTools_DataMapOfShapeShape& theDMFImF,
const TopTools_IndexedMapOfShape& theMFInv,
TopTools_MapOfShape& theMFToRem);
//! Removing valid splits according to results of intersection
void RemoveValidSplits (const TopTools_MapOfShape& theSpRem,
BOPAlgo_Builder& theGF,
TopTools_IndexedMapOfShape& theMERemoved);
//! Removing invalid splits according to the results of intersection
void RemoveInvalidSplits (const TopTools_MapOfShape& theSpRem,
BOPAlgo_Builder& theGF,
TopTools_IndexedMapOfShape& theMERemoved);
//! Filtering of the invalid faces
void FilterInvalidFaces (const TopTools_IndexedDataMapOfShapeListOfShape& theDMEF,
const TopTools_IndexedMapOfShape& theMERemoved);
//! Checks if the face is artificially invalid
Standard_Boolean CheckIfArtificial (const TopoDS_Shape& theF,
const TopTools_ListOfShape& theLFImages,
const TopoDS_Shape& theCE,
const TopTools_IndexedMapOfShape& theMapEInv,
TopTools_MapOfShape& theMENInv);
//! Looking for the faces that have to be rebuilt:
//! * Faces close to invalidity
//! * Faces containing some invalid parts
void FindFacesToRebuild();
//! Intersection of the faces that should be rebuild to resolve all invalidities
void IntersectFaces (TopTools_MapOfShape& theVertsToAvoid,
const Message_ProgressRange& theRange);
//! Preparation of the maps for analyzing intersections of the faces
void PrepareFacesForIntersection (const Standard_Boolean theLookVertToAvoid,
TopTools_IndexedDataMapOfShapeListOfShape& theFLE,
TopTools_DataMapOfShapeListOfShape& theMDone,
TopTools_DataMapOfShapeListOfShape& theDMSF,
TopTools_DataMapOfShapeListOfShape& theMEInfETrim,
TopTools_DataMapOfShapeListOfShape& theDMVEFull,
TopTools_IndexedDataMapOfShapeListOfShape& theDMEFInv);
//! Looking for the invalid vertices
void FindVerticesToAvoid (const TopTools_IndexedDataMapOfShapeListOfShape& theDMEFInv,
const TopTools_DataMapOfShapeListOfShape& theDMVEFull,
TopTools_MapOfShape& theMVRInv);
//! Looking for the faces around each invalidity for intersection
void FindFacesForIntersection (const TopoDS_Shape& theFInv,
const TopTools_IndexedMapOfShape& theME,
const TopTools_DataMapOfShapeListOfShape& theDMSF,
const TopTools_MapOfShape& theMVInvAll,
const Standard_Boolean theArtCase,
TopTools_IndexedMapOfShape& theMFAvoid,
TopTools_IndexedMapOfShape& theMFInt,
TopTools_IndexedMapOfShape& theMFIntExt,
TopTools_ListOfShape& theLFImInt);
//! Analyzing the common edges between splits of offset faces
void ProcessCommonEdges (const TopTools_ListOfShape& theLEC,
const TopTools_IndexedMapOfShape& theME,
const TopTools_DataMapOfShapeListOfShape& theMEInfETrim,
const TopTools_MapOfShape& theAllInvs,
const Standard_Boolean theForceUse,
TopTools_IndexedMapOfShape& theMECV,
TopTools_MapOfShape& theMECheckExt,
TopTools_DataMapOfShapeListOfShape& theDMEETrim,
TopTools_ListOfShape& theLFEi,
TopTools_ListOfShape& theLFEj,
TopTools_IndexedMapOfShape& theMEToInt);
//! Updating the already interfered faces
void UpdateIntersectedFaces (const TopoDS_Shape& theFInv,
const TopoDS_Shape& theFi,
const TopoDS_Shape& theFj,
const TopTools_ListOfShape& theLFInv,
const TopTools_ListOfShape& theLFImi,
const TopTools_ListOfShape& theLFImj,
const TopTools_ListOfShape& theLFEi,
const TopTools_ListOfShape& theLFEj,
TopTools_IndexedMapOfShape& theMEToInt);
//! Intersection of the pair of faces
void IntersectFaces (const TopoDS_Shape& theFInv,
const TopoDS_Shape& theFi,
const TopoDS_Shape& theFj,
const TopTools_ListOfShape& theLFInv,
const TopTools_ListOfShape& theLFImi,
const TopTools_ListOfShape& theLFImj,
TopTools_ListOfShape& theLFEi,
TopTools_ListOfShape& theLFEj,
TopTools_IndexedMapOfShape& theMECV,
TopTools_IndexedMapOfShape& theMEToInt);
//! Intersection of the new intersection edges among themselves
void IntersectAndTrimEdges (const TopTools_IndexedMapOfShape& theMFInt,
const TopTools_IndexedMapOfShape& theMEInt,
const TopTools_DataMapOfShapeListOfShape& theDMEETrim,
const TopTools_IndexedMapOfShape& theMSInv,
const TopTools_IndexedMapOfShape& theMVE,
const TopTools_MapOfShape& theVertsToAvoid,
const TopTools_MapOfShape& theNewVertsToAvoid,
const TopTools_MapOfShape& theMECheckExt,
const TopTools_DataMapOfShapeListOfShape* theSSInterfs,
TopTools_MapOfShape& theMVBounds,
TopTools_DataMapOfShapeListOfShape& theEImages);
//! Looking for the invalid edges by intersecting with invalid vertices
void GetInvalidEdges (const TopTools_MapOfShape& theVertsToAvoid,
const TopTools_MapOfShape& theMVBounds,
BOPAlgo_Builder& theGF,
TopTools_MapOfShape& theMEInv);
//! Making the new splits and updating the maps
void UpdateValidEdges (const TopTools_IndexedDataMapOfShapeListOfShape& theFLE,
const TopTools_IndexedDataMapOfShapeListOfShape& theOENEdges,
const TopTools_MapOfShape& theMVBounds,
const TopTools_MapOfShape& theMEInvOnArt,
TopTools_MapOfShape& theMECheckExt,
TopTools_MapOfShape& theVertsToAvoid,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_DataMapOfShapeListOfShape& theEETrim,
const Message_ProgressRange& theRange);
//! Trims intersection edges
void TrimNewIntersectionEdges (const TopTools_ListOfShape& theLE,
const TopTools_DataMapOfShapeListOfShape& theEETrim,
const TopTools_MapOfShape& theMVBounds,
TopTools_MapOfShape& theMECheckExt,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_MapOfShape& theMEB,
TopTools_MapOfShape& theMVOld,
TopTools_MapOfShape& theMENew,
TopTools_DataMapOfShapeListOfShape& theDMEOr,
TopTools_DataMapOfShapeListOfShape& theMELF);
//! Intersecting the trimmed edges to avoid self-intersections
void IntersectEdges (const TopTools_ListOfShape& theLA,
const TopTools_ListOfShape& theLE,
const TopTools_MapOfShape& theMVBounds,
const TopTools_MapOfShape& theVertsToAvoid,
TopTools_MapOfShape& theMENew,
TopTools_MapOfShape& theMECheckExt,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_DataMapOfShapeListOfShape& theDMEOr,
TopTools_DataMapOfShapeListOfShape& theMELF,
TopoDS_Shape& theSplits);
//! Getting edges from the splits of offset faces
void GetBounds (const TopTools_ListOfShape& theLFaces,
const TopTools_MapOfShape& theMEB,
TopoDS_Shape& theBounds);
//! Get bounding edges that should be updated
void GetBoundsToUpdate (const TopTools_ListOfShape& theLF,
const TopTools_MapOfShape& theMEB,
TopTools_ListOfShape& theLABounds,
TopTools_ListOfShape& theLAValid,
TopoDS_Shape& theBounds);
//! Filter new splits by intersection with bounds
void GetInvalidEdgesByBounds (const TopoDS_Shape& theSplits,
const TopoDS_Shape& theBounds,
const TopTools_MapOfShape& theMVOld,
const TopTools_MapOfShape& theMENew,
const TopTools_DataMapOfShapeListOfShape& theDMEOr,
const TopTools_DataMapOfShapeListOfShape& theMELF,
const TopTools_DataMapOfShapeListOfShape& theEImages,
const TopTools_MapOfShape& theMECheckExt,
const TopTools_MapOfShape& theMEInvOnArt,
TopTools_MapOfShape& theVertsToAvoid,
TopTools_MapOfShape& theMEInv);
//! Filter the images of edges from the invalid edges
void FilterSplits (const TopTools_ListOfShape& theLE,
const TopTools_MapOfShape& theMEFilter,
const Standard_Boolean theIsInv,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopoDS_Shape& theSplits);
//! Updating the maps of images and origins of the offset edges
void UpdateNewIntersectionEdges (const TopTools_ListOfShape& theLE,
const TopTools_DataMapOfShapeListOfShape& theMELF,
const TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_DataMapOfShapeListOfShape& theEETrim);
private:
//! Fill possible gaps (holes) in the splits of the offset faces
void FillGaps (const Message_ProgressRange& theRange);
//! Saving obtained results in history tools
void FillHistory();
private:
// Input data
const TopTools_ListOfShape* myFaces; //!< Input faces which have to be split
Handle(BRepAlgo_AsDes) myAsDes; //!< Ascendants/descendants of the edges faces
const BRepOffset_Analyse* myAnalyzer; //!< Analyzer of the input parameters
TopTools_DataMapOfShapeListOfShape* myEdgesOrigins; //!< Origins of the offset edges (binding between offset edge and original edge)
TopTools_DataMapOfShapeShape* myFacesOrigins; //!< Origins of the offset faces (binding between offset face and original face)
TopTools_DataMapOfShapeShape* myETrimEInf; //!< Binding between trimmed and infinite edge
// Intermediate data
TopTools_DataMapOfShapeListOfShape myOEImages; //!< Images of the offset edges
TopTools_DataMapOfShapeListOfShape myOEOrigins; //!< Origins of the splits of offset edges
TopTools_IndexedDataMapOfShapeListOfShape myOFImages; //!< Images of the offset edges
TopTools_IndexedMapOfShape myInvalidEdges; //!< Edges considered invalid (orientation is changed) in some split of face
TopTools_IndexedMapOfShape myValidEdges; //!< Edges considered valid (orientation is kept) in some split of face
TopTools_IndexedMapOfShape myInvertedEdges; //!< Edges considered inverted (vertices swapped) in some split of face
TopTools_IndexedMapOfShape myEdgesToAvoid; //!< Splits of edges to be avoided when building splits of faces
TopTools_MapOfShape myLastInvEdges; //!< Edges marked invalid on the current step and to be avoided on the next step
TopTools_MapOfShape myModifiedEdges; //!< Edges to be used for building splits
TopTools_IndexedMapOfShape myInsideEdges; //!< Edges located fully inside solids built from the splits of offset faces
TopTools_IndexedDataMapOfShapeListOfShape myInvalidFaces; //!< Invalid faces - splits of offset faces consisting of invalid edges
BRepOffset_DataMapOfShapeIndexedMapOfShape myArtInvalidFaces; //!< Artificially invalid faces - valid faces intentionally marked invalid
//! to be rebuilt on the future steps in the situations when invalid edges
//! are present, but invalid faces not
TopTools_DataMapOfShapeInteger myAlreadyInvFaces; //!< Count number of the same face being marked invalid to avoid infinite
//! rebuilding of the same face
TopTools_DataMapOfShapeListOfShape myFNewHoles; //!< Images of the hole faces of the original face
TopTools_DataMapOfShapeListOfShape mySSInterfs; //!< Intersection information, used to collect intersection pairs during rebuild
TopTools_DataMapOfShapeListOfShape mySSInterfsArt; //!< Intersection information, used to collect intersection pairs during rebuild
NCollection_DataMap <TopoDS_Shape,
BRepOffset_DataMapOfShapeMapOfShape,
TopTools_ShapeMapHasher> myIntersectionPairs; //!< All possible intersection pairs, used to avoid some of the intersections
TopTools_IndexedDataMapOfShapeListOfShape myFacesToRebuild; //!< Faces that have to be rebuilt (invalid and close to invalid faces)
TopTools_MapOfShape myFSelfRebAvoid; //!< Faces that have to be avoided when rebuilding splits of the same offset face
TopoDS_Shape mySolids; //!< Solids built from the splits of faces
// Auxiliary tools
Handle(IntTools_Context) myContext;
// Output
BRepAlgo_Image* myImage; //!< History of modifications
};
//=======================================================================
//function : BuildSplitsOfTrimmedFaces
//purpose :
//=======================================================================
void BRepOffset_BuildOffsetFaces::BuildSplitsOfTrimmedFaces (const Message_ProgressRange& theRange)
{
if (!hasData (myFaces))
{
return;
}
TopTools_DataMapOfShapeListOfShape anEdgesOrigins;
if (!myEdgesOrigins)
{
myEdgesOrigins = &anEdgesOrigins;
}
Message_ProgressScope aPS (theRange, "Building splits of trimmed faces", 5);
// Fuse all edges
IntersectTrimmedEdges (aPS.Next (1));
Message_ProgressScope aPSLoop (aPS.Next (4), NULL, myFaces->Extent());
for (TopTools_ListOfShape::Iterator aItLF (*myFaces); aItLF.More(); aItLF.Next())
{
if (!aPSLoop.More())
{
return;
}
const TopoDS_Face& aF = *(TopoDS_Face*)&aItLF.Value();
TopoDS_Shape aCE;
Standard_Boolean bFound = GetEdges (aF, aCE);
// split the face by the edges
if (!bFound)
{
if (!myImage->HasImage (aF))
{
myOFImages (myOFImages.Add (aF, TopTools_ListOfShape())).Append (aF);
}
continue;
}
TopTools_ListOfShape aLFImages;
BuildSplitsOfTrimmedFace (aF, aCE, aLFImages, aPSLoop.Next());
myOFImages.Add (aF, aLFImages);
}
// Fill history for faces and edges
FillHistory();
}
//=======================================================================
//function : BuildSplitsOfExtendedFaces
//purpose :
//=======================================================================
void BRepOffset_BuildOffsetFaces::BuildSplitsOfExtendedFaces (const Message_ProgressRange& theRange)
{
// Check input data
if (!hasData (myFaces) || !hasData (myEdgesOrigins) || !hasData (myFacesOrigins) || !hasData (myETrimEInf))
{
return;
}
Message_ProgressScope aPS (theRange, "Building splits of extended faces", 100.);
// Scope has to be added into a loop of undefined size.
// In general there are about 2 to 5 loops performed, each time
// decreasing complexity. So reserve for each next loop smaller time.
// Reserve also 4% on filling gaps after the faces are built.
Standard_Real aWhole = 100. - 4.;
// Fusing all trimmed offset edges to avoid self-intersections in the splits
IntersectTrimmedEdges (aPS.Next());
if (!aPS.More())
{
return;
}
// vertices to avoid
TopTools_MapOfShape aVertsToAvoid;
// Limit total number of attempts by 10. This should be extra, as each invalid face can be
// rebuilt only three times. So, in general, there are about 2-5 loops done.
const Standard_Integer aNbMaxAttempts = 10;
// First progress portion is the half of the whole. Each next step is half of the previous:
// 48%, 24%, 12%, 6% and so on. This way in three loops it will already be 84%,
// and in four - 90%. So even if the loop will stop earlier, the not advanced scope
// will be acceptable.
Standard_Real aPart = aWhole / 2.;
for (Standard_Integer iCount = 1; iCount <= aNbMaxAttempts; ++iCount, aPart /= 2.)
{
if (!aPS.More())
{
return;
}
// Clear the data before further faces construction
myInvalidFaces.Clear();
myArtInvalidFaces.Clear();
myInvalidEdges.Clear();
myInvertedEdges.Clear();
mySSInterfs.Clear();
mySSInterfsArt.Clear();
myIntersectionPairs.Clear();
mySolids.Nullify();
myFacesToRebuild.Clear();
myFSelfRebAvoid.Clear();
// Split progress range on
// * building faces basing on currently available edges and
// * rebuilding faces basing on edges classification
Message_ProgressScope aPSLoop (aPS.Next (aPart), NULL, 10.);
// Build splits of the faces having new intersection edges
BuildSplitsOfFaces (aPSLoop.Next (7.));
if (myInvalidFaces.IsEmpty())
{
break;
}
// Find faces to rebuild
FindFacesToRebuild();
if (myFacesToRebuild.IsEmpty())
{
break;
}
// Perform new intersections
myModifiedEdges.Clear();
IntersectFaces (aVertsToAvoid, aPSLoop.Next (3.));
}
// Fill possible gaps in the splits of offset faces to increase possibility of
// creating closed volume from these splits
FillGaps (aPS.Next (4.));
// Fill history for faces and edges
FillHistory();
}
//=======================================================================
//function : UpdateIntersectedEdges
//purpose : Saving connection from trimmed edges to not trimmed ones
//=======================================================================
void BRepOffset_BuildOffsetFaces::UpdateIntersectedEdges (const TopTools_ListOfShape& theLA,
BOPAlgo_Builder& theGF)
{
for (TopTools_ListOfShape::Iterator aItA (theLA); aItA.More(); aItA.Next())
{
const TopoDS_Shape& aS = aItA.Value();
const TopoDS_Shape* pEInf = myETrimEInf->Seek (aS);
if (!pEInf)
{
continue;
}
const TopTools_ListOfShape& aLSIm = theGF.Modified (aS);
if (aLSIm.IsEmpty())
{
continue;
}
for (TopTools_ListOfShape::Iterator aIt (aLSIm); aIt.More(); aIt.Next())
{
const TopoDS_Shape& aEIm = aIt.Value();
if (!myETrimEInf->IsBound (aEIm))
{
myETrimEInf->Bind (aEIm, *pEInf);
}
}
}
}
//=======================================================================
//function : IntersectTrimmedEdges
//purpose :
//=======================================================================
void BRepOffset_BuildOffsetFaces::IntersectTrimmedEdges (const Message_ProgressRange& theRange)
{
// get edges to intersect from descendants of the offset faces
TopTools_ListOfShape aLS;
//
Message_ProgressScope aPS (theRange, NULL, 2);
TopTools_ListIteratorOfListOfShape aItLF (*myFaces);
for (; aItLF.More(); aItLF.Next())
{
if (!aPS.More())
{
return;
}
const TopoDS_Face& aF = *(TopoDS_Face*)&aItLF.Value();
//
const TopTools_ListOfShape& aLE = myAsDes->Descendant (aF);
TopTools_ListIteratorOfListOfShape aItLE (aLE);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Edge& aE = *(TopoDS_Edge*)&aItLE.Value();
//
if (ProcessMicroEdge (aE, myContext))
{
continue;
}
//
if (myModifiedEdges.Add (aE))
{
aLS.Append (aE);
}
}
}
//
if (aLS.Extent() < 2)
{
// nothing to intersect
return;
}
//
// perform intersection of the edges
BOPAlgo_Builder aGFE;
aGFE.SetArguments (aLS);
aGFE.Perform (aPS.Next());
if (aGFE.HasErrors())
{
return;
}
//
TopTools_ListOfShape aLA;
// fill map with edges images
Message_ProgressScope aPSLoop (aPS.Next(), NULL, aLS.Size());
for (TopTools_ListOfShape::Iterator aIt (aLS); aIt.More(); aIt.Next(), aPSLoop.Next())
{
if (!aPSLoop.More())
{
return;
}
const TopoDS_Shape& aE = aIt.Value();
const TopTools_ListOfShape& aLEIm = aGFE.Modified (aE);
if (aLEIm.IsEmpty())
{
continue;
}
//
aLA.Append (aE);
// save images
myOEImages.Bind (aE, aLEIm);
// save origins
TopTools_ListIteratorOfListOfShape aItLE (aLEIm);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEIm = aItLE.Value();
if (TopTools_ListOfShape* pLEOr = myOEOrigins.ChangeSeek (aEIm))
{
AppendToList (*pLEOr, aE);
}
else
{
myOEOrigins.Bound (aEIm, TopTools_ListOfShape())->Append (aE);
}
}
}
//
UpdateOrigins (aLA, *myEdgesOrigins, aGFE);
UpdateIntersectedEdges (aLA, aGFE);
}
namespace
{
//=======================================================================
//function : CheckConnectionsOfFace
//purpose : Checks number of connections for theFace with theLF
// Returns true if number of connections more than 1
//=======================================================================
static Standard_Boolean checkConnectionsOfFace(const TopoDS_Shape& theFace,
const TopTools_ListOfShape& theLF)
{
TopTools_IndexedMapOfShape aShapeVert;
for (TopTools_ListOfShape::Iterator aFImIterator(theLF); aFImIterator.More(); aFImIterator.Next())
{
const TopoDS_Shape& aShape = aFImIterator.Value();
if (aShape.IsSame(theFace))
{
continue;
}
TopExp::MapShapes(aShape, TopAbs_VERTEX, aShapeVert);
}
Standard_Integer aNbConnections = 0;
TopTools_IndexedMapOfShape aFaceVertices;
TopExp::MapShapes(theFace, TopAbs_VERTEX, aFaceVertices);
for (TopTools_IndexedMapOfShape::Iterator aVertIter(aFaceVertices); aVertIter.More(); aVertIter.Next())
{
const TopoDS_Shape& aVert = aVertIter.Value();
if (aShapeVert.Contains(aVert))
{
++aNbConnections;
}
if (aNbConnections > 1)
{
return Standard_True;
}
}
return Standard_False;
}
}
//=======================================================================
//function : BuildSplitsOfFaces
//purpose : Building the splits of offset faces and
// looking for the invalid splits
//=======================================================================
void BRepOffset_BuildOffsetFaces::BuildSplitsOfFaces (const Message_ProgressRange& theRange)
{
BRep_Builder aBB;
Standard_Integer i, aNb;
//
// processed faces
TopTools_ListOfShape aLFDone;
// extended face - map of neutral edges, i.e. in one split - valid and in other - invalid
BRepOffset_DataMapOfShapeMapOfShape aDMFMNE;
// map of valid edges for each face
BRepOffset_DataMapOfShapeMapOfShape aDMFMVE;
// map of invalid edges for each face
BRepOffset_DataMapOfShapeIndexedMapOfShape aDMFMIE;
// map of valid inverted edges for the face
BRepOffset_DataMapOfShapeMapOfShape aDMFMVIE;
// map of splits to check for internals
TopTools_IndexedMapOfShape aMFToCheckInt;
// map of edges created from vertex and marked as invalid
TopTools_MapOfShape aMEdgeInvalidByVertex;
// map of edges created from vertex and marked as valid
TopTools_MapOfShape aMEdgeValidByVertex;
// connection map from old edges to new ones
TopTools_DataMapOfShapeListOfShape aDMEOrLEIm;
//
// Outer range
Message_ProgressScope aPSOuter (theRange, NULL, 10.);
// build splits of faces
Message_ProgressScope aPSBF (aPSOuter.Next (3.), "Building faces", 2 * myFaces->Extent());
TopTools_ListOfShape::Iterator aItLF (*myFaces);
for (; aItLF.More(); aItLF.Next(), aPSBF.Next())
{
if (!aPSBF.More())
{
return;
}
const TopoDS_Face& aF = *(TopoDS_Face*)&aItLF.Value();
//
TopTools_ListOfShape* pLFIm = myOFImages.ChangeSeek (aF);
if (pLFIm && pLFIm->IsEmpty())
{
continue;
}
// get edges by which the face should be split
TopoDS_Shape aCE;
TopTools_IndexedMapOfShape aMapEInv;
Standard_Boolean bFound = GetEdges (aF, aCE, &aMapEInv);
if (!bFound)
{
continue;
}
//
#ifdef OFFSET_DEBUG
// check the found edges on self-intersection
BRepAlgoAPI_Check aChecker (aCE);
if (!aChecker.IsValid())
{
std::cout << "Offset_i_c Error: set of edges to build faces is self-intersecting\n";
}
#endif
// build splits
TopTools_ListOfShape aLFImages;
BuildSplitsOfFace (aF, aCE, *myFacesOrigins, aLFImages);
//
if (aMapEInv.Extent())
{
// check if all possible faces are built
TopTools_MapOfShape aMENInv;
Standard_Boolean bArtificialCase = aLFImages.IsEmpty() ||
CheckIfArtificial (aF, aLFImages, aCE, aMapEInv, aMENInv);
//
// try to build splits using invalid edges
TopoDS_Compound aCE1;
aBB.MakeCompound (aCE1);
aBB.Add (aCE1, aCE);
for (i = 1; i <= aMapEInv.Extent(); ++i)
{
aBB.Add (aCE1, aMapEInv (i));
}
//
TopTools_ListOfShape aLFImages1;
BuildSplitsOfFace (aF, aCE1, *myFacesOrigins, aLFImages1);
//
// check if the rebuilding has added some new faces to the splits
for (TopTools_ListIteratorOfListOfShape aItLFIm (aLFImages1); aItLFIm.More();)
{
Standard_Boolean bAllInv = Standard_True;
// Additional check for artificial case
// if current image face consist only of edges from aMapEInv and aMENInv
// then recheck current face for the futher processing
Standard_Boolean aToReCheckFace = bArtificialCase;
const TopoDS_Shape& aFIm = aItLFIm.Value();
TopExp_Explorer aExpE (aFIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aE = aExpE.Current();
if (!aMapEInv.Contains (aE))
{
bAllInv = Standard_False;
if (!aMENInv.Contains (aE))
{
aToReCheckFace = Standard_False;
break;
}
}
}
// if current image face is to recheck then check number of connections for this face
// with other image faces for current face
if (!aExpE.More() && aToReCheckFace)
{
aToReCheckFace = checkConnectionsOfFace(aFIm, aLFImages1);
}
// do not delete image face from futher processing if aToReCheckFace is true
if (!aExpE.More() && !aToReCheckFace)
{
if (bAllInv)
{
aMFToCheckInt.Add (aFIm);
}
aLFImages1.Remove (aItLFIm);
}
else
{
aItLFIm.Next();
}
}
//
if (bArtificialCase)
{
if (aLFImages.Extent() == aLFImages1.Extent())
{
bArtificialCase = Standard_False;
}
else
{
aLFImages = aLFImages1;
}
}
//
if (bArtificialCase)
{
// make the face invalid
TopTools_IndexedMapOfShape aMEInv;
//
*pLFIm = aLFImages;
TopTools_ListIteratorOfListOfShape aItLFIm (aLFImages);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
TopExp_Explorer aExpE (aFIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aE = aExpE.Current();
if (aMapEInv.Contains (aE))
{
myInvalidEdges.Add (aE);
aMEInv.Add (aE);
}
else
{
myValidEdges.Add (aE);
}
}
}
//
myArtInvalidFaces.Bind (aF, aMEInv);
aDMFMIE.Bind (aF, aMEInv);
aLFDone.Append (aF);
//
continue;
}
}
// find invalid edges
FindInvalidEdges (aF, aLFImages, aDMFMVE, aDMFMNE, aDMFMIE, aDMFMVIE,
aDMEOrLEIm, aMEdgeInvalidByVertex, aMEdgeValidByVertex, aPSBF.Next());
// save the new splits
if (!pLFIm)
{
pLFIm = &myOFImages (myOFImages.Add (aF, TopTools_ListOfShape()));
}
else
{
pLFIm->Clear();
}
pLFIm->Append (aLFImages);
//
aLFDone.Append (aF);
}
//
if (myInvalidEdges.IsEmpty() && myArtInvalidFaces.IsEmpty() && aDMFMIE.IsEmpty())
{
return;
}
// Additional step to find invalid edges by checking unclassified edges
// in the splits of SD faces
FindInvalidEdges (aLFDone, aDMFMIE, aDMFMVE, aDMFMNE);
// Additional step to mark inverted edges located inside loops
// of invalid edges as invalid as well
MakeInvertedEdgesInvalid (aLFDone);
#ifdef OFFSET_DEBUG
// show invalid edges
TopoDS_Compound aCEInv1;
BRep_Builder().MakeCompound (aCEInv1);
Standard_Integer aNbEInv = myInvalidEdges.Extent();
for (i = 1; i <= aNbEInv; ++i)
{
const TopoDS_Shape& aE = myInvalidEdges (i);
BRep_Builder().Add (aCEInv1, aE);
}
// show valid edges
TopoDS_Compound aCEVal1;
BRep_Builder().MakeCompound (aCEVal1);
aNbEInv = myValidEdges.Extent();
for (i = 1; i <= aNbEInv; ++i)
{
const TopoDS_Shape& aE = myValidEdges (i);
BRep_Builder().Add (aCEVal1, aE);
}
// show inverted edges
TopoDS_Compound aCEInverted;
BRep_Builder().MakeCompound (aCEInverted);
for (i = 1; i <= myInvertedEdges.Extent(); ++i)
{
BRep_Builder().Add (aCEInverted, myInvertedEdges(i));
}
#endif
#ifdef OFFSET_DEBUG
// Show all obtained splits of faces
TopoDS_Compound aCFIm1;
BRep_Builder().MakeCompound (aCFIm1);
#endif
// Build Edge-Face connectivity map to find faces which removal
// may potentially lead to creation of the holes in the faces
// preventing from obtaining closed volume in the result
TopTools_IndexedDataMapOfShapeListOfShape anEFMap;
const Standard_Integer aNbF = myOFImages.Extent();
for (i = 1; i <= aNbF; ++i)
{
TopTools_ListIteratorOfListOfShape itLFIm (myOFImages (i));
for (; itLFIm.More(); itLFIm.Next())
{
TopExp::MapShapesAndAncestors (itLFIm.Value(), TopAbs_EDGE, TopAbs_FACE, anEFMap);
#ifdef OFFSET_DEBUG
BRep_Builder().Add (aCFIm1, itLFIm.Value());
#endif
}
}
TopTools_MapOfShape anEmptyMap;
// invalid faces inside the holes
TopTools_IndexedMapOfShape aMFInvInHole;
// all hole faces
TopoDS_Compound aFHoles;
aBB.MakeCompound (aFHoles);
// Find the faces containing only the inverted edges and the invalid ones
TopTools_ListOfShape anInvertedFaces;
Message_ProgressScope aPSIF (aPSOuter.Next (2.), "Checking validity of faces", aLFDone.Extent());
// find invalid faces
// considering faces containing only invalid edges as invalid
aItLF.Initialize (aLFDone);
for (; aItLF.More(); aItLF.Next(), aPSIF.Next())
{
if (!aPSIF.More())
{
return;
}
const TopoDS_Face& aF = TopoDS::Face (aItLF.Value());
TopTools_ListOfShape& aLFImages = myOFImages.ChangeFromKey (aF);
//
TopTools_ListOfShape aLFInv;
Standard_Boolean bArtificialCase = myArtInvalidFaces.IsBound (aF);
if (bArtificialCase)
{
aLFInv = aLFImages;
}
else
{
// neutral edges
const TopTools_MapOfShape* pMNE = aDMFMNE.ChangeSeek (aF);
if (!pMNE)
{
pMNE = &anEmptyMap;
}
// find faces inside holes wires
TopTools_MapOfShape aMFHoles;
const TopoDS_Face& aFOr = TopoDS::Face (myFacesOrigins->Find (aF));
FindFacesInsideHoleWires (aFOr, aF, aLFImages, aDMEOrLEIm, anEFMap, aMFHoles);
//
TopTools_MapIteratorOfMapOfShape aItMH (aMFHoles);
for (; aItMH.More(); aItMH.Next())
{
aBB.Add (aFHoles, aItMH.Value());
}
//
// find invalid faces
FindInvalidFaces (aLFImages, aDMFMVE, aDMFMIE, *pMNE, aMEdgeInvalidByVertex,
aMEdgeValidByVertex, aMFHoles, aMFInvInHole, aLFInv, anInvertedFaces);
}
//
if (aLFInv.Extent())
{
if (myAlreadyInvFaces.IsBound (aF))
{
if (myAlreadyInvFaces.Find (aF) > 2)
{
if (aLFInv.Extent() == aLFImages.Extent() && !bArtificialCase)
{
aLFImages.Clear();
}
//
aLFInv.Clear();
}
}
myInvalidFaces.Add (aF, aLFInv);
}
}
//
if (myInvalidFaces.IsEmpty())
{
myInvalidEdges.Clear();
return;
}
//
#ifdef OFFSET_DEBUG
// show invalid faces
TopoDS_Compound aCFInv1;
BRep_Builder().MakeCompound (aCFInv1);
Standard_Integer aNbFInv = myInvalidFaces.Extent();
for (i = 1; i <= aNbFInv; ++i)
{
const TopTools_ListOfShape& aLFInv = myInvalidFaces (i);
TopTools_ListIteratorOfListOfShape aItLFInv (aLFInv);
for (; aItLFInv.More(); aItLFInv.Next())
{
const TopoDS_Shape& aFIm = aItLFInv.Value();
BRep_Builder().Add (aCFInv1, aFIm);
}
}
#endif
//
TopTools_IndexedMapOfShape aMERemoved;
// remove invalid splits of faces using inverted edges
RemoveInvalidSplitsByInvertedEdges (aMERemoved);
if (myInvalidFaces.IsEmpty())
{
myInvalidEdges.Clear();
return;
}
//
// remove invalid splits from valid splits
RemoveInvalidSplitsFromValid (aDMFMVIE);
//
// remove inside faces
RemoveInsideFaces (anInvertedFaces, aMFToCheckInt, aMFInvInHole, aFHoles,
aMERemoved, myInsideEdges, aPSOuter.Next (5.));
//
// make compound of valid splits
TopoDS_Compound aCFIm;
aBB.MakeCompound (aCFIm);
//
aNb = myOFImages.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopTools_ListOfShape& aLFIm = myOFImages (i);
aItLF.Initialize (aLFIm);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
aBB.Add (aCFIm, aFIm);
}
}
//
TopTools_IndexedDataMapOfShapeListOfShape aDMEF;
TopExp::MapShapesAndAncestors (aCFIm, TopAbs_EDGE, TopAbs_FACE, aDMEF);
//
// filter maps of images and origins
FilterEdgesImages (aCFIm);
//
// filter invalid faces
FilterInvalidFaces (aDMEF, aMERemoved.Extent() ? myInsideEdges : aMERemoved);
aNb = myInvalidFaces.Extent();
if (!aNb)
{
myInvalidEdges.Clear();
return;
}
//
#ifdef OFFSET_DEBUG
// show invalid faces
TopoDS_Compound aCFInv;
BRep_Builder().MakeCompound (aCFInv);
aNbFInv = myInvalidFaces.Extent();
for (i = 1; i <= aNbFInv; ++i)
{
const TopTools_ListOfShape& aLFInv = myInvalidFaces (i);
TopTools_ListIteratorOfListOfShape aItLFInv (aLFInv);
for (; aItLFInv.More(); aItLFInv.Next())
{
const TopoDS_Shape& aFIm = aItLFInv.Value();
BRep_Builder().Add (aCFInv, aFIm);
}
}
#endif
//
// filter invalid edges
TopTools_MapOfShape aMEUseInRebuild;
FilterInvalidEdges (aDMFMIE, aMERemoved,
aMERemoved.Extent() ? myInsideEdges : aMERemoved,
aMEUseInRebuild);
//
// Check additionally validity of edges originated from vertices.
CheckEdgesCreatedByVertex();
#ifdef OFFSET_DEBUG
// show invalid edges
TopoDS_Compound aCEInv;
BRep_Builder().MakeCompound (aCEInv);
aNbEInv = myInvalidEdges.Extent();
for (i = 1; i <= aNbEInv; ++i)
{
const TopoDS_Shape& aE = myInvalidEdges (i);
BRep_Builder().Add (aCEInv, aE);
}
#endif
//
myLastInvEdges.Clear();
aNb = myInvalidEdges.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aE = myInvalidEdges (i);
myLastInvEdges.Add (aE);
if (!aMEUseInRebuild.Contains(aE))
{
myEdgesToAvoid.Add (aE);
}
}
//
aNb = myInvalidFaces.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
if (myAlreadyInvFaces.IsBound (aF))
{
myAlreadyInvFaces.ChangeFind (aF)++;
}
else
{
myAlreadyInvFaces.Bind (aF, 1);
}
}
}
//=======================================================================
//function : GetEdges
//purpose : Getting edges from AsDes map to build the splits of faces
//=======================================================================
Standard_Boolean BRepOffset_BuildOffsetFaces::GetEdges (const TopoDS_Face& theFace,
TopoDS_Shape& theEdges,
TopTools_IndexedMapOfShape* theInvMap)
{
// get boundary edges
TopTools_MapOfShape aMFBounds;
TopExp_Explorer aExp (theFace, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (const TopTools_ListOfShape* pLEIm = myOEImages.Seek (aE))
{
for (TopTools_ListOfShape::Iterator aItLE (*pLEIm); aItLE.More(); aItLE.Next())
{
aMFBounds.Add (aItLE.Value());
}
}
else
{
aMFBounds.Add (aE);
}
}
BRep_Builder aBB;
Standard_Boolean bFound (Standard_False), bUpdate (Standard_False);
// the resulting edges
TopoDS_Compound anEdges;
aBB.MakeCompound (anEdges);
// Fence map
TopTools_MapOfShape aMEFence;
// the edges by which the offset face should be split
const TopTools_ListOfShape& aLE = myAsDes->Descendant (theFace);
TopTools_ListIteratorOfListOfShape aItLE (aLE);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Edge& aE = TopoDS::Edge (aItLE.Value());
//
if (!bUpdate)
{
bUpdate = myModifiedEdges.Contains (aE);
}
//
const TopTools_ListOfShape* pLEIm = myOEImages.Seek (aE);
if (pLEIm)
{
TopTools_ListIteratorOfListOfShape aItLEIm (*pLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Edge& aEIm = TopoDS::Edge (aItLEIm.Value());
//
if (!aMEFence.Add (aEIm))
continue;
if (myEdgesToAvoid.Contains (aEIm))
{
if (theInvMap)
{
theInvMap->Add (aEIm);
}
if (!bUpdate)
{
bUpdate = myLastInvEdges.Contains (aEIm);
}
continue;
}
// check for micro edge
if (ProcessMicroEdge (aEIm, myContext))
{
continue;
}
//
aBB.Add (anEdges, aEIm);
if (!bFound)
{
bFound = !aMFBounds.Contains (aEIm);
}
//
if (!bUpdate)
{
bUpdate = myModifiedEdges.Contains (aEIm);
}
}
}
else
{
if (myEdgesToAvoid.Contains (aE))
{
if (theInvMap)
{
theInvMap->Add (aE);
}
if (!bUpdate)
{
bUpdate = myLastInvEdges.Contains (aE);
}
continue;
}
//
if (ProcessMicroEdge (aE, myContext))
{
continue;
}
aBB.Add (anEdges, aE);
if (!bFound)
{
bFound = !aMFBounds.Contains (aE);
}
}
}
//
theEdges = anEdges;
return bFound && bUpdate;
}
//=======================================================================
//function : CheckIfArtificial
//purpose : Checks if the face is artificially invalid
//=======================================================================
Standard_Boolean BRepOffset_BuildOffsetFaces::CheckIfArtificial (const TopoDS_Shape& theF,
const TopTools_ListOfShape& theLFImages,
const TopoDS_Shape& theCE,
const TopTools_IndexedMapOfShape& theMapEInv,
TopTools_MapOfShape& theMENInv)
{
// all boundary edges should be used
TopTools_IndexedMapOfShape aMEUsed;
TopTools_ListIteratorOfListOfShape aItLFIm (theLFImages);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
TopExp::MapShapes (aFIm, TopAbs_EDGE, aMEUsed);
TopExp::MapShapes (aFIm, TopAbs_VERTEX, aMEUsed);
}
//
TopTools_IndexedDataMapOfShapeListOfShape aMVE;
TopExp::MapShapesAndAncestors (theCE, TopAbs_VERTEX, TopAbs_EDGE, aMVE);
//
Standard_Integer i, aNb = theMapEInv.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aEInv = theMapEInv (i);
TopExp_Explorer aExpV (aEInv, TopAbs_VERTEX);
for (; aExpV.More(); aExpV.Next())
{
const TopoDS_Shape& aVEInv = aExpV.Current();
const TopTools_ListOfShape* pLENInv = aMVE.Seek (aVEInv);
if (pLENInv)
{
TopTools_ListIteratorOfListOfShape aItLEInv (*pLENInv);
for (; aItLEInv.More(); aItLEInv.Next())
{
const TopoDS_Shape& aENInv = aItLEInv.Value();
if (!aMEUsed.Contains (aENInv))
{
theMENInv.Add (aENInv);
}
}
}
}
}
//
if (theMENInv.IsEmpty())
{
return Standard_False;
}
//
TopTools_IndexedMapOfShape aMEFound;
TopExp::MapShapes (theCE, TopAbs_EDGE, aMEFound);
//
const TopTools_ListOfShape& aLE = myAsDes->Descendant (theF);
TopTools_ListIteratorOfListOfShape aItLE (aLE);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Edge& aE = TopoDS::Edge (aItLE.Value());
//
if (const TopTools_ListOfShape* pLEIm = myOEImages.Seek (aE))
{
Standard_Boolean bChecked = Standard_False;
TopTools_ListIteratorOfListOfShape aItLEIm (*pLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Edge& aEIm = TopoDS::Edge (aItLEIm.Value());
if (!aMEFound.Contains (aEIm) || theMENInv.Contains (aEIm))
{
continue;
}
//
bChecked = Standard_True;
if (aMEUsed.Contains (aEIm))
{
break;
}
}
//
if (bChecked && !aItLEIm.More())
{
break;
}
}
else
{
if (aMEFound.Contains (aE) && !theMENInv.Contains (aE) && !aMEUsed.Contains (aE))
{
break;
}
}
}
//
return aItLE.More();
}
//=======================================================================
//function : FindInvalidEdges
//purpose : Looking for the invalid edges
//=======================================================================
void BRepOffset_BuildOffsetFaces::FindInvalidEdges (const TopoDS_Face& theF,
const TopTools_ListOfShape& theLFImages,
BRepOffset_DataMapOfShapeMapOfShape& theDMFMVE,
BRepOffset_DataMapOfShapeMapOfShape& theDMFMNE,
BRepOffset_DataMapOfShapeIndexedMapOfShape& theDMFMIE,
BRepOffset_DataMapOfShapeMapOfShape& theDMFMVIE,
TopTools_DataMapOfShapeListOfShape& theDMEOrLEIm,
TopTools_MapOfShape& theEdgesInvalidByVertex,
TopTools_MapOfShape& theEdgesValidByVertex,
const Message_ProgressRange& theRange)
{
// Edge is considered as invalid in the following cases:
// 1. Its orientation on the face has changed comparing to the originals edge and face;
// 2. The vertices of the edge have changed places comparing to the originals edge and face.
//
// The edges created from vertices, i.e. as intersection between two faces connected only
// by VERTEX, will also be checked on validity. For these edges the correct orientation will
// be defined by the edges on the original face adjacent to the connection vertex
// original face
const TopoDS_Face& aFOr = *(TopoDS_Face*)&myFacesOrigins->Find (theF);
// invalid edges
TopTools_IndexedMapOfShape aMEInv;
// valid edges
TopTools_MapOfShape aMEVal;
// internal edges
TopTools_MapOfShape aMEInt;
//
// maps for checking the inverted edges
TopTools_IndexedDataMapOfShapeListOfShape aDMVE, aDMEF;
TopTools_IndexedMapOfShape aMEdges;
// back map from the original shapes to their offset images
TopTools_DataMapOfShapeListOfShape anImages;
//
Message_ProgressScope aPS (theRange, "Checking validity of edges", 2 * theLFImages.Extent());
TopTools_ListIteratorOfListOfShape aItLF (theLFImages);
for (; aItLF.More(); aItLF.Next(), aPS.Next())
{
if (!aPS.More())
{
return;
}
const TopoDS_Face& aFIm = *(TopoDS_Face*)&aItLF.Value();
//
TopExp_Explorer aExp (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
// keep all edges
aMEdges.Add (aE);
//
// keep connection from edges to faces
TopTools_ListOfShape* pLF = aDMEF.ChangeSeek (aE);
if (!pLF)
{
pLF = &aDMEF (aDMEF.Add (aE, TopTools_ListOfShape()));
}
AppendToList (*pLF, aFIm);
//
// keep connection from vertices to edges
TopoDS_Iterator aItV (aE);
for (; aItV.More(); aItV.Next())
{
const TopoDS_Shape& aV = aItV.Value();
//
TopTools_ListOfShape* pLE = aDMVE.ChangeSeek (aV);
if (!pLE)
{
pLE = &aDMVE (aDMVE.Add (aV, TopTools_ListOfShape()));
}
AppendToList (*pLE, aE);
}
// back map from original edges to their offset images
const TopTools_ListOfShape* pLOr = myEdgesOrigins->Seek (aE);
if (!pLOr)
continue;
for (TopTools_ListOfShape::Iterator itOr (*pLOr); itOr.More(); itOr.Next())
{
const TopoDS_Shape& aSOr = itOr.Value();
TopoDS_Shape aSInF;
if (!FindShape (aSOr, aFOr, myAnalyzer, aSInF))
continue;
TopTools_ListOfShape* pImages = anImages.ChangeSeek (aSInF);
if (!pImages)
pImages = anImages.Bound (aSInF, TopTools_ListOfShape());
AppendToList (*pImages, aE);
}
}
}
//
// the map will be used to find the edges on the original face
// adjacent to the same vertex. It will be filled at first necessity;
TopTools_IndexedDataMapOfShapeListOfShape aDMVEFOr;
//
aItLF.Initialize (theLFImages);
for (; aItLF.More(); aItLF.Next(), aPS.Next())
{
if (!aPS.More())
{
return;
}
const TopoDS_Face& aFIm = *(TopoDS_Face*)&aItLF.Value();
//
// valid edges for this split
TopTools_MapOfShape aMVE;
// invalid edges for this split
TopTools_IndexedMapOfShape aMIE;
//
TopExp_Explorer aExp (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Edge& aEIm = *(TopoDS_Edge*)&aExp.Current();
//
if (aEIm.Orientation() == TopAbs_INTERNAL)
{
aMEInt.Add (aEIm);
continue;
}
//
const TopTools_ListOfShape* pLEOr = myEdgesOrigins->Seek (aEIm);
if (!pLEOr || pLEOr->IsEmpty())
{
continue;
}
//
Standard_Integer aNbVOr = 0;
TopTools_ListIteratorOfListOfShape aItLEO (*pLEOr);
for (; aItLEO.More(); aItLEO.Next())
{
if (aItLEO.Value().ShapeType() == TopAbs_VERTEX)
{
++aNbVOr;
}
}
if (aNbVOr > 1 && (pLEOr->Extent() - aNbVOr) > 1)
continue;
//
TopTools_MapOfShape aME, aMV, aMF;
Standard_Boolean bInvalid = Standard_False, bChecked = Standard_False;
Standard_Integer aNbP = NbPoints (aEIm), aNbInv = 0;
Standard_Boolean bUseVertex = !aNbVOr ? Standard_False :
(aNbVOr == 1 &&
aDMEF.FindFromKey (aEIm).Extent() == 1 &&
!myOEOrigins.IsBound (aEIm));
//
aItLEO.Initialize (*pLEOr);
for (; aItLEO.More(); aItLEO.Next())
{
const TopoDS_Shape& aSOr = aItLEO.Value();
Standard_Boolean bVertex = (aSOr.ShapeType() == TopAbs_VERTEX);
//
TopoDS_Shape aEOrF;
if (bVertex)
{
// for some cases it is impossible to check the validity of the edge
if (!bUseVertex)
{
continue;
}
// find edges on the original face adjacent to this vertex
if (aDMVEFOr.IsEmpty())
{
// fill the map
TopExp::MapShapesAndAncestors (aFOr, TopAbs_VERTEX, TopAbs_EDGE, aDMVEFOr);
}
//
TopTools_ListOfShape* pLEFOr = aDMVEFOr.ChangeSeek (aSOr);
if (pLEFOr)
{
TopoDS_Compound aCEOr;
BRep_Builder().MakeCompound (aCEOr);
// Avoid classification of edges originated from vertices
// located between tangent edges
Standard_Boolean bAllTgt = Standard_True;
TopTools_ListIteratorOfListOfShape aItLEFOr (*pLEFOr);
gp_Vec aVRef = GetAverageTangent (aItLEFOr.Value(), aNbP);
for (; aItLEFOr.More(); aItLEFOr.Next())
{
const TopoDS_Shape& aEOr = aItLEFOr.Value();
BRep_Builder().Add (aCEOr, aEOr);
gp_Vec aVCur = GetAverageTangent (aEOr, aNbP);
if (!aVRef.IsParallel (aVCur, Precision::Angular()))
bAllTgt = Standard_False;
}
if (!bAllTgt)
aEOrF = aCEOr;
}
}
else
{
FindShape (aSOr, aFOr, myAnalyzer, aEOrF);
//
TopTools_ListOfShape* pLEIm = theDMEOrLEIm.ChangeSeek (aSOr);
if (!pLEIm)
{
pLEIm = theDMEOrLEIm.Bound (aSOr, TopTools_ListOfShape());
}
AppendToList (*pLEIm, aEIm);
}
//
if (aEOrF.IsNull())
{
// the edge has not been found
continue;
}
if (bVertex)
{
TopTools_MapOfShape aMVTotal;
Standard_Integer aNbChecked = 0;
// Just check if the original edges sharing the vertex do not share it any more.
for (TopoDS_Iterator it (aEOrF); it.More(); it.Next())
{
const TopoDS_Shape& aEOr = it.Value();
const TopTools_ListOfShape* aLIm = anImages.Seek (aEOr);
if (!aLIm)
continue;
++aNbChecked;
TopTools_IndexedDataMapOfShapeListOfShape aMVLoc;
for (TopTools_ListOfShape::Iterator itLIM (*aLIm); itLIM.More(); itLIM.Next())
TopExp::MapShapesAndAncestors (itLIM.Value(), TopAbs_VERTEX, TopAbs_EDGE, aMVLoc);
for (Standard_Integer i = 1; i <= aMVLoc.Extent(); ++i)
{
if (aMVLoc (i).Extent() > 1 && !aMVTotal.Add (aMVLoc.FindKey (i)))
{
bInvalid = Standard_True;
theEdgesInvalidByVertex.Add (aEIm);
break;
}
}
if (bInvalid)
break;
}
if (!bInvalid && aNbChecked < 2)
continue;
else
theEdgesValidByVertex.Add (aEIm);
}
else
{
//
// Check orientations of the image edge and original edge.
// In case the 3d curves are having the same direction the orientations
// must be the same. Otherwise the orientations should also be different.
//
// get average tangent vector for each curve taking into account
// the orientations of the edges, i.e. the edge is reversed
// the vector is reversed as well
gp_Vec aVSum1 = GetAverageTangent (aEIm, aNbP);
gp_Vec aVSum2 = GetAverageTangent (aEOrF, aNbP);
//
aVSum1.Normalize();
aVSum2.Normalize();
//
Standard_Real aCos = aVSum1.Dot (aVSum2);
if (Abs (aCos) < 0.9999)
{
continue;
}
//
aME.Add (aEOrF);
TopExp_Explorer aExpE (aEOrF, TopAbs_VERTEX);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aV = aExpE.Current();
aMV.Add (aV);
}
if (myAnalyzer)
{
for (TopTools_ListOfShape::Iterator itFA (myAnalyzer->Ancestors (aEOrF));
itFA.More(); itFA.Next())
aMF.Add (itFA.Value());
}
//
if (aCos < Precision::Confusion())
{
bInvalid = Standard_True;
aNbInv++;
}
}
bChecked = Standard_True;
}
//
if (!bChecked)
{
continue;
}
//
Standard_Boolean bLocalOnly = (aNbVOr > 1 && (pLEOr->Extent() - aNbVOr) > 1);
Standard_Integer aNbE = aME.Extent(), aNbV = aMV.Extent();
if (aNbE > 1 && aNbV == 2 * aNbE)
{
Standard_Boolean bSkip = Standard_True;
// It seems the edge originated from not connected edges and cannot be
// considered as correctly classified as it may fill some undesired parts.
// Still, allow the edge to be accounted for local analysis if it is:
// * originated from more than two faces
// * unanimously considered valid or invalid
// * not a boundary edge in the splits
if (aMF.Extent() > 2 && (aNbInv == 0 || aNbInv == aNbE))
{
if (theLFImages.Extent() > 2)
{
TopoDS_Iterator itV (aEIm);
for (; itV.More(); itV.Next())
{
TopTools_ListOfShape::Iterator itE (aDMVE.FindFromKey (itV.Value()));
for (; itE.More(); itE.Next())
if (aDMEF.FindFromKey (itE.Value()).Extent() < 2)
break;
if (itE.More())
break;
}
bSkip = itV.More();
}
}
if (bSkip)
continue;
else
bLocalOnly = Standard_True;
}
//
if (bInvalid)
{
if (!bLocalOnly)
myInvalidEdges.Add (aEIm);
aMIE.Add (aEIm);
aMEInv.Add (aEIm);
continue;
}
//
// check if the edge has been inverted
Standard_Boolean bInverted = !aNbE || bLocalOnly ? Standard_False :
CheckInverted (aEIm, aFOr, aDMVE, aMEdges);
//
if (!bInverted || !aNbVOr)
{
if (!bLocalOnly)
myValidEdges.Add (aEIm);
aMVE.Add (aEIm);
aMEVal.Add (aEIm);
}
}
//
// valid edges
if (aMVE.Extent())
{
theDMFMVE.Bind (aFIm, aMVE);
}
//
// invalid edges
if (aMIE.Extent())
{
theDMFMIE.Bind (aFIm, aMIE);
}
}
//
// process invalid edges:
// check for the inverted edges
TopTools_MapOfShape aMVIE;
// fill neutral edges
TopTools_MapOfShape aMNE;
//
Standard_Integer i, aNbEInv = aMEInv.Extent();
for (i = 1; i <= aNbEInv; ++i)
{
const TopoDS_Shape& aEIm = aMEInv (i);
//
// neutral edges - on the splits of the same offset face
// it is valid for one split and invalid for other
if (aMEVal.Contains (aEIm))
{
aMNE.Add (aEIm);
continue;
}
//
// the inverted images of the origins of invalid edges should also be invalid
if (!myInvertedEdges.Contains (aEIm))
{
continue;
}
//
const TopTools_ListOfShape* pLOEOr = myOEOrigins.Seek (aEIm);
if (!pLOEOr)
{
continue;
}
//
TopTools_ListIteratorOfListOfShape aItLOEOr (*pLOEOr);
for (; aItLOEOr.More(); aItLOEOr.Next())
{
const TopoDS_Shape& aOEOr = aItLOEOr.Value();
const TopTools_ListOfShape& aLEIm1 = myOEImages.Find (aOEOr);
//
TopTools_ListIteratorOfListOfShape aItLEIm1 (aLEIm1);
for (; aItLEIm1.More(); aItLEIm1.Next())
{
const TopoDS_Shape& aEIm1 = aItLEIm1.Value();
if (aMEdges.Contains (aEIm1) &&
!aMEInv.Contains (aEIm1) && !aMEInt.Contains (aEIm1) &&
myInvertedEdges.Contains (aEIm1))
{
myInvalidEdges.Add (aEIm1);
aMVIE.Add (aEIm1);
}
}
}
}
//
if (aMNE.Extent())
{
theDMFMNE.Bind (theF, aMNE);
}
//
if (aMVIE.Extent())
{
theDMFMVIE.Bind (theF, aMVIE);
}
}
namespace {
//=======================================================================
//function : addAsNeutral
//purpose : Adds as the edge into corresponding maps making it neutral
//=======================================================================
static void addAsNeutral (const TopoDS_Shape& theE,
const TopoDS_Shape& theFInv,
const TopoDS_Shape& theFVal,
BRepOffset_DataMapOfShapeIndexedMapOfShape& theLocInvEdges,
BRepOffset_DataMapOfShapeMapOfShape& theLocValidEdges)
{
TopTools_IndexedMapOfShape* pMEInv = theLocInvEdges.ChangeSeek (theFInv);
if (!pMEInv)
pMEInv = theLocInvEdges.Bound (theFInv, TopTools_IndexedMapOfShape());
pMEInv->Add (theE);
TopTools_MapOfShape* pMEVal = theLocValidEdges.ChangeSeek (theFVal);
if (!pMEVal)
pMEVal = theLocValidEdges.Bound (theFVal, TopTools_MapOfShape());
pMEVal->Add (theE);
}
}
//=======================================================================
//function : FindInvalidEdges
//purpose : Additional method to look for invalid edges
//=======================================================================
void BRepOffset_BuildOffsetFaces::FindInvalidEdges (const TopTools_ListOfShape& theLFOffset,
BRepOffset_DataMapOfShapeIndexedMapOfShape& theLocInvEdges,
BRepOffset_DataMapOfShapeMapOfShape& theLocValidEdges,
BRepOffset_DataMapOfShapeMapOfShape& theNeutralEdges)
{
// 1. Find edges unclassified in faces
// 2. Find SD faces in which the same edge is classified
// 3. Check if the edge is neutral in face in which it wasn't classified
NCollection_IndexedDataMap<TopoDS_Shape, TopTools_MapOfShape, TopTools_ShapeMapHasher> aMEUnclassified;
TopTools_DataMapOfShapeShape aFSplitFOffset;
// Avoid artificial faces
TopTools_MapOfShape aNewFaces;
if (myAnalyzer)
{
TopTools_MapOfShape aMapNewTmp;
for (TopTools_ListOfShape::Iterator it (myAnalyzer->NewFaces()); it.More(); it.Next())
aMapNewTmp.Add (it.Value());
for (TopTools_ListOfShape::Iterator it (theLFOffset); it.More(); it.Next())
{
const TopoDS_Shape& aFOffset = it.Value();
const TopoDS_Shape& aFOrigin = myFacesOrigins->Find (aFOffset);
if (aMapNewTmp.Contains (aFOrigin))
aNewFaces.Add (aFOffset);
}
}
TopTools_IndexedDataMapOfShapeListOfShape anEFMap;
for (TopTools_ListOfShape::Iterator itLFO (theLFOffset); itLFO.More(); itLFO.Next())
{
const TopoDS_Shape& aF = itLFO.Value();
if (aNewFaces.Contains (aF))
continue;
const TopTools_ListOfShape& aLFImages = myOFImages.FindFromKey (aF);
for (TopTools_ListOfShape::Iterator itLF (aLFImages); itLF.More(); itLF.Next())
{
const TopoDS_Shape& aFIm = itLF.Value();
TopExp::MapShapesAndAncestors (aFIm, TopAbs_EDGE, TopAbs_FACE, anEFMap);
const TopTools_IndexedMapOfShape* pMEInvalid = theLocInvEdges.Seek (aFIm);
const TopTools_MapOfShape* pMEValid = theLocValidEdges.Seek (aFIm);
for (TopExp_Explorer expE (aFIm, TopAbs_EDGE); expE.More(); expE.Next())
{
const TopoDS_Shape& aE = expE.Current();
if (myInvalidEdges.Contains (aE) != myValidEdges.Contains (aE))
{
// edge is classified in some face
if ((!pMEInvalid || !pMEInvalid->Contains (aE)) &&
(!pMEValid || !pMEValid->Contains (aE)))
{
// but not in the current one
TopTools_MapOfShape* pMap = aMEUnclassified.ChangeSeek (aE);
if (!pMap)
pMap = &aMEUnclassified (aMEUnclassified.Add (aE, TopTools_MapOfShape()));
pMap->Add (aFIm);
aFSplitFOffset.Bind (aFIm, aF);
}
}
}
}
}
if (aMEUnclassified.IsEmpty())
return;
// Analyze unclassified edges
const Standard_Integer aNbE = aMEUnclassified.Extent();
for (Standard_Integer iE = 1; iE <= aNbE; ++iE)
{
const TopoDS_Shape& aE = aMEUnclassified.FindKey (iE);
const TopTools_MapOfShape& aMFUnclassified = aMEUnclassified (iE);
const TopTools_ListOfShape& aLF = anEFMap.FindFromKey (aE);
for (TopTools_ListOfShape::Iterator itLF (aLF); itLF.More(); itLF.Next())
{
const TopoDS_Shape& aFClassified = itLF.Value();
if (aMFUnclassified.Contains (aFClassified))
continue;
BOPTools_Set anEdgeSetClass;
anEdgeSetClass.Add (aFClassified, TopAbs_EDGE);
TopoDS_Shape aEClassified;
FindShape (aE, aFClassified, NULL, aEClassified);
TopAbs_Orientation anOriClass = aEClassified.Orientation();
gp_Dir aDNClass;
BOPTools_AlgoTools3D::GetNormalToFaceOnEdge (TopoDS::Edge (aEClassified), TopoDS::Face (aFClassified), aDNClass);
const TopTools_IndexedMapOfShape* pMEInvalid = theLocInvEdges.Seek (aFClassified);
Standard_Boolean isInvalid = pMEInvalid && pMEInvalid->Contains (aE);
for (TopTools_MapOfShape::Iterator itM (aMFUnclassified); itM.More(); itM.Next())
{
const TopoDS_Shape& aFUnclassified = itM.Value();
BOPTools_Set anEdgeSetUnclass;
anEdgeSetUnclass.Add (aFUnclassified, TopAbs_EDGE);
if (anEdgeSetClass.IsEqual (anEdgeSetUnclass))
{
gp_Dir aDNUnclass;
BOPTools_AlgoTools3D::GetNormalToFaceOnEdge (TopoDS::Edge (aE), TopoDS::Face (aFUnclassified), aDNUnclass);
Standard_Boolean isSameOri = aDNClass.IsEqual (aDNUnclass, Precision::Angular());
// Among other splits of the same face find those where the edge is contained with different
// orientation
const TopoDS_Shape& aFOffset = aFSplitFOffset.Find (aFUnclassified);
const TopTools_ListOfShape& aLFSplits = myOFImages.FindFromKey (aFOffset);
TopTools_ListOfShape::Iterator itLFSp (aLFSplits);
for (; itLFSp.More(); itLFSp.Next())
{
const TopoDS_Shape& aFSp = itLFSp.Value();
if (!aFSp.IsSame (aFUnclassified) && aMFUnclassified.Contains (aFSp))
{
TopoDS_Shape aEUnclassified;
FindShape (aE, aFSp, NULL, aEUnclassified);
TopAbs_Orientation anOriUnclass = aEUnclassified.Orientation();
if (!isSameOri)
anOriUnclass = TopAbs::Reverse (anOriUnclass);
if (anOriClass != anOriUnclass)
{
// make the edge neutral for the face
TopTools_MapOfShape* pMENeutral = theNeutralEdges.ChangeSeek (aFOffset);
if (!pMENeutral)
pMENeutral = theNeutralEdges.Bound (aFOffset, TopTools_MapOfShape());
pMENeutral->Add (aE);
if (isInvalid && isSameOri)
{
// make edge invalid in aFUnclassified and valid in aFSp
addAsNeutral (aE, aFClassified, aFSp, theLocInvEdges, theLocValidEdges);
}
else
{
// make edge invalid in aFSp and valid in aFUnclassified
addAsNeutral (aE, aFSp, aFClassified, theLocInvEdges, theLocValidEdges);
}
}
}
}
if (itLFSp.More())
break;
}
}
}
}
}
//=======================================================================
//function : MakeInvertedEdgesInvalid
//purpose : Makes inverted edges located inside loop of invalid edges, invalid as well
//=======================================================================
void BRepOffset_BuildOffsetFaces::MakeInvertedEdgesInvalid (const TopTools_ListOfShape& theLFOffset)
{
if (myInvalidEdges.IsEmpty() || myInvertedEdges.IsEmpty())
return;
// Map all invalid edges
TopoDS_Compound aCBEInv;
BRep_Builder().MakeCompound (aCBEInv);
for (Standard_Integer i = 1; i <= myInvalidEdges.Extent(); ++i)
{
BRep_Builder().Add (aCBEInv, myInvalidEdges (i));
}
// Make loops of invalid edges
TopTools_ListOfShape aLCB;
BOPTools_AlgoTools::MakeConnexityBlocks (aCBEInv, TopAbs_VERTEX, TopAbs_EDGE, aLCB);
// Analyze each loop on closeness and use only closed ones
TopTools_DataMapOfShapeShape aDMVCB;
for (TopTools_ListOfShape::Iterator itLCB (aLCB); itLCB.More(); itLCB.Next())
{
const TopoDS_Shape& aCB = itLCB.Value();
TopTools_IndexedDataMapOfShapeListOfShape aDMVE;
TopExp::MapShapesAndAncestors (aCB, TopAbs_VERTEX, TopAbs_EDGE, aDMVE);
Standard_Boolean isClosed = Standard_True;
for (Standard_Integer iV = 1; iV <= aDMVE.Extent(); ++iV)
{
if (aDMVE (iV).Extent() != 2)
{
isClosed = Standard_False;
break;
}
}
if (!isClosed)
continue;
// Bind loop to each vertex of the loop
for (Standard_Integer iV = 1; iV <= aDMVE.Extent(); ++iV)
{
aDMVCB.Bind (aDMVE.FindKey (iV), aCB);
}
}
// Check if any inverted edges of offset faces are locked inside the loops of invalid edges.
// Make such edges invalid as well.
for (TopTools_ListOfShape::Iterator itLF (theLFOffset); itLF.More(); itLF.Next())
{
const TopTools_ListOfShape& aLFIm = myOFImages.FindFromKey (itLF.Value());
for (TopTools_ListOfShape::Iterator itLFIm (aLFIm); itLFIm.More(); itLFIm.Next())
{
for (TopExp_Explorer expE (itLFIm.Value(), TopAbs_EDGE); expE.More(); expE.Next())
{
const TopoDS_Edge& aE = TopoDS::Edge (expE.Current());
if (!myInvalidEdges.Contains (aE) && myInvertedEdges.Contains (aE))
{
const TopoDS_Shape* pCB1 = aDMVCB.Seek (TopExp::FirstVertex (aE));
const TopoDS_Shape* pCB2 = aDMVCB.Seek (TopExp::LastVertex (aE));
if (pCB1 && pCB2 && pCB1->IsSame (*pCB2))
{
myInvalidEdges.Add (aE);
}
}
}
}
}
}
//=======================================================================
//function : FindInvalidFaces
//purpose : Looking for the invalid faces by analyzing their invalid edges
//=======================================================================
void BRepOffset_BuildOffsetFaces::FindInvalidFaces (TopTools_ListOfShape& theLFImages,
const BRepOffset_DataMapOfShapeMapOfShape& theDMFMVE,
const BRepOffset_DataMapOfShapeIndexedMapOfShape& theDMFMIE,
const TopTools_MapOfShape& theMENeutral,
const TopTools_MapOfShape& theEdgesInvalidByVertex,
const TopTools_MapOfShape& theEdgesValidByVertex,
const TopTools_MapOfShape& theMFHoles,
TopTools_IndexedMapOfShape& theMFInvInHole,
TopTools_ListOfShape& theInvFaces,
TopTools_ListOfShape& theInvertedFaces)
{
// The face should be considered as invalid in the following cases:
// 1. It has been reverted, i.e. at least two not connected edges
// have changed orientation (i.e. invalid). In this case all edges,
// should be invalid for that face, because edges have also been reverted;
// 2. All checked edges of the face are invalid for this face;
// The face should be removed from the splits in the following cases:
// 1. All checked edges of the face are invalid for this one, but valid for
// some other face in this list of splits.
// The face will be kept in the following cases:
// 1. Some of the edges are valid for this face.
Standard_Boolean bHasValid, bAllValid, bAllInvalid, bHasReallyInvalid, bAllInvNeutral;
Standard_Boolean bValid, bValidLoc, bInvalid, bInvalidLoc, bNeutral, bInverted;
Standard_Boolean bIsInvalidByInverted, bHasInverted;
Standard_Integer aNbChecked;
//
Standard_Boolean bTreatInvertedAsInvalid = (theLFImages.Extent() == 1);
//
// neutral edges to remove
TopTools_IndexedMapOfShape aMENRem;
//
// faces for post treat
TopTools_ListOfShape aLFPT;
//
TopTools_IndexedDataMapOfShapeListOfShape aDMEF;
TopTools_ListIteratorOfListOfShape aItLF (theLFImages);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Face& aFIm = *(TopoDS_Face*)&aItLF.Value();
TopExp::MapShapesAndAncestors (aFIm, TopAbs_EDGE, TopAbs_FACE, aDMEF);
}
aItLF.Initialize (theLFImages);
for (; aItLF.More(); )
{
const TopoDS_Face& aFIm = *(TopoDS_Face*)&aItLF.Value();
//
// valid edges for this split
const TopTools_MapOfShape* pMVE = theDMFMVE.Seek (aFIm);
// invalid edges for this split
const TopTools_IndexedMapOfShape* pMIE = theDMFMIE.Seek (aFIm);
//
bHasValid = Standard_False;
bAllValid = Standard_True;
bAllInvalid = Standard_True;
bHasReallyInvalid = Standard_False;
bAllInvNeutral = Standard_True;
bIsInvalidByInverted = Standard_True;
bHasInverted = Standard_False;
aNbChecked = 0;
//
const TopoDS_Wire& aWIm = BRepTools::OuterWire (aFIm);
TopExp_Explorer aExp (aWIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aEIm = aExp.Current();
//
bValid = myValidEdges.Contains (aEIm);
bInvalid = myInvalidEdges.Contains (aEIm);
bNeutral = theMENeutral.Contains (aEIm);
//
if (!bValid && !bInvalid && !bNeutral)
{
// edge has not been checked for some reason
continue;
}
// skip not-boundary edges originated from vertex
if ((theEdgesInvalidByVertex.Contains (aEIm) ||
theEdgesValidByVertex.Contains (aEIm)) &&
aDMEF.FindFromKey (aEIm).Extent() != 1)
continue;
++aNbChecked;
//
bInvalidLoc = pMIE && pMIE->Contains (aEIm);
bHasReallyInvalid = bInvalid && bInvalidLoc && !bValid && !theEdgesInvalidByVertex.Contains (aEIm);
if (bHasReallyInvalid)
{
break;
}
//
bValidLoc = pMVE && pMVE->Contains (aEIm);
bInverted = myInvertedEdges.Contains (aEIm);
if (!bInvalid && !bInvalidLoc && bTreatInvertedAsInvalid)
{
bInvalid = bInverted;
}
//
if (bValidLoc && bNeutral)
{
bHasValid = Standard_True;
}
//
bAllValid &= bValidLoc;
bAllInvalid &= (bInvalid || bInvalidLoc);
bAllInvNeutral &= (bAllInvalid && bNeutral);
bIsInvalidByInverted &= (bInvalidLoc || bInverted);
bHasInverted |= bInverted;
}
//
if (!aNbChecked)
{
aItLF.Next();
continue;
}
//
if (!bHasReallyInvalid && (bAllInvNeutral && !bHasValid) && (aNbChecked > 1))
{
if (bHasInverted)
{
// The part seems to be filled due to overlapping of parts rather than
// due to multi-connection of faces. No need to remove the part.
aItLF.Next();
continue;
}
// remove edges from neutral
TopExp::MapShapes (aFIm, TopAbs_EDGE, aMENRem);
// remove face
theLFImages.Remove (aItLF);
continue;
}
//
if (bHasReallyInvalid || (bAllInvalid &&
!(bHasValid || bAllValid) &&
!(bAllInvNeutral && (aNbChecked == 1))))
{
theInvFaces.Append (aFIm);
if (theMFHoles.Contains (aFIm))
{
theMFInvInHole.Add (aFIm);
}
aItLF.Next();
continue;
}
//
if (theMFHoles.Contains (aFIm))
{
// remove edges from neutral
TopExp::MapShapes (aFIm, TopAbs_EDGE, aMENRem);
// remove face
theLFImages.Remove (aItLF);
continue;
}
//
if (bIsInvalidByInverted && !(bHasValid || bAllValid))
{
// The face contains only the inverted and locally invalid edges
theInvertedFaces.Append (aFIm);
}
if (!bAllInvNeutral)
{
aLFPT.Append (aFIm);
}
else
{
// remove edges from neutral
TopExp::MapShapes (aFIm, TopAbs_EDGE, aMENRem);
}
aItLF.Next();
}
//
if (aLFPT.IsEmpty() || aMENRem.IsEmpty())
{
return;
}
// check the splits once more
aItLF.Initialize (aLFPT);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Face& aFIm = *(TopoDS_Face*)&aItLF.Value();
//
// valid edges for this split
const TopTools_MapOfShape* pMVE = theDMFMVE.Seek (aFIm);
//
bHasValid = Standard_False;
bAllValid = Standard_True;
bAllInvalid = Standard_True;
//
const TopoDS_Wire& aWIm = BRepTools::OuterWire (aFIm);
TopExp_Explorer aExp (aWIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aEIm = aExp.Current();
//
bValid = myValidEdges.Contains (aEIm);
bInvalid = myInvalidEdges.Contains (aEIm);
bNeutral = theMENeutral.Contains (aEIm) && !aMENRem.Contains (aEIm);
bValidLoc = pMVE && pMVE->Contains (aEIm);
//
if (!bInvalid && bTreatInvertedAsInvalid)
{
bInvalid = myInvertedEdges.Contains (aEIm);
}
//
if (bValidLoc && bNeutral)
{
bHasValid = Standard_True;
}
//
bAllValid = bAllValid && bValidLoc;
bAllInvalid = bAllInvalid && bInvalid;
}
//
if (bAllInvalid && !bHasValid && !bAllValid)
{
theInvFaces.Append (aFIm);
}
}
}
//=======================================================================
//function : FindFacesInsideHoleWires
//purpose : Find faces inside holes wires from the original face
//=======================================================================
void BRepOffset_BuildOffsetFaces::FindFacesInsideHoleWires (const TopoDS_Face& theFOrigin,
const TopoDS_Face& theFOffset,
const TopTools_ListOfShape& theLFImages,
const TopTools_DataMapOfShapeListOfShape& theDMEOrLEIm,
const TopTools_IndexedDataMapOfShapeListOfShape& theEFMap,
TopTools_MapOfShape& theMFHoles)
{
if (theLFImages.IsEmpty())
{
return;
}
//
// find all hole wires in the original face
TopTools_ListOfShape aLHoleWires;
const TopoDS_Wire& anOuterWire = BRepTools::OuterWire (theFOrigin);
TopExp_Explorer aExpW (theFOrigin, TopAbs_WIRE);
for (; aExpW.More(); aExpW.Next())
{
const TopoDS_Wire& aHoleWire = TopoDS::Wire (aExpW.Current());
if (!aHoleWire.IsSame (anOuterWire) && aHoleWire.Orientation() != TopAbs_INTERNAL)
{
aLHoleWires.Append (aHoleWire);
}
}
//
if (aLHoleWires.IsEmpty())
{
// no holes in the face
return;
}
//
TopTools_ListOfShape* pLFNewHoles = myFNewHoles.ChangeSeek (theFOrigin);
//
if (!pLFNewHoles)
{
pLFNewHoles = myFNewHoles.Bound (theFOrigin, TopTools_ListOfShape());
}
if (pLFNewHoles->IsEmpty())
{
//
// find the faces representing holes in the images of the faces:
// 1. for each original hole wire try to build its image
// 2. build the new planar face from the images
//
// map vertices and edges of the splits
TopTools_IndexedMapOfShape aMESplits;
TopTools_ListIteratorOfListOfShape aItLF (theLFImages);
for (; aItLF.More(); aItLF.Next())
{
TopExp::MapShapes (aItLF.Value(), TopAbs_EDGE, aMESplits);
}
//
TopTools_ListIteratorOfListOfShape aItLW (aLHoleWires);
for (; aItLW.More(); aItLW.Next())
{
const TopoDS_Wire& aHoleWire = TopoDS::Wire (aItLW.Value());
// find images of all edges of the original wire
TopTools_IndexedMapOfShape aMEImWire;
TopoDS_Iterator aItE (aHoleWire);
for (; aItE.More(); aItE.Next())
{
const TopoDS_Shape& aEOr = aItE.Value();
const TopTools_ListOfShape* pLEIm = theDMEOrLEIm.Seek (aEOr);
if (!pLEIm || pLEIm->IsEmpty())
{
continue;
}
TopTools_ListIteratorOfListOfShape aItLEIm (*pLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
if (aMESplits.Contains (aEIm))
{
aMEImWire.Add (aEIm);
}
}
}
//
if (aMEImWire.IsEmpty())
{
continue;
}
//
// build new planar face using these edges
TopTools_ListOfShape aLE;
Standard_Integer i, aNbE = aMEImWire.Extent();
for (i = 1; i <= aNbE; ++i)
{
aLE.Append (aMEImWire (i).Oriented (TopAbs_FORWARD));
aLE.Append (aMEImWire (i).Oriented (TopAbs_REVERSED));
}
//
BOPAlgo_BuilderFace aBF;
aBF.SetFace (TopoDS::Face (theFOffset.Oriented (TopAbs_FORWARD)));
aBF.SetShapes (aLE);
aBF.Perform();
//
const TopTools_ListOfShape& aLFNew = aBF.Areas();
if (aLFNew.IsEmpty())
{
continue;
}
//
// check if outer edges in the new faces are not inverted
// because the inverted edges mean that the hole has been
// filled during offset and there will be no faces to remove
TopTools_IndexedDataMapOfShapeListOfShape aDMEFNew;
TopTools_ListIteratorOfListOfShape aItLFNew (aLFNew);
for (; aItLFNew.More(); aItLFNew.Next())
{
TopExp::MapShapesAndAncestors (aItLFNew.Value(), TopAbs_EDGE, TopAbs_FACE, aDMEFNew);
}
//
aNbE = aDMEFNew.Extent();
for (i = 1; i <= aNbE; ++i)
{
if (aDMEFNew (i).Extent() == 1)
{
const TopoDS_Shape& aE = aDMEFNew.FindKey (i);
if (myInvertedEdges.Contains (aE))
{
break;
}
}
}
//
if (i <= aNbE)
{
continue;
}
//
aItLFNew.Initialize (aLFNew);
for (; aItLFNew.More(); aItLFNew.Next())
{
pLFNewHoles->Append (aItLFNew.Value());
}
}
}
// Build Edge-Face map for splits of current offset face
TopTools_IndexedDataMapOfShapeListOfShape anEFSplitsMap;
// Build Edge-Face map for holes
TopTools_IndexedDataMapOfShapeListOfShape anEFHolesMap;
// among the splits of the offset face find those that are
// located inside the hole faces
TopTools_ListIteratorOfListOfShape aItLF (theLFImages);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Face& aFIm = TopoDS::Face (aItLF.Value());
TopExp::MapShapesAndAncestors (aFIm, TopAbs_EDGE, TopAbs_FACE, anEFSplitsMap);
// get the point inside the face and classify it relatively hole faces
gp_Pnt aP3D;
gp_Pnt2d aP2D;
Standard_Integer iErr = BOPTools_AlgoTools3D::PointInFace (aFIm, aP3D, aP2D, myContext);
if (iErr)
{
continue;
}
//
Standard_Real aTol = BRep_Tool::Tolerance (aFIm);
//
TopTools_ListIteratorOfListOfShape aItLFNew (*pLFNewHoles);
for (; aItLFNew.More(); aItLFNew.Next())
{
const TopoDS_Face& aFNew = TopoDS::Face (aItLFNew.Value());
if (myContext->IsValidPointForFace (aP3D, aFNew, aTol))
{
// the face is classified as IN
theMFHoles.Add (aFIm);
TopExp::MapShapesAndAncestors (aFIm, TopAbs_EDGE, TopAbs_FACE, anEFHolesMap);
break;
}
}
}
// Out of all found holes find those which cannot be removed
// by checking their connectivity to splits of other offset faces.
// These are the faces, which will create uncovered holes if removed.
const Standard_Integer aNbE = anEFHolesMap.Extent();
for (Standard_Integer i = 1; i <= aNbE; ++i)
{
const TopoDS_Shape& anEdge = anEFHolesMap.FindKey (i);
const TopTools_ListOfShape& aLFHoles = anEFHolesMap (i);
// Check if the edge is outer for holes
if (aLFHoles.Extent() != 1)
continue;
const TopoDS_Shape& aFHole = aLFHoles.First();
if (!theMFHoles.Contains (aFHole))
// Already removed
continue;
// Check if the edge is not outer for splits
const TopTools_ListOfShape& aLSplits = anEFSplitsMap.FindFromKey (anEdge);
if (aLSplits.Extent() == 1)
continue;
// Check if edge is only connected to splits of the current offset face
const TopTools_ListOfShape& aLFAll = theEFMap.FindFromKey (anEdge);
if (aLFAll.Extent() == 2)
// Avoid removal of the hole from the splits
theMFHoles.Remove (aFHole);
}
}
//=======================================================================
//function : CheckInverted
//purpose : Checks if the edge has been inverted
//=======================================================================
Standard_Boolean BRepOffset_BuildOffsetFaces::CheckInverted (const TopoDS_Edge& theEIm,
const TopoDS_Face& theFOr,
const TopTools_IndexedDataMapOfShapeListOfShape& theDMVE,
const TopTools_IndexedMapOfShape& theMEdges)
{
// It is necessary to compare the direction from first vertex
// to the last vertex on the original edge with the
// same direction on the new edge. If the directions
// will be different - the edge has been inverted.
//
TopoDS_Vertex aVI1, aVI2; // vertices on the offset edge
TopoDS_Vertex aVO1, aVO2; // vertices on the original edge
//
Standard_Integer i;
// find vertices of the offset shape
TopExp::Vertices (theEIm, aVI1, aVI2);
//
// find images
TopTools_ListOfShape aLEImages;
if (myOEOrigins.IsBound (theEIm))
{
TopoDS_Wire anImages;
BRep_Builder().MakeWire (anImages);
//
TopTools_MapOfShape aMImFence;
const TopTools_ListOfShape& aLOffsetOr = myOEOrigins.Find (theEIm);
TopTools_ListIteratorOfListOfShape aItOffset (aLOffsetOr);
for (; aItOffset.More(); aItOffset.Next())
{
const TopoDS_Shape& aEOffsetOr = aItOffset.Value();
const TopTools_ListOfShape& aLImages = myOEImages.Find (aEOffsetOr);
//
TopTools_ListIteratorOfListOfShape aItImages (aLImages);
for (; aItImages.More(); aItImages.Next())
{
const TopoDS_Edge& anIm = *(TopoDS_Edge*)&aItImages.Value();
if (theMEdges.Contains (anIm) && aMImFence.Add (anIm))
{
BRep_Builder().Add (anImages, anIm);
aLEImages.Append (anIm);
}
}
}
//
// find alone vertices
TopoDS_Vertex aVW1, aVW2;
TopTools_IndexedDataMapOfShapeListOfShape aDMImVE;
TopExp::MapShapesAndAncestors (anImages, TopAbs_VERTEX, TopAbs_EDGE, aDMImVE);
//
TopTools_ListOfShape aLVAlone;
Standard_Integer aNb = aDMImVE.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopTools_ListOfShape& aLImE = aDMImVE (i);
if (aLImE.Extent() == 1)
{
aLVAlone.Append (aDMImVE.FindKey (i));
}
}
//
if (aLVAlone.Extent() > 1)
{
aVW1 = *(TopoDS_Vertex*)&aLVAlone.First();
aVW2 = *(TopoDS_Vertex*)&aLVAlone.Last();
//
// check distances
const gp_Pnt& aPI1 = BRep_Tool::Pnt (aVI1);
const gp_Pnt& aPW1 = BRep_Tool::Pnt (aVW1);
const gp_Pnt& aPW2 = BRep_Tool::Pnt (aVW2);
//
Standard_Real aDist1 = aPI1.SquareDistance (aPW1);
Standard_Real aDist2 = aPI1.SquareDistance (aPW2);
//
if (aDist1 < aDist2)
{
aVI1 = aVW1;
aVI2 = aVW2;
}
else
{
aVI1 = aVW2;
aVI2 = aVW1;
}
}
}
else
{
aLEImages.Append (theEIm);
}
//
// Find edges connected to these vertices
const TopTools_ListOfShape& aLIE1 = theDMVE.FindFromKey (aVI1);
const TopTools_ListOfShape& aLIE2 = theDMVE.FindFromKey (aVI2);
//
// Find vertices on the original face corresponding to vertices on the offset edge
//
// find original edges for both lists
TopTools_ListOfShape aLOE1, aLOE2;
for (i = 0; i < 2; ++i)
{
const TopTools_ListOfShape& aLIE = !i ? aLIE1 : aLIE2;
TopTools_ListOfShape& aLOE = !i ? aLOE1 : aLOE2;
//
TopTools_MapOfShape aMFence;
//
TopTools_ListIteratorOfListOfShape aItLIE (aLIE);
for (; aItLIE.More(); aItLIE.Next())
{
const TopoDS_Shape& aEI = aItLIE.Value();
if (myEdgesOrigins->IsBound (aEI))
{
const TopTools_ListOfShape& aLEOrigins = myEdgesOrigins->Find (aEI);
//
TopTools_ListIteratorOfListOfShape aItLOE (aLEOrigins);
for (; aItLOE.More(); aItLOE.Next())
{
const TopoDS_Shape& aEO = aItLOE.Value();
if (aEO.ShapeType() == TopAbs_EDGE && aMFence.Add (aEO))
{
TopoDS_Shape aEOin;
if (FindShape (aEO, theFOr, NULL, aEOin))
{
AppendToList (aLOE, aEO);
}
}
}
}
}
}
//
if (aLOE1.Extent() < 2 || aLOE2.Extent() < 2)
{
return Standard_False;
}
//
// find vertices common for the max number of edges in the lists
for (i = 0; i < 2; ++i)
{
const TopTools_ListOfShape& aLOE = !i ? aLOE1 : aLOE2;
TopoDS_Vertex& aVO = !i ? aVO1 : aVO2;
TopTools_IndexedDataMapOfShapeListOfShape aDMVELoc;
for (TopTools_ListOfShape::Iterator itLOE (aLOE); itLOE.More(); itLOE.Next())
{
TopExp::MapShapesAndAncestors (itLOE.Value(), TopAbs_VERTEX, TopAbs_EDGE, aDMVELoc);
}
Standard_Integer aNbEMax = 0;
for (Standard_Integer j = 1; j <= aDMVELoc.Extent(); ++j)
{
Standard_Integer aNbE = aDMVELoc (j).Extent();
if (aNbE > 1 && aNbE > aNbEMax)
{
aVO = TopoDS::Vertex (aDMVELoc.FindKey (j));
aNbEMax = aNbE;
}
}
if (aVO.IsNull())
{
return Standard_False;
}
}
if (aVO1.IsSame (aVO2))
{
return Standard_False;
}
//
// check positions of the offset and original vertices
const gp_Pnt& aPI1 = BRep_Tool::Pnt (aVI1);
const gp_Pnt& aPI2 = BRep_Tool::Pnt (aVI2);
const gp_Pnt& aPO1 = BRep_Tool::Pnt (aVO1);
const gp_Pnt& aPO2 = BRep_Tool::Pnt (aVO2);
//
gp_Vec aVI (aPI1, aPI2);
gp_Vec aVO (aPO1, aPO2);
//
Standard_Real anAngle = aVI.Angle (aVO);
Standard_Boolean bInverted = Abs (anAngle - M_PI) < 1.e-4;
if (bInverted)
{
TopTools_ListIteratorOfListOfShape aItLEIm (aLEImages);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEInvr = aItLEIm.Value();
myInvertedEdges.Add (aEInvr);
}
}
return bInverted;
}
namespace {
//=======================================================================
//function : GetVerticesOnEdges
//purpose : Get vertices from the given shape belonging to the given edges
//=======================================================================
static void GetVerticesOnEdges (const TopoDS_Shape& theCB,
const TopTools_IndexedMapOfShape& theEdges,
TopTools_MapOfShape& theVerticesOnEdges,
TopTools_MapOfShape& theAllVertices)
{
TopExp_Explorer aExp (theCB, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
Standard_Boolean isOnGivenEdges = theEdges.Contains (aE);
for (TopoDS_Iterator aItV (aE); aItV.More(); aItV.Next())
{
theAllVertices.Add (aItV.Value());
if (isOnGivenEdges)
{
theVerticesOnEdges.Add (aItV.Value());
}
}
}
}
}
//=======================================================================
//function : CheckInvertedBlock
//purpose : Checks if it is possible to remove the block containing inverted edges
//=======================================================================
Standard_Boolean BRepOffset_BuildOffsetFaces::CheckInvertedBlock (const TopoDS_Shape& theCB,
const TopTools_ListOfShape& theLCBF,
BRepOffset_DataMapOfShapeMapOfShape& theDMCBVInverted,
BRepOffset_DataMapOfShapeMapOfShape& theDMCBVAll)
{
// For possible removal of the block:
// 1. There should be more than just one face in the block
if (theCB.NbChildren() < 2)
{
return Standard_False;
}
//
// 2. The block should at least contain two connected inverted edges with
// different origins (not just two images/splits of the same edge)
TopTools_MapOfShape aMECBInv;
TopoDS_Compound aCECBInv;
BRep_Builder().MakeCompound (aCECBInv);
//
TopExp_Explorer aExp (theCB, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (myInvertedEdges.Contains (aE))
{
if (aMECBInv.Add (aE))
{
BRep_Builder().Add (aCECBInv, aE);
}
}
}
//
if (aMECBInv.Extent() < 2)
{
return Standard_False;
}
//
// check that the edges are connected and different
TopTools_ListOfShape aLCBE;
BOPTools_AlgoTools::MakeConnexityBlocks (aCECBInv, TopAbs_VERTEX, TopAbs_EDGE, aLCBE);
//
TopTools_ListIteratorOfListOfShape aItLCBE (aLCBE);
for (; aItLCBE.More(); aItLCBE.Next())
{
const TopoDS_Shape& aCBE = aItLCBE.Value();
// count the unique edges in the block
Standard_Integer aNbUnique = 0;
TopTools_MapOfShape aMEOrigins;
TopoDS_Iterator aItE (aCBE);
for (; aItE.More(); aItE.Next())
{
const TopoDS_Shape& aE = aItE.Value();
const TopTools_ListOfShape* pLEOr = myOEOrigins.Seek (aE);
if (!pLEOr)
{
aMEOrigins.Add (aE);
++aNbUnique;
continue;
}
TopTools_ListIteratorOfListOfShape aItLEOr (*pLEOr);
for (; aItLEOr.More(); aItLEOr.Next())
{
const TopoDS_Shape& aEOr = aItLEOr.Value();
if (aMEOrigins.Add (aEOr))
{
++aNbUnique;
}
}
}
//
if (aNbUnique >= 2)
{
break;
}
}
//
if (!aItLCBE.More())
{
return Standard_False;
}
//
// 3. the block should not contain inverted edges which vertices
// are contained in other blocks
//
// collect vertices from inverted edges and compare them with
// vertices from other blocks
TopTools_MapOfShape* pMVInverted = theDMCBVInverted.ChangeSeek (theCB);
TopTools_MapOfShape* pMVAll = theDMCBVAll.ChangeSeek (theCB);
if (!pMVInverted)
{
pMVInverted = theDMCBVInverted.Bound (theCB, TopTools_MapOfShape());
pMVAll = theDMCBVAll.Bound (theCB, TopTools_MapOfShape());
//
GetVerticesOnEdges (theCB, myInvertedEdges, *pMVInverted, *pMVAll);
}
//
TopTools_ListIteratorOfListOfShape aItLCB1 (theLCBF);
for (; aItLCB1.More(); aItLCB1.Next())
{
const TopoDS_Shape& aCB1 = aItLCB1.Value();
if (aCB1.IsSame (theCB))
{
continue;
}
//
// collect vertices from inverted edges
TopTools_MapOfShape* pMVInverted1 = theDMCBVInverted.ChangeSeek (aCB1);
TopTools_MapOfShape* pMVAll1 = theDMCBVAll.ChangeSeek (aCB1);
if (!pMVInverted1)
{
pMVInverted1 = theDMCBVInverted.Bound (aCB1, TopTools_MapOfShape());
pMVAll1 = theDMCBVAll.Bound (aCB1, TopTools_MapOfShape());
//
GetVerticesOnEdges (aCB1, myInvertedEdges, *pMVInverted1, *pMVAll1);
}
//
if (pMVInverted->HasIntersection (*pMVAll1))
{
return Standard_False;
}
}
//
return Standard_True;
}
//=======================================================================
//function : RemoveInvalidSplitsByInvertedEdges
//purpose : Looking for the invalid faces containing inverted edges
// that can be safely removed
//=======================================================================
void BRepOffset_BuildOffsetFaces::RemoveInvalidSplitsByInvertedEdges (TopTools_IndexedMapOfShape& theMERemoved)
{
if (myInvertedEdges.IsEmpty())
{
return;
}
//
// check the faces on regularity, i.e. the splits of the same face
// should not be connected only by vertex. Such irregular splits
// will have to be rebuilt and cannot be removed.
//
BRep_Builder aBB;
TopTools_IndexedMapOfShape aMEAvoid;
TopTools_DataMapOfShapeListOfShape aDMVF;
Standard_Integer aNb = myOFImages.Extent(), i;
for (i = 1; i <= aNb; ++i)
{
const TopTools_ListOfShape& aLFIm = myOFImages (i);
//
TopoDS_Compound aCFIm;
aBB.MakeCompound (aCFIm);
//
TopTools_DataMapOfShapeListOfShape aDMEF;
TopTools_ListIteratorOfListOfShape aIt (aLFIm);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aF = aIt.Value();
aBB.Add (aCFIm, aF);
//
// make a map to use only outer edges
TopExp_Explorer aExp (aF, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
//
TopTools_ListOfShape* pLF = aDMEF.ChangeSeek (aE);
if (!pLF)
{
pLF = aDMEF.Bound (aE, TopTools_ListOfShape());
}
else
{
// internal edges should not be used
aMEAvoid.Add (aE);
}
AppendToList (*pLF, aF);
}
//
// fill connection map of the vertices of inverted edges to faces
aExp.Init (aF, TopAbs_VERTEX);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aV = aExp.Current();
//
TopTools_ListOfShape* pLF = aDMVF.ChangeSeek (aV);
if (!pLF)
{
pLF = aDMVF.Bound (aV, TopTools_ListOfShape());
}
AppendToList (*pLF, aF);
}
}
//
// for the splits to be regular they should form only one block
TopTools_ListOfShape aLCBF;
BOPTools_AlgoTools::MakeConnexityBlocks (aCFIm, TopAbs_EDGE, TopAbs_FACE, aLCBF);
if (aLCBF.Extent() == 1)
{
continue;
}
//
// check if the inverted edges create the irregularity
BRepOffset_DataMapOfShapeMapOfShape aDMCBVInverted, aDMCBVAll;
//
TopTools_ListIteratorOfListOfShape aItLCB (aLCBF);
for (; aItLCB.More(); aItLCB.Next())
{
const TopoDS_Shape& aCB = aItLCB.Value();
//
// check if it is possible to remove the block
if (!CheckInvertedBlock (aCB, aLCBF, aDMCBVInverted, aDMCBVAll))
{
// non of the edges in this block should be removed
TopExp::MapShapes (aCB, TopAbs_EDGE, aMEAvoid);
continue;
}
}
}
//
// all edges not included in aMEAvoid can be removed
TopTools_MapOfShape aMERem;
for (Standard_Integer iInverted = 1; iInverted <= myInvertedEdges.Extent(); ++iInverted)
{
const TopoDS_Shape& aE = myInvertedEdges (iInverted);
if (!aMEAvoid.Contains (aE))
{
TopoDS_Iterator aIt (aE);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aV = aIt.Value();
const TopTools_ListOfShape* pLF = aDMVF.Seek (aV);
if (pLF && (pLF->Extent() > 3))
{
aMERem.Add (aE);
break;
}
}
}
}
//
if (aMERem.IsEmpty())
{
return;
}
//
// all invalid faces containing these edges can be removed
TopTools_IndexedDataMapOfShapeListOfShape aInvFaces;
TopTools_MapOfShape aMFRem;
TopTools_IndexedMapOfShape aMFToUpdate;
aNb = myInvalidFaces.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
TopTools_ListOfShape& aLFIm = myInvalidFaces (i);
//
TopTools_ListIteratorOfListOfShape aIt (aLFIm);
for (; aIt.More(); )
{
const TopoDS_Shape& aFIm = aIt.Value();
//
// to be removed the face should have at least two not connected
// inverted edges
TopoDS_Compound aCEInv;
aBB.MakeCompound (aCEInv);
TopExp_Explorer aExp (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (aMERem.Contains (aE))
{
aBB.Add (aCEInv, aE);
}
}
//
// check connectivity
TopTools_ListOfShape aLCBE;
BOPTools_AlgoTools::MakeConnexityBlocks (aCEInv, TopAbs_VERTEX, TopAbs_EDGE, aLCBE);
//
if (aLCBE.Extent() >= 2)
{
aMFToUpdate.Add (aF);
aMFRem.Add (aFIm);
aLFIm.Remove (aIt);
}
else
{
aIt.Next();
}
}
//
if (aLFIm.Extent())
{
aInvFaces.Add (aF, aLFIm);
}
}
//
if (aMFRem.IsEmpty())
{
return;
}
//
myInvalidFaces = aInvFaces;
// remove from splits
aNb = aMFToUpdate.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = aMFToUpdate (i);
TopTools_ListOfShape& aLFIm = myOFImages.ChangeFromKey (aF);
//
TopTools_ListIteratorOfListOfShape aIt (aLFIm);
for (; aIt.More(); )
{
const TopoDS_Shape& aFIm = aIt.Value();
if (aMFRem.Contains (aFIm))
{
TopExp::MapShapes (aFIm, TopAbs_EDGE, theMERemoved);
aLFIm.Remove (aIt);
}
else
{
aIt.Next();
}
}
}
}
//=======================================================================
//function : RemoveInvalidSplitsFromValid
//purpose : Removing invalid splits of faces from valid
//=======================================================================
void BRepOffset_BuildOffsetFaces::RemoveInvalidSplitsFromValid (const BRepOffset_DataMapOfShapeMapOfShape& theDMFMVIE)
{
// Decide whether to remove the found invalid faces or not.
// The procedure is the following:
// 1. Make connexity blocks from invalid faces;
// 2. Find free edges in this blocks;
// 3. If all free edges are valid for the faces - remove block.
//
TopTools_MapOfShape aMFence, aMFToRem;
TopoDS_Compound aCFInv;
BRep_Builder aBB;
aBB.MakeCompound (aCFInv);
TopTools_ListIteratorOfListOfShape aItLF;
//
// make compound of invalid faces
TopTools_DataMapOfShapeShape aDMIFOF;
Standard_Integer i, aNb = myInvalidFaces.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
// artificially invalid faces should not be removed
if (myArtInvalidFaces.IsBound (aF))
{
continue;
}
const TopTools_ListOfShape& aLFInv = myInvalidFaces (i);
aItLF.Initialize (aLFInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
if (aMFence.Add (aFIm))
{
aBB.Add (aCFInv, aFIm);
aDMIFOF.Bind (aFIm, aF);
}
}
}
//
// make connexity blocks
TopTools_ListOfShape aLCBInv;
BOPTools_AlgoTools::MakeConnexityBlocks (aCFInv, TopAbs_EDGE, TopAbs_FACE, aLCBInv);
//
// analyze each block
aItLF.Initialize (aLCBInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aCB = aItLF.Value();
//
// if connexity block contains only one face - it should be removed;
TopExp_Explorer aExp (aCB, TopAbs_FACE);
aExp.Next();
if (aExp.More())
{
// check if there are valid images left
aExp.Init (aCB, TopAbs_FACE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aFIm = aExp.Current();
const TopoDS_Shape& aF = aDMIFOF.Find (aFIm);
//
const TopTools_ListOfShape& aLFIm = myOFImages.FindFromKey (aF);
const TopTools_ListOfShape& aLFInv = myInvalidFaces.FindFromKey (aF);
//
if (aLFIm.Extent() == aLFInv.Extent())
{
break;
}
}
}
//
if (!aExp.More())
{
aExp.Init (aCB, TopAbs_FACE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aF = aExp.Current();
aMFToRem.Add (aF);
}
continue;
}
//
// remove faces connected by inverted edges
TopTools_IndexedDataMapOfShapeListOfShape aDMEF;
TopExp::MapShapesAndAncestors (aCB, TopAbs_EDGE, TopAbs_FACE, aDMEF);
//
TopTools_DataMapOfShapeListOfShape aDMFF;
aExp.Init (aCB, TopAbs_FACE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aFCB = aExp.Current();
const TopoDS_Shape& aF = aDMIFOF.Find (aFCB);
TopTools_ListOfShape* pList = aDMFF.ChangeSeek (aF);
if (!pList)
pList = aDMFF.Bound (aF, TopTools_ListOfShape());
pList->Append (aFCB);
}
for (TopTools_DataMapOfShapeListOfShape::Iterator itM (aDMFF); itM.More(); itM.Next())
{
const TopoDS_Shape& aF = itM.Key();
const TopTools_MapOfShape* pValidInverted = theDMFMVIE.Seek (aF);
// either remove all of these faces or none.
const TopTools_ListOfShape& aLFCB = itM.Value();
TopTools_ListOfShape::Iterator itL (aLFCB);
for (; itL.More(); itL.Next())
{
const TopoDS_Shape& aFCB = itL.Value();
TopExp_Explorer aExpE (aFCB, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aECB = aExpE.Current();
if (pValidInverted && pValidInverted->Contains (aECB))
break;
if (aDMEF.FindFromKey (aECB).Extent() > 1)
{
if (!myInvertedEdges.Contains (aECB))
break;
}
}
if (!aExpE.More())
// if one removed - remove all
break;
}
if (itL.More())
{
for (itL.Initialize (aLFCB); itL.More(); itL.Next())
{
aMFToRem.Add (itL.Value());
}
}
}
}
//
if (aMFToRem.Extent())
{
// remove invalid faces from images
aNb = myInvalidFaces.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
TopTools_ListOfShape& aLFImages = myOFImages.ChangeFromKey (aF);
aItLF.Initialize (aLFImages);
for (; aItLF.More();)
{
const TopoDS_Shape& aFIm = aItLF.Value();
if (aMFToRem.Contains (aFIm))
{
aLFImages.Remove (aItLF);
}
else
{
aItLF.Next();
}
}
}
}
}
namespace {
//=======================================================================
//function : buildPairs
//purpose : builds pairs of shapes
//=======================================================================
static void buildPairs (const TopTools_IndexedMapOfShape& theSMap,
BRepOffset_DataMapOfShapeMapOfShape& theIntPairs)
{
const Standard_Integer aNbS = theSMap.Extent();
if (aNbS < 2)
return;
for (Standard_Integer it1 = 1; it1 <= aNbS; ++it1)
{
const TopoDS_Shape& aS = theSMap (it1);
if (!theIntPairs.IsBound (aS))
theIntPairs.Bind (aS, TopTools_MapOfShape());
}
for (Standard_Integer it1 = 1; it1 <= aNbS; ++it1)
{
const TopoDS_Shape& aS1 = theSMap (it1);
TopTools_MapOfShape& aMap1 = theIntPairs (aS1);
for (Standard_Integer it2 = it1 + 1; it2 <= aNbS; ++it2)
{
const TopoDS_Shape& aS2 = theSMap (it2);
aMap1.Add (aS2);
theIntPairs (aS2).Add (aS1);
}
}
}
//=======================================================================
//function : buildIntersectionPairs
//purpose : builds intersection pairs
//=======================================================================
static void buildIntersectionPairs (const TopTools_IndexedDataMapOfShapeListOfShape& myOFImages,
const TopTools_IndexedDataMapOfShapeListOfShape& myInvalidFaces,
const BOPAlgo_Builder& theBuilder,
const TopTools_MapOfShape& theMFRemoved,
const TopTools_DataMapOfShapeShape& theFOrigins,
NCollection_DataMap<TopoDS_Shape,
BRepOffset_DataMapOfShapeMapOfShape,
TopTools_ShapeMapHasher>& theIntPairs)
{
TopAbs_ShapeEnum aCType = TopAbs_VERTEX;
// Build connection map from vertices to faces
TopTools_IndexedDataMapOfShapeListOfShape aDMVF;
TopExp::MapShapesAndAncestors (theBuilder.Shape(), aCType, TopAbs_FACE, aDMVF);
const TopTools_DataMapOfShapeListOfShape& anImages = theBuilder.Images();
const TopTools_DataMapOfShapeListOfShape& anOrigins = theBuilder.Origins();
// Find all faces connected to the not removed faces and build intersection pairs among them.
// For removed faces intersect only those connected to each other.
for (Standard_Integer iF = 1; iF <= myInvalidFaces.Extent(); ++iF)
{
const TopoDS_Shape& aFInv = myInvalidFaces.FindKey (iF);
TopoDS_Compound aCF, aCFRem;
BRep_Builder().MakeCompound (aCF);
BRep_Builder().MakeCompound (aCFRem);
for (Standard_Integer iC = 0; iC < 2; ++iC)
{
const TopTools_ListOfShape& aLF = !iC ? myInvalidFaces (iF) : myOFImages.FindFromKey (aFInv);
for (TopTools_ListOfShape::Iterator it (aLF); it.More(); it.Next())
{
TopTools_ListOfShape aLFIm;
TakeModified (it.Value(), anImages, aLFIm);
for (TopTools_ListOfShape::Iterator itIm (aLFIm); itIm.More(); itIm.Next())
{
const TopoDS_Shape& aFIm = itIm.Value();
if (theMFRemoved.Contains (aFIm))
BRep_Builder().Add (aCFRem, aFIm);
else
BRep_Builder().Add (aCF, aFIm);
}
}
}
TopTools_ListOfShape aLCB;
BOPTools_AlgoTools::MakeConnexityBlocks (aCF, TopAbs_EDGE, TopAbs_FACE, aLCB);
if (aLCB.IsEmpty())
continue;
BRepOffset_DataMapOfShapeMapOfShape* pFInterMap =
theIntPairs.Bound (aFInv, BRepOffset_DataMapOfShapeMapOfShape());
// build pairs for not removed faces
for (TopTools_ListOfShape::Iterator itCB (aLCB); itCB.More(); itCB.Next())
{
const TopoDS_Shape& aCB = itCB.Value();
TopTools_IndexedMapOfShape aMFInter;
for (TopExp_Explorer exp (aCB, aCType); exp.More(); exp.Next())
{
const TopoDS_Shape& aCS = exp.Current();
const TopTools_ListOfShape* pLFV = aDMVF.Seek (aCS);
if (!pLFV)
continue;
for (TopTools_ListOfShape::Iterator itFV (*pLFV); itFV.More(); itFV.Next())
{
const TopoDS_Shape& aFConnected = itFV.Value();
TopTools_ListOfShape aLFOr;
TakeModified (aFConnected, anOrigins, aLFOr);
for (TopTools_ListOfShape::Iterator itOr (aLFOr); itOr.More(); itOr.Next())
{
const TopoDS_Shape* pFOr = theFOrigins.Seek (itOr.Value());
if (pFOr)
aMFInter.Add (*pFOr);
}
}
}
// build intersection pairs
buildPairs (aMFInter, *pFInterMap);
}
aLCB.Clear();
BOPTools_AlgoTools::MakeConnexityBlocks (aCFRem, TopAbs_EDGE, TopAbs_FACE, aLCB);
if (aLCB.IsEmpty())
continue;
for (TopTools_ListOfShape::Iterator itCB (aLCB); itCB.More(); itCB.Next())
{
const TopoDS_Shape& aCB = itCB.Value();
TopTools_IndexedDataMapOfShapeListOfShape aDMEF;
for (TopExp_Explorer exp (aCB, aCType); exp.More(); exp.Next())
{
const TopoDS_Shape& aCS = exp.Current();
const TopTools_ListOfShape* pLFV = aDMVF.Seek (aCS);
if (!pLFV)
continue;
for (TopTools_ListOfShape::Iterator itFV (*pLFV); itFV.More(); itFV.Next())
{
const TopoDS_Shape& aFConnected = itFV.Value();
TopExp::MapShapesAndAncestors (aFConnected, TopAbs_EDGE, TopAbs_FACE, aDMEF);
}
}
for (Standard_Integer iE = 1; iE <= aDMEF.Extent(); ++iE)
{
const TopTools_ListOfShape& aLFConnected = aDMEF (iE);
if (aLFConnected.Extent() < 2)
continue;
TopTools_IndexedMapOfShape aMFInter;
for (TopTools_ListOfShape::Iterator itLF (aLFConnected); itLF.More(); itLF.Next())
{
const TopoDS_Shape& aFConnected = itLF.Value();
TopTools_ListOfShape aLFOr;
TakeModified (aFConnected, anOrigins, aLFOr);
for (TopTools_ListOfShape::Iterator itOr (aLFOr); itOr.More(); itOr.Next())
{
const TopoDS_Shape* pFOr = theFOrigins.Seek (itOr.Value());
if (pFOr)
aMFInter.Add (*pFOr);
}
}
buildPairs (aMFInter, *pFInterMap);
}
}
}
}
}
//=======================================================================
//function : RemoveInsideFaces
//purpose : Looking for the inside faces that can be safely removed
//=======================================================================
void BRepOffset_BuildOffsetFaces::RemoveInsideFaces (const TopTools_ListOfShape& theInvertedFaces,
const TopTools_IndexedMapOfShape& theMFToCheckInt,
const TopTools_IndexedMapOfShape& theMFInvInHole,
const TopoDS_Shape& theFHoles,
TopTools_IndexedMapOfShape& theMERemoved,
TopTools_IndexedMapOfShape& theMEInside,
const Message_ProgressRange& theRange)
{
TopTools_ListOfShape aLS;
TopTools_MapOfShape aMFence;
TopTools_IndexedMapOfShape aMFInv;
TopTools_ListIteratorOfListOfShape aItLF;
TopTools_DataMapOfShapeShape aDMFImF;
//
Message_ProgressScope aPS (theRange, "Looking for inside faces", 10);
Standard_Integer i, aNb = myOFImages.Extent();
for (i = 1; i <= aNb; ++i)
{
if (!aPS.More())
{
return;
}
const TopoDS_Shape& aF = myOFImages.FindKey (i);
// to avoid intersection of the splits of the same
// offset faces among themselves make compound of the
// splits and use it as one argument
TopoDS_Compound aCFImi;
BRep_Builder().MakeCompound (aCFImi);
//
for (Standard_Integer j = 0; j < 2; ++j)
{
const TopTools_ListOfShape* pLFSp = !j ? myInvalidFaces.Seek (aF) : &myOFImages (i);
if (!pLFSp)
{
continue;
}
//
aItLF.Initialize (*pLFSp);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
if (aMFence.Add (aFIm))
{
BRep_Builder().Add (aCFImi, aFIm);
aDMFImF.Bind (aFIm, aF);
if (!j)
{
aMFInv.Add (aFIm);
}
}
}
}
//
aLS.Append (aCFImi);
}
//
// to make the solids more complete add for intersection also the faces
// consisting only of invalid edges and not included into splits
aNb = theMFToCheckInt.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aFSp = theMFToCheckInt (i);
if (aMFence.Add (aFSp))
{
aLS.Append (aFSp);
}
}
//
BOPAlgo_MakerVolume aMV;
aMV.SetArguments (aLS);
aMV.SetIntersect (Standard_True);
aMV.Perform (aPS.Next (9));
if (aMV.HasErrors())
return;
//
// get shapes connection for using in the rebuilding process
// for the cases in which some of the intersection left undetected
ShapesConnections (aDMFImF, aMV);
//
// find faces to remove
const TopoDS_Shape& aSols = aMV.Shape();
//
TopTools_IndexedDataMapOfShapeListOfShape aDMFS;
TopExp::MapShapesAndAncestors (aSols, TopAbs_FACE, TopAbs_SOLID, aDMFS);
//
aNb = aDMFS.Extent();
if (!aNb)
{
return;
}
//
// To use the created solids for classifications, firstly, it is necessary
// to check them on validity - the created solids should be complete,
// i.e. all faces should be included.
//
TopTools_MapOfShape aMFToRem;
// Check completeness
if (aMV.HasDeleted())
{
TopTools_IndexedMapOfShape aMEHoles;
TopExp::MapShapes (theFHoles, TopAbs_EDGE, aMEHoles);
// Map edges of the solids to check the connectivity
// of the removed invalid splits
TopTools_IndexedMapOfShape aMESols;
TopExp::MapShapes (aSols, TopAbs_EDGE, aMESols);
// perform additional check on faces
aNb = myOFImages.Extent();
for (i = 1; i <= aNb; ++i)
{
if (!aPS.More())
{
return;
}
const TopTools_ListOfShape& aLFIm = myOFImages (i);
if (aLFIm.IsEmpty())
{
continue;
}
const TopoDS_Shape& aF = myOFImages.FindKey (i);
Standard_Boolean bInvalid = myInvalidFaces.Contains (aF);
// For invalid faces it is allowed to be at least connected
// to the solids, otherwise the solids are considered as broken
Standard_Boolean bConnected = Standard_False;
Standard_Boolean bFaceKept = Standard_False;
aItLF.Initialize (aLFIm);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
if (!aMV.IsDeleted (aFIm))
{
bFaceKept = Standard_True;
continue;
}
//
TopExp_Explorer aExpE (aFIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
if (aMEHoles.Contains (aExpE.Current()))
{
bFaceKept = Standard_True;
aMFToRem.Add (aFIm);
break;
}
if (!bFaceKept && bInvalid && !bConnected)
bConnected = aMESols.Contains (aExpE.Current());
}
}
//
if (!bFaceKept && !bConnected)
{
return;
}
}
}
//
TopTools_IndexedMapOfShape aMEBoundary;
aNb = aDMFS.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aFIm = aDMFS.FindKey (i);
const TopTools_ListOfShape& aLSol = aDMFS (i);
if (aLSol.Extent() > 1)
{
aMFToRem.Add (aFIm);
}
else if (aFIm.Orientation() != TopAbs_INTERNAL)
{
TopExp::MapShapes (aFIm, TopAbs_EDGE, aMEBoundary);
}
}
// Tool for getting the splits of faces
const TopTools_DataMapOfShapeListOfShape& aMVIms = aMV.Images();
// update invalid faces with images
aNb = aMFInv.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aFInv = aMFInv (i);
TakeModified (aFInv, aMVIms, aMFInv);
}
// Take into account the faces invalid by inverted edges
for (TopTools_ListOfShape::Iterator itLF (theInvertedFaces); itLF.More(); itLF.Next())
TakeModified (itLF.Value(), aMVIms, aMFInv);
// check if the invalid faces inside the holes are really invalid:
// check its normal direction - if it has changed relatively the
// original face the offset face is invalid and should be kept for rebuilding
Standard_Integer aNbFH = theMFInvInHole.Extent();
for (i = 1; i <= aNbFH; ++i)
{
if (!aPS.More())
{
return;
}
const TopoDS_Shape& aFInv = theMFInvInHole (i);
TopTools_ListOfShape aLFInvIm = aMV.Modified (aFInv);
if (aLFInvIm.IsEmpty())
{
aLFInvIm.Append (aFInv);
}
//
const TopoDS_Shape* pFOffset = aDMFImF.Seek (aFInv);
if (!pFOffset)
{
continue;
}
TopTools_ListIteratorOfListOfShape aItLFInv (aLFInvIm);
for (; aItLFInv.More(); aItLFInv.Next())
{
const TopoDS_Shape& aFInvIm = aItLFInv.Value();
const TopTools_ListOfShape* pLSols = aDMFS.Seek (aFInvIm);
if (!pLSols || pLSols->Extent() != 1)
{
continue;
}
//
const TopoDS_Shape& aFSol = pLSols->First();
//
TopoDS_Shape aFx;
if (!FindShape (aFInvIm, aFSol, NULL, aFx))
{
continue;
}
//
if (BRepOffset_Tool::CheckPlanesNormals (TopoDS::Face (aFx), TopoDS::Face (*pFOffset)))
{
// the normal direction has not changed, thus the face can be removed
aMFToRem.Add (aFInvIm);
}
}
}
//
TopoDS_Compound aSolids;
BRep_Builder().MakeCompound (aSolids);
TopTools_MapOfShape aMFKeep;
//
TopExp_Explorer aExpS (aSols, TopAbs_SOLID);
for (; aExpS.More(); aExpS.Next())
{
if (!aPS.More())
{
return;
}
const TopoDS_Shape& aSol = aExpS.Current();
//
Standard_Boolean bAllInv (Standard_True), bAllRemoved (Standard_True);
for (TopExp_Explorer aExpF (aSol, TopAbs_FACE); aExpF.More(); aExpF.Next())
{
const TopoDS_Shape& aFS = aExpF.Current();
//
if (aFS.Orientation() == TopAbs_INTERNAL)
{
aMFToRem.Add (aFS);
continue;
}
if (aMFToRem.Contains (aFS))
continue;
bAllRemoved = false;
bAllInv &= aMFInv.Contains (aFS);
}
//
if (bAllInv && !bAllRemoved)
{
// remove invalid faces but keep those that have already been marked for removal
TopExp_Explorer aExpF (aSol, TopAbs_FACE);
for (; aExpF.More(); aExpF.Next())
{
const TopoDS_Shape& aFS = aExpF.Current();
//
if (aMFToRem.Contains (aFS))
{
if (!aMFKeep.Add (aFS))
{
aMFKeep.Remove (aFS);
}
}
else
{
aMFToRem.Add (aFS);
}
}
}
else
{
BRep_Builder().Add (aSolids, aSol);
mySolids = aSolids;
}
}
//
TopTools_MapIteratorOfMapOfShape aItM (aMFKeep);
for (; aItM.More(); aItM.Next())
{
aMFToRem.Remove (aItM.Value());
}
// Remove the invalid hanging parts external to the solids
RemoveHangingParts (aMV, aDMFImF, aMFInv, aMFToRem);
// Remove newly found internal and hanging faces
RemoveValidSplits (aMFToRem, aMV, theMERemoved);
RemoveInvalidSplits (aMFToRem, aMV, theMERemoved);
//
// Get inside faces from the removed ones comparing them with boundary edges
theMEInside.Clear();
aNb = theMERemoved.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aE = theMERemoved (i);
if (!aMEBoundary.Contains (aE))
{
theMEInside.Add (aE);
}
}
// build all possible intersection pairs basing on the intersection results
// taking into account removed faces.
if (aMFToRem.Extent())
buildIntersectionPairs (myOFImages, myInvalidFaces, aMV, aMFToRem, aDMFImF, myIntersectionPairs);
}
//=======================================================================
//function : ShapesConnections
//purpose : Looking for the connections between faces not to miss
// some necessary intersection
//=======================================================================
void BRepOffset_BuildOffsetFaces::ShapesConnections (const TopTools_DataMapOfShapeShape& theDMFOr,
BOPAlgo_Builder& theBuilder)
{
// Make connexity blocks from invalid edges to use the whole block
// to which the edge is connected instead of the single edge.
TopoDS_Compound aCEInv;
BRep_Builder().MakeCompound(aCEInv);
for (Standard_Integer i = 1; i <= myInvalidEdges.Extent(); ++i)
{
AddToContainer (myInvalidEdges(i), aCEInv);
}
TopTools_ListOfShape aLCB;
BOPTools_AlgoTools::MakeConnexityBlocks (aCEInv, TopAbs_VERTEX, TopAbs_EDGE, aLCB);
// Binding from the edge to the block
TopTools_DataMapOfShapeShape aECBMap;
for (TopTools_ListOfShape::Iterator itCB(aLCB); itCB.More(); itCB.Next())
{
for (TopoDS_Iterator itE(itCB.Value()); itE.More(); itE.Next())
{
aECBMap.Bind(itE.Value(), itCB.Value());
}
}
// update invalid edges with images and keep connection to original edge
TopTools_DataMapOfShapeListOfShape aDMEOr;
Standard_Integer aNb = myInvalidEdges.Extent();
for (Standard_Integer i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aEInv = myInvalidEdges (i);
const TopTools_ListOfShape& aLEIm = theBuilder.Modified (aEInv);
if (aLEIm.IsEmpty())
{
aDMEOr.Bound (aEInv, TopTools_ListOfShape())->Append (aEInv);
continue;
}
//
TopTools_ListIteratorOfListOfShape aItLEIm (aLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
AddToContainer (aEIm, aEInv, aDMEOr);
}
}
//
// get shapes connections for using in the rebuilding process
const BOPDS_PDS& pDS = theBuilder.PDS();
// analyze all Face/Face intersections
const BOPDS_VectorOfInterfFF& aFFs = pDS->InterfFF();
Standard_Integer iInt, aNbFF = aFFs.Length();
for (iInt = 0; iInt < aNbFF; ++iInt)
{
const BOPDS_InterfFF& aFF = aFFs (iInt);
const BOPDS_VectorOfCurve& aVNC = aFF.Curves();
Standard_Integer aNbC = aVNC.Length();
if (!aNbC)
{
continue;
}
//
const TopoDS_Shape& aFIm1 = pDS->Shape (aFF.Index1());
const TopoDS_Shape& aFIm2 = pDS->Shape (aFF.Index2());
//
const TopoDS_Shape* pF1 = theDMFOr.Seek (aFIm1);
const TopoDS_Shape* pF2 = theDMFOr.Seek (aFIm2);
//
if (!pF1 || !pF2)
{
continue;
}
//
if (pF1->IsSame (*pF2))
{
continue;
}
//
Standard_Boolean bInv1 = myInvalidFaces.Contains (*pF1);
Standard_Boolean bInv2 = myInvalidFaces.Contains (*pF2);
//
if (!bInv1 && !bInv2)
{
continue;
}
//
// check if it is real Face/Face intersection
TopTools_MapOfShape aMEInt;
for (Standard_Integer iC = 0; iC < aNbC; ++iC)
{
const BOPDS_Curve& aNC = aVNC (iC);
const BOPDS_ListOfPaveBlock& aLPB = aNC.PaveBlocks();
BOPDS_ListIteratorOfListOfPaveBlock aItLPB (aLPB);
for (; aItLPB.More(); aItLPB.Next())
{
const Handle(BOPDS_PaveBlock)& aPB = aItLPB.Value();
Standard_Integer nEInt;
if (aPB->HasEdge (nEInt))
{
const TopoDS_Shape& aEInt = pDS->Shape (nEInt);
aMEInt.Add (aEInt);
}
}
}
//
if (aMEInt.IsEmpty())
{
continue;
}
//
// check if invalid edges of the face are in the same splits with intersection edges
for (Standard_Integer i = 0; i < 2; ++i)
{
if ((!i && !bInv1) || (i && !bInv2))
{
continue;
}
//
const TopoDS_Shape& aF = !i ? *pF1 : *pF2;
const TopoDS_Shape& aFOp = !i ? *pF2 : *pF1;
const TopoDS_Shape& aFIm = !i ? aFIm1 : aFIm2;
//
Standard_Boolean bFound = Standard_False;
//
TopTools_ListOfShape aLFIm = theBuilder.Modified (aFIm);
if (aLFIm.IsEmpty())
{
aLFIm.Append (aFIm);
}
//
TopTools_ListIteratorOfListOfShape aItLFIm (aLFIm);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFImIm = aItLFIm.Value();
//
Standard_Boolean bInv (Standard_False), bInt (Standard_False);
TopExp_Explorer aExpE (aFImIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aE = aExpE.Current();
if (!bInv)
{
bInv = aDMEOr.IsBound (aE);
}
if (!bInt)
{
bInt = aMEInt.Contains (aE);
}
if (bInv && bInt)
{
break;
}
}
//
if (!bInt || !bInv)
{
continue;
}
//
bFound = Standard_True;
//
// append opposite face to all invalid edges in the split
aExpE.Init (aFImIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aE = aExpE.Current();
const TopTools_ListOfShape* pLEOr = aDMEOr.Seek (aE);
if (!pLEOr)
{
continue;
}
//
TopTools_ListIteratorOfListOfShape aItLE (*pLEOr);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEOr = aItLE.Value();
TopTools_ListOfShape* pLFE = mySSInterfs.ChangeSeek (aEOr);
if (!pLFE)
{
pLFE = mySSInterfs.Bound (aEOr, TopTools_ListOfShape());
}
AppendToList (*pLFE, aFOp);
}
}
}
if (bFound)
{
// save connection between offset faces
TopTools_ListOfShape* pLF = mySSInterfs.ChangeSeek (aF);
if (!pLF)
{
pLF = mySSInterfs.Bound (aF, TopTools_ListOfShape());
}
AppendToList (*pLF, aFOp);
}
}
// Treatment for the artificial case - check if one of the faces is artificially invalid
for (Standard_Integer iF = 0; iF < 2; ++iF)
{
const TopoDS_Shape& aFArt = !iF ? *pF1 : *pF2;
const TopoDS_Shape& aFOpposite = !iF ? *pF2 : *pF1;
if (!myArtInvalidFaces.IsBound (aFArt))
continue;
if (myInvalidFaces.Contains (aFOpposite) && !myArtInvalidFaces.IsBound (aFOpposite))
continue;
// Collect own invalid edges of the face and the invalid edges connected to those
// own invalid edges to be avoided in the check for intersection.
TopTools_IndexedMapOfShape aMEAvoid;
if (const TopTools_IndexedMapOfShape* pFEInv = myArtInvalidFaces.Seek (aFOpposite))
{
for (Standard_Integer iE = 1; iE <= pFEInv->Extent(); ++iE)
{
if (const TopoDS_Shape* pCB = aECBMap.Seek (pFEInv->FindKey(iE)))
{
TopExp::MapShapes (*pCB, TopAbs_EDGE, aMEAvoid);
}
}
}
else if (const TopTools_ListOfShape* pLFIm = myOFImages.Seek (aFOpposite))
{
for (TopTools_ListOfShape::Iterator itLFIm (*pLFIm); itLFIm.More(); itLFIm.Next())
{
for (TopExp_Explorer expE (itLFIm.Value(), TopAbs_EDGE); expE.More(); expE.Next())
{
if (const TopoDS_Shape* pCB = aECBMap.Seek (expE.Current()))
{
TopExp::MapShapes (*pCB, TopAbs_EDGE, aMEAvoid);
}
}
}
}
const TopoDS_Shape& aFArtIm = !iF ? aFIm1 : aFIm2;
const TopoDS_Shape& aFOppositeIm = !iF ? aFIm2 : aFIm1;
// Check if there are any intersections between edges of artificially
// invalid face and opposite face
const Standard_Integer nFOp = pDS->Index (aFOppositeIm);
for (TopExp_Explorer expE (aFArtIm, TopAbs_EDGE); expE.More(); expE.Next())
{
const TopoDS_Shape& aE = expE.Current();
if (!myInvalidEdges.Contains (aE) || myInvertedEdges.Contains (aE) || aMEAvoid.Contains (aE))
{
continue;
}
const Standard_Integer nE = pDS->Index (aE);
if (nE < 0)
{
continue;
}
if (!pDS->HasInterf(nE, nFOp))
{
continue;
}
TopTools_ListOfShape aLV;
const BOPDS_VectorOfInterfEF& aEFs = pDS->InterfEF();
for (Standard_Integer iEF = 0; iEF < aEFs.Size(); ++iEF)
{
const BOPDS_InterfEF& aEF = aEFs (iEF);
if (aEF.Contains (nE) && aEF.Contains(nFOp))
{
if (aEF.CommonPart().Type() == TopAbs_VERTEX)
aLV.Append (pDS->Shape (aEF.IndexNew()));
}
}
if (aLV.IsEmpty())
{
continue;
}
// Make sure that there is an opposite intersection exists, i.e. some of the edges
// of the opposite face intersect the artificially invalid face.
const Standard_Integer nFArt = pDS->Index (aFArtIm);
TopExp_Explorer expEOp (aFOppositeIm, TopAbs_EDGE);
for (; expEOp.More(); expEOp.Next())
{
const TopoDS_Shape& aEOp = expEOp.Current();
const Standard_Integer nEOp = pDS->Index (aEOp);
if (pDS->HasInterf(nEOp, nFArt))
{
break;
}
}
if (!expEOp.More())
{
continue;
}
// Intersection is present - add connection between offset faces.
AddToContainer (aFArt, aFOpposite, mySSInterfsArt);
// Add connection between edge and opposite face
AddToContainer (aE, aFOpposite, mySSInterfsArt);
// Along with the opposite face, save the intersection vertices to
// be used for trimming the intersection edge in the rebuilding process
for (TopTools_ListOfShape::Iterator itLV (aLV); itLV.More(); itLV.Next())
{
// Add connection to intersection vertex
AddToContainer (aE, itLV.Value(), mySSInterfsArt);
}
}
}
}
}
//=======================================================================
//function : RemoveHangingParts
//purpose : Remove isolated invalid hanging parts
//=======================================================================
void BRepOffset_BuildOffsetFaces::RemoveHangingParts (const BOPAlgo_MakerVolume& theMV,
const TopTools_DataMapOfShapeShape& theDMFImF,
const TopTools_IndexedMapOfShape& theMFInv,
TopTools_MapOfShape& theMFToRem)
{
// Map the faces of the result solids to filter them from avoided faces
TopTools_IndexedMapOfShape aMFS;
TopExp::MapShapes (theMV.Shape(), TopAbs_FACE, aMFS);
BRep_Builder aBB;
// Build compound of all faces not included into solids
TopoDS_Compound aCFHangs;
aBB.MakeCompound (aCFHangs);
// Tool for getting the splits of faces
const TopTools_DataMapOfShapeListOfShape& aMVIms = theMV.Images();
TopTools_ListIteratorOfListOfShape aItLArgs (theMV.Arguments());
for (; aItLArgs.More(); aItLArgs.Next())
{
TopExp_Explorer anExpF (aItLArgs.Value(), TopAbs_FACE);
for (; anExpF.More(); anExpF.Next())
{
const TopoDS_Shape& aF = anExpF.Current();
TakeModified (aF, aMVIms, aCFHangs, &aMFS);
}
}
// Make connexity blocks of all hanging parts and check that they are isolated
TopTools_ListOfShape aLCBHangs;
BOPTools_AlgoTools::MakeConnexityBlocks (aCFHangs, TopAbs_EDGE, TopAbs_FACE, aLCBHangs);
if (aLCBHangs.IsEmpty())
return;
// To be removed, the block should contain invalid splits of offset faces and should
// meet one of the following conditions:
// 1. The block should not be connected to any invalid parts (Faces or Edges)
// contained in solids;
// 2. The block should be isolated from other faces, i.e. it should consist of
// the splits of the single offset face.
// Map the edges and vertices of the result solids to check connectivity
// of the hanging blocks to invalid parts contained in solids
TopTools_IndexedDataMapOfShapeListOfShape aDMEF, aDMVE;
TopExp::MapShapesAndAncestors (theMV.Shape(), TopAbs_EDGE, TopAbs_FACE, aDMEF);
TopExp::MapShapesAndAncestors (theMV.Shape(), TopAbs_VERTEX, TopAbs_EDGE, aDMVE);
// Update invalid edges with intersection results
TopTools_MapOfShape aMEInv;
Standard_Integer i, aNbE = myInvalidEdges.Extent();
for (i = 1; i <= aNbE; ++i)
TakeModified (myInvalidEdges (i), aMVIms, aMEInv);
// Update inverted edges with intersection results
TopTools_MapOfShape aMEInverted;
for (Standard_Integer iInv = 1; iInv <= myInvertedEdges.Extent(); ++iInv)
TakeModified (myInvertedEdges (iInv), aMVIms, aMEInverted);
// Tool for getting the origins of the splits
const TopTools_DataMapOfShapeListOfShape& aMVOrs = theMV.Origins();
// Find hanging blocks to remove
TopTools_ListOfShape aBlocksToRemove;
TopTools_ListIteratorOfListOfShape aItLCBH (aLCBHangs);
for (; aItLCBH.More(); aItLCBH.Next())
{
const TopoDS_Shape& aCBH = aItLCBH.Value();
// Remove the block containing the inverted edges
Standard_Boolean bHasInverted = Standard_False;
TopExp_Explorer anExpE (aCBH, TopAbs_EDGE);
for (; anExpE.More() && !bHasInverted; anExpE.Next())
{
const TopoDS_Shape& aE = anExpE.Current();
bHasInverted = !aDMEF.Contains (aE) &&
aMEInverted.Contains (aE);
}
if (bHasInverted)
{
aBlocksToRemove.Append (aCBH);
continue;
}
// Check the block to contain invalid split
Standard_Boolean bHasInvalidFace = Standard_False;
// Check connectivity to invalid parts
Standard_Boolean bIsConnected = Standard_False;
TopTools_IndexedMapOfShape aBlockME;
TopExp::MapShapes (aCBH, TopAbs_EDGE, aBlockME);
// Map to collect all original faces
TopTools_MapOfShape aMOffsetF;
TopExp_Explorer anExpF (aCBH, TopAbs_FACE);
for (; anExpF.More(); anExpF.Next())
{
const TopoDS_Shape& aF = anExpF.Current();
// Check block to contain invalid face
if (!bHasInvalidFace)
bHasInvalidFace = theMFInv.Contains (aF);
// Check block for connectivity to invalid parts
if (!bIsConnected)
{
// check edges
anExpE.Init (aF, TopAbs_EDGE);
for (; anExpE.More() && !bIsConnected; anExpE.Next())
{
const TopoDS_Shape& aE = anExpE.Current();
const TopTools_ListOfShape* pLF = aDMEF.Seek (aE);
if (pLF)
{
TopTools_ListIteratorOfListOfShape aItLF (*pLF);
for (; aItLF.More() && !bIsConnected; aItLF.Next())
bIsConnected = theMFInv.Contains (aItLF.Value());
}
}
// check vertices
if (!bIsConnected)
{
TopExp_Explorer anExpV (aF, TopAbs_VERTEX);
for (; anExpV.More() && !bIsConnected; anExpV.Next())
{
const TopoDS_Shape& aV = anExpV.Current();
const TopTools_ListOfShape* pLE = aDMVE.Seek (aV);
if (pLE)
{
TopTools_ListIteratorOfListOfShape aItLE (*pLE);
for (; aItLE.More() && !bIsConnected; aItLE.Next())
bIsConnected = !aBlockME.Contains (aItLE.Value()) &&
aMEInv.Contains (aItLE.Value());
}
}
}
}
// Check block to be isolated
const TopTools_ListOfShape* pLFOr = aMVOrs.Seek (aF);
if (pLFOr)
{
TopTools_ListIteratorOfListOfShape aItLFOr (*pLFOr);
for (; aItLFOr.More(); aItLFOr.Next())
{
const TopoDS_Shape* pFOffset = theDMFImF.Seek (aItLFOr.Value());
if (pFOffset)
aMOffsetF.Add (*pFOffset);
}
}
else
{
const TopoDS_Shape* pFOffset = theDMFImF.Seek (aF);
if (pFOffset)
aMOffsetF.Add (*pFOffset);
}
}
Standard_Boolean bRemove = bHasInvalidFace &&
(!bIsConnected || aMOffsetF.Extent() == 1);
if (bRemove)
aBlocksToRemove.Append (aCBH);
}
// remove the invalidated blocks
aItLCBH.Initialize (aBlocksToRemove);
for (; aItLCBH.More(); aItLCBH.Next())
{
const TopoDS_Shape& aCBH = aItLCBH.Value();
TopExp_Explorer anExpF (aCBH, TopAbs_FACE);
for (; anExpF.More(); anExpF.Next())
theMFToRem.Add (anExpF.Current());
}
}
//=======================================================================
//function : RemoveValidSplits
//purpose : Removing valid splits according to results of intersection
//=======================================================================
void BRepOffset_BuildOffsetFaces::RemoveValidSplits (const TopTools_MapOfShape& theSpRem,
BOPAlgo_Builder& theGF,
TopTools_IndexedMapOfShape& theMERemoved)
{
Standard_Integer i, aNb = myOFImages.Extent();
if (!aNb)
{
return;
}
//
for (i = 1; i <= aNb; ++i)
{
TopTools_ListOfShape& aLSIm = myOFImages (i);
TopTools_ListIteratorOfListOfShape aIt (aLSIm);
for (; aIt.More(); )
{
const TopoDS_Shape& aSIm = aIt.Value();
if (theSpRem.Contains (aSIm))
{
TopExp::MapShapes (aSIm, TopAbs_EDGE, theMERemoved);
aLSIm.Remove (aIt);
continue;
}
//
// check if all its images are have to be removed
const TopTools_ListOfShape& aLSImIm = theGF.Modified (aSIm);
if (aLSImIm.Extent())
{
Standard_Boolean bAllRem = Standard_True;
TopTools_ListIteratorOfListOfShape aIt1 (aLSImIm);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aSImIm = aIt1.Value();
if (theSpRem.Contains (aSImIm))
{
TopExp::MapShapes (aSImIm, TopAbs_EDGE, theMERemoved);
}
else
{
bAllRem = Standard_False;
}
}
//
if (bAllRem)
{
TopExp::MapShapes (aSIm, TopAbs_EDGE, theMERemoved);
aLSIm.Remove (aIt);
continue;
}
}
aIt.Next();
}
}
}
//=======================================================================
//function : RemoveInvalidSplits
//purpose : Removing invalid splits according to the results of intersection
//=======================================================================
void BRepOffset_BuildOffsetFaces::RemoveInvalidSplits (const TopTools_MapOfShape& theSpRem,
BOPAlgo_Builder& theGF,
TopTools_IndexedMapOfShape& theMERemoved)
{
Standard_Integer i, aNb = myInvalidFaces.Extent();
if (!aNb)
{
return;
}
//
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aS = myInvalidFaces.FindKey (i);
Standard_Boolean bArt = myArtInvalidFaces.IsBound (aS);
//
TopTools_ListOfShape& aLSIm = myInvalidFaces (i);
TopTools_ListIteratorOfListOfShape aIt (aLSIm);
for (; aIt.More();)
{
const TopoDS_Shape& aSIm = aIt.Value();
if (theSpRem.Contains (aSIm))
{
TopExp::MapShapes (aSIm, TopAbs_EDGE, theMERemoved);
aLSIm.Remove (aIt);
continue;
}
//
// check if all its images are have to be removed
const TopTools_ListOfShape& aLSImIm = theGF.Modified (aSIm);
if (aLSImIm.IsEmpty())
{
aIt.Next();
continue;
}
//
Standard_Boolean bAllRem = Standard_True;
TopTools_IndexedMapOfShape aMERemoved;
TopTools_ListIteratorOfListOfShape aIt1 (aLSImIm);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aSImIm = aIt1.Value();
if (theSpRem.Contains (aSImIm))
{
TopExp::MapShapes (aSImIm, TopAbs_EDGE, aMERemoved);
}
else
{
bAllRem = Standard_False;
}
}
//
if (bAllRem)
{
aLSIm.Remove (aIt);
continue;
}
//
if (bArt)
{
aIt.Next();
continue;
}
//
// remove the face from invalid if all invalid edges of this face
// have been marked for removal
TopExp_Explorer aExpE (aSIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aEInv = aExpE.Current();
if (myInvalidEdges.Contains (aEInv) && !aMERemoved.Contains (aEInv))
{
break;
}
}
if (!aExpE.More())
{
TopExp::MapShapes (aSIm, TopAbs_EDGE, theMERemoved);
aLSIm.Remove (aIt);
}
else
{
aIt.Next();
}
}
}
}
//=======================================================================
//function : FilterEdgesImages
//purpose : Updating the maps of images and origins of the offset edges
//=======================================================================
void BRepOffset_BuildOffsetFaces::FilterEdgesImages (const TopoDS_Shape& theS)
{
// map edges
TopTools_IndexedMapOfShape aME;
TopExp::MapShapes (theS, TopAbs_EDGE, aME);
//
myOEOrigins.Clear();
TopTools_DataMapIteratorOfDataMapOfShapeListOfShape aItDM (myOEImages);
for (; aItDM.More(); aItDM.Next())
{
const TopoDS_Shape& aE = aItDM.Key();
TopTools_ListOfShape& aLEIm = aItDM.ChangeValue();
//
TopTools_ListIteratorOfListOfShape aIt (aLEIm);
for (; aIt.More(); )
{
const TopoDS_Shape& aEIm = aIt.Value();
// filter images
if (!aME.Contains (aEIm))
{
// remove the image
// edges with no images left should be kept in the map
// to avoid their usage when building the splits of faces
aLEIm.Remove (aIt);
continue;
}
//
// save origins
if (myOEOrigins.IsBound (aEIm))
{
AppendToList (myOEOrigins.ChangeFind (aEIm), aE);
}
else
{
TopTools_ListOfShape aLOr;
aLOr.Append (aE);
myOEOrigins.Bind (aEIm, aLOr);
}
//
aIt.Next();
}
}
}
//=======================================================================
//function : FilterInvalidFaces
//purpose : Filtering of the invalid faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::FilterInvalidFaces (const TopTools_IndexedDataMapOfShapeListOfShape& theDMEF,
const TopTools_IndexedMapOfShape& theMERemoved)
{
//
// filter invalid faces, considering faces having only valid
// images left with non-free edges as valid
// do not remove invalid faces if it creates free edges
//
TopTools_IndexedDataMapOfShapeListOfShape aReallyInvFaces;
// Edge-Face connexity map of all splits, both invalid and valid
TopTools_IndexedDataMapOfShapeListOfShape aDMEFAll;
TopTools_ListIteratorOfListOfShape aItLF;
//
const Standard_Integer aNb = myInvalidFaces.Extent();
for (Standard_Integer i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
const TopTools_ListOfShape& aLFInv = myInvalidFaces (i);
//
if (myArtInvalidFaces.IsBound (aF))
{
if (aLFInv.IsEmpty())
{
myArtInvalidFaces.UnBind (aF);
}
else
{
aReallyInvFaces.Add (aF, aLFInv);
}
continue;
}
//
if (aLFInv.IsEmpty())
{
continue;
}
//
TopTools_ListOfShape& aLFIm = myOFImages.ChangeFromKey (aF);
Standard_Boolean bInvalid = aLFIm.IsEmpty();
//
if (!bInvalid)
{
// check two lists on common splits
aItLF.Initialize (aLFInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFInv = aItLF.Value();
//
TopTools_ListIteratorOfListOfShape aItLFIm (aLFIm);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
//
if (aFInv.IsSame (aFIm))
{
break;
}
}
//
if (aItLFIm.More())
{
break;
}
}
//
bInvalid = aItLF.More();
}
//
if (!bInvalid)
{
// check for free edges
for (Standard_Integer j = 0; !bInvalid && j < 2; ++j)
{
const TopTools_ListOfShape& aLI = !j ? aLFIm : aLFInv;
aItLF.Initialize (aLI);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
//
TopExp_Explorer aExp (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (!theMERemoved.Contains (aE))
{
const TopTools_ListOfShape* pLEF = theDMEF.Seek (aE);
if (pLEF && pLEF->Extent() == 1)
{
break;
}
}
}
//
if (aExp.More())
{
break;
}
}
bInvalid = aItLF.More();
}
}
if (bInvalid)
{
if (aDMEFAll.IsEmpty())
{
aDMEFAll = theDMEF;
for (Standard_Integer iF = 1; iF <= aNb; ++iF)
for (TopTools_ListOfShape::Iterator itLFInv (myInvalidFaces (iF)); itLFInv.More(); itLFInv.Next())
TopExp::MapShapesAndAncestors (itLFInv.Value(), TopAbs_EDGE, TopAbs_FACE, aDMEFAll);
}
TopTools_MapOfShape aLocalSplits;
for (Standard_Integer j = 0; j < 2; ++j)
for (aItLF.Initialize ((!j ? aLFIm : aLFInv)); aItLF.More(); aItLF.Next())
aLocalSplits.Add (aItLF.Value());
// Check if all invalid edges are located inside the split and do not touch
// any other faces both invalid and valid
aItLF.Initialize (aLFInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
TopExp_Explorer aExp (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (myInvalidEdges.Contains (aE) && !theMERemoved.Contains (aE))
{
const TopTools_ListOfShape& aLF = aDMEFAll.FindFromKey (aE);
TopTools_ListOfShape::Iterator itLF (aLF);
for (; itLF.More(); itLF.Next())
{
if (!aLocalSplits.Contains (itLF.Value()))
break;
}
if (itLF.More())
break;
}
}
if (aExp.More())
break;
}
bInvalid = aItLF.More();
if (!bInvalid)
{
aItLF.Initialize (aLFInv);
for (; aItLF.More(); aItLF.Next())
AppendToList (aLFIm, aItLF.Value());
}
}
//
if (bInvalid)
{
aReallyInvFaces.Add (aF, aLFInv);
}
}
//
myInvalidFaces = aReallyInvFaces;
}
//=======================================================================
//function : CheckEdgesCreatedByVertex
//purpose : Checks additionally the unchecked edges originated from vertices
//=======================================================================
void BRepOffset_BuildOffsetFaces::CheckEdgesCreatedByVertex()
{
// Mark the unchecked edges contained in invalid faces as invalid
const Standard_Integer aNbF = myInvalidFaces.Extent();
for (Standard_Integer i = 1; i <= aNbF; ++i)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
if (myArtInvalidFaces.IsBound (aF))
continue;
const TopTools_ListOfShape& aLFIm = myInvalidFaces (i);
for (TopTools_ListOfShape::Iterator it (aLFIm); it.More(); it.Next())
{
const TopoDS_Shape& aFIm = it.Value();
for (TopExp_Explorer expE (aFIm, TopAbs_EDGE); expE.More(); expE.Next())
{
const TopoDS_Shape& aE = expE.Current();
if (myInvalidEdges.Contains (aE)
|| myValidEdges.Contains (aE))
{
continue;
}
// check if this edges is not created from vertex and mark it as invalid
const TopTools_ListOfShape* pLEOr = myEdgesOrigins->Seek (aE);
if (!pLEOr)
continue;
TopTools_ListOfShape::Iterator itLEO (*pLEOr);
for (; itLEO.More(); itLEO.Next())
{
if (itLEO.Value().ShapeType() != TopAbs_VERTEX)
break;
}
if (!itLEO.More())
{
myInvalidEdges.Add (aE);
}
}
}
}
}
//=======================================================================
//function : FilterInvalidEdges
//purpose : Filtering the invalid edges according to currently invalid faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::FilterInvalidEdges (const BRepOffset_DataMapOfShapeIndexedMapOfShape& theDMFMIE,
const TopTools_IndexedMapOfShape& theMERemoved,
const TopTools_IndexedMapOfShape& theMEInside,
TopTools_MapOfShape& theMEUseInRebuild)
{
TopoDS_Compound aCEInv;
TopTools_IndexedMapOfShape aMEInv;
BRep_Builder aBB;
aBB.MakeCompound (aCEInv);
TopTools_ListIteratorOfListOfShape aItLF;
//
Standard_Integer i, aNb = myInvalidFaces.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopTools_ListOfShape& aLFInv = myInvalidFaces (i);
aItLF.Initialize (aLFInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
TopExp::MapShapes (aFIm, TopAbs_EDGE, aMEInv);
//
TopExp_Explorer aExpE (aFIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aE = aExpE.Current();
if (myInvalidEdges.Contains (aE))
{
aBB.Add (aCEInv, aE);
}
}
}
}
//
// remove edges which have been marked for removal
TopTools_IndexedMapOfShape aMEInvToAvoid;
TopTools_ListOfShape aLCBE;
BOPTools_AlgoTools::MakeConnexityBlocks (aCEInv, TopAbs_VERTEX, TopAbs_EDGE, aLCBE);
//
TopTools_ListIteratorOfListOfShape aItLCBE (aLCBE);
for (; aItLCBE.More(); aItLCBE.Next())
{
const TopoDS_Shape& aCBE = aItLCBE.Value();
TopExp_Explorer aExpCB (aCBE, TopAbs_EDGE);
for (; aExpCB.More(); aExpCB.Next())
{
const TopoDS_Shape& aE = aExpCB.Current();
if (!theMERemoved.Contains (aE))
{
break;
}
}
//
if (!aExpCB.More())
{
TopExp::MapShapes (aCBE, TopAbs_EDGE, aMEInvToAvoid);
}
}
//
TopTools_IndexedMapOfShape aReallyInvEdges;
//
aNb = myInvalidFaces.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
if (myArtInvalidFaces.IsBound (aF))
{
if (const TopTools_IndexedMapOfShape* aMIE = theDMFMIE.Seek (aF))
{
const Standard_Integer aNbIE = aMIE->Extent();
for (Standard_Integer iE = 1; iE <= aNbIE; ++iE)
{
const TopoDS_Shape& aE = aMIE->FindKey (iE);
if (aMEInv.Contains (aE) && !aMEInvToAvoid.Contains (aE))
{
aReallyInvEdges.Add (aE);
}
}
}
}
else
{
const TopTools_ListOfShape& aLFInv = myInvalidFaces (i);
aItLF.Initialize (aLFInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
TopExp_Explorer aExpE (aFIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aE = aExpE.Current();
if (myInvalidEdges.Contains (aE) && !aMEInvToAvoid.Contains (aE))
{
aReallyInvEdges.Add (aE);
}
}
}
}
}
myInvalidEdges = aReallyInvEdges;
// Check if any of the currently invalid edges may be used for
// rebuilding splits of invalid faces.
// For that the edge should be inside and not connected to invalid
// boundary edges of the same origin.
aNb = myInvalidEdges.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aE = myInvalidEdges (i);
if (!theMEInside.Contains (aE) || !myValidEdges.Contains (aE))
{
continue;
}
const TopTools_ListOfShape* pEOrigins = myOEOrigins.Seek (aE);
if (!pEOrigins)
{
continue;
}
Standard_Boolean bHasInvOutside = Standard_False;
for (TopTools_ListOfShape::Iterator itEOr (*pEOrigins); !bHasInvOutside && itEOr.More(); itEOr.Next())
{
if (const TopTools_ListOfShape* pEIms = myOEImages.Seek (itEOr.Value()))
{
for (TopTools_ListOfShape::Iterator itEIms (*pEIms); !bHasInvOutside && itEIms.More(); itEIms.Next())
{
bHasInvOutside = myInvalidEdges.Contains (itEIms.Value()) && !theMEInside.Contains (itEIms.Value());
}
}
}
if (!bHasInvOutside)
{
theMEUseInRebuild.Add (aE);
}
}
}
//=======================================================================
//function : FindFacesToRebuild
//purpose : Looking for the faces that have to be rebuilt:
// 1. Faces close to invalidity
// 2. Faces containing some invalid parts
//=======================================================================
void BRepOffset_BuildOffsetFaces::FindFacesToRebuild()
{
Standard_Integer i, aNb = myOFImages.Extent();
if (!aNb)
{
return;
}
//
Standard_Boolean bRebuild;
TopTools_ListIteratorOfListOfShape aItLF;
TopTools_ListOfShape aLEValid;
TopTools_MapOfShape aMFence, aMEReb, aMFReb;
TopExp_Explorer aExp;
//
TopTools_DataMapOfShapeListOfShape aDMFLV;
// get edges from invalid faces
aNb = myInvalidFaces.Extent();
for (i = 1; i <= aNb; i++)
{
const TopoDS_Shape& aF = myInvalidFaces.FindKey (i);
aMFence.Clear();
TopTools_ListOfShape aLVAvoid;
const TopTools_ListOfShape& aLFIm = myInvalidFaces (i);
aItLF.Initialize (aLFIm);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
aExp.Init (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
aMEReb.Add (aE);
if (myInvalidEdges.Contains (aE))
{
TopExp_Explorer aExpV (aE, TopAbs_VERTEX);
for (; aExpV.More(); aExpV.Next())
{
const TopoDS_Shape& aV = aExpV.Current();
if (aMFence.Add (aV))
{
aLVAvoid.Append (aV);
aMEReb.Add (aV);
}
}
}
}
}
//
if (aLVAvoid.Extent())
{
aDMFLV.Bind (aF, aLVAvoid);
}
//
const TopTools_ListOfShape* pLF = !myArtInvalidFaces.IsBound(aF) ? mySSInterfs.Seek (aF) : mySSInterfsArt.Seek(aF);
if (pLF)
{
TopTools_ListIteratorOfListOfShape aItLFE (*pLF);
for (; aItLFE.More(); aItLFE.Next())
{
const TopoDS_Shape& aFE = aItLFE.Value();
aMFReb.Add (aFE);
}
}
}
//
// get face to rebuild
aNb = myOFImages.Extent();
for (i = 1; i <= aNb; i++)
{
const TopoDS_Shape& aF = myOFImages.FindKey (i);
const TopTools_ListOfShape& aLFIm = myOFImages (i);
TopTools_MapOfShape aMVAvoid;
if (aDMFLV.IsBound (aF))
{
const TopTools_ListOfShape& aLVAvoid = aDMFLV.Find (aF);
TopTools_ListIteratorOfListOfShape aItLV (aLVAvoid);
for (; aItLV.More(); aItLV.Next())
{
const TopoDS_Shape& aV = aItLV.Value();
aMVAvoid.Add (aV);
}
}
//
bRebuild = aMFReb.Contains (aF);
aLEValid.Clear();
aMFence.Clear();
//
aItLF.Initialize (aLFIm);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
aExp.Init (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Edge& anEIm = TopoDS::Edge (aExp.Current());
if (!myInvalidEdges.Contains (anEIm))
{
if (aMFence.Add (anEIm))
{
aLEValid.Append (anEIm);
}
}
//
if (!bRebuild)
{
bRebuild = aMEReb.Contains (anEIm);
}
//
if (!bRebuild)
{
// check vertices
TopExp_Explorer aExpV (anEIm, TopAbs_VERTEX);
for (; aExpV.More() && !bRebuild; aExpV.Next())
{
const TopoDS_Shape& aV = aExpV.Current();
if (!aMVAvoid.Contains (aV))
{
bRebuild = aMEReb.Contains (aV);
}
}
}
}
}
//
if (!bRebuild)
{
bRebuild = aLFIm.Extent() && myInvalidFaces.Contains (aF);
if (bRebuild)
{
myFSelfRebAvoid.Add (aF);
}
}
//
if (bRebuild)
{
myFacesToRebuild.Add (aF, aLEValid);
}
}
}
namespace
{
//=======================================================================
//function : mapShapes
//purpose : Collect theVecShapes into theMap with setted theType
//=======================================================================
template<class Container>
static void mapShapes (const Container& theVecShapes,
const TopAbs_ShapeEnum theType,
TopTools_MapOfShape& theMap)
{
for (const auto& aShape : theVecShapes)
{
for (TopExp_Explorer anExp(aShape, theType); anExp.More(); anExp.Next())
{
theMap.Add(anExp.Current());
}
}
}
}
//=======================================================================
//function : IntersectFaces
//purpose : Intersection of the faces that should be rebuild to resolve all invalidities
//=======================================================================
void BRepOffset_BuildOffsetFaces::IntersectFaces (TopTools_MapOfShape& theVertsToAvoid,
const Message_ProgressRange& theRange)
{
Standard_Integer aNbFR = myFacesToRebuild.Extent();
if (!aNbFR)
{
return;
}
Message_ProgressScope aPSOuter (theRange, "Rebuilding invalid faces", 10);
//
Standard_Integer i, j, k, aNbInv;
TopTools_ListIteratorOfListOfShape aItLF, aItLE;
//
// get vertices from invalid edges
TopTools_MapOfShape aMVInv, aMVInvAll;
aNbInv = myInvalidEdges.Extent();
for (i = 1; i <= aNbInv; ++i)
{
const TopoDS_Shape& aEInv = myInvalidEdges (i);
Standard_Boolean bValid = myValidEdges.Contains (aEInv);
for (TopExp_Explorer aExp (aEInv, TopAbs_VERTEX); aExp.More(); aExp.Next())
{
const TopoDS_Shape& aV = aExp.Current();
aMVInvAll.Add (aV);
if (!bValid)
{
aMVInv.Add (aV);
}
}
}
//
Standard_Boolean bLookVertToAvoid = (aMVInv.Extent() > 0);
//
TopTools_DataMapOfShapeListOfShape aDMSF, aMDone, aMEInfETrim, aDMVEFull;
TopTools_IndexedDataMapOfShapeListOfShape aFLE, aDMEFInv;
//
// Add all faces to rebuild to outgoing map <aFLE>,
// plus link edges and vertices to the faces to
// define intersection faces
PrepareFacesForIntersection (bLookVertToAvoid, aFLE, aMDone, aDMSF, aMEInfETrim, aDMVEFull, aDMEFInv);
// Find vertices to avoid while trimming the edges.
// These vertices are taken from the invalid edges common between
// splits of different invalid, but not artificially, faces.
// Additional condition for these vertices is that all
// edges adjacent to this vertex must be either invalid
// or contained in invalid faces
TopTools_MapOfShape aMVRInv = theVertsToAvoid;
FindVerticesToAvoid (aDMEFInv, aDMVEFull, aMVRInv);
//
aPSOuter.Next();
if (!aPSOuter.More())
{
return;
}
// The faces should be intersected selectively -
// intersect only faces neighboring to the same invalid face
// and connected to its invalid edges;
// when dealing with artificially invalid faces for intersection to be
// complete we need to use not only invalid edges, but also the
// edges connected to invalid ones
// find blocks of artificially invalid faces
TopTools_DataMapOfShapeShape aDMFImF;
TopoDS_Compound aCFArt;
BRep_Builder().MakeCompound (aCFArt);
BRepOffset_DataMapOfShapeIndexedMapOfShape::Iterator aItM (myArtInvalidFaces);
for (; aItM.More(); aItM.Next())
{
const TopoDS_Shape& aF = aItM.Key();
const TopTools_ListOfShape& aLFInv = myInvalidFaces.FindFromKey (aF);
aItLF.Initialize (aLFInv);
for (; aItLF.More(); aItLF.Next())
{
BRep_Builder().Add (aCFArt, aItLF.Value());
aDMFImF.Bind (aItLF.Value(), aF);
}
}
//
// make connexity blocks
TopTools_ListOfShape aLCBArt;
BOPTools_AlgoTools::MakeConnexityBlocks (aCFArt, TopAbs_VERTEX, TopAbs_FACE, aLCBArt);
//
// alone edges
TopTools_MapOfShape aMEAlone, aMEInvOnArt;
//
Message_ProgressScope aPSArt (aPSOuter.Next(), NULL, aLCBArt.Extent());
TopTools_ListIteratorOfListOfShape aItLCBArt (aLCBArt);
for (; aItLCBArt.More(); aItLCBArt.Next(), aPSArt.Next())
{
if (!aPSArt.More())
{
return;
}
const TopoDS_Shape& aCB = aItLCBArt.Value();
//
// check if aCB contains splits of only one offset face
TopTools_MapOfShape aMFArt;
TopExp_Explorer aExpF (aCB, TopAbs_FACE);
for (; aExpF.More(); aExpF.Next())
{
aMFArt.Add (aDMFImF.Find (aExpF.Current()));
}
//
Standard_Boolean bAlone = (aMFArt.Extent() == 1);
//
// vertices on invalid edges
TopTools_MapOfShape aMVEInv;
TopTools_MapOfShape aMFence;
// edges that should not be marked as alone - edges having same origins as invalid ones
TopTools_MapOfShape aMEAvoid;
// map to find alone edges by looking for free vertices
TopTools_IndexedDataMapOfShapeListOfShape aDMVEVal;
//
TopExp_Explorer aExpE (aCB, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aE = aExpE.Current();
if (myInvalidEdges.Contains (aE))
{
aMEInvOnArt.Add (aE);
for (TopoDS_Iterator aItV (aE); aItV.More(); aItV.Next())
{
aMVEInv.Add (aItV.Value());
}
//
if (bAlone)
{
const TopTools_ListOfShape* pLEOr = myOEOrigins.Seek (aE);
if (pLEOr)
{
TopTools_ListIteratorOfListOfShape aItLEOr (*pLEOr);
for (; aItLEOr.More(); aItLEOr.Next())
{
TopTools_ListIteratorOfListOfShape aItLEIm (myOEImages.Find (aItLEOr.Value()));
for (; aItLEIm.More(); aItLEIm.Next())
{
aMEAvoid.Add (aItLEIm.Value());
}
}
}
}
continue;
}
//
if (aMFence.Add (aE))
{
TopExp::MapShapesAndAncestors (aE, TopAbs_VERTEX, TopAbs_EDGE, aDMVEVal);
}
}
//
// find edges with free vertices
Standard_Integer aNbV = aDMVEVal.Extent();
for (i = 1; i <= aNbV; ++i)
{
const TopoDS_Shape& aV = aDMVEVal.FindKey (i);
if (!aMVEInv.Contains (aV))
{
continue;
}
//
const TopTools_ListOfShape& aLEV = aDMVEVal (i);
if (aLEV.Extent() > 1)
{
continue;
}
//
const TopoDS_Shape& aE = aLEV.First();
if (aMEAvoid.Contains (aE))
{
continue;
}
//
aMEAlone.Add (aE);
//
// if this alone edge adds nothing to the intersection list
// it means that the origin of this edge has been split and we need to
// add the neighboring images of the same origins
if (aDMSF.Find (aE).Extent() > 1)
{
continue;
}
//
// check also its vertices
TopoDS_Iterator aItE (aE);
for (; aItE.More(); aItE.Next())
{
const TopoDS_Shape& aVE = aItE.Value();
if (aDMSF.Find (aVE).Extent() > 2)
{
break;
}
}
//
if (aItE.More())
{
continue;
}
//
// the edge is useless - look for other images
const TopTools_ListOfShape* pLEOr = myOEOrigins.Seek (aE);
if (!pLEOr)
{
continue;
}
//
TopTools_ListIteratorOfListOfShape aItLEOr (*pLEOr);
for (; aItLEOr.More(); aItLEOr.Next())
{
const TopoDS_Shape& aEOr = aItLEOr.Value();
//
const TopTools_ListOfShape& aLEIm = myOEImages.Find (aEOr);
TopTools_ListIteratorOfListOfShape aItLEIm (aLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
//
if (aMFence.Contains (aEIm))
{
aMEAlone.Add (aEIm);
}
}
}
}
}
//
// Get all invalidities from all faces to be used for avoiding
// repeated usage of the common edges
TopTools_MapOfShape aMAllInvs;
aNbInv = myInvalidFaces.Extent();
for (k = 1; k <= aNbInv; ++k)
{
TopTools_ListIteratorOfListOfShape aIt (myInvalidFaces (k));
for (; aIt.More(); aIt.Next())
{
TopExp_Explorer aExp (aIt.Value(), TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (myInvalidEdges.Contains (aE) || aMEAlone.Contains (aE))
{
aMAllInvs.Add (aE);
TopoDS_Iterator aItV (aE);
for (; aItV.More(); aItV.Next())
{
aMAllInvs.Add (aItV.Value());
}
}
}
}
}
//
// Bounding vertices of not trimmed edges
TopTools_MapOfShape aMVBounds;
//
TopTools_MapOfShape aMECheckExt;
// Save connections between not trimmed edge and its trimmed parts
TopTools_DataMapOfShapeListOfShape aDMEETrim;
// Splits of the new edges
TopTools_DataMapOfShapeListOfShape aEImages;
BRep_Builder aBB;
// Keep connection between blocks of invalid edges to the lists of
// found edges to be intersected for its treatment
TopTools_IndexedDataMapOfShapeListOfShape aDMOENEdges;
aNbInv = myInvalidFaces.Extent();
Message_ProgressScope aPSInter (aPSOuter.Next (5), NULL, aNbInv);
for (k = 1; k <= aNbInv; ++k)
{
if (!aPSInter.More())
{
return;
}
const TopoDS_Shape& aFInv = myInvalidFaces.FindKey (k);
Standard_Boolean bSelfRebAvoid = myFSelfRebAvoid.Contains (aFInv);
const TopTools_ListOfShape& aLFInv = myInvalidFaces (k);
//
TopTools_ListOfShape aLCB;
if (aLFInv.Extent() > 1)
{
// make compound of invalid faces
TopoDS_Compound aCFInv;
aBB.MakeCompound (aCFInv);
//
TopTools_ListIteratorOfListOfShape aIt (aLFInv);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aFIm = aIt.Value();
aBB.Add (aCFInv, aFIm);
}
//
// make connexity blocks
BOPTools_AlgoTools::MakeConnexityBlocks (aCFInv, TopAbs_EDGE, TopAbs_FACE, aLCB);
}
else
{
aLCB = aLFInv;
}
//
Message_ProgressScope aPSCB (aPSInter.Next(), NULL, aLCB.Extent());
Standard_Boolean bArtificial = myArtInvalidFaces.IsBound (aFInv);
TopTools_ListIteratorOfListOfShape aItLCB (aLCB);
for (; aItLCB.More(); aItLCB.Next())
{
if (!aPSCB.More())
{
return;
}
const TopoDS_Shape& aCBInv = aItLCB.Value();
//
TopTools_MapOfShape aMEFence;
//
TopoDS_Compound aCBE;
aBB.MakeCompound (aCBE);
//
// remember inside edges and vertices to further check
TopTools_MapOfShape anInsideEdges;
TopTools_MapOfShape anInsideVertices;
TopExp_Explorer aExp(aCBInv, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (myInvalidEdges.Contains (aE) || (bArtificial && aMEAlone.Contains (aE)))
{
if (aMEFence.Add (aE))
{
aBB.Add (aCBE, aE);
if (!myEdgesToAvoid.Contains(aE) && myInvalidEdges.Contains(aE))
{
anInsideEdges.Add(aE);
TopoDS_Iterator anIt(aE);
for (; anIt.More(); anIt.Next())
{
anInsideVertices.Add(anIt.Value());
}
}
}
}
}
//
// make connexity blocks of edges
TopTools_ListOfShape aLCBE;
BOPTools_AlgoTools::MakeConnexityBlocks (aCBE, TopAbs_VERTEX, TopAbs_EDGE, aLCBE);
//
Message_ProgressScope aPSCBE (aPSCB.Next(), NULL, aLCBE.Extent());
TopTools_ListIteratorOfListOfShape aItLCBE (aLCBE);
for (; aItLCBE.More(); aItLCBE.Next())
{
if (!aPSCBE.More())
{
return;
}
const TopoDS_Shape& aCBELoc = aItLCBE.Value();
//
// map of edges and vertices of processing invalidity
TopTools_IndexedMapOfShape aME;
// map of vertices to trim the new edges
TopTools_IndexedMapOfShape aMECV;
TopExp::MapShapes (aCBELoc, TopAbs_EDGE, aME);
aMECV = aME;
TopExp::MapShapes (aCBELoc, TopAbs_VERTEX, aME);
TopTools_IndexedMapOfShape aMFInt;
// additional faces for intersection
TopTools_IndexedMapOfShape aMFIntExt;
// splits of faces for intersection
TopTools_ListOfShape aLFInt;
// faces to avoid intersection
TopTools_IndexedMapOfShape aMFAvoid;
//
FindFacesForIntersection (aFInv, aME, aDMSF, aMVInvAll, bArtificial, aMFAvoid, aMFInt, aMFIntExt, aLFInt);
if (aMFInt.Extent() < 3)
{
// nothing to intersect
aPSCBE.Next();
continue;
}
//
const BRepOffset_DataMapOfShapeMapOfShape* pMFInter = myIntersectionPairs.Seek (aFInv);
// intersect the faces, but do not intersect the invalid ones
// among each other (except for the artificially invalid faces)
TopTools_IndexedMapOfShape aMEToInt;
Standard_Integer aNb = aMFInt.Extent();
Message_ProgressScope aPSIntPair (aPSCBE.Next(), NULL, aNb);
for (i = 1; i <= aNb; ++i, aPSIntPair.Next())
{
if (!aPSIntPair.More())
{
return;
}
const TopoDS_Face& aFi = TopoDS::Face (aMFInt (i));
if (bSelfRebAvoid && aFi.IsSame (aFInv))
{
continue;
}
//
const TopTools_ListOfShape* aLFImi = myOFImages.Seek (aFi);
if (!aLFImi)
continue;
//
TopTools_ListOfShape* aLFEi = aFLE.ChangeSeek (aFi);
if (!aLFEi)
continue;
//
TopTools_ListOfShape& aLFDone = aMDone.ChangeFind (aFi);
//
const TopTools_MapOfShape* pInterFi = !pMFInter ? 0 : pMFInter->Seek (aFi);
if (pMFInter && !pInterFi)
continue;
// create map of edges and vertices for aLFImi
TopTools_MapOfShape aMEVIm;
mapShapes(*aLFImi, TopAbs_EDGE, aMEVIm);
mapShapes(*aLFImi, TopAbs_VERTEX, aMEVIm);
Standard_Boolean isIContainsE = aMEVIm.HasIntersection(anInsideEdges);
Standard_Boolean isIContainsV = aMEVIm.HasIntersection(anInsideVertices);
for (j = i + 1; j <= aNb; ++j)
{
const TopoDS_Face& aFj = TopoDS::Face (aMFInt (j));
if (bSelfRebAvoid && aFj.IsSame (aFInv))
{
continue;
}
//
if (pInterFi && !pInterFi->Contains (aFj))
continue;
const TopTools_ListOfShape* aLFImj = myOFImages.Seek(aFj);
if (!aLFImj)
continue;
//
TopTools_ListOfShape* aLFEj = aFLE.ChangeSeek (aFj);
if (!aLFEj)
continue;
// create map of edges and vertices for aLFImi
aMEVIm.Clear();
mapShapes(*aLFImj, TopAbs_EDGE, aMEVIm);
mapShapes(*aLFImj, TopAbs_VERTEX, aMEVIm);
// check images of both faces contain anInsideEdges and anInsideVertices
// not process if false and true
Standard_Boolean isJContainsE = aMEVIm.HasIntersection(anInsideEdges);
Standard_Boolean isJContainsV = aMEVIm.HasIntersection(anInsideVertices);
// Check if one face is connected to inside edge then
// the other must be also connected
if ((isIContainsE && !isJContainsV) ||
(isJContainsE && !isIContainsV))
{
TopTools_ListOfShape aLVC;
// it is necessary to process the images if they already have
// common vertices
FindCommonParts(*aLFImi, *aLFImj, aLVC, TopAbs_VERTEX);
if (aLVC.IsEmpty())
continue;
}
//
// if there are some common edges between faces
// we should use these edges and do not intersect again.
TopTools_ListOfShape aLEC;
FindCommonParts (*aLFImi, *aLFImj, aLEC);
//
if (aLEC.Extent())
{
// no need to intersect if we have common edges between faces
Standard_Boolean bForceUse = aMFIntExt.Contains (aFi) || aMFIntExt.Contains (aFj);
ProcessCommonEdges (aLEC, aME, aMEInfETrim, aMAllInvs, bForceUse, aMECV, aMECheckExt, aDMEETrim, *aLFEi, *aLFEj, aMEToInt);
// Add common vertices not belonging to the common edges for trimming the intersection edges
TopTools_IndexedMapOfShape aMVOnCE;
for (TopTools_ListOfShape::Iterator itE (aLEC); itE.More(); itE.Next())
{
TopExp::MapShapes (itE.Value(), TopAbs_VERTEX, aMVOnCE);
}
TopTools_ListOfShape aLEV;
FindCommonParts (*aLFImi, *aLFImj, aLEV, TopAbs_VERTEX);
for (TopTools_ListOfShape::Iterator itV (aLEV); itV.More(); itV.Next())
{
if (!aMVOnCE.Contains (itV.Value()))
{
aMECV.Add (itV.Value());
}
}
continue;
}
//
// check if both these faces are invalid and sharing edges
if (myInvalidFaces.Contains (aFi) && myInvalidFaces.Contains (aFj) &&
!myArtInvalidFaces.IsBound (aFi) && !myArtInvalidFaces.IsBound (aFj))
{
continue;
}
//
// check if these two faces have already been treated
aItLE.Initialize (aLFDone);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aF = aItLE.Value();
if (aF.IsSame (aFj))
{
break;
}
}
//
if (aItLE.More())
{
// use intersection line obtained on the previous steps
// plus, find new origins for these lines
UpdateIntersectedFaces (aFInv, aFi, aFj, aLFInv, *aLFImi, *aLFImj, *aLFEi, *aLFEj, aMEToInt);
continue;
}
//
if (aMFAvoid.Contains (aFi) || aMFAvoid.Contains (aFj))
{
continue;
}
//
aLFDone.Append (aFj);
aMDone.ChangeFind (aFj).Append (aFi);
//
IntersectFaces (aFInv, aFi, aFj, aLFInv, *aLFImi, *aLFImj, *aLFEi, *aLFEj, aMECV, aMEToInt);
}
}
//
// intersect and trim edges for this chain
IntersectAndTrimEdges (aMFInt, aMEToInt, aDMEETrim, aME, aMECV,
aMVInv, aMVRInv, aMECheckExt, bArtificial ? &mySSInterfsArt : 0,
aMVBounds, aEImages);
//
Standard_Integer iE, aNbEToInt = aMEToInt.Extent();
for (iE = 1; iE <= aNbEToInt; ++iE)
{
const TopoDS_Shape& aEInt = aMEToInt (iE);
TopExp_Explorer anExpE (aCBELoc, TopAbs_EDGE);
for (; anExpE.More(); anExpE.Next())
{
const TopoDS_Shape& aE = anExpE.Current();
TopTools_ListOfShape* pLEToInt = aDMOENEdges.ChangeSeek (aE);
if (!pLEToInt)
pLEToInt = &aDMOENEdges (aDMOENEdges.Add (aE, TopTools_ListOfShape()));
AppendToList (*pLEToInt, aEInt);
}
}
}
}
}
//
// filter the obtained edges
UpdateValidEdges (aFLE, aDMOENEdges, aMVBounds, aMEInvOnArt, aMECheckExt,
theVertsToAvoid, aEImages, aDMEETrim, aPSOuter.Next (3));
}
//=======================================================================
//function : PrepareFacesForIntersection
//purpose : Preparation of the maps for analyzing intersections of the faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::PrepareFacesForIntersection (const Standard_Boolean theLookVertToAvoid,
TopTools_IndexedDataMapOfShapeListOfShape& theFLE,
TopTools_DataMapOfShapeListOfShape& theMDone,
TopTools_DataMapOfShapeListOfShape& theDMSF,
TopTools_DataMapOfShapeListOfShape& theMEInfETrim,
TopTools_DataMapOfShapeListOfShape& theDMVEFull,
TopTools_IndexedDataMapOfShapeListOfShape& theDMEFInv)
{
Standard_Integer i, aNb = myFacesToRebuild.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = myFacesToRebuild.FindKey (i);
//
TopTools_ListOfShape aLE;
theFLE.Add (aF, aLE);
theMDone.Bind (aF, aLE);
//
const TopTools_ListOfShape& aLFIm = myOFImages.FindFromKey (aF);
TopTools_ListIteratorOfListOfShape aItLF (aLFIm);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFIm = aItLF.Value();
TopExp_Explorer aExp (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
// save connection to untrimmed face
TopTools_ListOfShape* pLF = theDMSF.ChangeSeek (aE);
if (!pLF)
{
pLF = theDMSF.Bound (aE, TopTools_ListOfShape());
}
AppendToList (*pLF, aF);
//
// save connection to untrimmed edge
const TopoDS_Shape& aEInf = myETrimEInf->Find (aE);
TopTools_ListOfShape* pLETrim = theMEInfETrim.ChangeSeek (aEInf);
if (!pLETrim)
{
pLETrim = theMEInfETrim.Bound (aEInf, TopTools_ListOfShape());
}
AppendToList (*pLETrim, aE);
//
TopExp_Explorer aExpV (aE, TopAbs_VERTEX);
for (; aExpV.More(); aExpV.Next())
{
const TopoDS_Shape& aV = aExpV.Current();
// save connection to face
TopTools_ListOfShape* pLFV = theDMSF.ChangeSeek (aV);
if (!pLFV)
{
pLFV = theDMSF.Bound (aV, TopTools_ListOfShape());
}
AppendToList (*pLFV, aF);
//
if (theLookVertToAvoid)
{
// save connection to edges
TopTools_ListOfShape* pLEV = theDMVEFull.ChangeSeek (aV);
if (!pLEV)
{
pLEV = theDMVEFull.Bound (aV, TopTools_ListOfShape());
}
AppendToList (*pLEV, aE);
}
}
}
}
//
if (theLookVertToAvoid)
{
// get edges of invalid faces (from invalid splits only)
const TopTools_ListOfShape* pLFInv = myInvalidFaces.Seek (aF);
if (!pLFInv || myArtInvalidFaces.IsBound (aF))
{
continue;
}
//
aItLF.Initialize (*pLFInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aFInv = aItLF.Value();
TopExp_Explorer aExp (aFInv, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
TopTools_ListOfShape* pLF = theDMEFInv.ChangeSeek (aE);
if (!pLF)
{
pLF = &theDMEFInv (theDMEFInv.Add (aE, TopTools_ListOfShape()));
}
AppendToList (*pLF, aF);
}
}
}
}
}
//=======================================================================
//function : FindVerticesToAvoid
//purpose : Looking for the invalid vertices
//=======================================================================
void BRepOffset_BuildOffsetFaces::FindVerticesToAvoid (const TopTools_IndexedDataMapOfShapeListOfShape& theDMEFInv,
const TopTools_DataMapOfShapeListOfShape& theDMVEFull,
TopTools_MapOfShape& theMVRInv)
{
TopTools_MapOfShape aMFence;
Standard_Integer i, aNb = theDMEFInv.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopTools_ListOfShape& aLFInv = theDMEFInv (i);
if (aLFInv.Extent() == 1)
{
continue;
}
//
const TopoDS_Shape& aE = theDMEFInv.FindKey (i);
if (!myInvalidEdges.Contains (aE) || myValidEdges.Contains (aE))
{
continue;
}
if (!aMFence.Add (aE))
continue;
TopTools_IndexedDataMapOfShapeListOfShape aMVEEdges;
// Do not check the splitting vertices, but check only the ending ones
const TopTools_ListOfShape* pLEOr = myOEOrigins.Seek (aE);
if (pLEOr)
{
TopTools_ListIteratorOfListOfShape aItLEOr (*pLEOr);
for (; aItLEOr.More(); aItLEOr.Next())
{
const TopTools_ListOfShape& aLEIm = myOEImages.Find (aItLEOr.Value());
TopTools_ListIteratorOfListOfShape aItLEIm (aLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
aMFence.Add (aItLEIm.Value());
TopExp::MapShapesAndAncestors (aItLEIm.Value(), TopAbs_VERTEX, TopAbs_EDGE, aMVEEdges);
}
}
}
else
{
TopExp::MapShapesAndAncestors (aE, TopAbs_VERTEX, TopAbs_EDGE, aMVEEdges);
}
Standard_Integer j, aNbV = aMVEEdges.Extent();
for (j = 1; j <= aNbV; ++j)
{
if (aMVEEdges (j).Extent() != 1)
continue;
const TopoDS_Shape& aV = aMVEEdges.FindKey (j);
if (!aMFence.Add (aV))
continue;
const TopTools_ListOfShape* pLE = theDMVEFull.Seek (aV);
if (!pLE)
{
// isolated vertex
theMVRInv.Add (aV);
continue;
}
//
// If all edges sharing the vertex are either invalid or
// the vertex is connected to at least two inverted edges
// mark the vertex to be avoided in the new splits
Standard_Integer iNbEInverted = 0;
Standard_Boolean bAllEdgesInv = Standard_True;
TopTools_ListIteratorOfListOfShape aItLE (*pLE);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEV = aItLE.Value();
if (myInvertedEdges.Contains (aEV))
++iNbEInverted;
if (bAllEdgesInv)
bAllEdgesInv = myInvalidEdges.Contains (aEV);
}
if (iNbEInverted > 1 || bAllEdgesInv)
{
theMVRInv.Add (aV);
}
}
}
}
//=======================================================================
//function : FindFacesForIntersection
//purpose : Looking for the faces around each invalidity for intersection
//=======================================================================
void BRepOffset_BuildOffsetFaces::FindFacesForIntersection (const TopoDS_Shape& theFInv,
const TopTools_IndexedMapOfShape& theME,
const TopTools_DataMapOfShapeListOfShape& theDMSF,
const TopTools_MapOfShape& theMVInvAll,
const Standard_Boolean theArtCase,
TopTools_IndexedMapOfShape& theMFAvoid,
TopTools_IndexedMapOfShape& theMFInt,
TopTools_IndexedMapOfShape& theMFIntExt,
TopTools_ListOfShape& theLFImInt)
{
Standard_Integer i, aNbE = theME.Extent();
//
TopTools_IndexedMapOfShape aMShapes;
//
for (i = 1; i <= aNbE; ++i)
{
const TopoDS_Shape& aS = theME (i);
if (!theDMSF.IsBound (aS))
{
continue;
}
//
// in artificial case we intersect the faces which are close to invalidity
Standard_Boolean bAvoid = theArtCase ?
((aS.ShapeType() == TopAbs_VERTEX) && !theMVInvAll.Contains (aS)) : Standard_False;
//
const TopTools_ListOfShape& aLF = theDMSF.Find (aS);
TopTools_ListIteratorOfListOfShape aItLF (aLF);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aF = aItLF.Value();
if (theMFInt.Contains (aF))
{
continue;
}
//
if (bAvoid && myArtInvalidFaces.IsBound (aF))
{
theMFAvoid.Add (aF);
}
//
theMFInt.Add (aF);
//
Standard_Boolean bUse = !aF.IsSame (theFInv);
//
const TopTools_ListOfShape& aLFIm = myOFImages.FindFromKey (aF);
TopTools_ListIteratorOfListOfShape aItLFIm (aLFIm);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
theLFImInt.Append (aFIm);
if (bUse)
{
TopExp::MapShapes (aFIm, TopAbs_EDGE, aMShapes);
}
}
}
}
//
const TopTools_DataMapOfShapeListOfShape& aSSInterfsMap = theArtCase ? mySSInterfsArt : mySSInterfs;
const TopTools_ListOfShape* pLFInv = aSSInterfsMap.Seek (theFInv);
if (!pLFInv)
{
return;
}
//
TopTools_MapOfShape aMF;
TopTools_ListIteratorOfListOfShape aItLF (*pLFInv);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aF = aItLF.Value();
aMF.Add (aF);
}
//
// the faces should be unique in each place
TopoDS_Compound aCF;
BRep_Builder().MakeCompound (aCF);
//
TopTools_IndexedMapOfShape aMFToAdd;
TopTools_DataMapOfShapeShape aDMFOr;
//
for (i = 1; i <= aNbE; ++i)
{
const TopoDS_Shape& aS = theME (i);
const TopTools_ListOfShape* pLF = aSSInterfsMap.Seek (aS);
if (!pLF)
{
continue;
}
//
aItLF.Initialize (*pLF);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aF = aItLF.Value();
if (theMFInt.Contains (aF) || aMFToAdd.Contains (aF) || !aMF.Contains (aF))
{
continue;
}
//
// check if the face has some connection to already added for intersection faces
const TopTools_ListOfShape& aLFIm = myOFImages.FindFromKey (aF);
if (!theArtCase)
{
TopTools_ListIteratorOfListOfShape aItLFIm (aLFIm);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
TopExp_Explorer aExp (aFIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
if (aMShapes.Contains (aExp.Current()))
{
break;
}
}
if (aExp.More())
{
break;
}
}
if (!aItLFIm.More())
{
continue;
}
}
//
aMFToAdd.Add (aF);
TopTools_ListIteratorOfListOfShape aItLFIm (aLFIm);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
aDMFOr.Bind (aFIm, aF);
BRep_Builder().Add (aCF, aFIm);
}
}
}
//
if (aMFToAdd.IsEmpty())
{
return;
}
//
TopTools_ListOfShape aLCB;
BOPTools_AlgoTools::MakeConnexityBlocks (aCF, TopAbs_EDGE, TopAbs_FACE, aLCB);
//
if ((aLCB.Extent() == 1) && (aMFToAdd.Extent() > 1))
{
return;
}
//
TopTools_ListIteratorOfListOfShape aItLCB (aLCB);
for (; aItLCB.More(); aItLCB.Next())
{
const TopoDS_Shape& aCB = aItLCB.Value();
aMFToAdd.Clear();
TopExp_Explorer aExpF (aCB, TopAbs_FACE);
for (; aExpF.More(); aExpF.Next())
{
const TopoDS_Shape& aFIm = aExpF.Current();
aMFToAdd.Add (aDMFOr.Find (aFIm));
}
//
if (aMFToAdd.Extent() == 1)
{
const TopoDS_Shape& aF = aMFToAdd (1);
//
theMFInt.Add (aF);
theMFIntExt.Add (aF);
//
const TopTools_ListOfShape& aLFIm = myOFImages.FindFromKey (aF);
TopTools_ListIteratorOfListOfShape aItLFIm (aLFIm);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
theLFImInt.Append (aFIm);
}
}
}
}
//=======================================================================
//function : ProcessCommonEdges
//purpose : Analyzing the common edges between splits of offset faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::ProcessCommonEdges (const TopTools_ListOfShape& theLEC,
const TopTools_IndexedMapOfShape& theME,
const TopTools_DataMapOfShapeListOfShape& theMEInfETrim,
const TopTools_MapOfShape& theAllInvs,
const Standard_Boolean theForceUse,
TopTools_IndexedMapOfShape& theMECV,
TopTools_MapOfShape& theMECheckExt,
TopTools_DataMapOfShapeListOfShape& theDMEETrim,
TopTools_ListOfShape& theLFEi,
TopTools_ListOfShape& theLFEj,
TopTools_IndexedMapOfShape& theMEToInt)
{
TopTools_ListOfShape aLEC;
// process common edges
TopTools_ListIteratorOfListOfShape aItLE (theLEC);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEC = aItLE.Value();
//
// check first if common edges are valid
if (myInvalidEdges.Contains (aEC) && !myValidEdges.Contains (aEC))
{
continue;
}
//
// common edge should have connection to current invalidity
if (theME.Contains (aEC))
{
aLEC.Append (aEC);
continue;
}
//
TopoDS_Iterator aItV (aEC);
for (; aItV.More(); aItV.Next())
{
const TopoDS_Shape& aVE = aItV.Value();
if (theME.Contains (aVE))
{
aLEC.Append (aEC);
break;
}
}
}
//
Standard_Boolean bUseOnlyInf = aLEC.IsEmpty();
if (bUseOnlyInf)
{
if (theForceUse)
{
aLEC = theLEC;
}
else
{
aItLE.Initialize (theLEC);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEC = aItLE.Value();
// check if all images of the origin of this edge
// are not connected to any invalidity
const TopoDS_Shape& aEInt = myETrimEInf->Find (aEC);
const TopTools_ListOfShape& aLVE = theMEInfETrim.Find (aEInt);
TopTools_ListIteratorOfListOfShape aItLVE (aLVE);
for (; aItLVE.More(); aItLVE.Next())
{
const TopoDS_Shape& aECx = aItLVE.Value();
if (theAllInvs.Contains (aECx) || myInvalidEdges.Contains (aECx))
{
return;
}
//
TopoDS_Iterator aItV (aECx);
for (; aItV.More(); aItV.Next())
{
if (theAllInvs.Contains (aItV.Value()))
{
return;
}
}
}
// use only one element
if (aLEC.IsEmpty())
{
aLEC.Append (aEC);
}
}
}
}
//
aItLE.Initialize (aLEC);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEC = aItLE.Value();
//
const TopoDS_Shape& aEInt = myETrimEInf->Find (aEC);
if (!bUseOnlyInf)
{
// find the edges of the same original edge
// and take their vertices as well
const TopTools_ListOfShape& aLVE = theMEInfETrim.Find (aEInt);
TopTools_ListIteratorOfListOfShape aItLVE (aLVE);
for (; aItLVE.More(); aItLVE.Next())
{
const TopoDS_Shape& aECx = aItLVE.Value();
//
const TopTools_ListOfShape* pLEOr = myOEOrigins.Seek (aECx);
if (!pLEOr || (pLEOr->Extent() == 1))
{
TopExp::MapShapes (aECx, TopAbs_VERTEX, theMECV);
}
}
//
// bind unlimited edge to its trimmed part in face to update maps of
// images and origins in the future
TopTools_ListOfShape* pLTAdded = theDMEETrim.ChangeSeek (aEInt);
if (!pLTAdded)
{
pLTAdded = theDMEETrim.Bound (aEInt, TopTools_ListOfShape());
}
AppendToList (*pLTAdded, aEC);
}
else if (!theForceUse)
{
theMECheckExt.Add (aEInt);
}
//
AppendToList (theLFEi, aEInt);
AppendToList (theLFEj, aEInt);
theMEToInt.Add (aEInt);
}
}
namespace {
//=======================================================================
//function : FindOrigins
//purpose : Looking for the origin edges
//=======================================================================
static void FindOrigins (const TopTools_ListOfShape& theLFIm1,
const TopTools_ListOfShape& theLFIm2,
const TopTools_IndexedMapOfShape& theME,
const TopTools_DataMapOfShapeListOfShape& theOrigins,
TopTools_ListOfShape& theLEOr)
{
TopTools_MapOfShape aMFence;
for (Standard_Integer i = 0; i < 2; ++i)
{
const TopTools_ListOfShape& aLF = !i ? theLFIm1 : theLFIm2;
TopTools_ListOfShape::Iterator aIt (aLF);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aF = aIt.Value();
//
TopExp_Explorer aExp (aF, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
//
if (theME.Contains (aE) && theOrigins.IsBound (aE))
{
const TopTools_ListOfShape& aLEOr = theOrigins.Find (aE);
TopTools_ListOfShape::Iterator aItLE (aLEOr);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEOr = aItLE.Value();
//
if (aMFence.Add (aEOr) && (aEOr.ShapeType() == TopAbs_EDGE))
{
theLEOr.Append (aEOr);
}
}
}
}
}
}
}
}
//=======================================================================
//function : UpdateIntersectedFaces
//purpose : Updating the already interfered faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::UpdateIntersectedFaces (const TopoDS_Shape& theFInv,
const TopoDS_Shape& theFi,
const TopoDS_Shape& theFj,
const TopTools_ListOfShape& theLFInv,
const TopTools_ListOfShape& theLFImi,
const TopTools_ListOfShape& theLFImj,
const TopTools_ListOfShape& theLFEi,
const TopTools_ListOfShape& theLFEj,
TopTools_IndexedMapOfShape& theMEToInt)
{
// Find common edges in these two lists
TopTools_MapOfShape aMEi;
TopTools_ListIteratorOfListOfShape aItLE (theLFEi);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aE = aItLE.Value();
aMEi.Add (aE);
}
//
// find origins
TopTools_IndexedMapOfShape aMEToFindOrigins;
TopTools_ListOfShape aLEToFindOrigins;
if (!theFi.IsSame (theFInv))
{
FindCommonParts (theLFImi, theLFInv, aLEToFindOrigins);
}
if (!theFj.IsSame (theFInv))
{
FindCommonParts (theLFImj, theLFInv, aLEToFindOrigins);
}
//
TopTools_ListOfShape aLEOrInit;
aItLE.Initialize (aLEToFindOrigins);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEC = aItLE.Value();
aMEToFindOrigins.Add (aEC);
}
//
FindOrigins (theLFImi, theLFImj, aMEToFindOrigins, *myEdgesOrigins, aLEOrInit);
//
aItLE.Initialize (theLFEj);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aE = aItLE.Value();
if (aMEi.Contains (aE))
{
theMEToInt.Add (aE);
if (aLEOrInit.Extent())
{
if (myEdgesOrigins->IsBound (aE))
{
TopTools_ListOfShape& aLEOr = myEdgesOrigins->ChangeFind (aE);
TopTools_ListIteratorOfListOfShape aItLEOr (aLEOrInit);
for (; aItLEOr.More(); aItLEOr.Next())
{
const TopoDS_Shape& aEOr = aItLEOr.Value();
AppendToList (aLEOr, aEOr);
}
}
else
{
myEdgesOrigins->Bind (aE, aLEOrInit);
}
}
}
}
}
//=======================================================================
//function : IntersectFaces
//purpose : Intersection of the pair of faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::IntersectFaces (const TopoDS_Shape& theFInv,
const TopoDS_Shape& theFi,
const TopoDS_Shape& theFj,
const TopTools_ListOfShape& theLFInv,
const TopTools_ListOfShape& theLFImi,
const TopTools_ListOfShape& theLFImj,
TopTools_ListOfShape& theLFEi,
TopTools_ListOfShape& theLFEj,
TopTools_IndexedMapOfShape& theMECV,
TopTools_IndexedMapOfShape& theMEToInt)
{
// intersect faces
TopAbs_State aSide = TopAbs_OUT;
TopTools_ListOfShape aLInt1, aLInt2;
TopoDS_Edge aNullEdge;
TopoDS_Face aNullFace;
BRepOffset_Tool::Inter3D (TopoDS::Face (theFi), TopoDS::Face (theFj), aLInt1, aLInt2, aSide,
aNullEdge, aNullFace, aNullFace);
//
if (aLInt1.IsEmpty())
{
return;
}
//
// find common vertices for trimming edges
TopTools_ListOfShape aLCV;
TopTools_ListIteratorOfListOfShape aItLE;
FindCommonParts (theLFImi, theLFImj, aLCV, TopAbs_VERTEX);
if (aLCV.Extent() > 1)
{
aItLE.Initialize (aLCV);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aCV = aItLE.Value();
theMECV.Add (aCV);
}
}
//
// find origins
TopTools_IndexedMapOfShape aMEToFindOrigins;
TopTools_ListOfShape aLEToFindOrigins;
if (!theFi.IsSame (theFInv))
{
FindCommonParts (theLFImi, theLFInv, aLEToFindOrigins);
}
if (!theFj.IsSame (theFInv))
{
FindCommonParts (theLFImj, theLFInv, aLEToFindOrigins);
}
TopTools_ListOfShape aLEOrInit;
aItLE.Initialize (aLEToFindOrigins);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEC = aItLE.Value();
aMEToFindOrigins.Add (aEC);
}
//
FindOrigins (theLFImi, theLFImj, aMEToFindOrigins, *myEdgesOrigins, aLEOrInit);
//
aItLE.Initialize (aLInt1);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEInt = aItLE.Value();
theLFEi.Append (aEInt);
theLFEj.Append (aEInt);
//
if (aLEOrInit.Extent())
{
myEdgesOrigins->Bind (aEInt, aLEOrInit);
}
//
theMEToInt.Add (aEInt);
}
}
//=======================================================================
//function : IntersectAndTrimEdges
//purpose : Intersection of the new intersection edges among themselves
//=======================================================================
void BRepOffset_BuildOffsetFaces::IntersectAndTrimEdges (const TopTools_IndexedMapOfShape& theMFInt,
const TopTools_IndexedMapOfShape& theMEInt,
const TopTools_DataMapOfShapeListOfShape& theDMEETrim,
const TopTools_IndexedMapOfShape& theMSInv,
const TopTools_IndexedMapOfShape& theMVE,
const TopTools_MapOfShape& theVertsToAvoid,
const TopTools_MapOfShape& theNewVertsToAvoid,
const TopTools_MapOfShape& theMECheckExt,
const TopTools_DataMapOfShapeListOfShape* theSSInterfs,
TopTools_MapOfShape& theMVBounds,
TopTools_DataMapOfShapeListOfShape& theEImages)
{
Standard_Integer i, aNb = theMEInt.Extent();
if (!aNb)
{
return;
}
//
TopTools_ListOfShape aLArgs;
TopTools_MapOfShape aMFence;
TopTools_ListIteratorOfListOfShape aIt, aIt1;
TopExp_Explorer aExp;
//
// get vertices from the splits of intersected faces.
// vertices are taken from the edges close to invalidity
//
TopTools_IndexedDataMapOfShapeListOfShape aDMVE;
aNb = theMFInt.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aF = theMFInt (i);
const TopTools_ListOfShape& aLE = myFacesToRebuild.FindFromKey (aF);
//
aIt.Initialize (aLE);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
TopExp::MapShapesAndAncestors (aE, TopAbs_VERTEX, TopAbs_EDGE, aDMVE);
//
aExp.Init (aE, TopAbs_VERTEX);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aV1 = aExp.Current();
if (!theVertsToAvoid.Contains (aV1) && theMVE.Contains (aV1) && aMFence.Add (aV1))
{
aLArgs.Append (aV1);
}
}
}
}
//
aNb = theMSInv.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aS = theMSInv(i);
// edge case
if (theSSInterfs)
{
if (const TopTools_ListOfShape* pLV = theSSInterfs->Seek (aS))
{
// Add vertices from intersection info to trim section edges of artificial faces
for (TopTools_ListOfShape::Iterator itLV (*pLV); itLV.More(); itLV.Next())
{
if (itLV.Value().ShapeType() == TopAbs_VERTEX)
{
aLArgs.Append (itLV.Value());
}
}
}
}
// vertex case
if (const TopTools_ListOfShape* pLVE = aDMVE.ChangeSeek(aS))
{
aIt.Initialize(*pLVE);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
//
aExp.Init(aE, TopAbs_VERTEX);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aV1 = aExp.Current();
if (!theVertsToAvoid.Contains(aV1) && aMFence.Add(aV1))
{
aLArgs.Append(aV1);
}
}
}
}
}
//
// bounding vertices of untrimmed edges
TopTools_ListOfShape aLVBounds;
// new intersection edges
TopTools_ListOfShape aLENew;
// get edges to intersect
TopTools_ListOfShape aLEInt;
// Common intersection edges. Should be intersected separately
TopTools_ListOfShape aLCE;
//
aNb = theMEInt.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aE = theMEInt (i);
if (theMECheckExt.Contains (aE))
{
// avoid trimming of the intersection edges by additional common edges
aLCE.Append (aE);
continue;
}
//
if (!theDMEETrim.IsBound (aE))
{
aLENew.Append (aE);
}
//
aLEInt.Append (aE);
aLArgs.Append (aE);
//
aExp.Init (aE, TopAbs_VERTEX);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aV = aExp.Current();
aLVBounds.Append (aV);
}
}
//
// Intersect Edges
BOPAlgo_Builder aGF;
aGF.SetArguments (aLArgs);
aGF.Perform();
if (aGF.HasErrors())
{
return;
}
//
// update vertices to avoid with SD vertices
aIt.Initialize (aLVBounds);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aV = aIt.Value();
const TopTools_ListOfShape& aLVIm = aGF.Modified (aV);
if (aLVIm.IsEmpty())
{
theMVBounds.Add (aV);
}
else
{
const TopoDS_Shape& aVIm = aLVIm.First();
theMVBounds.Add (aVIm);
}
}
//
// find invalid splits of edges
TopTools_MapOfShape aMEInv;
GetInvalidEdges (theNewVertsToAvoid, theMVBounds, aGF, aMEInv);
//
BRep_Builder aBB;
// get valid splits to intersect with the commons
TopoDS_Compound aCEIm;
aBB.MakeCompound (aCEIm);
//
// remove the splits containing vertices from invalid edges
aIt.Initialize (aLEInt);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
//
TopTools_ListOfShape aLEIm = aGF.Modified (aE);
if (aLEIm.IsEmpty())
{
continue;
}
//
aIt1.Initialize (aLEIm);
for (; aIt1.More(); )
{
const TopoDS_Shape& aEIm = aIt1.Value();
//
if (aMEInv.Contains (aEIm))
{
aLEIm.Remove (aIt1);
}
else
{
aBB.Add (aCEIm, aEIm);
aIt1.Next();
}
}
//
if (aLEIm.Extent())
{
TopTools_ListOfShape* pLEIm = theEImages.ChangeSeek (aE);
if (!pLEIm)
{
pLEIm = theEImages.Bound (aE, TopTools_ListOfShape());
}
pLEIm->Append (aLEIm);
}
}
//
if (!aLCE.Extent())
{
return;
}
//
// trim common edges by other intersection edges
BOPAlgo_Builder aGFCE;
aGFCE.SetArguments (aLCE);
aGFCE.AddArgument (aCEIm);
aGFCE.Perform();
//
if (aGFCE.HasErrors())
{
return;
}
//
const BOPDS_PDS& pDS = aGFCE.PDS();
TopTools_ListIteratorOfListOfShape aItLCE (aLCE);
for (; aItLCE.More(); aItLCE.Next())
{
const TopoDS_Shape& aE = aItLCE.Value();
TopTools_ListOfShape aLEIm = aGFCE.Modified (aE);
if (aLEIm.IsEmpty())
{
continue;
}
//
// check if it's not coincide with some intersection edge
BOPDS_ListIteratorOfListOfPaveBlock aItLPB (pDS->PaveBlocks (pDS->Index (aE)));
for (; aItLPB.More(); aItLPB.Next())
{
if (pDS->IsCommonBlock (aItLPB.Value()))
{
// find with what it is a common
const BOPDS_ListOfPaveBlock& aLPBC = pDS->CommonBlock (aItLPB.Value())->PaveBlocks();
BOPDS_ListIteratorOfListOfPaveBlock aItLPBC (aLPBC);
for (; aItLPBC.More(); aItLPBC.Next())
{
const TopoDS_Shape& aEC = pDS->Shape (aItLPBC.Value()->OriginalEdge());
if (!theMECheckExt.Contains (aEC))
{
break;
}
}
if (aItLPBC.More())
{
break;
}
}
}
if (aItLPB.More())
{
// avoid creation of unnecessary splits from commons which
// coincide with intersection edges
continue;
}
//
// save the images
TopTools_ListOfShape* pLEIm = theEImages.ChangeSeek (aE);
if (!pLEIm)
{
pLEIm = theEImages.Bound (aE, TopTools_ListOfShape());
}
pLEIm->Append (aLEIm);
//
// save bounding vertices
for (TopoDS_Iterator aItV (aE); aItV.More(); aItV.Next())
{
const TopoDS_Shape& aV = aItV.Value();
const TopTools_ListOfShape& aLVIm = aGFCE.Modified (aV);
theMVBounds.Add (aLVIm.IsEmpty() ? aV : aLVIm.First());
}
}
}
//=======================================================================
//function : GetInvalidEdges
//purpose : Looking for the invalid edges by intersecting with invalid vertices
//=======================================================================
void BRepOffset_BuildOffsetFaces::GetInvalidEdges (const TopTools_MapOfShape& theVertsToAvoid,
const TopTools_MapOfShape& theMVBounds,
BOPAlgo_Builder& theGF,
TopTools_MapOfShape& theMEInv)
{
if (theVertsToAvoid.IsEmpty())
{
return;
}
//
TopTools_ListIteratorOfListOfShape aIt, aIt1;
// get vertices created with intersection edges
const TopoDS_Shape& aRes = theGF.Shape();
TopTools_IndexedDataMapOfShapeListOfShape aDMVE;
TopExp::MapShapesAndAncestors (aRes, TopAbs_VERTEX, TopAbs_EDGE, aDMVE);
//
const BOPDS_PDS& pDS = theGF.PDS();
//
// find invalid splits of edges
// check if the vertex is invalid:
// a. it may be the vertex SD with the vertices to avoid
// b. or it may be the vertex which is created by the intersection
// of only existing edges, i.e. no new intersection edges goes
// through this vertex
//
TopTools_MapOfShape aMVInv;
Standard_Integer i, aNb = aDMVE.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Vertex& aV = TopoDS::Vertex (aDMVE.FindKey (i));
if (theMVBounds.Contains (aV))
{
continue;
}
//
Standard_Integer nV = pDS->Index (aV);
if ((nV >= 0) && !pDS->IsNewShape (nV))
{
continue;
}
//
TopTools_MapIteratorOfMapOfShape aItM (theVertsToAvoid);
for (; aItM.More(); aItM.Next())
{
const TopoDS_Vertex& aVInv = *(TopoDS_Vertex*)&aItM.Value();
Standard_Integer iFlag = BOPTools_AlgoTools::ComputeVV (aV, aVInv);
if (!iFlag)
{
aMVInv.Add (aV);
break;
}
}
//
if (aItM.More())
{
const TopTools_ListOfShape& aLVE = aDMVE.FindFromKey (aV);
aIt.Initialize (aLVE);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
theMEInv.Add (aE);
}
}
}
}
//=======================================================================
//function : UpdateValidEdges
//purpose : Making the new splits and updating the maps
//=======================================================================
void BRepOffset_BuildOffsetFaces::UpdateValidEdges (const TopTools_IndexedDataMapOfShapeListOfShape& theFLE,
const TopTools_IndexedDataMapOfShapeListOfShape& theOENEdges,
const TopTools_MapOfShape& theMVBounds,
const TopTools_MapOfShape& theMEInvOnArt,
TopTools_MapOfShape& theMECheckExt,
TopTools_MapOfShape& theVertsToAvoid,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_DataMapOfShapeListOfShape& theEETrim,
const Message_ProgressRange& theRange)
{
Message_ProgressScope aPSOuter (theRange, "Updating edges", 10);
// update images and origins of edges, plus update AsDes
//
// new edges
TopTools_ListOfShape aLE;
// back connection from edges to faces
TopTools_DataMapOfShapeListOfShape aMELF;
//
TopTools_MapOfShape aMETmp;
Standard_Integer i, aNb = theFLE.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Face& aF = TopoDS::Face (theFLE.FindKey (i));
//
const TopTools_ListOfShape& aLEInt = theFLE (i);
TopTools_ListIteratorOfListOfShape aItLE (aLEInt);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aE = aItLE.Value();
if ((theMECheckExt.Contains (aE) || aMETmp.Contains (aE)) && !theEImages.IsBound (aE))
{
theMECheckExt.Remove (aE);
aMETmp.Add (aE);
continue;
}
TopTools_ListOfShape* pLF = aMELF.ChangeSeek (aE);
if (!pLF)
{
pLF = aMELF.Bound (aE, TopTools_ListOfShape());
aLE.Append (aE);
}
pLF->Append (aF);
}
}
//
if (aLE.IsEmpty())
{
return;
}
//
// bounding edges, that are going to be replaced
TopTools_MapOfShape aMEB;
//
// new intersection edges
TopTools_MapOfShape aMENew;
// map of old vertices
TopTools_MapOfShape aMVOld;
// back connection to untrimmed edges
TopTools_DataMapOfShapeListOfShape aDMEOr;
//
// trim the new intersection edges
TrimNewIntersectionEdges (aLE, theEETrim, theMVBounds, theMECheckExt,
theEImages, aMEB, aMVOld, aMENew, aDMEOr, aMELF);
//
if (theEImages.IsEmpty())
{
// No new splits is preserved
// update intersection edges and exit
UpdateNewIntersectionEdges (aLE, aMELF, theEImages, theEETrim);
return;
}
aPSOuter.Next();
if (!aPSOuter.More())
{
return;
}
BRep_Builder aBB;
// Make connexity blocks of the invalid edges
// and treat each block separately
// Compound of all invalid edges to make the blocks
TopoDS_Compound aCEAll;
aBB.MakeCompound (aCEAll);
Standard_Integer aNbE = theOENEdges.Extent();
for (i = 1; i <= aNbE; ++i)
aBB.Add (aCEAll, theOENEdges.FindKey (i));
// Separate the edges into blocks
TopTools_ListOfShape aLBlocks;
BOPTools_AlgoTools::MakeConnexityBlocks (aCEAll, TopAbs_VERTEX, TopAbs_EDGE, aLBlocks);
// Perform intersection of the new splits for each block
// Intersected splits
TopTools_IndexedDataMapOfShapeListOfShape aMBlocksSp;
Message_ProgressScope aPSB (aPSOuter.Next(), NULL, aLBlocks.Extent());
TopTools_ListIteratorOfListOfShape aItLB (aLBlocks);
for (; aItLB.More(); aItLB.Next(), aPSB.Next())
{
if (!aPSB.More())
{
return;
}
const TopoDS_Shape& aBlock = aItLB.Value();
// Get the list of new edges for the block
TopTools_ListOfShape aBlockLENew;
{
// Fence map
TopTools_MapOfShape aMEFence;
TopExp_Explorer anExpE (aBlock, TopAbs_EDGE);
for (; anExpE.More(); anExpE.Next())
{
const TopoDS_Shape& aE = anExpE.Current();
const TopTools_ListOfShape& aLEInt = theOENEdges.FindFromKey (aE);
TopTools_ListIteratorOfListOfShape aItLEInt (aLEInt);
for (; aItLEInt.More(); aItLEInt.Next())
{
if (aMEFence.Add (aItLEInt.Value()))
aBlockLENew.Append (aItLEInt.Value());
}
}
}
if (aBlockLENew.IsEmpty())
continue;
// Get the splits of new edges to intersect
TopTools_ListOfShape aLSplits;
TopTools_ListIteratorOfListOfShape aItLE (aBlockLENew);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aE = aItLE.Value();
TopTools_ListOfShape* pLEIm = theEImages.ChangeSeek (aE);
if (!pLEIm || pLEIm->IsEmpty())
continue;
TopTools_ListIteratorOfListOfShape aItLEIm (*pLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
aLSplits.Append (aItLEIm.Value());
}
if (aLSplits.IsEmpty())
continue;
TopoDS_Shape aCE;
if (aLSplits.Extent() > 1)
// Intersect the new splits among themselves to avoid self-intersections
IntersectEdges (aLSplits, aBlockLENew, theMVBounds, theVertsToAvoid,
aMENew, theMECheckExt, theEImages, aDMEOr, aMELF, aCE);
else
aCE = aLSplits.First();
aMBlocksSp.Add (aCE, aBlockLENew);
}
// Perform filtering of the edges in two steps:
// - Check each block separately using localized bounds
// taken only from the splits of faces of the current block;
// - Intersect all splits together and filter the splits by all bounds.
// FIRST STAGE - separate treatment of the blocks
// Valid splits to be preserved on the first stage
TopTools_MapOfShape aMEVal;
// Blocks of valid edges on the first stage
TopTools_ListOfShape aLValBlocks;
Standard_Integer aNbB = aMBlocksSp.Extent();
Message_ProgressScope aPSBSp (aPSOuter.Next(), NULL, aNbB);
for (i = 1; i <= aNbB; ++i, aPSBSp.Next())
{
if (!aPSBSp.More())
{
return;
}
const TopoDS_Shape& aCE = aMBlocksSp.FindKey (i);
const TopTools_ListOfShape& aBlockLENew = aMBlocksSp (i);
// Get all participating faces to get the bounds
TopTools_ListOfShape aLFaces;
TopTools_ListIteratorOfListOfShape aItLE (aBlockLENew);
for (; aItLE.More(); aItLE.Next())
{
const TopTools_ListOfShape* pLF = aMELF.Seek (aItLE.Value());
if (!pLF)
continue;
TopTools_ListIteratorOfListOfShape aItLF (*pLF);
for (; aItLF.More(); aItLF.Next())
AppendToList (aLFaces, aItLF.Value());
}
// Localized bounds of the splits of the offset faces
// to filter the new splits of the current block
TopoDS_Shape aFilterBounds;
GetBounds (aLFaces, aMEB, aFilterBounds);
// Filter the splits by bounds
TopTools_MapOfShape aMEInvLoc;
GetInvalidEdgesByBounds (aCE, aFilterBounds, aMVOld, aMENew, aDMEOr, aMELF, theEImages,
theMECheckExt, theMEInvOnArt, theVertsToAvoid, aMEInvLoc);
// Keep only valid edges of the block
TopoDS_Compound aCEVal;
aBB.MakeCompound (aCEVal);
Standard_Boolean bKept = Standard_False;
TopExp_Explorer anExpE (aCE, TopAbs_EDGE);
for (; anExpE.More(); anExpE.Next())
{
const TopoDS_Shape& aESp = anExpE.Current();
if (!aMEInvLoc.Contains (aESp) && aMEVal.Add (aESp))
{
aBB.Add (aCEVal, aESp);
bKept = Standard_True;
}
}
if (bKept)
aLValBlocks.Append (aCEVal);
}
// Filter the images of edges after the first filtering stage
TopoDS_Shape aSplits1;
FilterSplits (aLE, aMEVal, Standard_False, theEImages, aSplits1);
if (aLValBlocks.IsEmpty())
{
// update intersection edges
UpdateNewIntersectionEdges (aLE, aMELF, theEImages, theEETrim);
return;
}
aPSOuter.Next();
if (!aPSOuter.More())
{
return;
}
// SECOND STAGE - Filter the remaining splits together
// Add for intersection already removed new edges using them
// as markers for other invalid edges
aNbB = aMBlocksSp.Extent();
for (i = 1; i <= aNbB; ++i)
{
const TopoDS_Shape& aCE = aMBlocksSp.FindKey (i);
for (TopExp_Explorer anExp (aCE, TopAbs_EDGE); anExp.More(); anExp.Next())
{
const TopoDS_Shape& aEIm = anExp.Current();
if (aMENew.Contains (aEIm) && !aMEVal.Contains (aEIm))
aLValBlocks.Append (aEIm);
}
}
if (aLValBlocks.Extent() > 1)
// intersect the new splits among themselves to avoid self-intersections
IntersectEdges (aLValBlocks, aLE, theMVBounds, theVertsToAvoid, aMENew,
theMECheckExt, theEImages, aDMEOr, aMELF, aSplits1);
else
aSplits1 = aLValBlocks.First();
aPSOuter.Next();
if (!aPSOuter.More())
{
return;
}
// Get all faces to get the bounds from their splits
TopTools_ListOfShape aLFaces;
for (i = 1; i <= myOFImages.Extent(); ++i)
aLFaces.Append (myOFImages.FindKey (i));
// Bounds of the splits of the offset faces to filter the new splits
TopoDS_Shape aFilterBounds;
GetBounds (aLFaces, aMEB, aFilterBounds);
// Filter the splits by intersection with bounds
TopTools_MapOfShape aMEInv;
GetInvalidEdgesByBounds (aSplits1, aFilterBounds, aMVOld, aMENew, aDMEOr, aMELF,
theEImages, theMECheckExt, theMEInvOnArt, theVertsToAvoid, aMEInv);
// Filter the images of edges after the second filtering stage
// and combine all valid edges into a single compound
TopoDS_Shape aSplits;
FilterSplits (aLE, aMEInv, Standard_True, theEImages, aSplits);
aPSOuter.Next();
if (!aPSOuter.More())
{
return;
}
// get bounds to update
// we need to update the edges of all the affected faces
TopTools_ListOfShape aLF;
// prepare the vertices from new splits of edges
TopTools_IndexedMapOfShape aMVSp;
TopExp::MapShapes (aSplits, TopAbs_VERTEX, aMVSp);
//
Standard_Integer aNbF = myOFImages.Extent();
for (i = 1; i <= aNbF; ++i)
{
const TopoDS_Shape& aF = myOFImages.FindKey (i);
if (theFLE.Contains (aF))
{
aLF.Append (aF);
continue;
}
//
// check the splits of faces to have vertices from splits
const TopTools_ListOfShape& aLFIm = myOFImages (i);
TopTools_ListIteratorOfListOfShape aItLFIm (aLFIm);
for (; aItLFIm.More(); aItLFIm.Next())
{
const TopoDS_Shape& aFIm = aItLFIm.Value();
//
TopExp_Explorer aExpV (aFIm, TopAbs_VERTEX);
for (; aExpV.More(); aExpV.Next())
{
const TopoDS_Shape& aV = aExpV.Current();
if (aMVSp.Contains (aV))
{
break;
}
}
//
if (aExpV.More())
{
break;
}
}
//
if (aItLFIm.More())
{
aLF.Append (aF);
}
}
//
// get bounds from splits of faces of aLF
TopoDS_Shape aBounds;
TopTools_ListOfShape aLAValid, aLABounds;
GetBoundsToUpdate (aLF, aMEB, aLABounds, aLAValid, aBounds);
//
// Intersect valid splits with bounds and update both
BOPAlgo_Builder aGF;
aGF.AddArgument (aBounds);
aGF.AddArgument (aSplits);
aGF.Perform (aPSOuter.Next (3));
//
// update splits
UpdateImages (aLE, theEImages, aGF, myModifiedEdges);
//
// update new intersection edges
UpdateNewIntersectionEdges (aLE, aMELF, theEImages, theEETrim);
//
// update bounds
UpdateImages (aLAValid, myOEImages, aGF, myModifiedEdges);
UpdateOrigins (aLABounds, myOEOrigins, aGF);
UpdateOrigins (aLABounds, *myEdgesOrigins, aGF);
UpdateIntersectedEdges (aLABounds, aGF);
//
// update the EdgesToAvoid with the splits
TopTools_IndexedMapOfShape aNewEdges;
const TopTools_ListOfShape* pSplitsIm = aGF.Images().Seek (aSplits);
if (pSplitsIm)
{
TopTools_ListIteratorOfListOfShape aItSpIm (*pSplitsIm);
for (; aItSpIm.More(); aItSpIm.Next())
{
TopExp::MapShapes (aItSpIm.Value(), TopAbs_EDGE, aNewEdges);
}
}
TopoDS_Compound anInsideEdges;
BRep_Builder().MakeCompound (anInsideEdges);
for (Standard_Integer iE = 1; iE <= myInsideEdges.Extent(); ++iE)
{
BRep_Builder().Add (anInsideEdges, myInsideEdges (iE));
}
//
// Rebuild the map of edges to avoid, using the intersection results
TopTools_IndexedMapOfShape aMEAvoid;
TopoDS_Compound aCEAvoid;
BRep_Builder().MakeCompound (aCEAvoid);
// GF's data structure
const BOPDS_PDS& pDS = aGF.PDS();
aNbE = myEdgesToAvoid.Extent();
for (i = 1; i <= aNbE; ++i)
{
const TopoDS_Shape& aE = myEdgesToAvoid (i);
const TopTools_ListOfShape& aLEIm = aGF.Modified (aE);
// Only untouched and fully coinciding edges should be kept in the avoid map
Standard_Boolean bKeep = aLEIm.IsEmpty();
if (aLEIm.Extent() == 1 && aE.IsSame (aLEIm.First()))
{
const BOPDS_ListOfPaveBlock& aLPB = pDS->PaveBlocks (pDS->Index (aE));
if (aLPB.Extent() == 1)
{
const Handle(BOPDS_PaveBlock)& aPB = aLPB.First();
const Handle(BOPDS_CommonBlock)& aCB = pDS->CommonBlock (aPB);
if (!aCB.IsNull())
{
const BOPDS_ListOfPaveBlock& aLPBCB = aCB->PaveBlocks();
BOPDS_ListIteratorOfListOfPaveBlock aItLPB (aLPBCB);
for (; aItLPB.More(); aItLPB.Next())
{
if (pDS->PaveBlocks (aItLPB.Value()->OriginalEdge()).Extent() > 1)
break;
}
bKeep = !aItLPB.More();
}
}
}
if (bKeep)
{
// keep the original edge
AddToContainer (aE, aMEAvoid);
continue;
}
TopTools_ListIteratorOfListOfShape aItLEIm (aLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
if (!aNewEdges.Contains (aEIm))
{
AddToContainer(aEIm, aCEAvoid);
}
}
}
Standard_Boolean isCut = Standard_False;
if (aCEAvoid.NbChildren() > 0)
{
// Perform intersection with the small subset of the edges to make
// it possible to use the inside edges for building new splits.
BOPAlgo_BOP aBOP;
aBOP.AddArgument (aCEAvoid);
aBOP.AddTool (anInsideEdges);
aBOP.SetOperation (BOPAlgo_CUT);
aBOP.Perform();
isCut = !aBOP.HasErrors();
if (isCut)
{
for (TopoDS_Iterator itCE (aCEAvoid); itCE.More(); itCE.Next())
{
if (!aBOP.IsDeleted (itCE.Value()))
{
aMEAvoid.Add (itCE.Value());
}
}
}
}
if (!isCut)
{
TopExp::MapShapes (aCEAvoid, TopAbs_EDGE, aMEAvoid);
}
myEdgesToAvoid = aMEAvoid;
}
//=======================================================================
//function : TrimNewIntersectionEdges
//purpose :
//=======================================================================
void BRepOffset_BuildOffsetFaces::TrimNewIntersectionEdges (const TopTools_ListOfShape& theLE,
const TopTools_DataMapOfShapeListOfShape& theEETrim,
const TopTools_MapOfShape& theMVBounds,
TopTools_MapOfShape& theMECheckExt,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_MapOfShape& theMEB,
TopTools_MapOfShape& theMVOld,
TopTools_MapOfShape& theMENew,
TopTools_DataMapOfShapeListOfShape& theDMEOr,
TopTools_DataMapOfShapeListOfShape& theMELF)
{
TopTools_ListIteratorOfListOfShape aIt, aIt1;
aIt.Initialize (theLE);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
//
Standard_Boolean bCheckExt = theMECheckExt.Remove (aE);
//
Standard_Boolean bOld = theEETrim.IsBound (aE);
if (bOld)
{
const TopTools_ListOfShape& aLET = theEETrim.Find (aE);
aIt1.Initialize (aLET);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aET = aIt1.Value();
theMEB.Add (aET);
TopExp_Explorer aExpV (aET, TopAbs_VERTEX);
for (; aExpV.More(); aExpV.Next())
{
const TopoDS_Shape& aV = aExpV.Current();
theMVOld.Add (aV);
}
}
}
//
if (!theEImages.IsBound (aE))
{
continue;
}
//
TopTools_ListOfShape& aLEIm = theEImages.ChangeFind (aE);
if (aLEIm.IsEmpty())
{
theEImages.UnBind (aE);
continue;
}
//
TopoDS_Shape aCEIm;
TopTools_MapOfShape aMEVBounds;
//
if (aLEIm.Extent() > 1)
{
TopTools_IndexedMapOfShape aMV;
// fuse these parts
BOPAlgo_Builder aGFE;
TopTools_ListIteratorOfListOfShape aItLEIm (aLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
aGFE.AddArgument (aEIm);
TopExp::MapShapes (aEIm, TopAbs_VERTEX, aMV);
}
//
// add two bounding vertices of this edge to the operation
TopoDS_Vertex aV1, aV2;
TopExp::Vertices (TopoDS::Edge (aE), aV1, aV2);
//
aGFE.AddArgument (aV1);
aGFE.AddArgument (aV2);
aMV.Add (aV1);
aMV.Add (aV2);
//
aGFE.Perform();
if (!aGFE.HasErrors())
{
// get images of bounding vertices to remove splits containing them
// in case some of the bounding edges has been interfered
// during operation it is necessary to update their images as well
Standard_Integer iV, aNbV = aMV.Extent();
for (iV = 1; iV <= aNbV; ++iV)
{
const TopoDS_Shape& aV = aMV (iV);
if (theMVBounds.Contains (aV) || aV.IsSame (aV1) || aV.IsSame (aV2))
{
const TopTools_ListOfShape& aLVIm = aGFE.Modified (aV);
aMEVBounds.Add (aLVIm.IsEmpty() ? aV : aLVIm.First());
}
}
//
aCEIm = aGFE.Shape();
}
}
else
{
aCEIm = aLEIm.First();
}
//
aLEIm.Clear();
//
TopExp_Explorer aExp (aCEIm, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aEIm = aExp.Current();
//
// check the split not to contain bounding vertices
TopoDS_Iterator aItV (aEIm);
for (; aItV.More(); aItV.Next())
{
const TopoDS_Shape& aV = aItV.Value();
if (aMEVBounds.Contains (aV) || theMVBounds.Contains (aV))
{
break;
}
}
//
if (!aItV.More())
{
aLEIm.Append (aEIm);
//
theDMEOr.Bound (aEIm, TopTools_ListOfShape())->Append (aE);
}
}
//
if (aLEIm.IsEmpty())
{
theEImages.UnBind (aE);
}
else
{
const TopTools_ListOfShape& aLFE = theMELF.Find (aE);
TopTools_ListIteratorOfListOfShape aItLEIm (aLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
TopTools_ListOfShape* pLFEIm = theMELF.ChangeSeek (aEIm);
if (!pLFEIm)
{
pLFEIm = theMELF.Bound (aEIm, TopTools_ListOfShape());
}
TopTools_ListIteratorOfListOfShape aItLF (aLFE);
for (; aItLF.More(); aItLF.Next())
{
AppendToList (*pLFEIm, aItLF.Value());
}
//
if (bCheckExt)
{
theMECheckExt.Add (aEIm);
}
else if (!bOld)
{
theMENew.Add (aEIm);
}
}
}
}
}
//=======================================================================
//function : IntersectEdges
//purpose : Intersecting the trimmed edges to avoid self-intersections
//=======================================================================
void BRepOffset_BuildOffsetFaces::IntersectEdges (const TopTools_ListOfShape& theLA,
const TopTools_ListOfShape& theLE,
const TopTools_MapOfShape& theMVBounds,
const TopTools_MapOfShape& theVertsToAvoid,
TopTools_MapOfShape& theMENew,
TopTools_MapOfShape& theMECheckExt,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_DataMapOfShapeListOfShape& theDMEOr,
TopTools_DataMapOfShapeListOfShape& theMELF,
TopoDS_Shape& theSplits)
{
BOPAlgo_Builder aGFA;
aGFA.SetArguments (theLA);
aGFA.Perform();
if (aGFA.HasErrors())
{
// just copy input to the result
TopoDS_Compound aSp;
BRep_Builder aBB;
aBB.MakeCompound (aSp);
TopTools_ListIteratorOfListOfShape anIt (theLA);
for (; anIt.More(); anIt.Next())
{
const TopoDS_Shape& aE = anIt.Value();
aBB.Add (aSp, aE);
}
theSplits = aSp;
return;
}
//
UpdateImages (theLE, theEImages, aGFA, myModifiedEdges);
//
// compound of valid splits
theSplits = aGFA.Shape();
//
TopTools_ListIteratorOfListOfShape aIt, aIt1;
// prepare list of edges to update
TopTools_ListOfShape aLEInput;
for (aIt.Initialize (theLA); aIt.More(); aIt.Next())
{
TopExp_Explorer anExpE (aIt.Value(), TopAbs_EDGE);
for (; anExpE.More(); anExpE.Next())
aLEInput.Append (anExpE.Current());
}
// update new edges
aIt.Initialize (aLEInput);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
if (!theMENew.Contains (aE))
continue;
const TopTools_ListOfShape& aLEIm = aGFA.Modified (aE);
if (aLEIm.IsEmpty())
continue;
theMENew.Remove (aE);
aIt1.Initialize (aLEIm);
for (; aIt1.More(); aIt1.Next())
theMENew.Add (aIt1.Value());
}
//
// update edges after intersection for extended checking
aIt.Initialize (aLEInput);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
const TopTools_ListOfShape& aLEIm = aGFA.Modified (aE);
if (aLEIm.IsEmpty())
{
continue;
}
//
if (theMECheckExt.Contains (aE))
{
aIt1.Initialize (aLEIm);
for (; aIt1.More(); aIt1.Next())
{
theMECheckExt.Add (aIt1.Value());
}
theMECheckExt.Remove (aE);
}
//
const TopTools_ListOfShape& aLFE = theMELF.Find (aE);
aIt1.Initialize (aLEIm);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aEIm = aIt1.Value();
TopTools_ListOfShape* pLFEIm = theMELF.ChangeSeek (aEIm);
if (!pLFEIm)
{
pLFEIm = theMELF.Bound (aEIm, TopTools_ListOfShape());
}
TopTools_ListIteratorOfListOfShape aItLF (aLFE);
for (; aItLF.More(); aItLF.Next())
{
AppendToList (*pLFEIm, aItLF.Value());
}
}
}
//
TopTools_MapOfShape aMEInv;
GetInvalidEdges (theVertsToAvoid, theMVBounds, aGFA, aMEInv);
if (aMEInv.Extent())
{
// update shape
TopoDS_Compound aSp;
BRep_Builder aBB;
aBB.MakeCompound (aSp);
TopExp_Explorer aExp (theSplits, TopAbs_EDGE);
for (; aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (!aMEInv.Contains (aE))
{
aBB.Add (aSp, aE);
}
}
theSplits = aSp;
}
//
// update origins
UpdateOrigins (aLEInput, theDMEOr, aGFA);
}
//=======================================================================
//function : GetBounds
//purpose : Getting edges from the splits of offset faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::GetBounds (const TopTools_ListOfShape& theLFaces,
const TopTools_MapOfShape& theMEB,
TopoDS_Shape& theBounds)
{
BRep_Builder aBB;
// Make compound of edges contained in the splits of faces
TopoDS_Compound aBounds;
aBB.MakeCompound (aBounds);
// Fence map
TopTools_MapOfShape aMFence;
TopTools_ListIteratorOfListOfShape aItLF (theLFaces);
for (; aItLF.More(); aItLF.Next())
{
const TopTools_ListOfShape* pLFIm = myOFImages.Seek (aItLF.Value());
if (!pLFIm)
continue;
TopTools_ListIteratorOfListOfShape aIt (*pLFIm);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aFIm = aIt.Value();
//
TopExp_Explorer aExpE (aFIm, TopAbs_EDGE);
for (; aExpE.More(); aExpE.Next())
{
const TopoDS_Shape& aEIm = aExpE.Current();
if (!theMEB.Contains (aEIm) && aMFence.Add (aEIm))
{
aBB.Add (aBounds, aEIm);
}
}
}
}
theBounds = aBounds;
}
//=======================================================================
//function : GetBoundsToUpdate
//purpose : Get bounding edges that should be updated
//=======================================================================
void BRepOffset_BuildOffsetFaces::GetBoundsToUpdate (const TopTools_ListOfShape& theLF,
const TopTools_MapOfShape& theMEB,
TopTools_ListOfShape& theLABounds,
TopTools_ListOfShape& theLAValid,
TopoDS_Shape& theBounds)
{
// get all edges
TopoDS_Compound aBounds;
BRep_Builder aBB;
aBB.MakeCompound (aBounds);
//
TopTools_MapOfShape aMAValid, aMFence;
//
TopTools_ListIteratorOfListOfShape aItLF (theLF);
for (; aItLF.More(); aItLF.Next())
{
const TopoDS_Shape& aF = aItLF.Value();
//
TopTools_IndexedMapOfShape aMDE;
const TopTools_ListOfShape& aLFDes = myAsDes->Descendant (aF);
TopTools_ListIteratorOfListOfShape aItLFDes (aLFDes);
for (; aItLFDes.More(); aItLFDes.Next())
{
const TopoDS_Shape& aED = aItLFDes.Value();
const TopTools_ListOfShape* pLEDIm = myOEImages.Seek (aED);
if (!pLEDIm)
{
aMDE.Add (aED);
continue;
}
//
TopTools_ListIteratorOfListOfShape aItLEDIm (*pLEDIm);
for (; aItLEDIm.More(); aItLEDIm.Next())
{
const TopoDS_Shape& aEDIm = aItLEDIm.Value();
aMDE.Add (aEDIm);
}
}
//
Standard_Integer j, aNbE = aMDE.Extent();
for (j = 1; j <= aNbE; ++j)
{
const TopoDS_Edge& aEIm = TopoDS::Edge (aMDE (j));
//
if (!theMEB.Contains (aEIm) && aMFence.Add (aEIm))
{
aBB.Add (aBounds, aEIm);
theLABounds.Append (aEIm);
}
//
const TopTools_ListOfShape* pLO = myOEOrigins.Seek (aEIm);
if (pLO)
{
TopTools_ListIteratorOfListOfShape aItLO (*pLO);
for (; aItLO.More(); aItLO.Next())
{
const TopoDS_Shape& aEO = aItLO.Value();
//
if (aMAValid.Add (aEO))
{
theLAValid.Append (aEO);
}
}
}
else
{
if (aMAValid.Add (aEIm))
{
theLAValid.Append (aEIm);
}
}
}
}
theBounds = aBounds;
}
//=======================================================================
//function : GetInvalidEdgesByBounds
//purpose : Filter new splits by intersection with bounds
//=======================================================================
void BRepOffset_BuildOffsetFaces::GetInvalidEdgesByBounds (const TopoDS_Shape& theSplits,
const TopoDS_Shape& theBounds,
const TopTools_MapOfShape& theMVOld,
const TopTools_MapOfShape& theMENew,
const TopTools_DataMapOfShapeListOfShape& theDMEOr,
const TopTools_DataMapOfShapeListOfShape& theMELF,
const TopTools_DataMapOfShapeListOfShape& theEImages,
const TopTools_MapOfShape& theMECheckExt,
const TopTools_MapOfShape& theMEInvOnArt,
TopTools_MapOfShape& theVertsToAvoid,
TopTools_MapOfShape& theMEInv)
{
// map splits to check the vertices of edges
TopTools_IndexedDataMapOfShapeListOfShape aDMVE;
TopExp::MapShapesAndAncestors (theSplits, TopAbs_VERTEX, TopAbs_EDGE, aDMVE);
//
BOPAlgo_Section aSec;
aSec.AddArgument (theSplits);
aSec.AddArgument (theBounds);
//
aSec.Perform();
//
// invalid vertices
TopTools_IndexedMapOfShape aMVInv;
// vertices to check additionally by classification relatively to solid
TopTools_MapOfShape aMVCheckAdd;
// collect parts for removal
const BOPDS_PDS& pDS = aSec.PDS();
//
// check edge/edge intersections
const BOPDS_VectorOfInterfEE& aEEs = pDS->InterfEE();
Standard_Integer i, aNb = aEEs.Length();
for (i = 0; i < aNb; ++i)
{
const BOPDS_InterfEE& aEE = aEEs (i);
//
const TopoDS_Shape& aE1 = pDS->Shape (aEE.Index1());
const TopoDS_Shape& aE2 = pDS->Shape (aEE.Index2());
//
if (!aEE.HasIndexNew())
{
if (theMECheckExt.Contains (aE1) && (aEE.CommonPart().Type() == TopAbs_EDGE))
{
theMEInv.Add (aE1);
}
continue;
}
//
if (myInvalidEdges.Contains (aE2))
{
theMEInv.Add (aE1);
}
//
if (theMEInvOnArt.Contains (aE2))
{
// avoid checking of the vertices of the split edge intersected by
// the invalid edge from artificial face
TopoDS_Vertex aV1, aV2;
TopExp::Vertices (TopoDS::Edge (aE2), aV1, aV2);
if (aDMVE.Contains (aV1) && aDMVE.Contains (aV2))
{
continue;
}
}
//
// add vertices of all images of the edge from splits for checking
const TopTools_ListOfShape& aLEOr = theDMEOr.Find (aE1);
TopTools_ListIteratorOfListOfShape aItLEOr (aLEOr);
for (; aItLEOr.More(); aItLEOr.Next())
{
const TopoDS_Shape& aEOr = aItLEOr.Value();
//
const TopTools_ListOfShape* pLEIm = theEImages.Seek (aEOr);
if (!pLEIm)
continue;
TopTools_ListIteratorOfListOfShape aItLEIm (*pLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
//
TopoDS_Iterator aItV (aEIm);
for (; aItV.More(); aItV.Next())
{
const TopoDS_Shape& aV = aItV.Value();
if (!theMVOld.Contains (aV))
{
aMVInv.Add (aV);
aMVCheckAdd.Add (aV);
}
}
}
}
}
//
// to avoid unnecessary filling of parts due to extra trim of the edges
// process Edge/Edge interferences of type EDGE, i.e. common blocks and check
// not the bounding vertices of the edges, but check the edge itself
// to be lying on some face
//
// all common blocks are contained in the result of SECTION operation
// between sets of edges
const TopoDS_Shape& aSecR = aSec.Shape();
//
TopTools_IndexedMapOfShape aMSSec;
TopExp::MapShapes (aSecR, aMSSec);
//
const TopTools_DataMapOfShapeListOfShape& anIm = aSec.Images();
for (TopExp_Explorer aExp (theSplits, TopAbs_EDGE); aExp.More(); aExp.Next())
{
const TopoDS_Shape& aE = aExp.Current();
if (aSec.IsDeleted (aE))
{
// no common blocks for this edge
continue;
}
//
const TopTools_ListOfShape* pLEIm = anIm.Seek (aE);
if (!pLEIm)
{
// no splits, i.e. completely coincides with some edge from boundary
continue;
}
//
TopTools_ListIteratorOfListOfShape aItLEIm (*pLEIm);
for (; aItLEIm.More(); aItLEIm.Next())
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
if (!aMSSec.Contains (aEIm))
{
// the edge included in section only partially.
// the part not included in section may be excessive
//
// check vertices of this edge - if one of them is new
// the edge might be removed
TopoDS_Vertex aV1, aV2;
TopExp::Vertices (TopoDS::Edge (aEIm), aV1, aV2);
if (!theMVOld.Contains (aV1) || !theMVOld.Contains (aV2))
{
// add this edge for checking by making new vertex in the middle of the edge
TopoDS_Vertex aV;
Standard_Real f, l;
const Handle(Geom_Curve)& aC = BRep_Tool::Curve (TopoDS::Edge (aEIm), f, l);
BRep_Builder().MakeVertex (aV, aC->Value ((f + l) * 0.5), Precision::Confusion());
// and adding this vertex for checking
aDMVE.ChangeFromIndex (aDMVE.Add (aV, TopTools_ListOfShape())).Append (aE);
aMVInv.Add (aV);
break;
}
}
}
}
//
// Add for check also the edges created from common between splits
// of offset faces edges not connected to any invalidity.
// These edges may also accidentally fill some part.
TopTools_MapIteratorOfMapOfShape aItM (theMECheckExt);
for (; aItM.More(); aItM.Next())
{
const TopoDS_Shape& aE = aItM.Value();
//
// make new vertex in the middle of the edge
TopoDS_Vertex aV;
Standard_Real f, l;
const Handle(Geom_Curve)& aC = BRep_Tool::Curve (TopoDS::Edge (aE), f, l);
BRep_Builder().MakeVertex (aV, aC->Value ((f + l) * 0.5), Precision::Confusion());
// add this vertex for checking
aDMVE.ChangeFromIndex (aDMVE.Add (aV, TopTools_ListOfShape())).Append (aE);
aMVInv.Add (aV);
}
//
// add for check also the vertices connected only to new or old edges
aNb = aDMVE.Extent();
for (i = 1; i <= aNb; ++i)
{
const TopoDS_Shape& aV = aDMVE.FindKey (i);
if (theMVOld.Contains (aV))
{
continue;
}
//
Standard_Boolean bNew = Standard_False, bOld = Standard_False;
const TopTools_ListOfShape& aLEx = aDMVE (i);
TopTools_ListIteratorOfListOfShape aIt (aLEx);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
if (theMECheckExt.Contains (aE))
{
continue;
}
//
if (theMENew.Contains (aE))
{
bNew = Standard_True;
}
else
{
bOld = Standard_True;
}
//
if (bNew && bOld)
{
break;
}
}
//
if (!bNew || !bOld)
{
aMVInv.Add (aV);
aMVCheckAdd.Remove (aV);
}
}
//
// perform the checking of the vertices
Standard_Integer iv, aNbIV = aMVInv.Extent();
for (iv = 1; iv <= aNbIV; ++iv)
{
const TopoDS_Vertex& aV = TopoDS::Vertex (aMVInv (iv));
if (theMVOld.Contains (aV))
{
continue;
}
//
const TopTools_ListOfShape* pLEInv = aDMVE.Seek (aV);
if (!pLEInv)
{
continue;
}
// find faces by the edges to check the vertex
TopTools_IndexedMapOfShape aMF;
TopTools_ListIteratorOfListOfShape aItLE (*pLEInv);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aE = aItLE.Value();
const TopTools_ListOfShape& aLF = theMELF.Find (aE);
TopTools_ListIteratorOfListOfShape aItLF (aLF);
for (; aItLF.More(); aItLF.Next())
{
aMF.Add (aItLF.Value());
}
}
//
// check the vertex to belong to some split of the faces
Standard_Boolean bInvalid = Standard_True;
//
Standard_Integer aNbF = aMF.Extent();
for (i = 1; i <= aNbF && bInvalid; ++i)
{
const TopoDS_Face& aF = TopoDS::Face (aMF (i));
const TopTools_ListOfShape& aLFIm = myOFImages.FindFromKey (aF);
//
TopTools_ListIteratorOfListOfShape aItLF (aLFIm);
for (; aItLF.More() && bInvalid; aItLF.Next())
{
const TopoDS_Face& aFIm = TopoDS::Face (aItLF.Value());
TopExp_Explorer aExp (aFIm, TopAbs_VERTEX);
for (; aExp.More() && bInvalid; aExp.Next())
{
const TopoDS_Shape& aVF = aExp.Current();
bInvalid = !aVF.IsSame (aV);
}
}
//
if (bInvalid)
{
Standard_Real U, V, aTol;
Standard_Integer iStatus = myContext->ComputeVF (aV, aF, U, V, aTol);
if (!iStatus)
{
// classify the point relatively faces
gp_Pnt2d aP2d (U, V);
aItLF.Initialize (aLFIm);
for (; aItLF.More() && bInvalid; aItLF.Next())
{
const TopoDS_Face& aFIm = TopoDS::Face (aItLF.Value());
bInvalid = !myContext->IsPointInOnFace (aFIm, aP2d);
}
}
}
}
//
if (bInvalid && aMVCheckAdd.Contains (aV))
{
// the vertex is invalid for all faces
// check the same vertex for the solids
const gp_Pnt& aP = BRep_Tool::Pnt (aV);
Standard_Real aTolV = BRep_Tool::Tolerance (aV);
//
TopExp_Explorer aExpS (mySolids, TopAbs_SOLID);
for (; aExpS.More() && bInvalid; aExpS.Next())
{
const TopoDS_Solid& aSol = TopoDS::Solid (aExpS.Current());
BRepClass3d_SolidClassifier& aSC = myContext->SolidClassifier (aSol);
aSC.Perform (aP, aTolV);
bInvalid = (aSC.State() == TopAbs_OUT);
}
}
//
if (bInvalid)
{
theVertsToAvoid.Add (aV);
aItLE.Initialize (*pLEInv);
for (; aItLE.More(); aItLE.Next())
{
theMEInv.Add (aItLE.Value());
}
}
}
}
//=======================================================================
//function : FilterSplits
//purpose : Filter the images of edges from the invalid edges
//=======================================================================
void BRepOffset_BuildOffsetFaces::FilterSplits (const TopTools_ListOfShape& theLE,
const TopTools_MapOfShape& theMEFilter,
const Standard_Boolean theIsInv,
TopTools_DataMapOfShapeListOfShape& theEImages,
TopoDS_Shape& theSplits)
{
TopoDS_Compound aSplits;
BRep_Builder().MakeCompound (aSplits);
TopTools_MapOfShape aMFence;
TopTools_ListIteratorOfListOfShape aItLE (theLE);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aE = aItLE.Value();
TopTools_ListOfShape* pLEIm = theEImages.ChangeSeek (aE);
if (!pLEIm)
continue;
TopTools_ListIteratorOfListOfShape aItLEIm (*pLEIm);
for (; aItLEIm.More();)
{
const TopoDS_Shape& aEIm = aItLEIm.Value();
if (theMEFilter.Contains (aEIm) == theIsInv)
{
pLEIm->Remove (aItLEIm);
continue;
}
if (aMFence.Add (aEIm))
BRep_Builder().Add (aSplits, aEIm);
aItLEIm.Next();
}
if (pLEIm->IsEmpty())
theEImages.UnBind (aE);
}
theSplits = aSplits;
}
//=======================================================================
//function : UpdateNewIntersectionEdges
//purpose : Updating the maps of images and origins of the offset edges
//=======================================================================
void BRepOffset_BuildOffsetFaces::UpdateNewIntersectionEdges (const TopTools_ListOfShape& theLE,
const TopTools_DataMapOfShapeListOfShape& theMELF,
const TopTools_DataMapOfShapeListOfShape& theEImages,
TopTools_DataMapOfShapeListOfShape& theEETrim)
{
TopTools_ListOfShape aLEImEmpty;
TopTools_ListIteratorOfListOfShape aIt, aIt1;
// update global maps of images and origins with new splits
aIt.Initialize (theLE);
for (; aIt.More(); aIt.Next())
{
const TopoDS_Shape& aE = aIt.Value();
//
if (!theEImages.IsBound (aE))
{
TopTools_ListOfShape* pLET = theEETrim.ChangeSeek (aE);
if (!pLET)
{
continue;
}
//
TopTools_ListIteratorOfListOfShape aItLET (*pLET);
for (; aItLET.More();)
{
const TopoDS_Shape& aET = aItLET.Value();
if (!myInvalidEdges.Contains (aET) && !myInvertedEdges.Contains (aET))
{
pLET->Remove (aItLET);
}
else
{
aItLET.Next();
}
}
//
if (pLET->IsEmpty())
{
continue;
}
}
// new images
const TopTools_ListOfShape& aLENew =
theEImages.IsBound (aE) ? theEImages.Find (aE) : aLEImEmpty;
//
// save connection to untrimmed edge for the next steps
aIt1.Initialize (aLENew);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aET = aIt1.Value();
myETrimEInf->Bind (aET, aE);
myModifiedEdges.Add (aET);
}
//
// check if it is existing edge
if (!theEETrim.IsBound (aE))
{
const TopTools_ListOfShape& aLF = theMELF.Find (aE);
// the edge is new
// add this edge to AsDes
aIt1.Initialize (aLF);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aF = aIt1.Value();
myAsDes->Add (aF, aE);
}
//
// add aE to the images
myOEImages.Bind (aE, aLENew);
myModifiedEdges.Add (aE);
//
// add to origins
TopTools_ListIteratorOfListOfShape aItNew (aLENew);
for (; aItNew.More(); aItNew.Next())
{
const TopoDS_Shape& aENew = aItNew.Value();
if (myOEOrigins.IsBound (aENew))
{
TopTools_ListOfShape& aEOrigins = myOEOrigins.ChangeFind (aENew);
AppendToList (aEOrigins, aE);
}
else
{
TopTools_ListOfShape aEOrigins;
aEOrigins.Append (aE);
myOEOrigins.Bind (aENew, aEOrigins);
}
}
//
// update connection to initial origins
if (myEdgesOrigins->IsBound (aE))
{
const TopTools_ListOfShape& aLEOrInit = myEdgesOrigins->Find (aE);
aIt1.Initialize (aLENew);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aENew = aIt1.Value();
if (myEdgesOrigins->IsBound (aENew))
{
TopTools_ListOfShape& aLENewOr = myEdgesOrigins->ChangeFind (aENew);
TopTools_ListIteratorOfListOfShape aItOrInit (aLEOrInit);
for (; aItOrInit.More(); aItOrInit.Next())
{
const TopoDS_Shape& aEOr = aItOrInit.Value();
AppendToList (aLENewOr, aEOr);
}
}
else
{
myEdgesOrigins->Bind (aENew, aLEOrInit);
}
}
}
//
continue;
}
//
// old images
const TopTools_ListOfShape& aLEOld = theEETrim.Find (aE);
//
// list of initial origins
TopTools_ListOfShape anInitOrigins;
//
// it is necessary to replace the old edges with new ones
aIt1.Initialize (aLEOld);
for (; aIt1.More(); aIt1.Next())
{
const TopoDS_Shape& aEOld = aIt1.Value();
//
if (myOEOrigins.IsBound (aEOld))
{
// get its origins
const TopTools_ListOfShape& aEOrigins = myOEOrigins.Find (aEOld);
//
TopTools_ListIteratorOfListOfShape aItOr (aEOrigins);
for (; aItOr.More(); aItOr.Next())
{
const TopoDS_Shape& aEOr = aItOr.Value();
//
myModifiedEdges.Add (aEOr);
//
TopTools_ListOfShape& aEImages = myOEImages.ChangeFind (aEOr);
//
// remove old edge from images
TopTools_ListIteratorOfListOfShape aItIm (aEImages);
for (; aItIm.More(); )
{
const TopoDS_Shape& aEIm = aItIm.Value();
if (aEIm.IsSame (aEOld))
{
aEImages.Remove (aItIm);
}
else
{
aItIm.Next();
}
}
//
// add new images
TopTools_ListIteratorOfListOfShape aItNew (aLENew);
for (; aItNew.More(); aItNew.Next())
{
const TopoDS_Shape& aENew = aItNew.Value();
AppendToList (aEImages, aENew);
if (myOEOrigins.IsBound (aENew))
{
TopTools_ListOfShape& aENewOrigins = myOEOrigins.ChangeFind (aENew);
AppendToList (aENewOrigins, aEOr);
}
else
{
TopTools_ListOfShape aENewOrigins;
aENewOrigins.Append (aEOr);
myOEOrigins.Bind (aENew, aENewOrigins);
}
}
}
}
else
{
// add to images
myOEImages.Bind (aEOld, aLENew);
//
myModifiedEdges.Add (aEOld);
//
// add to origins
TopTools_ListIteratorOfListOfShape aItNew (aLENew);
for (; aItNew.More(); aItNew.Next())
{
const TopoDS_Shape& aENew = aItNew.Value();
if (myOEOrigins.IsBound (aENew))
{
TopTools_ListOfShape& aEOrigins = myOEOrigins.ChangeFind (aENew);
AppendToList (aEOrigins, aEOld);
}
else
{
TopTools_ListOfShape aEOrigins;
aEOrigins.Append (aEOld);
myOEOrigins.Bind (aENew, aEOrigins);
}
}
}
//
// update connection to initial shape
if (myEdgesOrigins->IsBound (aEOld))
{
const TopTools_ListOfShape& aLEOrInit = myEdgesOrigins->Find (aEOld);
TopTools_ListIteratorOfListOfShape aItEOrInit (aLEOrInit);
for (; aItEOrInit.More(); aItEOrInit.Next())
{
const TopoDS_Shape& aEOrInit = aItEOrInit.Value();
AppendToList (anInitOrigins, aEOrInit);
}
}
}
//
if (anInitOrigins.Extent())
{
TopTools_ListIteratorOfListOfShape aItNew (aLENew);
for (; aItNew.More(); aItNew.Next())
{
const TopoDS_Shape& aENew = aItNew.Value();
if (myEdgesOrigins->IsBound (aENew))
{
TopTools_ListOfShape& aLENewOr = myEdgesOrigins->ChangeFind (aENew);
TopTools_ListIteratorOfListOfShape aItOrInit (anInitOrigins);
for (; aItOrInit.More(); aItOrInit.Next())
{
const TopoDS_Shape& aEOr = aItOrInit.Value();
AppendToList (aLENewOr, aEOr);
}
}
else
{
myEdgesOrigins->Bind (aENew, anInitOrigins);
}
}
}
}
}
//=======================================================================
//function : FillGaps
//purpose : Fill possible gaps (holes) in the splits of the offset faces
//=======================================================================
void BRepOffset_BuildOffsetFaces::FillGaps (const Message_ProgressRange& theRange)
{
Standard_Integer aNbF = myOFImages.Extent();
if (!aNbF)
return;
Message_ProgressScope aPS (theRange, "Filling gaps", 2 * aNbF);
// Check the splits of offset faces on the free edges and fill the gaps (holes)
// in created splits, otherwise the closed volume will not be possible to create.
// Map the splits of faces to find free edges
TopTools_IndexedDataMapOfShapeListOfShape anEFMap;
for (Standard_Integer i = 1; i <= aNbF; ++i, aPS.Next())
{
if (!aPS.More())
{
return;
}
TopTools_ListIteratorOfListOfShape itLF (myOFImages (i));
for (; itLF.More(); itLF.Next())
TopExp::MapShapesAndAncestors (itLF.Value(), TopAbs_EDGE, TopAbs_FACE, anEFMap);
}
// Analyze images of each offset face on the presence of free edges
// and try to fill the holes
for (Standard_Integer i = 1; i <= aNbF; ++i, aPS.Next())
{
if (!aPS.More())
{
return;
}
TopTools_ListOfShape& aLFImages = myOFImages (i);
if (aLFImages.IsEmpty())
continue;
// Collect all edges from the splits
TopoDS_Compound anEdges;
BRep_Builder().MakeCompound (anEdges);
// Collect all free edges into a map with reverted orientation
TopTools_MapOfOrientedShape aFreeEdgesMap;
TopTools_ListIteratorOfListOfShape itLF (aLFImages);
for (; itLF.More(); itLF.Next())
{
const TopoDS_Shape& aFIm = itLF.Value();
TopExp_Explorer anExpE (aFIm, TopAbs_EDGE);
for (; anExpE.More(); anExpE.Next())
{
const TopoDS_Shape& aE = anExpE.Current();
if (aE.Orientation() != TopAbs_FORWARD &&
aE.Orientation() != TopAbs_REVERSED)
// Skip internals
continue;
const TopTools_ListOfShape& aLF = anEFMap.FindFromKey (aE);
if (aLF.Extent() == 1)
aFreeEdgesMap.Add (aE.Reversed());
BRep_Builder().Add (anEdges, aE);
}
}
if (aFreeEdgesMap.IsEmpty())
// No free edges
continue;
// Free edges are found - fill the gaps by creating new splits
// of the face using these free edges
const TopoDS_Shape& aF = myOFImages.FindKey (i);
// Build new splits using all kept edges and among new splits
// find those containing free edges
TopTools_ListOfShape aLFNew;
TopTools_DataMapOfShapeShape aDummy;
BuildSplitsOfFace (TopoDS::Face (aF), anEdges, aDummy, aLFNew);
// Find faces filling holes
itLF.Initialize (aLFNew);
for (; itLF.More(); itLF.Next())
{
const TopoDS_Shape& aFNew = itLF.Value();
TopExp_Explorer anExpE (aFNew, TopAbs_EDGE);
for (; anExpE.More(); anExpE.Next())
{
const TopoDS_Shape& aE = anExpE.Current();
if (aFreeEdgesMap.Contains (aE))
{
// Add face to splits
aLFImages.Append (aFNew);
break;
}
}
}
}
}
//=======================================================================
//function : FillHistory
//purpose : Saving obtained results in history tools
//=======================================================================
void BRepOffset_BuildOffsetFaces::FillHistory()
{
Standard_Integer aNbF = myOFImages.Extent();
if (!aNbF)
{
return;
}
#ifdef OFFSET_DEBUG
// Build compound of faces to see preliminary result
TopoDS_Compound aDFaces;
BRep_Builder().MakeCompound (aDFaces);
#endif
// Map of kept edges
TopTools_IndexedMapOfShape anEdgesMap;
// Fill history for faces
for (Standard_Integer i = 1; i <= aNbF; ++i)
{
const TopTools_ListOfShape& aLFImages = myOFImages (i);
if (aLFImages.IsEmpty())
{
continue;
}
// Add the splits to history map
const TopoDS_Shape& aF = myOFImages.FindKey (i);
if (myImage->HasImage (aF))
myImage->Add (aF, aLFImages);
else
myImage->Bind (aF, aLFImages);
// Collect edges from splits
TopTools_ListIteratorOfListOfShape itLF (aLFImages);
for (; itLF.More(); itLF.Next())
{
const TopoDS_Shape& aFIm = itLF.Value();
TopExp::MapShapes (aFIm, TopAbs_EDGE, anEdgesMap);
#ifdef OFFSET_DEBUG
BRep_Builder().Add (aDFaces, aFIm);
#endif
}
}
// Fill history for edges (iteration by the map is safe because the
// order is not important here)
TopTools_DataMapIteratorOfDataMapOfShapeListOfShape aItEIm (myOEImages);
for (; aItEIm.More(); aItEIm.Next())
{
const TopoDS_Shape& aE = aItEIm.Key();
const TopTools_ListOfShape& aLEIm = aItEIm.Value();
Standard_Boolean bHasImage = myImage->HasImage (aE);
TopTools_ListIteratorOfListOfShape aItLE (aLEIm);
for (; aItLE.More(); aItLE.Next())
{
const TopoDS_Shape& aEIm = aItLE.Value();
if (anEdgesMap.Contains (aEIm))
{
if (bHasImage)
{
myImage->Add (aE, aEIm);
}
else
{
myImage->Bind (aE, aEIm);
bHasImage = Standard_True;
}
}
}
}
}
//=======================================================================
//function : BuildSplitsOfTrimmedFaces
//purpose : Building splits of already trimmed faces
//=======================================================================
void BRepOffset_MakeOffset::BuildSplitsOfTrimmedFaces (const TopTools_ListOfShape& theLF,
const Handle(BRepAlgo_AsDes)& theAsDes,
BRepAlgo_Image& theImage,
const Message_ProgressRange& theRange)
{
BRepOffset_BuildOffsetFaces aBFTool (theImage);
aBFTool.SetFaces (theLF);
aBFTool.SetAsDesInfo (theAsDes);
aBFTool.BuildSplitsOfTrimmedFaces (theRange);
}
//=======================================================================
//function : BuildSplitsOfExtendedFaces
//purpose : Building splits of not-trimmed offset faces.
// For the cases in which invalidity will be found,
// these invalidities will be rebuilt.
//=======================================================================
void BRepOffset_MakeOffset::BuildSplitsOfExtendedFaces (const TopTools_ListOfShape& theLF,
const BRepOffset_Analyse& theAnalyse,
const Handle(BRepAlgo_AsDes)& theAsDes,
TopTools_DataMapOfShapeListOfShape& theEdgesOrigins,
TopTools_DataMapOfShapeShape& theFacesOrigins,
TopTools_DataMapOfShapeShape& theETrimEInf,
BRepAlgo_Image& theImage,
const Message_ProgressRange& theRange)
{
BRepOffset_BuildOffsetFaces aBFTool (theImage);
aBFTool.SetFaces (theLF);
aBFTool.SetAsDesInfo (theAsDes);
aBFTool.SetAnalysis (theAnalyse);
aBFTool.SetEdgesOrigins (theEdgesOrigins);
aBFTool.SetFacesOrigins (theFacesOrigins);
aBFTool.SetInfEdges (theETrimEInf);
aBFTool.BuildSplitsOfExtendedFaces (theRange);
}