// 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef OFFSET_DEBUG #include #endif typedef NCollection_DataMap BRepOffset_DataMapOfShapeMapOfShape; typedef NCollection_DataMap 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 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 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 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 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 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 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& 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 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 , // 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); }