#include <GeomAPI_Face.h>
#include <GeomAPI_Pln.h>
#include <GeomAPI_Pnt.h>
+#include <GeomAPI_Wire.h>
#include <Bnd_Box.hxx>
-#include <BOPTools.hxx>
#include <BRep_Builder.hxx>
#include <BRepAdaptor_Curve.hxx>
+#include <BRepAlgo.hxx>
#include <BRepAlgo_FaceRestrictor.hxx>
#include <BRepBndLib.hxx>
#include <BRepBuilderAPI_FindPlane.hxx>
#include <BRepExtrema_ExtCF.hxx>
#include <BRepGProp.hxx>
#include <BRepTools.hxx>
+#include <BRepTools_WireExplorer.hxx>
#include <BRepTopAdaptor_FClass2d.hxx>
#include <BRepClass_FaceClassifier.hxx>
#include <Geom2d_Curve.hxx>
#include <TopoDS_Shell.hxx>
#include <TopoDS_Vertex.hxx>
#include <TopoDS.hxx>
+#include <TopExp.hxx>
#include <TopExp_Explorer.hxx>
#include <TopTools_ListIteratorOfListOfShape.hxx>
return 0.0;
}
const Standard_Real anEps = 1.e-6;
- BRepGProp::VolumeProperties(aShape, aGProps, anEps);
+ if (aShape.ShapeType() <= TopAbs_SOLID)
+ BRepGProp::VolumeProperties(aShape, aGProps, anEps);
+ else
+ BRepGProp::SurfaceProperties(aShape, aGProps, anEps);
+ return aGProps.Mass();
+}
+
+//==================================================================================================
+double GeomAlgoAPI_ShapeTools::area (const std::shared_ptr<GeomAPI_Shape> theShape)
+{
+ GProp_GProps aGProps;
+ if(!theShape.get()) {
+ return 0.0;
+ }
+ const TopoDS_Shape& aShape = theShape->impl<TopoDS_Shape>();
+ if(aShape.IsNull()) {
+ return 0.0;
+ }
+ const Standard_Real anEps = 1.e-6;
+
+ BRepGProp::SurfaceProperties(aShape, aGProps, anEps);
return aGProps.Mass();
}
BRepGProp::LinearProperties(aShape, aGProps);
aCentre = aGProps.CentreOfMass();
} else {
- BRepGProp::SurfaceProperties(aShape, aGProps);
+ const Standard_Real anEps = 1.e-6;
+ BRepGProp::SurfaceProperties(aShape, aGProps, anEps);
aCentre = aGProps.CentreOfMass();
}
return std::shared_ptr<GeomAPI_Pnt>(new GeomAPI_Pnt(aCentre.X(), aCentre.Y(), aCentre.Z()));
}
// Map subshapes and shapes.
- BOPCol_IndexedDataMapOfShapeListOfShape aMapSA;
- BOPTools::MapShapesAndAncestors(aShapesComp, aTS, aTA, aMapSA);
+ TopTools_IndexedDataMapOfShapeListOfShape aMapSA;
+ TopExp::MapShapesAndAncestors(aShapesComp, aTS, aTA, aMapSA);
if(aMapSA.IsEmpty()) {
return aResult;
}
// Get all shapes with common subshapes and free shapes.
NCollection_Map<TopoDS_Shape> aFreeShapes;
NCollection_Vector<NCollection_Map<TopoDS_Shape>> aShapesWithCommonSubshapes;
- for(BOPCol_IndexedDataMapOfShapeListOfShape::Iterator
+ for(TopTools_IndexedDataMapOfShapeListOfShape::Iterator
anIter(aMapSA); anIter.More(); anIter.Next()) {
const TopoDS_Shape& aShape = anIter.Key();
- BOPCol_ListOfShape& aListOfShape = anIter.ChangeValue();
+ TopTools_ListOfShape& aListOfShape = anIter.ChangeValue();
if(aListOfShape.IsEmpty()) {
continue;
}
} else {
NCollection_List<TopoDS_Shape> aTempList;
NCollection_Map<TopoDS_Shape> aTempMap;
- const TopoDS_Shape& aF = aListOfShape.First();
- const TopoDS_Shape& aL = aListOfShape.Last();
- aTempList.Append(aF);
- aTempList.Append(aL);
- aTempMap.Add(aF);
- aTempMap.Add(aL);
- aFreeShapes.Remove(aF);
- aFreeShapes.Remove(aL);
+ for (TopTools_ListOfShape::Iterator aListIt(aListOfShape); aListIt.More(); aListIt.Next()) {
+ aTempList.Append(aListIt.Value());
+ aTempMap.Add(aListIt.Value());
+ aFreeShapes.Remove(aListIt.Value());
+ }
aListOfShape.Clear();
for(NCollection_List<TopoDS_Shape>::Iterator
aTempIter(aTempList); aTempIter.More(); aTempIter.Next()) {
const TopoDS_Shape& aTempShape = aTempIter.Value();
- for(BOPCol_IndexedDataMapOfShapeListOfShape::Iterator
+ for(TopTools_IndexedDataMapOfShapeListOfShape::Iterator
anIter(aMapSA); anIter.More(); anIter.Next()) {
- BOPCol_ListOfShape& aTempListOfShape = anIter.ChangeValue();
+ TopTools_ListOfShape& aTempListOfShape = anIter.ChangeValue();
if(aTempListOfShape.IsEmpty()) {
continue;
} else if(aTempListOfShape.Size() == 1 && aTempListOfShape.First() == aTempShape) {
aTempListOfShape.Clear();
} else if(aTempListOfShape.Size() > 1) {
- if(aTempListOfShape.First() == aTempShape) {
- const TopoDS_Shape& aTL = aTempListOfShape.Last();
- if(aTempMap.Add(aTL)) {
- aTempList.Append(aTL);
- aFreeShapes.Remove(aTL);
+ TopTools_ListOfShape::Iterator anIt1(aTempListOfShape);
+ for (; anIt1.More(); anIt1.Next()) {
+ if (anIt1.Value() == aTempShape) {
+ TopTools_ListOfShape::Iterator anIt2(aTempListOfShape);
+ for (; anIt2.More(); anIt2.Next())
+ {
+ if (anIt2.Value() != anIt1.Value()) {
+ if (aTempMap.Add(anIt2.Value())) {
+ aTempList.Append(anIt2.Value());
+ aFreeShapes.Remove(anIt2.Value());
+ }
+ }
+ }
+ aTempListOfShape.Clear();
+ break;
}
- aTempListOfShape.Clear();
- } else if(aTempListOfShape.Last() == aTempShape) {
- const TopoDS_Shape& aTF = aTempListOfShape.First();
- if(aTempMap.Add(aTF)) {
- aTempList.Append(aTF);
- aFreeShapes.Remove(aTF);
- }
- aTempListOfShape.Clear();
}
}
}
{
GeomShapePtr aResult = theCompound;
- if(!theCompound.get()) {
+ if (!theCompound.get()) {
return aResult;
}
addSimpleShapeToList(anInShape, anUngroupedShapes);
// Iterate over all shapes and find shapes with shared vertices.
- TopTools_ListOfShape aMapOrder;
- BOPCol_DataMapOfShapeListOfShape aVertexShapesMap;
- for(NCollection_List<TopoDS_Shape>::Iterator aShapesIt(anUngroupedShapes);
- aShapesIt.More();
- aShapesIt.Next()) {
+ TopTools_ListOfShape allVertices;
+ TopTools_DataMapOfShapeListOfShape aVertexShapesMap;
+ for (NCollection_List<TopoDS_Shape>::Iterator aShapesIt(anUngroupedShapes);
+ aShapesIt.More();
+ aShapesIt.Next()) {
const TopoDS_Shape& aShape = aShapesIt.Value();
- for(TopExp_Explorer aShapeExp(aShape, TopAbs_VERTEX);
- aShapeExp.More();
- aShapeExp.Next()) {
+ for (TopExp_Explorer aShapeExp(aShape, TopAbs_VERTEX);
+ aShapeExp.More();
+ aShapeExp.Next()) {
const TopoDS_Shape& aVertex = aShapeExp.Current();
if (!aVertexShapesMap.IsBound(aVertex)) {
NCollection_List<TopoDS_Shape> aList;
aList.Append(aShape);
- aMapOrder.Append(aVertex);
+ allVertices.Append(aVertex);
aVertexShapesMap.Bind(aVertex, aList);
- } else {
- if(!aVertexShapesMap.ChangeFind(aVertex).Contains(aShape)) {
+ }
+ else {
+ if (!aVertexShapesMap.ChangeFind(aVertex).Contains(aShape)) {
aVertexShapesMap.ChangeFind(aVertex).Append(aShape);
}
}
}
// Iterate over the map and group shapes.
- NCollection_Vector<TopTools_ListOfShape> aGroups;
- while (!aMapOrder.IsEmpty()) {
+ NCollection_Vector<TopTools_ListOfShape> aGroups; // groups of shapes connected by vertices
+ while (!allVertices.IsEmpty()) {
// Get first group of shapes in map, and then unbind it.
- const TopoDS_Shape& aKey = aMapOrder.First();
- TopTools_ListOfShape aGroupedShapes = aVertexShapesMap.Find(aKey);
+ const TopoDS_Shape& aKey = allVertices.First();
+ TopTools_ListOfShape aConnectedShapes = aVertexShapesMap.Find(aKey);
aVertexShapesMap.UnBind(aKey);
- aMapOrder.Remove(aKey);
+ allVertices.Remove(aKey);
// Iterate over shapes in this group and add to it shapes from groups in map.
- for(TopTools_ListOfShape::Iterator aGroupIt(aGroupedShapes);
- aGroupIt.More(); aGroupIt.Next()) {
- const TopoDS_Shape& aGroupedShape = aGroupIt.Value();
+ for (TopTools_ListOfShape::Iterator aConnectedIt(aConnectedShapes);
+ aConnectedIt.More(); aConnectedIt.Next()) {
+ const TopoDS_Shape& aConnected = aConnectedIt.Value();
TopTools_ListOfShape aKeysToUnbind;
- for(TopTools_ListOfShape::Iterator aKeysIt(aMapOrder);
- aKeysIt.More();
- aKeysIt.Next()) {
- const TopTools_ListOfShape& aGroupInMap = aVertexShapesMap(aKeysIt.Value());
- if(!aGroupInMap.Contains(aGroupedShape)) {
- // Group in map does not containt shape from our group, so go to the next group in map.
+ for (TopTools_ListOfShape::Iterator aKeysIt(allVertices);
+ aKeysIt.More();
+ aKeysIt.Next()) {
+ const TopTools_ListOfShape& anOtherConnected = aVertexShapesMap(aKeysIt.Value());
+ if (!anOtherConnected.Contains(aConnected)) {
+ // Other connected group does not containt shape from our connected group
continue;
}
- // Iterate over shape in group in map, and add new shapes into our group.
- for(TopTools_ListOfShape::Iterator aGroupInMapIt(aGroupInMap);
- aGroupInMapIt.More();
- aGroupInMapIt.Next()) {
- const TopoDS_Shape& aShape = aGroupInMapIt.Value();
- if (!aGroupedShapes.Contains(aShape)) {
- aGroupedShapes.Append(aShape);
+ // Other is connected to our, so add them to our connected
+ for (TopTools_ListOfShape::Iterator anOtherIt(anOtherConnected);
+ anOtherIt.More();
+ anOtherIt.Next()) {
+ const TopoDS_Shape& aShape = anOtherIt.Value();
+ if (!aConnectedShapes.Contains(aShape)) {
+ aConnectedShapes.Append(aShape);
}
}
// Save key to unbind from this map.
aKeysToUnbind.Append(aKeysIt.Value());
}
// Unbind groups from map that we added to our group.
- for(TopTools_ListOfShape::Iterator aKeysIt(aKeysToUnbind);
- aKeysIt.More();
- aKeysIt.Next()) {
+ for (TopTools_ListOfShape::Iterator aKeysIt(aKeysToUnbind);
+ aKeysIt.More();
+ aKeysIt.Next()) {
aVertexShapesMap.UnBind(aKeysIt.Value());
- aMapOrder.Remove(aKeysIt.Value());
+ allVertices.Remove(aKeysIt.Value());
}
}
- // Sort shapes.
+ // Sort shapes from the most complicated to the simplest ones
TopTools_ListOfShape aSortedGroup;
- for(int aST = TopAbs_COMPOUND; aST <= TopAbs_SHAPE; ++aST) {
- TopTools_ListOfShape::Iterator anIt(aGroupedShapes);
+ for (int aST = TopAbs_COMPOUND; aST <= TopAbs_SHAPE; ++aST) {
+ TopTools_ListOfShape::Iterator anIt(aConnectedShapes);
while (anIt.More()) {
- if(anIt.Value().ShapeType() == aST) {
+ if (anIt.Value().ShapeType() == aST) {
aSortedGroup.Append(anIt.Value());
- aGroupedShapes.Remove(anIt);
- } else {
+ aConnectedShapes.Remove(anIt);
+ }
+ else {
anIt.Next();
}
}
}
aBOP.Perform();
-#ifdef USE_OCCT_720
if (aBOP.HasErrors())
return;
-#else
- if (aBOP.ErrorStatus())
- return;
-#endif
// Collect splits
const TopTools_ListOfShape& aSplits = aBOP.Modified(aBaseEdge);
}
aBOP.Perform();
-#ifdef USE_OCCT_720
if (aBOP.HasErrors())
return;
-#else
- if (aBOP.ErrorStatus())
- return;
-#endif
// Collect splits
const TopTools_ListOfShape& aSplits = aBOP.Modified(aBaseEdge);
aCentreOfMassPoint.Z()-aPoint.Z()));
return aDir;
}
+
+//==================================================================================================
+static TopoDS_Wire fixParametricGaps(const TopoDS_Wire& theWire)
+{
+ TopoDS_Wire aFixedWire;
+ Handle(Geom_Curve) aPrevCurve;
+ double aPrevLastParam = 0.0;
+
+ BRep_Builder aBuilder;
+ aBuilder.MakeWire(aFixedWire);
+
+ BRepTools_WireExplorer aWExp(theWire);
+ for (; aWExp.More(); aWExp.Next()) {
+ TopoDS_Edge anEdge = aWExp.Current();
+ double aFirst, aLast;
+ Handle(Geom_Curve) aCurve = BRep_Tool::Curve(anEdge, aFirst, aLast);
+ if (aCurve == aPrevCurve) {
+ // if parametric gap occurs, create new edge based on the copied curve
+ aCurve = Handle(Geom_Curve)::DownCast(aCurve->Copy());
+ TopoDS_Vertex aV1, aV2;
+ TopExp::Vertices(anEdge, aV1, aV2);
+ anEdge = TopoDS::Edge(anEdge.EmptyCopied());
+ aBuilder.UpdateEdge(anEdge, aCurve, BRep_Tool::Tolerance(anEdge));
+ aBuilder.Add(anEdge, aV1);
+ aBuilder.Add(anEdge, aV2);
+ }
+
+ aBuilder.Add(aFixedWire, anEdge);
+
+ aPrevCurve = aCurve;
+ aPrevLastParam = aLast;
+ }
+
+ return aFixedWire;
+}
+
+std::shared_ptr<GeomAPI_Edge> GeomAlgoAPI_ShapeTools::wireToEdge(
+ const std::shared_ptr<GeomAPI_Wire>& theWire)
+{
+ GeomEdgePtr anEdge;
+ if (theWire) {
+ TopoDS_Wire aWire = theWire->impl<TopoDS_Wire>();
+ // Workaround: when concatenate a wire consisting of two edges based on the same B-spline curve
+ // (non-periodic, but having equal start and end points), first of which is placed at the end
+ // on the curve and second is placed at the start, this workaround copies second curve to avoid
+ // treating these edges as a single curve by setting trim parameters.
+ aWire = fixParametricGaps(aWire);
+ TopoDS_Edge aNewEdge = BRepAlgo::ConcatenateWireC0(aWire);
+ anEdge = GeomEdgePtr(new GeomAPI_Edge);
+ anEdge->setImpl(new TopoDS_Edge(aNewEdge));
+ }
+ return anEdge;
+}