mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-04 18:06:22 +03:00
404 lines
22 KiB
C++
404 lines
22 KiB
C++
// Copyright (c) 2017-2019 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.
|
|
|
|
#ifndef _RWGltf_CafWriter_HeaderFiler
|
|
#define _RWGltf_CafWriter_HeaderFiler
|
|
|
|
#include <TColStd_IndexedDataMapOfStringString.hxx>
|
|
#include <TColStd_MapOfAsciiString.hxx>
|
|
#include <TDF_LabelSequence.hxx>
|
|
#include <TopTools_ShapeMapHasher.hxx>
|
|
#include <RWGltf_DracoParameters.hxx>
|
|
#include <RWGltf_GltfBufferView.hxx>
|
|
#include <RWGltf_GltfFace.hxx>
|
|
#include <RWGltf_WriterTrsfFormat.hxx>
|
|
#include <RWMesh_CoordinateSystemConverter.hxx>
|
|
#include <RWMesh_NameFormat.hxx>
|
|
#include <XCAFPrs_Style.hxx>
|
|
#include <Poly_Triangle.hxx>
|
|
|
|
#include <memory>
|
|
|
|
class Message_ProgressRange;
|
|
class RWMesh_FaceIterator;
|
|
class RWGltf_GltfOStreamWriter;
|
|
class RWGltf_GltfMaterialMap;
|
|
class RWGltf_GltfSceneNodeMap;
|
|
class TDocStd_Document;
|
|
|
|
//! glTF writer context from XCAF document.
|
|
class RWGltf_CafWriter : public Standard_Transient
|
|
{
|
|
DEFINE_STANDARD_RTTIEXT(RWGltf_CafWriter, Standard_Transient)
|
|
public:
|
|
|
|
//! Mesh
|
|
struct Mesh
|
|
{
|
|
std::vector<Graphic3d_Vec3> NodesVec; //!< vector for mesh nodes
|
|
std::vector<Graphic3d_Vec3> NormalsVec; //!< vector for mesh normals
|
|
std::vector<Graphic3d_Vec2> TexCoordsVec; //!< vector for mesh texture UV coordinates
|
|
std::vector<Poly_Triangle> IndicesVec; //!< vector for mesh indices
|
|
};
|
|
|
|
//! Main constructor.
|
|
//! @param theFile [in] path to output glTF file
|
|
//! @param theIsBinary [in] flag to write into binary glTF format (.glb)
|
|
Standard_EXPORT RWGltf_CafWriter (const TCollection_AsciiString& theFile,
|
|
Standard_Boolean theIsBinary);
|
|
|
|
//! Destructor.
|
|
Standard_EXPORT virtual ~RWGltf_CafWriter();
|
|
|
|
//! Return transformation from OCCT to glTF coordinate system.
|
|
const RWMesh_CoordinateSystemConverter& CoordinateSystemConverter() const { return myCSTrsf; }
|
|
|
|
//! Return transformation from OCCT to glTF coordinate system.
|
|
RWMesh_CoordinateSystemConverter& ChangeCoordinateSystemConverter() { return myCSTrsf; }
|
|
|
|
//! Set transformation from OCCT to glTF coordinate system.
|
|
void SetCoordinateSystemConverter (const RWMesh_CoordinateSystemConverter& theConverter) { myCSTrsf = theConverter; }
|
|
|
|
//! Return flag to write into binary glTF format (.glb), specified within class constructor.
|
|
bool IsBinary() const { return myIsBinary; }
|
|
|
|
//! Return preferred transformation format for writing into glTF file; RWGltf_WriterTrsfFormat_Compact by default.
|
|
RWGltf_WriterTrsfFormat TransformationFormat() const { return myTrsfFormat; }
|
|
|
|
//! Set preferred transformation format for writing into glTF file.
|
|
void SetTransformationFormat (RWGltf_WriterTrsfFormat theFormat) { myTrsfFormat = theFormat; }
|
|
|
|
//! Return name format for exporting Nodes; RWMesh_NameFormat_InstanceOrProduct by default.
|
|
RWMesh_NameFormat NodeNameFormat() const { return myNodeNameFormat; }
|
|
|
|
//! Set name format for exporting Nodes.
|
|
void SetNodeNameFormat (RWMesh_NameFormat theFormat) { myNodeNameFormat = theFormat; }
|
|
|
|
//! Return name format for exporting Meshes; RWMesh_NameFormat_Product by default.
|
|
RWMesh_NameFormat MeshNameFormat() const { return myMeshNameFormat; }
|
|
|
|
//! Set name format for exporting Meshes.
|
|
void SetMeshNameFormat (RWMesh_NameFormat theFormat) { myMeshNameFormat = theFormat; }
|
|
|
|
//! Return TRUE to export UV coordinates even if there are no mapped texture; FALSE by default.
|
|
bool IsForcedUVExport() const { return myIsForcedUVExport; }
|
|
|
|
//! Set flag to export UV coordinates even if there are no mapped texture; FALSE by default.
|
|
void SetForcedUVExport (bool theToForce) { myIsForcedUVExport = theToForce; }
|
|
|
|
//! Return default material definition to be used for nodes with only color defined.
|
|
const XCAFPrs_Style& DefaultStyle() const { return myDefaultStyle; }
|
|
|
|
//! Set default material definition to be used for nodes with only color defined.
|
|
void SetDefaultStyle (const XCAFPrs_Style& theStyle) { myDefaultStyle = theStyle; }
|
|
|
|
//! Return flag to write image textures into GLB file (binary gltf export); TRUE by default.
|
|
//! When set to FALSE, texture images will be written as separate files.
|
|
//! Has no effect on writing into non-binary format.
|
|
Standard_Boolean ToEmbedTexturesInGlb() { return myToEmbedTexturesInGlb; }
|
|
|
|
//! 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; }
|
|
|
|
//! Return Draco parameters
|
|
const RWGltf_DracoParameters& CompressionParameters() const { return myDracoParameters; }
|
|
|
|
//! Set Draco parameters
|
|
void SetCompressionParameters(const RWGltf_DracoParameters& theDracoParameters) { myDracoParameters = theDracoParameters; }
|
|
|
|
//! Write glTF file and associated binary file.
|
|
//! Triangulation data should be precomputed within shapes!
|
|
//! @param theDocument [in] input document
|
|
//! @param theRootLabels [in] list of root shapes to export
|
|
//! @param theLabelFilter [in] optional filter with document nodes to export,
|
|
//! with keys defined by XCAFPrs_DocumentExplorer::DefineChildId() and filled recursively
|
|
//! (leaves and parent assembly nodes at all levels);
|
|
//! when not NULL, all nodes not included into the map will be ignored
|
|
//! @param theFileInfo [in] map with file metadata to put into glTF header section
|
|
//! @param theProgress [in] optional progress indicator
|
|
//! @return FALSE on file writing failure
|
|
Standard_EXPORT virtual bool Perform (const Handle(TDocStd_Document)& theDocument,
|
|
const TDF_LabelSequence& theRootLabels,
|
|
const TColStd_MapOfAsciiString* theLabelFilter,
|
|
const TColStd_IndexedDataMapOfStringString& theFileInfo,
|
|
const Message_ProgressRange& theProgress);
|
|
|
|
//! Write glTF file and associated binary file.
|
|
//! Triangulation data should be precomputed within shapes!
|
|
//! @param theDocument [in] input document
|
|
//! @param theFileInfo [in] map with file metadata to put into glTF header section
|
|
//! @param theProgress [in] optional progress indicator
|
|
//! @return FALSE on file writing failure
|
|
Standard_EXPORT virtual bool Perform (const Handle(TDocStd_Document)& theDocument,
|
|
const TColStd_IndexedDataMapOfStringString& theFileInfo,
|
|
const Message_ProgressRange& theProgress);
|
|
|
|
protected:
|
|
|
|
//! Write binary data file with triangulation data.
|
|
//! Triangulation data should be precomputed within shapes!
|
|
//! @param theDocument [in] input document
|
|
//! @param theRootLabels [in] list of root shapes to export
|
|
//! @param theLabelFilter [in] optional filter with document nodes to export
|
|
//! @param theProgress [in] optional progress indicator
|
|
//! @return FALSE on file writing failure
|
|
Standard_EXPORT virtual bool writeBinData (const Handle(TDocStd_Document)& theDocument,
|
|
const TDF_LabelSequence& theRootLabels,
|
|
const TColStd_MapOfAsciiString* theLabelFilter,
|
|
const Message_ProgressRange& theProgress);
|
|
|
|
//! Write JSON file with glTF structure (should be called after writeBinData()).
|
|
//! @param theDocument [in] input document
|
|
//! @param theRootLabels [in] list of root shapes to export
|
|
//! @param theLabelFilter [in] optional filter with document nodes to export
|
|
//! @param theFileInfo [in] map with file metadata to put into glTF header section
|
|
//! @param theProgress [in] optional progress indicator
|
|
//! @return FALSE on file writing failure
|
|
Standard_EXPORT virtual bool writeJson (const Handle(TDocStd_Document)& theDocument,
|
|
const TDF_LabelSequence& theRootLabels,
|
|
const TColStd_MapOfAsciiString* theLabelFilter,
|
|
const TColStd_IndexedDataMapOfStringString& theFileInfo,
|
|
const Message_ProgressRange& theProgress);
|
|
|
|
protected:
|
|
|
|
//! Return TRUE if face mesh should be skipped (e.g. because it is invalid or empty).
|
|
Standard_EXPORT virtual Standard_Boolean toSkipFaceMesh (const RWMesh_FaceIterator& theFaceIter);
|
|
|
|
//! Generate name for specified labels.
|
|
//! @param[in] theFormat name format to apply
|
|
//! @param[in] theLabel instance label
|
|
//! @param[in] theRefLabel product label
|
|
Standard_EXPORT virtual TCollection_AsciiString formatName (RWMesh_NameFormat theFormat,
|
|
const TDF_Label& theLabel,
|
|
const TDF_Label& theRefLabel) const;
|
|
|
|
//! Write mesh nodes into binary file.
|
|
//! @param theGltfFace [out] glTF face definition
|
|
//! @param theBinFile [out] output file to write into
|
|
//! @param theFaceIter [in] current face to write
|
|
//! @param theAccessorNb [in] [out] last accessor index
|
|
//! @param theMesh [in] [out] mesh
|
|
Standard_EXPORT virtual void saveNodes (RWGltf_GltfFace& theGltfFace,
|
|
std::ostream& theBinFile,
|
|
const RWMesh_FaceIterator& theFaceIter,
|
|
Standard_Integer& theAccessorNb,
|
|
const std::shared_ptr<RWGltf_CafWriter::Mesh>& theMesh) const;
|
|
|
|
//! Write mesh normals into binary file.
|
|
//! @param theGltfFace [out] glTF face definition
|
|
//! @param theBinFile [out] output file to write into
|
|
//! @param theFaceIter [in] current face to write
|
|
//! @param theAccessorNb [in] [out] last accessor index
|
|
//! @param theMesh [in] [out] mesh
|
|
Standard_EXPORT virtual void saveNormals (RWGltf_GltfFace& theGltfFace,
|
|
std::ostream& theBinFile,
|
|
RWMesh_FaceIterator& theFaceIter,
|
|
Standard_Integer& theAccessorNb,
|
|
const std::shared_ptr<RWGltf_CafWriter::Mesh>& theMesh) const;
|
|
|
|
//! Write mesh texture UV coordinates into binary file.
|
|
//! @param theGltfFace [out] glTF face definition
|
|
//! @param theBinFile [out] output file to write into
|
|
//! @param theFaceIter [in] current face to write
|
|
//! @param theAccessorNb [in] [out] last accessor index
|
|
//! @param theMesh [in] [out] mesh
|
|
Standard_EXPORT virtual void saveTextCoords (RWGltf_GltfFace& theGltfFace,
|
|
std::ostream& theBinFile,
|
|
const RWMesh_FaceIterator& theFaceIter,
|
|
Standard_Integer& theAccessorNb,
|
|
const std::shared_ptr<RWGltf_CafWriter::Mesh>& theMesh) const;
|
|
|
|
//! Write mesh indexes into binary file.
|
|
//! @param theGltfFace [out] glTF face definition
|
|
//! @param theBinFile [out] output file to write into
|
|
//! @param theFaceIter [in] current face to write
|
|
//! @param theAccessorNb [in] [out] last accessor index
|
|
//! @param theMesh [in] [out] mesh
|
|
Standard_EXPORT virtual void saveIndices (RWGltf_GltfFace& theGltfFace,
|
|
std::ostream& theBinFile,
|
|
const RWMesh_FaceIterator& theFaceIter,
|
|
Standard_Integer& theAccessorNb,
|
|
const std::shared_ptr<RWGltf_CafWriter::Mesh>& theMesh);
|
|
|
|
protected:
|
|
|
|
//! Write bufferView for vertex positions within RWGltf_GltfRootElement_Accessors section
|
|
//! @param theGltfFace [in] face definition to write
|
|
Standard_EXPORT virtual void writePositions (const RWGltf_GltfFace& theGltfFace);
|
|
|
|
//! Write bufferView for vertex normals within RWGltf_GltfRootElement_Accessors section
|
|
//! @param theGltfFace [in] face definition to write
|
|
Standard_EXPORT virtual void writeNormals (const RWGltf_GltfFace& theGltfFace);
|
|
|
|
//! Write bufferView for vertex texture coordinates within RWGltf_GltfRootElement_Accessors section
|
|
//! @param theGltfFace [in] face definition to write
|
|
Standard_EXPORT virtual void writeTextCoords (const RWGltf_GltfFace& theGltfFace);
|
|
|
|
//! Write bufferView for triangle indexes within RWGltf_GltfRootElement_Accessors section.
|
|
//! @param theGltfFace [in] face definition to write
|
|
Standard_EXPORT virtual void writeIndices (const RWGltf_GltfFace& theGltfFace);
|
|
|
|
protected:
|
|
|
|
//! Write RWGltf_GltfRootElement_Accessors section.
|
|
//! @param theSceneNodeMap [in] ordered map of scene nodes
|
|
Standard_EXPORT virtual void writeAccessors (const RWGltf_GltfSceneNodeMap& theSceneNodeMap);
|
|
|
|
//! Write RWGltf_GltfRootElement_Animations section (reserved).
|
|
Standard_EXPORT virtual void writeAnimations();
|
|
|
|
//! Write RWGltf_GltfRootElement_Asset section.
|
|
//! @param theFileInfo [in] optional metadata to write into file header
|
|
Standard_EXPORT virtual void writeAsset (const TColStd_IndexedDataMapOfStringString& theFileInfo);
|
|
|
|
//! Write RWGltf_GltfRootElement_BufferViews section.
|
|
//! @param theBinDataBufferId [in] index of binary buffer with vertex data
|
|
Standard_EXPORT virtual void writeBufferViews (const Standard_Integer theBinDataBufferId);
|
|
|
|
//! Write RWGltf_GltfRootElement_Buffers section.
|
|
Standard_EXPORT virtual void writeBuffers();
|
|
|
|
//! Write RWGltf_GltfRootElement_ExtensionsUsed/RWGltf_GltfRootElement_ExtensionsRequired sections (reserved).
|
|
Standard_EXPORT virtual void writeExtensions();
|
|
|
|
//! Write RWGltf_GltfRootElement_Images section.
|
|
//! @param theSceneNodeMap [in] ordered map of scene nodes
|
|
//! @param theMaterialMap [out] map of materials, filled with image files used by textures
|
|
Standard_EXPORT virtual void writeImages (const RWGltf_GltfSceneNodeMap& theSceneNodeMap);
|
|
|
|
//! Write RWGltf_GltfRootElement_Materials section.
|
|
//! @param theSceneNodeMap [in] ordered map of scene nodes
|
|
//! @param theMaterialMap [out] map of materials, filled with materials
|
|
Standard_EXPORT virtual void writeMaterials (const RWGltf_GltfSceneNodeMap& theSceneNodeMap);
|
|
|
|
//! Write RWGltf_GltfRootElement_Meshes section.
|
|
//! @param theSceneNodeMap [in] ordered map of scene nodes
|
|
//! @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] theDracoBufInd draco buffer index
|
|
//! @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,
|
|
const int theDracoBufInd,
|
|
bool& theToStartPrims);
|
|
|
|
//! Write RWGltf_GltfRootElement_Nodes section.
|
|
//! @param theDocument [in] input document
|
|
//! @param theRootLabels [in] list of root shapes to export
|
|
//! @param theLabelFilter [in] optional filter with document nodes to export
|
|
//! @param theSceneNodeMap [in] ordered map of scene nodes
|
|
//! @param theSceneRootNodeInds [out] sequence of scene nodes pointing to root shapes (to be used for writeScenes())
|
|
Standard_EXPORT virtual void writeNodes (const Handle(TDocStd_Document)& theDocument,
|
|
const TDF_LabelSequence& theRootLabels,
|
|
const TColStd_MapOfAsciiString* theLabelFilter,
|
|
const RWGltf_GltfSceneNodeMap& theSceneNodeMap,
|
|
NCollection_Sequence<Standard_Integer>& theSceneRootNodeInds);
|
|
|
|
//! Write RWGltf_GltfRootElement_Samplers section.
|
|
Standard_EXPORT virtual void writeSamplers();
|
|
|
|
//! Write RWGltf_GltfRootElement_Scene section.
|
|
//! @param theDefSceneId [in] index of default scene (0)
|
|
Standard_EXPORT virtual void writeScene (const Standard_Integer theDefSceneId);
|
|
|
|
//! Write RWGltf_GltfRootElement_Scenes section.
|
|
//! @param theSceneRootNodeInds [in] sequence of scene nodes pointing to root shapes
|
|
Standard_EXPORT virtual void writeScenes (const NCollection_Sequence<Standard_Integer>& theSceneRootNodeInds);
|
|
|
|
//! Write RWGltf_GltfRootElement_Skins section (reserved).
|
|
Standard_EXPORT virtual void writeSkins();
|
|
|
|
//! Write RWGltf_GltfRootElement_Textures section.
|
|
//! @param theSceneNodeMap [in] ordered map of scene nodes
|
|
//! @param theMaterialMap [out] map of materials, filled with textures
|
|
Standard_EXPORT virtual void writeTextures (const RWGltf_GltfSceneNodeMap& theSceneNodeMap);
|
|
|
|
protected:
|
|
|
|
//! Shape + Style pair.
|
|
struct RWGltf_StyledShape
|
|
{
|
|
TopoDS_Shape Shape;
|
|
XCAFPrs_Style Style;
|
|
|
|
RWGltf_StyledShape() {}
|
|
explicit RWGltf_StyledShape (const TopoDS_Shape& theShape) : Shape (theShape) {}
|
|
explicit RWGltf_StyledShape (const TopoDS_Shape& theShape,
|
|
const XCAFPrs_Style& theStyle) : Shape (theShape), Style (theStyle) {}
|
|
public:
|
|
//! Computes a hash code.
|
|
static Standard_Integer HashCode (const RWGltf_StyledShape& theShape, Standard_Integer theUpperBound)
|
|
{
|
|
return theShape.Shape.HashCode (theUpperBound);
|
|
}
|
|
//! Equality comparison.
|
|
static Standard_Boolean IsEqual (const RWGltf_StyledShape& theS1, const RWGltf_StyledShape& theS2)
|
|
{
|
|
return theS1.Shape.IsSame (theS2.Shape)
|
|
&& theS1.Style.IsEqual(theS2.Style);
|
|
}
|
|
};
|
|
|
|
typedef NCollection_IndexedDataMap<RWGltf_StyledShape, Handle(RWGltf_GltfFaceList), RWGltf_StyledShape> ShapeToGltfFaceMap;
|
|
|
|
protected:
|
|
|
|
TCollection_AsciiString myFile; //!< output glTF file
|
|
TCollection_AsciiString myBinFileNameFull; //!< output file with binary data (full path)
|
|
TCollection_AsciiString myBinFileNameShort; //!< output file with binary data (short path)
|
|
RWGltf_WriterTrsfFormat myTrsfFormat; //!< transformation format to write into glTF file
|
|
RWMesh_NameFormat myNodeNameFormat; //!< name format for exporting Nodes
|
|
RWMesh_NameFormat myMeshNameFormat; //!< name format for exporting Meshes
|
|
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
|
|
|
|
std::shared_ptr<RWGltf_GltfOStreamWriter> myWriter; //!< JSON writer
|
|
Handle(RWGltf_GltfMaterialMap) myMaterialMap; //!< map of defined materials
|
|
RWGltf_GltfBufferView myBuffViewPos; //!< current buffer view with nodes positions
|
|
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
|
|
ShapeToGltfFaceMap myBinDataMap; //!< map for TopoDS_Face to glTF face (merging duplicates)
|
|
int64_t myBinDataLen64; //!< length of binary file
|
|
|
|
std::vector<RWGltf_GltfBufferView> myBuffViewsDraco; //!< vector of buffers view with compression data
|
|
RWGltf_DracoParameters myDracoParameters; //!< Draco parameters
|
|
};
|
|
|
|
#endif // _RWGltf_CafWriter_HeaderFiler
|