1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-03 17:56:21 +03:00

0032530: Data Exchange, RWGltf_CafWriter - add option merging Faces within the Part

Added RWGltf_CafWriter::ToMergeFaces() property disabled by default.
RWMesh_MaterialMap - fixed creation of texture folder within working dir ".".
XCAFDoc_VisMaterial::FillMaterialAspect() - added clamping of too small shininess values.

Added options -mergefaces and -splitindices16 to WriteGltf for new feature.
Added -systemCoordSys option to WriteGltf for consistency with WriteObj.
This commit is contained in:
kgv 2021-08-11 18:48:23 +03:00 committed by bugmaster
parent fd42e7645d
commit 11c23250dc
17 changed files with 568 additions and 480 deletions

View File

@ -13,6 +13,7 @@
#include <RWGltf_CafWriter.hxx>
#include <BRep_Builder.hxx>
#include <gp_Quaternion.hxx>
#include <Message.hxx>
#include <Message_Messenger.hxx>
@ -23,6 +24,7 @@
#include <OSD_Path.hxx>
#include <Poly_Triangulation.hxx>
#include <RWGltf_GltfAccessorLayout.hxx>
#include <RWGltf_GltfArrayType.hxx>
#include <RWGltf_GltfMaterialMap.hxx>
#include <RWGltf_GltfPrimitiveMode.hxx>
#include <RWGltf_GltfRootElement.hxx>
@ -32,6 +34,7 @@
#include <TDataStd_Name.hxx>
#include <TDF_Tool.hxx>
#include <TDocStd_Document.hxx>
#include <TopoDS_Compound.hxx>
#include <XCAFDoc_DocumentTool.hxx>
#include <XCAFDoc_ShapeTool.hxx>
#include <XCAFPrs_DocumentExplorer.hxx>
@ -95,6 +98,8 @@ RWGltf_CafWriter::RWGltf_CafWriter (const TCollection_AsciiString& theFile,
myIsBinary (theIsBinary),
myIsForcedUVExport (false),
myToEmbedTexturesInGlb (true),
myToMergeFaces (false),
myToSplitIndices16 (false),
myBinDataLen64 (0)
{
myCSTrsf.SetOutputLengthUnit (1.0); // meters
@ -146,11 +151,19 @@ void RWGltf_CafWriter::saveNodes (RWGltf_GltfFace& theGltfFace,
const RWMesh_FaceIterator& theFaceIter,
Standard_Integer& theAccessorNb) const
{
theGltfFace.NodePos.Id = theAccessorNb++;
theGltfFace.NodePos.Count = theFaceIter.NbNodes();
theGltfFace.NodePos.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewPos.ByteOffset;
theGltfFace.NodePos.Type = RWGltf_GltfAccessorLayout_Vec3;
theGltfFace.NodePos.ComponentType = RWGltf_GltfAccessorCompType_Float32;
if (theGltfFace.NodePos.Id == RWGltf_GltfAccessor::INVALID_ID)
{
theGltfFace.NodePos.Id = theAccessorNb++;
theGltfFace.NodePos.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewPos.ByteOffset;
theGltfFace.NodePos.Type = RWGltf_GltfAccessorLayout_Vec3;
theGltfFace.NodePos.ComponentType = RWGltf_GltfAccessorCompType_Float32;
}
else
{
const int64_t aPos = theGltfFace.NodePos.ByteOffset + myBuffViewPos.ByteOffset + theGltfFace.NodePos.Count * sizeof(Graphic3d_Vec3);
Standard_ASSERT_RAISE (aPos == (int64_t )theBinFile.tellp(), "wrong offset");
}
theGltfFace.NodePos.Count += theFaceIter.NbNodes();
const Standard_Integer aNodeUpper = theFaceIter.NodeUpper();
for (Standard_Integer aNodeIter = theFaceIter.NodeLower(); aNodeIter <= aNodeUpper; ++aNodeIter)
@ -176,11 +189,19 @@ void RWGltf_CafWriter::saveNormals (RWGltf_GltfFace& theGltfFace,
return;
}
theGltfFace.NodeNorm.Id = theAccessorNb++;
theGltfFace.NodeNorm.Count = theFaceIter.NbNodes();
theGltfFace.NodeNorm.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewNorm.ByteOffset;
theGltfFace.NodeNorm.Type = RWGltf_GltfAccessorLayout_Vec3;
theGltfFace.NodeNorm.ComponentType = RWGltf_GltfAccessorCompType_Float32;
if (theGltfFace.NodeNorm.Id == RWGltf_GltfAccessor::INVALID_ID)
{
theGltfFace.NodeNorm.Id = theAccessorNb++;
theGltfFace.NodeNorm.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewNorm.ByteOffset;
theGltfFace.NodeNorm.Type = RWGltf_GltfAccessorLayout_Vec3;
theGltfFace.NodeNorm.ComponentType = RWGltf_GltfAccessorCompType_Float32;
}
else
{
const int64_t aPos = theGltfFace.NodeNorm.ByteOffset + myBuffViewNorm.ByteOffset + theGltfFace.NodeNorm.Count * sizeof(Graphic3d_Vec3);
Standard_ASSERT_RAISE (aPos == (int64_t )theBinFile.tellp(), "wrong offset");
}
theGltfFace.NodeNorm.Count += theFaceIter.NbNodes();
const Standard_Integer aNodeUpper = theFaceIter.NodeUpper();
for (Standard_Integer aNodeIter = theFaceIter.NodeLower(); aNodeIter <= aNodeUpper; ++aNodeIter)
@ -222,11 +243,20 @@ void RWGltf_CafWriter::saveTextCoords (RWGltf_GltfFace& theGltfFace,
}
}
theGltfFace.NodeUV.Id = theAccessorNb++;
theGltfFace.NodeUV.Count = theFaceIter.NbNodes();
theGltfFace.NodeUV.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewTextCoord.ByteOffset;
theGltfFace.NodeUV.Type = RWGltf_GltfAccessorLayout_Vec2;
theGltfFace.NodeUV.ComponentType = RWGltf_GltfAccessorCompType_Float32;
if (theGltfFace.NodeUV.Id == RWGltf_GltfAccessor::INVALID_ID)
{
theGltfFace.NodeUV.Id = theAccessorNb++;
theGltfFace.NodeUV.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewTextCoord.ByteOffset;
theGltfFace.NodeUV.Type = RWGltf_GltfAccessorLayout_Vec2;
theGltfFace.NodeUV.ComponentType = RWGltf_GltfAccessorCompType_Float32;
}
else
{
const int64_t aPos = theGltfFace.NodeUV.ByteOffset + myBuffViewTextCoord.ByteOffset + theGltfFace.NodeUV.Count * sizeof(Graphic3d_Vec2);
Standard_ASSERT_RAISE (aPos == (int64_t )theBinFile.tellp(), "wrong offset");
}
theGltfFace.NodeUV.Count += theFaceIter.NbNodes();
const Standard_Integer aNodeUpper = theFaceIter.NodeUpper();
for (Standard_Integer aNodeIter = theFaceIter.NodeLower(); aNodeIter <= aNodeUpper; ++aNodeIter)
{
@ -245,22 +275,36 @@ void RWGltf_CafWriter::saveIndices (RWGltf_GltfFace& theGltfFace,
const RWMesh_FaceIterator& theFaceIter,
Standard_Integer& theAccessorNb)
{
theGltfFace.Indices.Id = theAccessorNb++;
theGltfFace.Indices.Count = theFaceIter.NbTriangles() * 3;
theGltfFace.Indices.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewInd.ByteOffset;
theGltfFace.Indices.Type = RWGltf_GltfAccessorLayout_Scalar;
theGltfFace.Indices.ComponentType = theGltfFace.NodePos.Count > std::numeric_limits<uint16_t>::max()
? RWGltf_GltfAccessorCompType_UInt32
: RWGltf_GltfAccessorCompType_UInt16;
if (theGltfFace.Indices.Id == RWGltf_GltfAccessor::INVALID_ID)
{
theGltfFace.Indices.Id = theAccessorNb++;
theGltfFace.Indices.ByteOffset = (int64_t )theBinFile.tellp() - myBuffViewInd.ByteOffset;
theGltfFace.Indices.Type = RWGltf_GltfAccessorLayout_Scalar;
theGltfFace.Indices.ComponentType = theGltfFace.NodePos.Count > std::numeric_limits<uint16_t>::max()
? RWGltf_GltfAccessorCompType_UInt32
: RWGltf_GltfAccessorCompType_UInt16;
}
else
{
const int64_t aRefPos = (int64_t )theBinFile.tellp();
const int64_t aPos = theGltfFace.Indices.ByteOffset
+ myBuffViewInd.ByteOffset
+ theGltfFace.Indices.Count * (theGltfFace.Indices.ComponentType == RWGltf_GltfAccessorCompType_UInt32 ? sizeof(uint32_t) : sizeof(uint16_t));
Standard_ASSERT_RAISE (aPos == aRefPos, "wrong offset");
}
const Standard_Integer aNodeFirst = theGltfFace.NbIndexedNodes - theFaceIter.ElemLower();
theGltfFace.NbIndexedNodes += theFaceIter.NbNodes();
theGltfFace.Indices.Count += theFaceIter.NbTriangles() * 3;
const Standard_Integer anElemLower = theFaceIter.ElemLower();
const Standard_Integer anElemUpper = theFaceIter.ElemUpper();
for (Standard_Integer anElemIter = anElemLower; anElemIter <= anElemUpper; ++anElemIter)
{
Poly_Triangle aTri = theFaceIter.TriangleOriented (anElemIter);
aTri(1) -= anElemLower;
aTri(2) -= anElemLower;
aTri(3) -= anElemLower;
aTri(1) += aNodeFirst;
aTri(2) += aNodeFirst;
aTri(3) += aNodeFirst;
if (theGltfFace.Indices.ComponentType == RWGltf_GltfAccessorCompType_UInt16)
{
writeTriangle16 (theBinFile, NCollection_Vec3<uint16_t>((uint16_t)aTri(1), (uint16_t)aTri(2), (uint16_t)aTri(3)));
@ -270,16 +314,6 @@ void RWGltf_CafWriter::saveIndices (RWGltf_GltfFace& theGltfFace,
writeTriangle32 (theBinFile, Graphic3d_Vec3i (aTri(1), aTri(2), aTri(3)));
}
}
if (theGltfFace.Indices.ComponentType == RWGltf_GltfAccessorCompType_UInt16)
{
// alignment by 4 bytes
int64_t aContentLen64 = (int64_t)theBinFile.tellp();
while (aContentLen64 % 4 != 0)
{
theBinFile.write (" ", 1);
++aContentLen64;
}
}
}
// =======================================================================
@ -369,11 +403,16 @@ bool RWGltf_CafWriter::writeBinData (const Handle(TDocStd_Document)& theDocument
}
Message_ProgressScope aPSentryBin (theProgress, "Binary data", 4);
const RWGltf_GltfArrayType anArrTypes[4] =
{
RWGltf_GltfArrayType_Position,
RWGltf_GltfArrayType_Normal,
RWGltf_GltfArrayType_TCoord0,
RWGltf_GltfArrayType_Indices
};
Standard_Integer aNbAccessors = 0;
// write positions
myBuffViewPos.ByteOffset = aBinFile->tellp();
// dispatch faces
NCollection_DataMap<XCAFPrs_Style, Handle(RWGltf_GltfFace), XCAFPrs_Style> aMergedFaces;
for (XCAFPrs_DocumentExplorer aDocExplorer (theDocument, theRootLabels, XCAFPrs_DocumentExplorerFlags_OnlyLeafNodes);
aDocExplorer.More() && aPSentryBin.More(); aDocExplorer.Next())
{
@ -385,149 +424,165 @@ bool RWGltf_CafWriter::writeBinData (const Handle(TDocStd_Document)& theDocument
}
// transformation will be stored at scene nodes
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More() && aPSentryBin.More(); aFaceIter.Next())
aMergedFaces.Clear (false);
RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style);
if (myToMergeFaces)
{
if (myBinDataMap.IsBound (aFaceIter.Face())
|| toSkipFaceMesh (aFaceIter))
if (myBinDataMap.Contains (aFaceIter.ExploredShape()))
{
continue;
}
RWGltf_GltfFace aGltfFace;
saveNodes (aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
if (!aBinFile->good())
Handle(RWGltf_GltfFaceList) aGltfFaceList = new RWGltf_GltfFaceList();
myBinDataMap.Add (aFaceIter.ExploredShape(), aGltfFaceList);
for (; aFaceIter.More() && aPSentryBin.More(); aFaceIter.Next())
{
Message::SendFail (TCollection_AsciiString ("File '") + myBinFileNameFull + "' can not be written");
return false;
}
if (toSkipFaceMesh (aFaceIter))
{
continue;
}
myBinDataMap.Bind (aFaceIter.Face(), aGltfFace);
Handle(RWGltf_GltfFace) aGltfFace;
if (!aMergedFaces.Find (aFaceIter.FaceStyle(), aGltfFace))
{
aGltfFace = new RWGltf_GltfFace();
aGltfFaceList->Append (aGltfFace);
aGltfFace->Shape = aFaceIter.Face();
aGltfFace->Style = aFaceIter.FaceStyle();
aGltfFace->NbIndexedNodes = aFaceIter.NbNodes();
aMergedFaces.Bind (aFaceIter.FaceStyle(), aGltfFace);
}
else if (myToSplitIndices16
&& aGltfFace->NbIndexedNodes < std::numeric_limits<uint16_t>::max()
&& (aGltfFace->NbIndexedNodes + aFaceIter.NbNodes()) >= std::numeric_limits<uint16_t>::max())
{
aMergedFaces.UnBind (aFaceIter.FaceStyle());
aGltfFace = new RWGltf_GltfFace();
aGltfFaceList->Append (aGltfFace);
aGltfFace->Shape = aFaceIter.Face();
aGltfFace->Style = aFaceIter.FaceStyle();
aGltfFace->NbIndexedNodes = aFaceIter.NbNodes();
aMergedFaces.Bind (aFaceIter.FaceStyle(), aGltfFace);
}
else
{
if (aGltfFace->Shape.ShapeType() != TopAbs_COMPOUND)
{
TopoDS_Shape anOldShape = aGltfFace->Shape;
TopoDS_Compound aComp;
BRep_Builder().MakeCompound (aComp);
BRep_Builder().Add (aComp, anOldShape);
aGltfFace->Shape = aComp;
}
BRep_Builder().Add (aGltfFace->Shape, aFaceIter.Face());
aGltfFace->NbIndexedNodes += aFaceIter.NbNodes();
}
}
}
else
{
for (; aFaceIter.More() && aPSentryBin.More(); aFaceIter.Next())
{
if (toSkipFaceMesh (aFaceIter)
|| myBinDataMap.Contains (aFaceIter.Face()))
{
continue;
}
Handle(RWGltf_GltfFaceList) aGltfFaceList = new RWGltf_GltfFaceList();
Handle(RWGltf_GltfFace) aGltfFace = new RWGltf_GltfFace();
aGltfFace->Shape = aFaceIter.Face();
aGltfFace->Style = aFaceIter.FaceStyle();
aGltfFaceList->Append (aGltfFace);
myBinDataMap.Add (aFaceIter.Face(), aGltfFaceList);
}
}
}
myBuffViewPos.ByteLength = (int64_t )aBinFile->tellp() - myBuffViewPos.ByteOffset;
if (!aPSentryBin.More())
Standard_Integer aNbAccessors = 0;
NCollection_Map<Handle(RWGltf_GltfFaceList)> aWrittenFaces;
for (Standard_Integer aTypeIter = 0; aTypeIter < 4; ++aTypeIter)
{
return false;
}
aPSentryBin.Next();
// write normals
myBuffViewNorm.ByteOffset = aBinFile->tellp();
for (XCAFPrs_DocumentExplorer aDocExplorer (theDocument, theRootLabels, XCAFPrs_DocumentExplorerFlags_OnlyLeafNodes);
aDocExplorer.More() && aPSentryBin.More(); aDocExplorer.Next())
{
const XCAFPrs_DocumentNode& aDocNode = aDocExplorer.Current();
if (theLabelFilter != NULL
&& !theLabelFilter->Contains (aDocNode.Id))
const RWGltf_GltfArrayType anArrType = (RWGltf_GltfArrayType )anArrTypes[aTypeIter];
RWGltf_GltfBufferView* aBuffView = NULL;
switch (anArrType)
{
continue;
case RWGltf_GltfArrayType_Position: aBuffView = &myBuffViewPos; break;
case RWGltf_GltfArrayType_Normal: aBuffView = &myBuffViewNorm; break;
case RWGltf_GltfArrayType_TCoord0: aBuffView = &myBuffViewTextCoord; break;
case RWGltf_GltfArrayType_Indices: aBuffView = &myBuffViewInd; break;
default: break;
}
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More() && aPSentryBin.More(); aFaceIter.Next())
aBuffView->ByteOffset = aBinFile->tellp();
aWrittenFaces.Clear (false);
for (ShapeToGltfFaceMap::Iterator aBinDataIter (myBinDataMap); aBinDataIter.More() && aPSentryBin.More(); aBinDataIter.Next())
{
if (toSkipFaceMesh (aFaceIter))
const Handle(RWGltf_GltfFaceList)& aGltfFaceList = aBinDataIter.Value();
if (!aWrittenFaces.Add (aGltfFaceList)) // skip repeating faces
{
continue;
}
RWGltf_GltfFace& aGltfFace = myBinDataMap.ChangeFind (aFaceIter.Face());
if (aGltfFace.NodeNorm.Id != RWGltf_GltfAccessor::INVALID_ID)
for (RWGltf_GltfFaceList::Iterator aGltfFaceIter (*aGltfFaceList); aGltfFaceIter.More() && aPSentryBin.More(); aGltfFaceIter.Next())
{
continue;
}
const Handle(RWGltf_GltfFace)& aGltfFace = aGltfFaceIter.Value();
for (RWMesh_FaceIterator aFaceIter (aGltfFace->Shape, aGltfFace->Style); aFaceIter.More() && aPSentryBin.More(); aFaceIter.Next())
{
switch (anArrType)
{
case RWGltf_GltfArrayType_Position:
{
aGltfFace->NbIndexedNodes = 0; // reset to zero before RWGltf_GltfArrayType_Indices step
saveNodes (*aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
break;
}
case RWGltf_GltfArrayType_Normal:
{
saveNormals (*aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
break;
}
case RWGltf_GltfArrayType_TCoord0:
{
saveTextCoords (*aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
break;
}
case RWGltf_GltfArrayType_Indices:
{
saveIndices (*aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
break;
}
default:
{
break;
}
}
saveNormals (aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
if (!aBinFile->good())
{
Message::SendFail (TCollection_AsciiString ("File '") + myBinFileNameFull + "' cannot be written");
return false;
}
}
if (!aBinFile->good())
{
Message::SendFail (TCollection_AsciiString ("File '") + myBinFileNameFull + "' can not be written");
return false;
// add alignment by 4 bytes (might happen on RWGltf_GltfAccessorCompType_UInt16 indices)
int64_t aContentLen64 = (int64_t)aBinFile->tellp();
while (aContentLen64 % 4 != 0)
{
aBinFile->write (" ", 1);
++aContentLen64;
}
}
}
}
myBuffViewNorm.ByteLength = (int64_t )aBinFile->tellp() - myBuffViewNorm.ByteOffset;
if (!aPSentryBin.More())
{
return false;
}
aPSentryBin.Next();
// write texture coordinates
myBuffViewTextCoord.ByteOffset = aBinFile->tellp();
for (XCAFPrs_DocumentExplorer aDocExplorer (theDocument, theRootLabels, XCAFPrs_DocumentExplorerFlags_OnlyLeafNodes);
aDocExplorer.More() && aPSentryBin.More(); aDocExplorer.Next())
{
const XCAFPrs_DocumentNode& aDocNode = aDocExplorer.Current();
if (theLabelFilter != NULL
&& !theLabelFilter->Contains (aDocNode.Id))
aBuffView->ByteLength = (int64_t )aBinFile->tellp() - aBuffView->ByteOffset;
if (!aPSentryBin.More())
{
continue;
return false;
}
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More() && aPSentryBin.More(); aFaceIter.Next())
{
if (toSkipFaceMesh (aFaceIter))
{
continue;
}
RWGltf_GltfFace& aGltfFace = myBinDataMap.ChangeFind (aFaceIter.Face());
if (aGltfFace.NodeUV.Id != RWGltf_GltfAccessor::INVALID_ID)
{
continue;
}
saveTextCoords (aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
if (!aBinFile->good())
{
Message::SendFail (TCollection_AsciiString ("File '") + myBinFileNameFull + "' can not be written");
return false;
}
}
aPSentryBin.Next();
}
myBuffViewTextCoord.ByteLength = (int64_t )aBinFile->tellp() - myBuffViewTextCoord.ByteOffset;
if (!aPSentryBin.More())
{
return false;
}
aPSentryBin.Next();
// write indices
myBuffViewInd.ByteOffset = aBinFile->tellp();
for (XCAFPrs_DocumentExplorer aDocExplorer (theDocument, theRootLabels, XCAFPrs_DocumentExplorerFlags_OnlyLeafNodes);
aDocExplorer.More() && aPSentryBin.More(); aDocExplorer.Next())
{
const XCAFPrs_DocumentNode& aDocNode = aDocExplorer.Current();
if (theLabelFilter != NULL
&& !theLabelFilter->Contains (aDocNode.Id))
{
continue;
}
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More() && aPSentryBin.More(); aFaceIter.Next())
{
if (toSkipFaceMesh (aFaceIter))
{
continue;
}
RWGltf_GltfFace& aGltfFace = myBinDataMap.ChangeFind (aFaceIter.Face());
if (aGltfFace.Indices.Id != RWGltf_GltfAccessor::INVALID_ID)
{
continue;
}
saveIndices (aGltfFace, *aBinFile, aFaceIter, aNbAccessors);
if (!aBinFile->good())
{
Message::SendFail (TCollection_AsciiString ("File '") + myBinFileNameFull + "' can not be written");
return false;
}
}
}
myBuffViewInd.ByteLength = (int64_t )aBinFile->tellp() - myBuffViewInd.ByteOffset;
if (myIsBinary
&& myToEmbedTexturesInGlb)
@ -579,7 +634,7 @@ bool RWGltf_CafWriter::writeBinData (const Handle(TDocStd_Document)& theDocument
aBinFile->flush();
if (!aBinFile->good())
{
Message::SendFail (TCollection_AsciiString ("File '") + myBinFileNameFull + "' can not be written");
Message::SendFail (TCollection_AsciiString ("File '") + myBinFileNameFull + "' cannot be written");
return false;
}
aBinFile.reset();
@ -788,7 +843,7 @@ bool RWGltf_CafWriter::writeJson (const Handle(TDocStd_Document)& theDocument,
// function : writeAccessors
// purpose :
// =======================================================================
void RWGltf_CafWriter::writeAccessors (const RWGltf_GltfSceneNodeMap& theSceneNodeMap)
void RWGltf_CafWriter::writeAccessors (const RWGltf_GltfSceneNodeMap& )
{
#ifdef HAVE_RAPIDJSON
Standard_ProgramError_Raise_if (myWriter.get() == NULL, "Internal error: RWGltf_CafWriter::writeAccessors()");
@ -796,74 +851,61 @@ void RWGltf_CafWriter::writeAccessors (const RWGltf_GltfSceneNodeMap& theSceneNo
myWriter->Key (RWGltf_GltfRootElementName (RWGltf_GltfRootElement_Accessors));
myWriter->StartArray();
NCollection_Map<TopoDS_Shape, TopTools_ShapeMapHasher> aWrittenFaces;
for (RWGltf_GltfSceneNodeMap::Iterator aSceneNodeIter (theSceneNodeMap); aSceneNodeIter.More(); aSceneNodeIter.Next())
const RWGltf_GltfArrayType anArrTypes[4] =
{
const XCAFPrs_DocumentNode& aDocNode = aSceneNodeIter.Value();
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More(); aFaceIter.Next())
RWGltf_GltfArrayType_Position,
RWGltf_GltfArrayType_Normal,
RWGltf_GltfArrayType_TCoord0,
RWGltf_GltfArrayType_Indices
};
NCollection_Map<Handle(RWGltf_GltfFaceList)> aWrittenFaces;
for (Standard_Integer aTypeIter = 0; aTypeIter < 4; ++aTypeIter)
{
const RWGltf_GltfArrayType anArrType = (RWGltf_GltfArrayType )anArrTypes[aTypeIter];
aWrittenFaces.Clear (false);
for (ShapeToGltfFaceMap::Iterator aBinDataIter (myBinDataMap); aBinDataIter.More(); aBinDataIter.Next())
{
if (!aWrittenFaces.Add (aFaceIter.Face()) // skip repeating faces
|| toSkipFaceMesh (aFaceIter))
const Handle(RWGltf_GltfFaceList)& aGltfFaceList = aBinDataIter.Value();
if (!aWrittenFaces.Add (aGltfFaceList)) // skip repeating faces
{
continue;
}
const RWGltf_GltfFace& aGltfFace = myBinDataMap.Find (aFaceIter.Face());
writePositions (aGltfFace);
}
}
aWrittenFaces.Clear();
for (RWGltf_GltfSceneNodeMap::Iterator aSceneNodeIter (theSceneNodeMap); aSceneNodeIter.More(); aSceneNodeIter.Next())
{
const XCAFPrs_DocumentNode& aDocNode = aSceneNodeIter.Value();
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More(); aFaceIter.Next())
{
if (!aWrittenFaces.Add (aFaceIter.Face()) // skip repeating faces
|| toSkipFaceMesh (aFaceIter))
for (RWGltf_GltfFaceList::Iterator aFaceIter (*aGltfFaceList); aFaceIter.More(); aFaceIter.Next())
{
continue;
const Handle(RWGltf_GltfFace)& aGltfFace = aFaceIter.Value();
switch (anArrType)
{
case RWGltf_GltfArrayType_Position:
{
writePositions (*aGltfFace);
break;
}
case RWGltf_GltfArrayType_Normal:
{
writeNormals (*aGltfFace);
break;
}
case RWGltf_GltfArrayType_TCoord0:
{
writeTextCoords (*aGltfFace);
break;
}
case RWGltf_GltfArrayType_Indices:
{
writeIndices (*aGltfFace);
break;
}
default:
{
break;
}
}
}
const RWGltf_GltfFace& aGltfFace = myBinDataMap.Find (aFaceIter.Face());
writeNormals (aGltfFace);
}
}
aWrittenFaces.Clear();
for (RWGltf_GltfSceneNodeMap::Iterator aSceneNodeIter (theSceneNodeMap); aSceneNodeIter.More(); aSceneNodeIter.Next())
{
const XCAFPrs_DocumentNode& aDocNode = aSceneNodeIter.Value();
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More(); aFaceIter.Next())
{
if (!aWrittenFaces.Add (aFaceIter.Face()) // skip repeating faces
|| toSkipFaceMesh (aFaceIter))
{
continue;
}
const RWGltf_GltfFace& aGltfFace = myBinDataMap.Find (aFaceIter.Face());
writeTextCoords (aGltfFace);
}
}
aWrittenFaces.Clear();
for (RWGltf_GltfSceneNodeMap::Iterator aSceneNodeIter (theSceneNodeMap); aSceneNodeIter.More(); aSceneNodeIter.Next())
{
const XCAFPrs_DocumentNode& aDocNode = aSceneNodeIter.Value();
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More(); aFaceIter.Next())
{
if (!aWrittenFaces.Add (aFaceIter.Face()) // skip repeating faces
|| toSkipFaceMesh (aFaceIter))
{
continue;
}
const RWGltf_GltfFace& aGltfFace = myBinDataMap.Find (aFaceIter.Face());
writeIndices (aGltfFace);
}
}
myWriter->EndArray();
#else
(void )theSceneNodeMap;
#endif
}
@ -1280,6 +1322,67 @@ void RWGltf_CafWriter::writeMaterials (const RWGltf_GltfSceneNodeMap& theSceneNo
#endif
}
// =======================================================================
// function : writePrimArray
// purpose :
// =======================================================================
void RWGltf_CafWriter::writePrimArray (const RWGltf_GltfFace& theGltfFace,
const TCollection_AsciiString& theName,
bool& theToStartPrims)
{
#ifdef HAVE_RAPIDJSON
if (theToStartPrims)
{
theToStartPrims = false;
myWriter->StartObject();
if (!theName.IsEmpty())
{
myWriter->Key ("name");
myWriter->String (theName.ToCString());
}
myWriter->Key ("primitives");
myWriter->StartArray();
}
const TCollection_AsciiString aMatId = myMaterialMap->FindMaterial (theGltfFace.Style);
myWriter->StartObject();
{
myWriter->Key ("attributes");
myWriter->StartObject();
{
if (theGltfFace.NodeNorm.Id != RWGltf_GltfAccessor::INVALID_ID)
{
myWriter->Key ("NORMAL");
myWriter->Int (theGltfFace.NodeNorm.Id);
}
myWriter->Key ("POSITION");
myWriter->Int (theGltfFace.NodePos.Id);
if (theGltfFace.NodeUV.Id != RWGltf_GltfAccessor::INVALID_ID)
{
myWriter->Key ("TEXCOORD_0");
myWriter->Int (theGltfFace.NodeUV.Id);
}
}
myWriter->EndObject();
myWriter->Key ("indices");
myWriter->Int (theGltfFace.Indices.Id);
if (!aMatId.IsEmpty())
{
myWriter->Key ("material");
myWriter->Int (aMatId.IntegerValue());
}
myWriter->Key ("mode");
myWriter->Int (RWGltf_GltfPrimitiveMode_Triangles);
}
myWriter->EndObject();
#else
(void )theGltfFace;
(void )theName;
(void )theToStartPrims;
#endif
}
// =======================================================================
// function : writeMeshes
// purpose :
@ -1292,6 +1395,7 @@ void RWGltf_CafWriter::writeMeshes (const RWGltf_GltfSceneNodeMap& theSceneNodeM
myWriter->Key (RWGltf_GltfRootElementName (RWGltf_GltfRootElement_Meshes));
myWriter->StartArray();
NCollection_Map<Handle(RWGltf_GltfFaceList)> aWrittenFaces;
for (RWGltf_GltfSceneNodeMap::Iterator aSceneNodeIter (theSceneNodeMap); aSceneNodeIter.More(); aSceneNodeIter.Next())
{
const XCAFPrs_DocumentNode& aDocNode = aSceneNodeIter.Value();
@ -1299,59 +1403,48 @@ void RWGltf_CafWriter::writeMeshes (const RWGltf_GltfSceneNodeMap& theSceneNodeM
bool toStartPrims = true;
Standard_Integer aNbFacesInNode = 0;
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More(); aFaceIter.Next(), ++aNbFacesInNode)
aWrittenFaces.Clear (false);
if (myToMergeFaces)
{
if (toSkipFaceMesh (aFaceIter))
TopoDS_Shape aShape;
if (!XCAFDoc_ShapeTool::GetShape (aDocNode.RefLabel, aShape)
|| aShape.IsNull())
{
continue;
}
if (toStartPrims)
Handle(RWGltf_GltfFaceList) aGltfFaceList;
aShape.Location (TopLoc_Location());
myBinDataMap.FindFromKey (aShape, aGltfFaceList);
if (!aWrittenFaces.Add (aGltfFaceList))
{
toStartPrims = false;
myWriter->StartObject();
if (!aNodeName.IsEmpty())
{
myWriter->Key ("name");
myWriter->String (aNodeName.ToCString());
}
myWriter->Key ("primitives");
myWriter->StartArray();
continue;
}
const RWGltf_GltfFace& aGltfFace = myBinDataMap.Find (aFaceIter.Face());
const TCollection_AsciiString aMatId = myMaterialMap->FindMaterial (aFaceIter.FaceStyle());
myWriter->StartObject();
for (RWGltf_GltfFaceList::Iterator aFaceGroupIter (*aGltfFaceList); aFaceGroupIter.More(); aFaceGroupIter.Next())
{
myWriter->Key ("attributes");
myWriter->StartObject();
{
if (aGltfFace.NodeNorm.Id != RWGltf_GltfAccessor::INVALID_ID)
{
myWriter->Key ("NORMAL");
myWriter->Int (aGltfFace.NodeNorm.Id);
}
myWriter->Key ("POSITION");
myWriter->Int (aGltfFace.NodePos.Id);
if (aGltfFace.NodeUV.Id != RWGltf_GltfAccessor::INVALID_ID)
{
myWriter->Key ("TEXCOORD_0");
myWriter->Int (aGltfFace.NodeUV.Id);
}
}
myWriter->EndObject();
myWriter->Key ("indices");
myWriter->Int (aGltfFace.Indices.Id);
if (!aMatId.IsEmpty())
{
myWriter->Key ("material");
myWriter->Int (aMatId.IntegerValue());
}
myWriter->Key ("mode");
myWriter->Int (RWGltf_GltfPrimitiveMode_Triangles);
const Handle(RWGltf_GltfFace)& aGltfFace = aFaceGroupIter.Value();
writePrimArray (*aGltfFace, aNodeName, toStartPrims);
}
}
else
{
for (RWMesh_FaceIterator aFaceIter (aDocNode.RefLabel, TopLoc_Location(), true, aDocNode.Style); aFaceIter.More(); aFaceIter.Next(), ++aNbFacesInNode)
{
if (toSkipFaceMesh (aFaceIter))
{
continue;
}
const Handle(RWGltf_GltfFaceList)& aGltfFaceList = myBinDataMap.FindFromKey (aFaceIter.Face());
if (!aWrittenFaces.Add (aGltfFaceList))
{
continue;
}
const Handle(RWGltf_GltfFace)& aGltfFace = aGltfFaceList->First();
writePrimArray (*aGltfFace, aNodeName, toStartPrims);
}
myWriter->EndObject();
}
if (!toStartPrims)

View File

@ -99,6 +99,21 @@ public:
//! Set flag to write image textures into GLB file (binary gltf export).
void SetToEmbedTexturesInGlb (Standard_Boolean theToEmbedTexturesInGlb) { myToEmbedTexturesInGlb = theToEmbedTexturesInGlb; }
//! Return flag to merge faces within a single part; FALSE by default.
bool ToMergeFaces() const { return myToMergeFaces; }
//! Set flag to merge faces within a single part.
//! May reduce JSON size thanks to smaller number of primitive arrays.
void SetMergeFaces (bool theToMerge) { myToMergeFaces = theToMerge; }
//! Return flag to prefer keeping 16-bit indexes while merging face; FALSE by default.
bool ToSplitIndices16() const { return myToSplitIndices16; }
//! Set flag to prefer keeping 16-bit indexes while merging face.
//! Has effect only with ToMergeFaces() option turned ON.
//! May reduce binary data size thanks to smaller triangle indexes.
void SetSplitIndices16 (bool theToSplit) { myToSplitIndices16 = theToSplit; }
//! Write glTF file and associated binary file.
//! Triangulation data should be precomputed within shapes!
//! @param theDocument [in] input document
@ -262,6 +277,14 @@ protected:
//! @param theMaterialMap [in] map of materials
Standard_EXPORT virtual void writeMeshes (const RWGltf_GltfSceneNodeMap& theSceneNodeMap);
//! Write a primitive array to RWGltf_GltfRootElement_Meshes section.
//! @param[in] theGltfFace face to write
//! @param[in] theName primitive array name
//! @param[in,out] theToStartPrims flag indicating that primitive array has been started
Standard_EXPORT virtual void writePrimArray (const RWGltf_GltfFace& theGltfFace,
const TCollection_AsciiString& theName,
bool& theToStartPrims);
//! Write RWGltf_GltfRootElement_Nodes section.
//! @param theDocument [in] input document
//! @param theRootLabels [in] list of root shapes to export
@ -293,6 +316,10 @@ protected:
//! @param theMaterialMap [out] map of materials, filled with textures
Standard_EXPORT virtual void writeTextures (const RWGltf_GltfSceneNodeMap& theSceneNodeMap);
protected:
typedef NCollection_IndexedDataMap<TopoDS_Shape, Handle(RWGltf_GltfFaceList), TopTools_ShapeMapHasher> ShapeToGltfFaceMap;
protected:
TCollection_AsciiString myFile; //!< output glTF file
@ -304,6 +331,8 @@ protected:
Standard_Boolean myIsBinary; //!< flag to write into binary glTF format (.glb)
Standard_Boolean myIsForcedUVExport; //!< export UV coordinates even if there are no mapped texture
Standard_Boolean myToEmbedTexturesInGlb; //!< flag to write image textures into GLB file
Standard_Boolean myToMergeFaces; //!< flag to merge faces within a single part
Standard_Boolean myToSplitIndices16; //!< flag to prefer keeping 16-bit indexes while merging face
RWMesh_CoordinateSystemConverter myCSTrsf; //!< transformation from OCCT to glTF coordinate system
XCAFPrs_Style myDefaultStyle; //!< default material definition to be used for nodes with only color defined
@ -314,8 +343,7 @@ protected:
RWGltf_GltfBufferView myBuffViewNorm; //!< current buffer view with nodes normals
RWGltf_GltfBufferView myBuffViewTextCoord; //!< current buffer view with nodes UV coordinates
RWGltf_GltfBufferView myBuffViewInd; //!< current buffer view with triangulation indexes
NCollection_DataMap<TopoDS_Shape, RWGltf_GltfFace,
TopTools_ShapeMapHasher> myBinDataMap; //!< map for TopoDS_Face to glTF face (merging duplicates)
ShapeToGltfFaceMap myBinDataMap; //!< map for TopoDS_Face to glTF face (merging duplicates)
int64_t myBinDataLen64; //!< length of binary file
};

View File

@ -15,15 +15,27 @@
#ifndef _RWGltf_GltfFace_HeaderFile
#define _RWGltf_GltfFace_HeaderFile
#include <NCollection_List.hxx>
#include <NCollection_Shared.hxx>
#include <RWGltf_GltfAccessor.hxx>
#include <TopoDS_Shape.hxx>
#include <XCAFPrs_Style.hxx>
//! Low-level glTF data structure holding single Face (one primitive array) definition.
struct RWGltf_GltfFace
class RWGltf_GltfFace : public Standard_Transient
{
public:
RWGltf_GltfAccessor NodePos; //!< accessor for nodal positions
RWGltf_GltfAccessor NodeNorm; //!< accessor for nodal normals
RWGltf_GltfAccessor NodeUV; //!< accessor for nodal UV texture coordinates
RWGltf_GltfAccessor Indices; //!< accessor for indexes
TopoDS_Shape Shape; //!< original Face or face list
XCAFPrs_Style Style; //!< face style
Standard_Integer NbIndexedNodes; //!< transient variable for merging several faces into one while writing Indices
RWGltf_GltfFace() : NbIndexedNodes (0) {}
};
typedef NCollection_Shared<NCollection_List<Handle(RWGltf_GltfFace)>> RWGltf_GltfFaceList;
#endif // _RWGltf_GltfFace_HeaderFile

View File

@ -53,6 +53,28 @@ RWMesh_FaceIterator::RWMesh_FaceIterator (const TDF_Label& theLabel,
Next();
}
// =======================================================================
// function : RWMesh_FaceIterator
// purpose :
// =======================================================================
RWMesh_FaceIterator::RWMesh_FaceIterator (const TopoDS_Shape& theShape,
const XCAFPrs_Style& theStyle)
: myDefStyle (theStyle),
myToMapColors (true),
mySLTool (1, 1e-12),
myHasNormals (false),
myIsMirrored (false),
myHasFaceColor (false)
{
if (theShape.IsNull())
{
return;
}
myFaceIter.Init (theShape, TopAbs_FACE);
Next();
}
// =======================================================================
// function : dispatchStyles
// purpose :

View File

@ -39,6 +39,13 @@ public:
const Standard_Boolean theToMapColors = false,
const XCAFPrs_Style& theStyle = XCAFPrs_Style());
//! Auxiliary constructor.
Standard_EXPORT RWMesh_FaceIterator (const TopoDS_Shape& theShape,
const XCAFPrs_Style& theStyle = XCAFPrs_Style());
//! Return explored shape.
const TopoDS_Shape& ExploredShape() const { return myFaceIter.ExploredShape(); }
//! Return true if iterator points to the valid triangulation.
bool More() const { return !myPolyTriang.IsNull(); }

View File

@ -38,6 +38,10 @@ RWMesh_MaterialMap::RWMesh_MaterialMap (const TCollection_AsciiString& theFile)
TCollection_AsciiString aFileName, aFileExt;
OSD_Path::FolderAndFileFromPath (theFile, myFolder, aFileName);
OSD_Path::FileNameAndExtension (aFileName, myShortFileNameBase, aFileExt);
if (myFolder.IsEmpty())
{
myFolder = ".";
}
}
// =======================================================================
@ -216,6 +220,7 @@ bool RWMesh_MaterialMap::CreateTextureFolder()
OSD_Directory aResDir (aResFolderPath);
if (!aResDir.Exists())
{
Message::SendFail() << "Failed to create textures folder '" << myFolder << "'";
return false;
}
const OSD_Protection aParentProt = aResDir.Protection();
@ -233,6 +238,7 @@ bool RWMesh_MaterialMap::CreateTextureFolder()
if (aTexDir.Failed())
{
// fallback to the same folder as output model file
Message::SendFail() << "Failed to create textures folder '" << myTexFolder << "'";
myTexFolder = myFolder;
myTexFolderShort.Clear();
return true;

View File

@ -2,5 +2,4 @@ TopExp.cxx
TopExp.hxx
TopExp_Explorer.cxx
TopExp_Explorer.hxx
TopExp_Explorer.lxx
TopExp_Stack.hxx

View File

@ -17,14 +17,11 @@
#define No_Standard_NoMoreObject
#define No_Standard_NoSuchObject
#include <TopExp_Explorer.hxx>
#include <Standard.hxx>
#include <Standard_NoMoreObject.hxx>
#include <Standard_NoSuchObject.hxx>
#include <TopAbs.hxx>
#include <TopExp_Explorer.hxx>
#include <TopoDS_Iterator.hxx>
#include <TopoDS_Shape.hxx>
// macro to compare two types of shapes
// always True if the first one is SHAPE

View File

@ -17,19 +17,8 @@
#ifndef _TopExp_Explorer_HeaderFile
#define _TopExp_Explorer_HeaderFile
#include <Standard.hxx>
#include <Standard_DefineAlloc.hxx>
#include <Standard_Handle.hxx>
#include <TopExp_Stack.hxx>
#include <Standard_Integer.hxx>
#include <TopoDS_Shape.hxx>
#include <Standard_Boolean.hxx>
#include <TopAbs_ShapeEnum.hxx>
class Standard_NoMoreObject;
class Standard_NoSuchObject;
class TopoDS_Shape;
//! An Explorer is a Tool to visit a Topological Data
//! Structure form the TopoDS package.
@ -117,10 +106,9 @@ public:
//! ToFind it has no effect on the search.
Standard_EXPORT void Init (const TopoDS_Shape& S, const TopAbs_ShapeEnum ToFind, const TopAbs_ShapeEnum ToAvoid = TopAbs_SHAPE);
//! Returns True if there are more shapes in the
//! exploration.
Standard_Boolean More() const;
//! Returns True if there are more shapes in the exploration.
Standard_Boolean More() const { return hasMore; }
//! Moves to the next Shape in the exploration.
//! Exceptions
//! Standard_NoMoreObject if there are no more shapes to explore.
@ -135,15 +123,17 @@ public:
//! Exceptions
//! Standard_NoSuchObject if this explorer has no more shapes to explore.
Standard_EXPORT const TopoDS_Shape& Current() const;
//! Reinitialize the exploration with the original
//! arguments.
//! Reinitialize the exploration with the original arguments.
Standard_EXPORT void ReInit();
//! Return explored shape.
const TopoDS_Shape& ExploredShape() const { return myShape; }
//! Returns the current depth of the exploration. 0 is
//! the shape to explore itself.
Standard_Integer Depth() const;
Standard_Integer Depth() const { return myTop; }
//! Clears the content of the explorer. It will return
//! False on More().
void Clear();
@ -154,19 +144,8 @@ public:
Destroy();
}
protected:
private:
TopExp_Stack myStack;
Standard_Integer myTop;
Standard_Integer mySizeOfStack;
@ -175,14 +154,21 @@ private:
TopAbs_ShapeEnum toFind;
TopAbs_ShapeEnum toAvoid;
};
#include <TopoDS_Iterator.hxx>
#include <TopExp_Explorer.lxx>
inline void TopExp_Explorer::Clear()
{
hasMore = Standard_False;
if (myTop > 0)
{
for (int i = 1; i <= myTop; i++)
{
myStack[i].~TopoDS_Iterator();
}
}
myTop = 0;
}
#endif // _TopExp_Explorer_HeaderFile

View File

@ -1,54 +0,0 @@
// Created on: 1993-01-18
// Created by: Remi LEQUETTE
// Copyright (c) 1993-1999 Matra Datavision
// Copyright (c) 1999-2014 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and/or modify it under
// the terms of the GNU Lesser General Public License version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#include <TopoDS_Iterator.hxx>
//=======================================================================
//function : More
//purpose :
//=======================================================================
inline Standard_Boolean TopExp_Explorer::More()const
{
return hasMore;
}
//=======================================================================
//function : Depth
//purpose :
//=======================================================================
inline Standard_Integer TopExp_Explorer::Depth()const
{
return myTop;
}
//=======================================================================
//function : Clear
//purpose :
//=======================================================================
inline void TopExp_Explorer::Clear()
{
hasMore = Standard_False;
if(myTop >0) {
for(int i=1;i<= myTop; i++)
myStack[i].~TopoDS_Iterator();
}
myTop = 0;
}

View File

@ -19,7 +19,6 @@ TopoDS_HShape.hxx
TopoDS_HShape.lxx
TopoDS_Iterator.cxx
TopoDS_Iterator.hxx
TopoDS_Iterator.lxx
TopoDS_ListIteratorOfListOfShape.hxx
TopoDS_ListOfShape.hxx
TopoDS_LockedShape.hxx

View File

@ -16,11 +16,10 @@
#define No_Standard_NoSuchObject
#include <TopoDS_Iterator.hxx>
#include <Standard_NoMoreObject.hxx>
#include <Standard_NoSuchObject.hxx>
#include <TopoDS_Iterator.hxx>
#include <TopoDS_Shape.hxx>
//=======================================================================
//function : Initialize

View File

@ -17,19 +17,11 @@
#ifndef _TopoDS_Iterator_HeaderFile
#define _TopoDS_Iterator_HeaderFile
#include <Standard.hxx>
#include <Standard_DefineAlloc.hxx>
#include <Standard_Handle.hxx>
#include <Standard_NoSuchObject.hxx>
#include <TopoDS_Shape.hxx>
#include <TopoDS_ListIteratorOfListOfShape.hxx>
#include <TopAbs_Orientation.hxx>
#include <TopLoc_Location.hxx>
#include <Standard_Boolean.hxx>
class Standard_NoMoreObject;
class Standard_NoSuchObject;
class TopoDS_Shape;
//! Iterates on the underlying shape underlying a given
//! TopoDS_Shape object, providing access to its
@ -42,10 +34,9 @@ public:
DEFINE_STANDARD_ALLOC
//! Creates an empty Iterator.
TopoDS_Iterator();
TopoDS_Iterator() {}
//! Creates an Iterator on <S> sub-shapes.
//! Note:
//! - If cumOri is true, the function composes all
@ -53,8 +44,13 @@ public:
//! - If cumLoc is true, the function multiplies all
//! sub-shapes by the location of S, i.e. it applies to
//! each sub-shape the transformation that is associated with S.
TopoDS_Iterator(const TopoDS_Shape& S, const Standard_Boolean cumOri = Standard_True, const Standard_Boolean cumLoc = Standard_True);
TopoDS_Iterator (const TopoDS_Shape& S,
const Standard_Boolean cumOri = Standard_True,
const Standard_Boolean cumLoc = Standard_True)
{
Initialize (S, cumOri,cumLoc);
}
//! Initializes this iterator with shape S.
//! Note:
//! - If cumOri is true, the function composes all
@ -66,8 +62,8 @@ public:
//! Returns true if there is another sub-shape in the
//! shape which this iterator is scanning.
Standard_Boolean More() const;
Standard_Boolean More() const { return myShapes.More(); }
//! Moves on to the next sub-shape in the shape which
//! this iterator is scanning.
//! Exceptions
@ -78,34 +74,19 @@ public:
//! this iterator is scanning.
//! Exceptions
//! Standard_NoSuchObject if there is no current sub-shape.
const TopoDS_Shape& Value() const;
protected:
const TopoDS_Shape& Value() const
{
Standard_NoSuchObject_Raise_if(!More(),"TopoDS_Iterator::Value");
return myShape;
}
private:
TopoDS_Shape myShape;
TopoDS_ListIteratorOfListOfShape myShapes;
TopAbs_Orientation myOrientation;
TopLoc_Location myLocation;
};
#include <TopoDS_Iterator.lxx>
#endif // _TopoDS_Iterator_HeaderFile

View File

@ -1,58 +0,0 @@
// Created on: 1993-01-21
// Created by: Remi LEQUETTE
// Copyright (c) 1993-1999 Matra Datavision
// Copyright (c) 1999-2014 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and/or modify it under
// the terms of the GNU Lesser General Public License version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#include <Standard_NoSuchObject.hxx>
//=======================================================================
//function : TopoDS_Iterator
//purpose :
//=======================================================================
inline TopoDS_Iterator::TopoDS_Iterator()
{}
//=======================================================================
//function : TopoDS_Iterator
//purpose :
//=======================================================================
inline TopoDS_Iterator::TopoDS_Iterator(const TopoDS_Shape& S,
const Standard_Boolean cumOri,
const Standard_Boolean cumLoc)
{
Initialize(S,cumOri,cumLoc);
}
//=======================================================================
//function : More
//purpose :
//=======================================================================
inline Standard_Boolean TopoDS_Iterator::More() const
{
return myShapes.More();
}
//=======================================================================
//function : Value
//purpose :
//=======================================================================
inline const TopoDS_Shape& TopoDS_Iterator::Value() const
{
Standard_NoSuchObject_Raise_if(!More(),"TopoDS_Iterator::Value");
return myShape;
}

View File

@ -240,6 +240,11 @@ void XCAFDoc_VisMaterial::FillMaterialAspect (Graphic3d_MaterialAspect& theAspec
theAspect.SetAlpha (myPbrMat.BaseColor.Alpha());
theAspect.SetSpecularColor(Quantity_Color (Graphic3d_Vec3 (myPbrMat.Metallic)));
theAspect.SetShininess (1.0f - myPbrMat.Roughness);
if (theAspect.Shininess() < 0.01f)
{
// clamp too small shininess values causing visual artifacts on corner view angles
theAspect.SetShininess (0.01f);
}
theAspect.SetEmissiveColor (Quantity_Color (myPbrMat.EmissiveFactor.cwiseMin (Graphic3d_Vec3 (1.0f))));
}

View File

@ -148,6 +148,27 @@ static bool parseNameFormat (const char* theArg,
return true;
}
//! Parse RWMesh_CoordinateSystem enumeration.
static bool parseCoordinateSystem (const char* theArg,
RWMesh_CoordinateSystem& theSystem)
{
TCollection_AsciiString aCSStr (theArg);
aCSStr.LowerCase();
if (aCSStr == "zup")
{
theSystem = RWMesh_CoordinateSystem_Zup;
}
else if (aCSStr == "yup")
{
theSystem = RWMesh_CoordinateSystem_Yup;
}
else
{
return Standard_False;
}
return Standard_True;
}
//=============================================================================
//function : ReadGltf
//purpose : Reads glTF file
@ -333,7 +354,9 @@ static Standard_Integer WriteGltf (Draw_Interpretor& theDI,
Handle(TDocStd_Application) anApp = DDocStd::GetApplication();
TColStd_IndexedDataMapOfStringString aFileInfo;
RWGltf_WriterTrsfFormat aTrsfFormat = RWGltf_WriterTrsfFormat_Compact;
RWMesh_CoordinateSystem aSystemCoordSys = RWMesh_CoordinateSystem_Zup;
bool toForceUVExport = false, toEmbedTexturesInGlb = true;
bool toMergeFaces = false, toSplitIndices16 = false;
RWMesh_NameFormat aNodeNameFormat = RWMesh_NameFormat_InstanceOrProduct;
RWMesh_NameFormat aMeshNameFormat = RWMesh_NameFormat_Product;
for (Standard_Integer anArgIter = 1; anArgIter < theNbArgs; ++anArgIter)
@ -360,6 +383,40 @@ static Standard_Integer WriteGltf (Draw_Interpretor& theDI,
++anArgIter;
}
}
else if (anArgCase == "-mergefaces")
{
toMergeFaces = true;
if (anArgIter + 1 < theNbArgs
&& Draw::ParseOnOff (theArgVec[anArgIter + 1], toMergeFaces))
{
++anArgIter;
}
}
else if (anArgCase == "-splitindices16"
|| anArgCase == "-splitindexes16"
|| anArgCase == "-splitindices"
|| anArgCase == "-splitindexes"
|| anArgCase == "-splitind")
{
toSplitIndices16 = true;
if (anArgIter + 1 < theNbArgs
&& Draw::ParseOnOff (theArgVec[anArgIter + 1], toSplitIndices16))
{
++anArgIter;
}
}
else if (anArgIter + 1 < theNbArgs
&& (anArgCase == "-systemcoordinatesystem"
|| anArgCase == "-systemcoordsystem"
|| anArgCase == "-systemcoordsys"
|| anArgCase == "-syscoordsys"))
{
if (!parseCoordinateSystem (theArgVec[++anArgIter], aSystemCoordSys))
{
Message::SendFail() << "Syntax error: unknown coordinate system '" << theArgVec[anArgIter] << "'";
return 1;
}
}
else if (anArgCase == "-trsfformat"
&& anArgIter + 1 < theNbArgs)
{
@ -456,8 +513,10 @@ static Standard_Integer WriteGltf (Draw_Interpretor& theDI,
aWriter.SetMeshNameFormat (aMeshNameFormat);
aWriter.SetForcedUVExport (toForceUVExport);
aWriter.SetToEmbedTexturesInGlb (toEmbedTexturesInGlb);
aWriter.SetMergeFaces (toMergeFaces);
aWriter.SetSplitIndices16 (toSplitIndices16);
aWriter.ChangeCoordinateSystemConverter().SetInputLengthUnit (aSystemUnitFactor);
aWriter.ChangeCoordinateSystemConverter().SetInputCoordinateSystem (RWMesh_CoordinateSystem_Zup);
aWriter.ChangeCoordinateSystemConverter().SetInputCoordinateSystem (aSystemCoordSys);
aWriter.Perform (aDoc, aFileInfo, aProgress->Start());
return 0;
}
@ -550,27 +609,6 @@ static Standard_Integer readstl(Draw_Interpretor& theDI,
return 0;
}
//! Parse RWMesh_CoordinateSystem enumeration.
static Standard_Boolean parseCoordinateSystem (const char* theArg,
RWMesh_CoordinateSystem& theSystem)
{
TCollection_AsciiString aCSStr (theArg);
aCSStr.LowerCase();
if (aCSStr == "zup")
{
theSystem = RWMesh_CoordinateSystem_Zup;
}
else if (aCSStr == "yup")
{
theSystem = RWMesh_CoordinateSystem_Yup;
}
else
{
return Standard_False;
}
return Standard_True;
}
//=============================================================================
//function : ReadObj
//purpose : Reads OBJ file
@ -1985,14 +2023,18 @@ void XSDRAWSTLVRML::InitCommands (Draw_Interpretor& theCommands)
"\n\t\t: Same as ReadGltf but reads glTF file into a shape instead of a document.",
__FILE__, ReadGltf, g);
theCommands.Add ("WriteGltf",
"WriteGltf Doc file [-trsfFormat {compact|TRS|mat4}=compact]"
"WriteGltf Doc file [-trsfFormat {compact|TRS|mat4}]=compact"
"\n\t\t: [-systemCoordSys {Zup|Yup}]=Zup"
"\n\t\t: [-comments Text] [-author Name]"
"\n\t\t: [-forceUVExport] [-texturesSeparate]"
"\n\t\t: [-nodeNameFormat {empty|product|instance|instOrProd|prodOrInst|prodAndInst|verbose}=instOrProd]"
"\n\t\t: [-meshNameFormat {empty|product|instance|instOrProd|prodOrInst|prodAndInst|verbose}=product]"
"\n\t\t: [-forceUVExport]=0 [-texturesSeparate]=0 [-mergeFaces]=0 [-splitIndices16]=0"
"\n\t\t: [-nodeNameFormat {empty|product|instance|instOrProd|prodOrInst|prodAndInst|verbose}]=instOrProd"
"\n\t\t: [-meshNameFormat {empty|product|instance|instOrProd|prodOrInst|prodAndInst|verbose}]=product"
"\n\t\t: Write XDE document into glTF file."
"\n\t\t: -trsfFormat preferred transformation format"
"\n\t\t: -forceUVExport always export UV coordinates"
"\n\t\t: -systemCoordSys system coordinate system; Zup when not specified"
"\n\t\t: -mergeFaces merge Faces within the same Mesh"
"\n\t\t: -splitIndices16 split Faces to keep 16-bit indices when -mergeFaces is enabled"
"\n\t\t: -forceUVExport always export UV coordinates"
"\n\t\t: -texturesSeparate write textures to separate files"
"\n\t\t: -nodeNameFormat name format for Nodes"
"\n\t\t: -meshNameFormat name format for Meshes",

View File

@ -0,0 +1,24 @@
puts "========"
puts "0032530: Data Exchange, RWGltf_CafWriter - add option merging Faces within the Part"
puts "========"
Close D0 -silent
ReadStep D0 [locate_data_file as1-oc-214-mat.stp]
XGetOneShape ss D0
incmesh ss 1.0
set aTmpGltf1 "${imagedir}/${casename}_tmp1.glb"
set aTmpGltf2 "${imagedir}/${casename}_tmp2.glb"
lappend occ_tmp_files $aTmpGltf1
lappend occ_tmp_files $aTmpGltf2
WriteGltf D0 "$aTmpGltf1"
WriteGltf D0 "$aTmpGltf2" -mergeFaces
ReadGltf D1 "$aTmpGltf1"
XGetOneShape s1 D1
checknbshapes s1 -face 160 -compound 28
ReadGltf D "$aTmpGltf2"
XGetOneShape s2 D
checknbshapes s2 -face 18 -compound 10