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

0030953: Data Exchange - implement export of mesh data into glTF 2.0 format

Added new class RWGltf_CafWriter for exporting XCAF document into glTF file
as well as Draw Harness command WriteGltf.

Added auxiliary method OSD_Path::FileNameAndExtension() splitting file name into Name and Extension.
This commit is contained in:
kgv 2019-11-19 02:09:09 +03:00 committed by bugmaster
parent df8c7e3e18
commit 01b2f506d6
26 changed files with 3620 additions and 12 deletions

View File

@ -1675,3 +1675,34 @@ void OSD_Path::FolderAndFileFromPath (const TCollection_AsciiString& theFilePath
theFileName.Clear();
}
}
// =======================================================================
// function : FileNameAndExtension
// purpose :
// =======================================================================
void OSD_Path::FileNameAndExtension (const TCollection_AsciiString& theFilePath,
TCollection_AsciiString& theName,
TCollection_AsciiString& theExtension)
{
const Standard_Integer THE_EXT_MAX_LEN = 20; // this method is supposed to be used with normal extension
const Standard_Integer aLen = theFilePath.Length();
for (Standard_Integer anExtLen = 1; anExtLen < aLen && anExtLen < THE_EXT_MAX_LEN; ++anExtLen)
{
if (theFilePath.Value (aLen - anExtLen) == '.')
{
const Standard_Integer aNameUpper = aLen - anExtLen - 1;
if (aNameUpper < 1)
{
break;
}
theName = theFilePath.SubString (1, aNameUpper);
theExtension = theFilePath.SubString (aLen - anExtLen + 1, aLen);
theExtension.LowerCase();
return;
}
}
theName = theFilePath;
theExtension.Clear();
}

View File

@ -215,6 +215,18 @@ public:
TCollection_AsciiString& theFolder,
TCollection_AsciiString& theFileName);
//! Return file extension from the name in lower case.
//! Extension is expected to be within 20-symbols length, and determined as file name tail after last dot.
//! Example: IN theFilePath ='Image.sbs.JPG'
//! OUT theName ='Image.sbs'
//! OUT theFileName ='jpg'
//! @param theFilePath [in] file path
//! @param theName [out] file name without extension
//! @param theExtension [out] file extension in lower case and without dot
Standard_EXPORT static void FileNameAndExtension (const TCollection_AsciiString& theFilePath,
TCollection_AsciiString& theName,
TCollection_AsciiString& theExtension);
//! Detect absolute DOS-path also used in Windows.
//! The total path length is limited to 256 characters.
//! Sample path:

View File

@ -15,9 +15,16 @@ RWGltf_MaterialCommon.hxx
RWGltf_MaterialMetallicRoughness.hxx
RWGltf_CafReader.cxx
RWGltf_CafReader.hxx
RWGltf_CafWriter.cxx
RWGltf_CafWriter.hxx
RWGltf_GltfMaterialMap.cxx
RWGltf_GltfMaterialMap.hxx
RWGltf_GltfJsonParser.cxx
RWGltf_GltfJsonParser.pxx
RWGltf_GltfOStreamWriter.hxx
RWGltf_GltfSceneNodeMap.hxx
RWGltf_PrimitiveArrayReader.cxx
RWGltf_PrimitiveArrayReader.hxx
RWGltf_TriangulationReader.cxx
RWGltf_TriangulationReader.hxx
RWGltf_WriterTrsfFormat.hxx

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,294 @@
// 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_GltfBufferView.hxx>
#include <RWGltf_GltfFace.hxx>
#include <RWGltf_WriterTrsfFormat.hxx>
#include <RWMesh_CoordinateSystemConverter.hxx>
#include <XCAFPrs_Style.hxx>
#include <memory>
class Message_ProgressIndicator;
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:
//! 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 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; }
//! 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 Handle(Message_ProgressIndicator)& 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 Handle(Message_ProgressIndicator)& 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 Handle(Message_ProgressIndicator)& 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 Handle(Message_ProgressIndicator)& 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);
//! 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
Standard_EXPORT virtual void saveNodes (RWGltf_GltfFace& theGltfFace,
std::ostream& theBinFile,
const RWMesh_FaceIterator& theFaceIter,
Standard_Integer& theAccessorNb) 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
Standard_EXPORT virtual void saveNormals (RWGltf_GltfFace& theGltfFace,
std::ostream& theBinFile,
RWMesh_FaceIterator& theFaceIter,
Standard_Integer& theAccessorNb) 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
Standard_EXPORT virtual void saveTextCoords (RWGltf_GltfFace& theGltfFace,
std::ostream& theBinFile,
const RWMesh_FaceIterator& theFaceIter,
Standard_Integer& theAccessorNb) 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
Standard_EXPORT virtual void saveIndices (RWGltf_GltfFace& theGltfFace,
std::ostream& theBinFile,
const RWMesh_FaceIterator& theFaceIter,
Standard_Integer& theAccessorNb);
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,
RWGltf_GltfMaterialMap& theMaterialMap);
//! 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,
RWGltf_GltfMaterialMap& theMaterialMap);
//! 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,
const RWGltf_GltfMaterialMap& theMaterialMap);
//! 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 (const RWGltf_GltfMaterialMap& theMaterialMap);
//! 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,
RWGltf_GltfMaterialMap& theMaterialMap);
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
Standard_Boolean myIsBinary; //!< flag to write into binary glTF format (.glb)
Standard_Boolean myIsForcedUVExport; //!< export UV coordinates even if there are no mapped texture
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
opencascade::std::shared_ptr<RWGltf_GltfOStreamWriter>
myWriter; //!< JSON writer
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
NCollection_DataMap<TopoDS_Shape, RWGltf_GltfFace,
TopTools_ShapeMapHasher> myBinDataMap; //!< map for TopoDS_Face to glTF face (merging duplicates)
int64_t myBinDataLen64; //!< length of binary file
};
#endif // _RWGltf_CafWriter_HeaderFiler

View File

@ -1780,7 +1780,7 @@ bool RWGltf_GltfJsonParser::Parse (const Handle(Message_ProgressIndicator)& theP
}
return true;
#else
Message::DefaultMessenger()->Send ("Error: glTF reader is unavailable - OCCT has been built without RapidJSON support.", Message_Fail);
Message::DefaultMessenger()->Send ("Error: glTF reader is unavailable - OCCT has been built without RapidJSON support [HAVE_RAPIDJSON undefined].", Message_Fail);
return false;
#endif
}

View File

@ -0,0 +1,461 @@
// 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.
#include <RWGltf_GltfMaterialMap.hxx>
#include <RWGltf_GltfRootElement.hxx>
#ifdef HAVE_RAPIDJSON
#include <RWGltf_GltfOStreamWriter.hxx>
#endif
// =======================================================================
// function : baseColorTexture
// purpose :
// =======================================================================
const Handle(Image_Texture)& RWGltf_GltfMaterialMap::baseColorTexture (const Handle(XCAFDoc_VisMaterial)& theMat)
{
static const Handle(Image_Texture) THE_NULL_TEXTURE;
if (theMat.IsNull())
{
return THE_NULL_TEXTURE;
}
else if (theMat->HasPbrMaterial()
&& !theMat->PbrMaterial().BaseColorTexture.IsNull())
{
return theMat->PbrMaterial().BaseColorTexture;
}
else if (theMat->HasCommonMaterial()
&& !theMat->CommonMaterial().DiffuseTexture.IsNull())
{
return theMat->CommonMaterial().DiffuseTexture;
}
return THE_NULL_TEXTURE;
}
// =======================================================================
// function : RWGltf_GltfMaterialMap
// purpose :
// =======================================================================
RWGltf_GltfMaterialMap::RWGltf_GltfMaterialMap (const TCollection_AsciiString& theFile,
const Standard_Integer theDefSamplerId)
: RWMesh_MaterialMap (theFile),
myWriter (NULL),
myDefSamplerId (theDefSamplerId),
myNbImages (0)
{
myMatNameAsKey = false;
}
// =======================================================================
// function : ~RWGltf_GltfMaterialMap
// purpose :
// =======================================================================
RWGltf_GltfMaterialMap::~RWGltf_GltfMaterialMap()
{
//
}
// =======================================================================
// function : AddImages
// purpose :
// =======================================================================
void RWGltf_GltfMaterialMap::AddImages (RWGltf_GltfOStreamWriter* theWriter,
const XCAFPrs_Style& theStyle,
Standard_Boolean& theIsStarted)
{
if (theWriter == NULL
|| theStyle.Material().IsNull()
|| theStyle.Material()->IsEmpty())
{
return;
}
addImage (theWriter, baseColorTexture (theStyle.Material()), theIsStarted);
addImage (theWriter, theStyle.Material()->PbrMaterial().MetallicRoughnessTexture, theIsStarted);
addImage (theWriter, theStyle.Material()->PbrMaterial().NormalTexture, theIsStarted);
addImage (theWriter, theStyle.Material()->PbrMaterial().EmissiveTexture, theIsStarted);
addImage (theWriter, theStyle.Material()->PbrMaterial().OcclusionTexture, theIsStarted);
}
// =======================================================================
// function : addImage
// purpose :
// =======================================================================
void RWGltf_GltfMaterialMap::addImage (RWGltf_GltfOStreamWriter* theWriter,
const Handle(Image_Texture)& theTexture,
Standard_Boolean& theIsStarted)
{
#ifdef HAVE_RAPIDJSON
if (theTexture.IsNull()
|| myImageMap.IsBound1 (theTexture)
|| myImageFailMap.Contains (theTexture))
{
return;
}
TCollection_AsciiString aGltfImgKey = myNbImages;
++myNbImages;
for (; myImageMap.IsBound2 (aGltfImgKey); ++myNbImages)
{
aGltfImgKey = myNbImages;
}
TCollection_AsciiString aTextureUri;
if (!CopyTexture (aTextureUri, theTexture, aGltfImgKey))
{
myImageFailMap.Add (theTexture);
return;
}
myImageMap.Bind (theTexture, aGltfImgKey);
if (!theIsStarted)
{
theWriter->Key (RWGltf_GltfRootElementName (RWGltf_GltfRootElement_Images));
theWriter->StartArray();
theIsStarted = true;
}
theWriter->StartObject();
{
theWriter->Key ("uri");
theWriter->String (aTextureUri.ToCString());
}
theWriter->EndObject();
#else
(void )theWriter;
(void )theTexture;
(void )theIsStarted;
#endif
}
// =======================================================================
// function : AddMaterial
// purpose :
// =======================================================================
void RWGltf_GltfMaterialMap::AddMaterial (RWGltf_GltfOStreamWriter* theWriter,
const XCAFPrs_Style& theStyle,
Standard_Boolean& theIsStarted)
{
#ifdef HAVE_RAPIDJSON
if (theWriter == NULL
|| ((theStyle.Material().IsNull() || theStyle.Material()->IsEmpty())
&& !theStyle.IsSetColorSurf()))
{
return;
}
if (!theIsStarted)
{
theWriter->Key (RWGltf_GltfRootElementName (RWGltf_GltfRootElement_Materials));
theWriter->StartArray();
theIsStarted = true;
}
myWriter = theWriter;
AddMaterial (theStyle);
myWriter = NULL;
#else
(void )theWriter;
(void )theStyle;
(void )theIsStarted;
#endif
}
// =======================================================================
// function : AddTextures
// purpose :
// =======================================================================
void RWGltf_GltfMaterialMap::AddTextures (RWGltf_GltfOStreamWriter* theWriter,
const XCAFPrs_Style& theStyle,
Standard_Boolean& theIsStarted)
{
if (theWriter == NULL
|| theStyle.Material().IsNull()
|| theStyle.Material()->IsEmpty())
{
return;
}
addTexture (theWriter, baseColorTexture (theStyle.Material()), theIsStarted);
addTexture (theWriter, theStyle.Material()->PbrMaterial().MetallicRoughnessTexture, theIsStarted);
addTexture (theWriter, theStyle.Material()->PbrMaterial().NormalTexture, theIsStarted);
addTexture (theWriter, theStyle.Material()->PbrMaterial().EmissiveTexture, theIsStarted);
addTexture (theWriter, theStyle.Material()->PbrMaterial().OcclusionTexture, theIsStarted);
}
// =======================================================================
// function : addTexture
// purpose :
// =======================================================================
void RWGltf_GltfMaterialMap::addTexture (RWGltf_GltfOStreamWriter* theWriter,
const Handle(Image_Texture)& theTexture,
Standard_Boolean& theIsStarted)
{
#ifdef HAVE_RAPIDJSON
if (theTexture.IsNull()
|| myTextureMap.Contains (theTexture)
|| !myImageMap .IsBound1 (theTexture))
{
return;
}
const TCollection_AsciiString anImgKey = myImageMap.Find1 (theTexture);
myTextureMap.Add (theTexture);
if (anImgKey.IsEmpty())
{
return;
}
if (!theIsStarted)
{
theWriter->Key (RWGltf_GltfRootElementName (RWGltf_GltfRootElement_Textures));
theWriter->StartArray();
theIsStarted = true;
}
theWriter->StartObject();
{
theWriter->Key ("sampler");
theWriter->Int (myDefSamplerId); // mandatory field by specs
theWriter->Key ("source");
theWriter->Int (anImgKey.IntegerValue());
}
theWriter->EndObject();
#else
(void )theWriter;
(void )theTexture;
(void )theIsStarted;
#endif
}
// =======================================================================
// function : AddMaterial
// purpose :
// =======================================================================
TCollection_AsciiString RWGltf_GltfMaterialMap::AddMaterial (const XCAFPrs_Style& theStyle)
{
return RWMesh_MaterialMap::AddMaterial (theStyle);
}
// =======================================================================
// function : DefineMaterial
// purpose :
// =======================================================================
void RWGltf_GltfMaterialMap::DefineMaterial (const XCAFPrs_Style& theStyle,
const TCollection_AsciiString& /*theKey*/,
const TCollection_AsciiString& theName)
{
#ifdef HAVE_RAPIDJSON
if (myWriter == NULL)
{
Standard_ProgramError::Raise ("RWGltf_GltfMaterialMap::DefineMaterial() should be called with JSON Writer");
return;
}
XCAFDoc_VisMaterialPBR aPbrMat;
const bool hasMaterial = !theStyle.Material().IsNull()
&& !theStyle.Material()->IsEmpty();
if (hasMaterial)
{
aPbrMat = theStyle.Material()->ConvertToPbrMaterial();
}
else if (!myDefaultStyle.Material().IsNull()
&& myDefaultStyle.Material()->HasPbrMaterial())
{
aPbrMat = myDefaultStyle.Material()->PbrMaterial();
}
if (theStyle.IsSetColorSurf())
{
aPbrMat.BaseColor.SetRGB (theStyle.GetColorSurf());
if (theStyle.GetColorSurfRGBA().Alpha() < 1.0f)
{
aPbrMat.BaseColor.SetAlpha (theStyle.GetColorSurfRGBA().Alpha());
}
}
myWriter->StartObject();
{
myWriter->Key ("name");
myWriter->String (theName.ToCString());
myWriter->Key ("pbrMetallicRoughness");
myWriter->StartObject();
{
myWriter->Key ("baseColorFactor");
myWriter->StartArray();
{
myWriter->Double (aPbrMat.BaseColor.GetRGB().Red());
myWriter->Double (aPbrMat.BaseColor.GetRGB().Green());
myWriter->Double (aPbrMat.BaseColor.GetRGB().Blue());
myWriter->Double (aPbrMat.BaseColor.Alpha());
}
myWriter->EndArray();
if (const Handle(Image_Texture)& aBaseTexture = baseColorTexture (theStyle.Material()))
{
if (myImageMap.IsBound1 (aBaseTexture))
{
myWriter->Key ("baseColorTexture");
myWriter->StartObject();
{
myWriter->Key ("index");
const TCollection_AsciiString& anImageIdx = myImageMap.Find1 (aBaseTexture);
if (!anImageIdx.IsEmpty())
{
myWriter->Int (anImageIdx.IntegerValue());
}
}
myWriter->EndObject();
}
}
if (hasMaterial
|| aPbrMat.Metallic != 1.0f)
{
myWriter->Key ("metallicFactor");
myWriter->Double (aPbrMat.Metallic);
}
if (!aPbrMat.MetallicRoughnessTexture.IsNull()
&& myImageMap.IsBound1 (aPbrMat.MetallicRoughnessTexture))
{
myWriter->Key ("metallicRoughnessTexture");
myWriter->StartObject();
{
myWriter->Key ("index");
const TCollection_AsciiString& anImageIdx = myImageMap.Find1 (aPbrMat.MetallicRoughnessTexture);
if (!anImageIdx.IsEmpty())
{
myWriter->Int (anImageIdx.IntegerValue());
}
}
myWriter->EndObject();
}
if (hasMaterial
|| aPbrMat.Roughness != 1.0f)
{
myWriter->Key ("roughnessFactor");
myWriter->Double (aPbrMat.Roughness);
}
}
myWriter->EndObject();
if (theStyle.Material().IsNull()
|| theStyle.Material()->IsDoubleSided())
{
myWriter->Key ("doubleSided");
myWriter->Bool (true);
}
const Graphic3d_AlphaMode anAlphaMode = !theStyle.Material().IsNull() ? theStyle.Material()->AlphaMode() : Graphic3d_AlphaMode_BlendAuto;
switch (anAlphaMode)
{
case Graphic3d_AlphaMode_BlendAuto:
{
if (aPbrMat.BaseColor.Alpha() < 1.0f)
{
myWriter->Key ("alphaMode");
myWriter->String ("BLEND");
}
break;
}
case Graphic3d_AlphaMode_Opaque:
{
break;
}
case Graphic3d_AlphaMode_Mask:
{
myWriter->Key ("alphaMode");
myWriter->String ("MASK");
break;
}
case Graphic3d_AlphaMode_Blend:
{
myWriter->Key ("alphaMode");
myWriter->String ("BLEND");
break;
}
}
if (!theStyle.Material().IsNull()
&& theStyle.Material()->AlphaCutOff() != 0.5f)
{
myWriter->Key ("alphaCutoff");
myWriter->Double (theStyle.Material()->AlphaCutOff());
}
if (aPbrMat.EmissiveFactor != Graphic3d_Vec3 (0.0f, 0.0f, 0.0f))
{
myWriter->Key ("emissiveFactor");
myWriter->StartArray();
{
myWriter->Double (aPbrMat.EmissiveFactor.r());
myWriter->Double (aPbrMat.EmissiveFactor.g());
myWriter->Double (aPbrMat.EmissiveFactor.b());
}
myWriter->EndArray();
}
if (!aPbrMat.EmissiveTexture.IsNull()
&& myImageMap.IsBound1 (aPbrMat.EmissiveTexture))
{
myWriter->Key ("emissiveTexture");
myWriter->StartObject();
{
myWriter->Key ("index");
const TCollection_AsciiString& anImageIdx = myImageMap.Find1 (aPbrMat.EmissiveTexture);
if (!anImageIdx.IsEmpty())
{
myWriter->Int (anImageIdx.IntegerValue());
}
}
myWriter->EndObject();
}
if (!aPbrMat.NormalTexture.IsNull()
&& myImageMap.IsBound1 (aPbrMat.NormalTexture))
{
myWriter->Key ("normalTexture");
myWriter->StartObject();
{
myWriter->Key ("index");
const TCollection_AsciiString& anImageIdx = myImageMap.Find1 (aPbrMat.NormalTexture);
if (!anImageIdx.IsEmpty())
{
myWriter->Int (anImageIdx.IntegerValue());
}
}
myWriter->EndObject();
}
if (!aPbrMat.OcclusionTexture.IsNull()
&& myImageMap.IsBound1 (aPbrMat.OcclusionTexture))
{
myWriter->Key ("occlusionTexture");
myWriter->StartObject();
{
myWriter->Key ("index");
const TCollection_AsciiString& anImageIdx = myImageMap.Find1 (aPbrMat.OcclusionTexture);
if (!anImageIdx.IsEmpty())
{
myWriter->Int (anImageIdx.IntegerValue());
}
}
myWriter->EndObject();
}
}
myWriter->EndObject();
#else
(void )theStyle;
(void )theName;
#endif
}

View File

@ -0,0 +1,88 @@
// 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_GltfMaterialMap_HeaderFile
#define _RWGltf_GltfMaterialMap_HeaderFile
#include <RWMesh_MaterialMap.hxx>
class RWGltf_GltfOStreamWriter;
//! Material manager for exporting into glTF format.
class RWGltf_GltfMaterialMap : public RWMesh_MaterialMap
{
public:
//! Main constructor.
Standard_EXPORT RWGltf_GltfMaterialMap (const TCollection_AsciiString& theFile,
const Standard_Integer theDefSamplerId);
//! Destructor.
Standard_EXPORT virtual ~RWGltf_GltfMaterialMap();
//! Add material images.
Standard_EXPORT void AddImages (RWGltf_GltfOStreamWriter* theWriter,
const XCAFPrs_Style& theStyle,
Standard_Boolean& theIsStarted);
//! Add material.
Standard_EXPORT void AddMaterial (RWGltf_GltfOStreamWriter* theWriter,
const XCAFPrs_Style& theStyle,
Standard_Boolean& theIsStarted);
//! Add material textures.
Standard_EXPORT void AddTextures (RWGltf_GltfOStreamWriter* theWriter,
const XCAFPrs_Style& theStyle,
Standard_Boolean& theIsStarted);
//! Return extent of images map.
Standard_Integer NbImages() const { return myImageMap.Extent(); }
//! Return extent of textures map.
Standard_Integer NbTextures() const { return myTextureMap.Extent(); }
public:
//! Return base color texture.
Standard_EXPORT static const Handle(Image_Texture)& baseColorTexture (const Handle(XCAFDoc_VisMaterial)& theMat);
protected:
//! Add texture image.
Standard_EXPORT void addImage (RWGltf_GltfOStreamWriter* theWriter,
const Handle(Image_Texture)& theTexture,
Standard_Boolean& theIsStarted);
//! Add texture.
Standard_EXPORT void addTexture (RWGltf_GltfOStreamWriter* theWriter,
const Handle(Image_Texture)& theTexture,
Standard_Boolean& theIsStarted);
//! Add material
Standard_EXPORT virtual TCollection_AsciiString AddMaterial (const XCAFPrs_Style& theStyle) Standard_OVERRIDE;
//! Virtual method actually defining the material (e.g. export to the file).
Standard_EXPORT virtual void DefineMaterial (const XCAFPrs_Style& theStyle,
const TCollection_AsciiString& theKey,
const TCollection_AsciiString& theName) Standard_OVERRIDE;
protected:
RWGltf_GltfOStreamWriter* myWriter;
NCollection_DoubleMap<Handle(Image_Texture), TCollection_AsciiString, Image_Texture, TCollection_AsciiString> myImageMap;
NCollection_Map<Handle(Image_Texture), Image_Texture> myTextureMap;
Standard_Integer myDefSamplerId;
Standard_Integer myNbImages;
};
#endif // _RWGltf_GltfMaterialMap_HeaderFile

View File

@ -0,0 +1,29 @@
// Copyright (c) 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_GltfOStreamWriter_HeaderFile
#define _RWGltf_GltfOStreamWriter_HeaderFile
#include <rapidjson/prettywriter.h>
#include <rapidjson/ostreamwrapper.h>
//! rapidjson::Writer wrapper for forward declaration.
class RWGltf_GltfOStreamWriter : public rapidjson::Writer<rapidjson::OStreamWrapper>
{
public:
//! Main constructor.
RWGltf_GltfOStreamWriter (rapidjson::OStreamWrapper& theOStream)
: rapidjson::Writer<rapidjson::OStreamWrapper> (theOStream) {}
};
#endif // _RWGltf_GltfOStreamWriter_HeaderFile

View File

@ -0,0 +1,48 @@
// Copyright (c) 2018-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_GltfSceneNodeMap_HeaderFile
#define _RWGltf_GltfSceneNodeMap_HeaderFile
#include <NCollection_IndexedMap.hxx>
#include <XCAFPrs_DocumentExplorer.hxx>
//! Indexed map of scene nodes with custom search algorithm.
class RWGltf_GltfSceneNodeMap : public NCollection_IndexedMap<XCAFPrs_DocumentNode, XCAFPrs_DocumentNode>
{
public:
//! Empty constructor.
RWGltf_GltfSceneNodeMap() {}
//! Find index from document node string identifier.
Standard_Integer FindIndex (const TCollection_AsciiString& theNodeId) const
{
if (IsEmpty())
{
return 0;
}
for (IndexedMapNode* aNode1Iter = (IndexedMapNode* )myData1[::HashCode (theNodeId, NbBuckets())]; aNode1Iter != NULL; aNode1Iter = (IndexedMapNode* )aNode1Iter->Next())
{
if (::IsEqual (aNode1Iter->Key1().Id, theNodeId))
{
return aNode1Iter->Index();
}
}
return 0;
}
};
#endif // _RWGltf_GltfSceneNodeMap_HeaderFile

View File

@ -0,0 +1,26 @@
// 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_WriterTrsfFormat_HeaderFile
#define _RWGltf_WriterTrsfFormat_HeaderFile
//! Transformation format.
enum RWGltf_WriterTrsfFormat
{
RWGltf_WriterTrsfFormat_Compact = 0, //!< automatically choose most compact representation between Mat4 and TRS
RWGltf_WriterTrsfFormat_Mat4 = 1, //!< 4x4 transformation Matrix
RWGltf_WriterTrsfFormat_TRS = 2, //!< transformation decomposed into Translation vector, Rotation quaternion and Scale factor (T * R * S)
};
enum { RWGltf_WriterTrsfFormat_LOWER = 0, RWGltf_WriterTrsfFormat_UPPER = RWGltf_WriterTrsfFormat_TRS }; // aliases
#endif // _RWGltf_WriterTrsfFormat_HeaderFile

View File

@ -3,4 +3,8 @@ RWMesh_CoordinateSystemConverter.cxx
RWMesh_CoordinateSystemConverter.hxx
RWMesh_CafReader.cxx
RWMesh_CafReader.hxx
RWMesh_FaceIterator.cxx
RWMesh_FaceIterator.hxx
RWMesh_MaterialMap.cxx
RWMesh_MaterialMap.hxx
RWMesh_NodeAttributes.hxx

View File

@ -0,0 +1,240 @@
// 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.
#include <RWMesh_FaceIterator.hxx>
#include <BRepAdaptor_Surface.hxx>
#include <BRep_Tool.hxx>
#include <TopExp.hxx>
#include <TopoDS.hxx>
#include <XCAFDoc_ShapeTool.hxx>
#include <XCAFPrs.hxx>
// =======================================================================
// function : RWMesh_FaceIterator
// purpose :
// =======================================================================
RWMesh_FaceIterator::RWMesh_FaceIterator (const TDF_Label& theLabel,
const TopLoc_Location& theLocation,
const Standard_Boolean theToMapColors,
const XCAFPrs_Style& theStyle)
: myDefStyle (theStyle),
myToMapColors (theToMapColors),
mySLTool (1, 1e-12),
myNodes (NULL),
myNormals (NULL),
myNodeUVs (NULL),
myHasNormals (false),
myIsMirrored (false),
myHasFaceColor (false)
{
TopoDS_Shape aShape;
if (!XCAFDoc_ShapeTool::GetShape (theLabel, aShape)
|| aShape.IsNull())
{
return;
}
aShape.Location (theLocation);
myFaceIter.Init (aShape, TopAbs_FACE);
if (theToMapColors)
{
dispatchStyles (theLabel, theLocation, theStyle);
}
Next();
}
// =======================================================================
// function : dispatchStyles
// purpose :
// =======================================================================
void RWMesh_FaceIterator::dispatchStyles (const TDF_Label& theLabel,
const TopLoc_Location& theLocation,
const XCAFPrs_Style& theStyle)
{
TopLoc_Location aDummyLoc;
XCAFPrs_IndexedDataMapOfShapeStyle aStyles;
XCAFPrs::CollectStyleSettings (theLabel, aDummyLoc, aStyles);
Standard_Integer aNbTypes[TopAbs_SHAPE] = {};
for (Standard_Integer aTypeIter = TopAbs_FACE; aTypeIter >= TopAbs_COMPOUND; --aTypeIter)
{
if (aTypeIter != TopAbs_FACE
&& aNbTypes[aTypeIter] == 0)
{
continue;
}
for (XCAFPrs_IndexedDataMapOfShapeStyle::Iterator aStyleIter (aStyles); aStyleIter.More(); aStyleIter.Next())
{
const TopoDS_Shape& aKeyShape = aStyleIter.Key();
const TopAbs_ShapeEnum aKeyShapeType = aKeyShape.ShapeType();
if (aTypeIter == TopAbs_FACE)
{
++aNbTypes[aKeyShapeType];
}
if (aTypeIter != aKeyShapeType)
{
continue;
}
XCAFPrs_Style aCafStyle = aStyleIter.Value();
if (!aCafStyle.IsSetColorCurv()
&& theStyle.IsSetColorCurv())
{
aCafStyle.SetColorCurv (theStyle.GetColorCurv());
}
if (!aCafStyle.IsSetColorSurf()
&& theStyle.IsSetColorSurf())
{
aCafStyle.SetColorSurf (theStyle.GetColorSurfRGBA());
}
if (aCafStyle.Material().IsNull()
&& !theStyle.Material().IsNull())
{
aCafStyle.SetMaterial (theStyle.Material());
}
TopoDS_Shape aKeyShapeLocated = aKeyShape.Located (theLocation);
if (aKeyShapeType == TopAbs_FACE)
{
myStyles.Bind (aKeyShapeLocated, aCafStyle);
}
else
{
for (TopExp_Explorer aFaceIter (aKeyShapeLocated, TopAbs_FACE); aFaceIter.More(); aFaceIter.Next())
{
if (!myStyles.IsBound (aFaceIter.Current()))
{
myStyles.Bind (aFaceIter.Current(), aCafStyle);
}
}
}
}
}
}
// =======================================================================
// function : normal
// purpose :
// =======================================================================
gp_Dir RWMesh_FaceIterator::normal (Standard_Integer theNode)
{
gp_Dir aNormal (gp::DZ());
if (myNormals != NULL)
{
const Standard_Integer aNodeIndex = theNode - myNodes->Lower();
const Graphic3d_Vec3 aNormVec3 (myNormals->Value (myNormals->Lower() + aNodeIndex * 3),
myNormals->Value (myNormals->Lower() + aNodeIndex * 3 + 1),
myNormals->Value (myNormals->Lower() + aNodeIndex * 3 + 2));
if (aNormVec3.Modulus() != 0.0f)
{
aNormal.SetCoord (aNormVec3.x(), aNormVec3.y(), aNormVec3.z());
}
}
else if (myHasNormals
&& myNodeUVs != NULL)
{
const gp_XY& anUV = myNodeUVs->Value (theNode).XY();
mySLTool.SetParameters (anUV.X(), anUV.Y());
if (mySLTool.IsNormalDefined())
{
aNormal = mySLTool.Normal();
}
}
return aNormal;
}
// =======================================================================
// function : Next
// purpose :
// =======================================================================
void RWMesh_FaceIterator::Next()
{
for (; myFaceIter.More(); myFaceIter.Next())
{
myFace = TopoDS::Face (myFaceIter.Current());
myPolyTriang = BRep_Tool::Triangulation (myFace, myFaceLocation);
myTrsf = myFaceLocation.Transformation();
if (myPolyTriang.IsNull()
|| myPolyTriang->Triangles().Length() == 0)
{
resetFace();
continue;
}
initFace();
myFaceIter.Next();
return;
}
resetFace();
}
// =======================================================================
// function : initFace
// purpose :
// =======================================================================
void RWMesh_FaceIterator::initFace()
{
myHasNormals = false;
myHasFaceColor = false;
myIsMirrored = myTrsf.VectorialPart().Determinant() < 0.0;
myNormals = NULL;
myNodeUVs = NULL;
myNodes = &myPolyTriang->Nodes();
if (myPolyTriang->HasNormals())
{
myNormals = &myPolyTriang->Normals();
myHasNormals = true;
}
if (myPolyTriang->HasUVNodes())
{
myNodeUVs = &myPolyTriang->UVNodes();
if (!myHasNormals)
{
TopoDS_Face aFaceFwd = TopoDS::Face (myFace.Oriented (TopAbs_FORWARD));
aFaceFwd.Location (TopLoc_Location());
TopLoc_Location aLoc;
if (!BRep_Tool::Surface (aFaceFwd, aLoc).IsNull())
{
myFaceAdaptor.Initialize (aFaceFwd, false);
mySLTool.SetSurface (myFaceAdaptor);
myHasNormals = true;
}
}
}
if (!myToMapColors)
{
return;
}
if (!myStyles.Find (myFace, myFaceStyle))
{
myFaceStyle = myDefStyle;
}
if (!myFaceStyle.Material().IsNull())
{
myHasFaceColor = true;
myFaceColor = myFaceStyle.Material()->BaseColor();
}
else if (myFaceStyle.IsSetColorSurf())
{
myHasFaceColor = true;
myFaceColor = myFaceStyle.GetColorSurfRGBA();
}
}

View File

@ -0,0 +1,205 @@
// 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 _RWMesh_FaceIterator_HeaderFile
#define _RWMesh_FaceIterator_HeaderFile
#include <BRepLProp_SLProps.hxx>
#include <gp_Trsf.hxx>
#include <NCollection_DataMap.hxx>
#include <Poly_Array1OfTriangle.hxx>
#include <Poly_Triangulation.hxx>
#include <TopExp_Explorer.hxx>
#include <TopoDS_Face.hxx>
#include <TopTools_IndexedDataMapOfShapeListOfShape.hxx>
#include <XCAFPrs_Style.hxx>
#include <algorithm>
class TDF_Label;
//! Auxiliary class to iterate through triangulated faces.
class RWMesh_FaceIterator
{
public:
//! Main constructor.
Standard_EXPORT RWMesh_FaceIterator (const TDF_Label& theLabel,
const TopLoc_Location& theLocation,
const Standard_Boolean theToMapColors = false,
const XCAFPrs_Style& theStyle = XCAFPrs_Style());
//! Return true if iterator points to the valid triangulation.
bool More() const { return !myPolyTriang.IsNull(); }
//! Find next value.
Standard_EXPORT void Next();
//! Return current face.
const TopoDS_Face& Face() const { return myFace; }
//! Return current face triangulation.
const Handle(Poly_Triangulation)& Triangulation() const { return myPolyTriang; }
//! Return true if mesh data is defined.
bool IsEmptyMesh() const
{
return myPolyTriang.IsNull()
|| (myPolyTriang->NbNodes() < 1 && myPolyTriang->NbTriangles() < 1);
}
public:
//! Return face material.
const XCAFPrs_Style& FaceStyle() const { return myFaceStyle; }
//! Return TRUE if face color is set.
bool HasFaceColor() const { return myHasFaceColor; }
//! Return face color.
const Quantity_ColorRGBA& FaceColor() const { return myFaceColor; }
public:
//! Return number of elements of specific type for the current face.
Standard_Integer NbTriangles() const { return myPolyTriang->NbTriangles(); }
//! Lower element index in current triangulation.
Standard_Integer ElemLower() const { return myPolyTriang->Triangles().Lower(); }
//! Upper element index in current triangulation.
Standard_Integer ElemUpper() const { return myPolyTriang->Triangles().Upper(); }
//! Return triangle with specified index with applied Face orientation.
Poly_Triangle TriangleOriented (Standard_Integer theElemIndex) const
{
Poly_Triangle aTri = triangle (theElemIndex);
if ((myFace.Orientation() == TopAbs_REVERSED) ^ myIsMirrored)
{
return Poly_Triangle (aTri.Value (1), aTri.Value (3), aTri.Value (2));
}
return aTri;
}
public:
//! Return true if triangulation has defined normals.
bool HasNormals() const { return myHasNormals; }
//! Return true if triangulation has defined normals.
bool HasTexCoords() const { return myNodeUVs != NULL; }
//! Return normal at specified node index with face transformation applied and face orientation applied.
gp_Dir NormalTransformed (Standard_Integer theNode)
{
gp_Dir aNorm = normal (theNode);
if (myTrsf.Form() != gp_Identity)
{
aNorm.Transform (myTrsf);
}
if (myFace.Orientation() == TopAbs_REVERSED)
{
aNorm.Reverse();
}
return aNorm;
}
//! Return number of nodes for the current face.
Standard_Integer NbNodes() const
{
return !myPolyTriang.IsNull()
? myPolyTriang->Nodes().Length()
: 0;
}
//! Lower node index in current triangulation.
Standard_Integer NodeLower() const { return myPolyTriang->Nodes().Lower(); }
//! Upper node index in current triangulation.
Standard_Integer NodeUpper() const { return myPolyTriang->Nodes().Upper(); }
//! Return the node with specified index with applied transformation.
gp_Pnt NodeTransformed (const Standard_Integer theNode) const
{
gp_Pnt aNode = node (theNode);
aNode.Transform (myTrsf);
return aNode;
}
//! Return texture coordinates for the node.
gp_Pnt2d NodeTexCoord (const Standard_Integer theNode) const
{
return myNodeUVs != NULL ? myNodeUVs->Value (theNode) : gp_Pnt2d();
}
public:
//! Return the node with specified index with applied transformation.
gp_Pnt node (const Standard_Integer theNode) const { return myPolyTriang->Nodes().Value (theNode); }
//! Return normal at specified node index without face transformation applied.
Standard_EXPORT gp_Dir normal (Standard_Integer theNode);
//! Return triangle with specified index.
Poly_Triangle triangle (Standard_Integer theElemIndex) const { return myPolyTriang->Triangles().Value (theElemIndex); }
private:
//! Dispatch face styles.
void dispatchStyles (const TDF_Label& theLabel,
const TopLoc_Location& theLocation,
const XCAFPrs_Style& theStyle);
//! Reset information for current face.
void resetFace()
{
myPolyTriang.Nullify();
myFace.Nullify();
myNodes = NULL;
myNormals = NULL;
myNodeUVs = NULL;
myHasNormals = false;
myHasFaceColor = false;
myFaceColor = Quantity_ColorRGBA();
myFaceStyle = XCAFPrs_Style();
}
//! Initialize face properties.
void initFace();
private:
NCollection_DataMap<TopoDS_Shape, XCAFPrs_Style, TopTools_ShapeMapHasher>
myStyles; //!< Face -> Style map
XCAFPrs_Style myDefStyle; //!< default style for faces without dedicated style
Standard_Boolean myToMapColors; //!< flag to dispatch styles
TopExp_Explorer myFaceIter; //!< face explorer
TopoDS_Face myFace; //!< current face
Handle(Poly_Triangulation) myPolyTriang; //!< triangulation of current face
TopLoc_Location myFaceLocation; //!< current face location
BRepLProp_SLProps mySLTool; //!< auxiliary tool for fetching normals from surface
BRepAdaptor_Surface myFaceAdaptor; //!< surface adaptor for fetching normals from surface
const TColgp_Array1OfPnt* myNodes; //!< node positions of current face
const TShort_Array1OfShortReal* myNormals; //!< node normals of current face
const TColgp_Array1OfPnt2d* myNodeUVs; //!< node UV coordinates of current face
Standard_Boolean myHasNormals; //!< flag indicating that current face has normals
gp_Trsf myTrsf; //!< current face transformation
Standard_Boolean myIsMirrored; //!< flag indicating that face triangles should be mirrored
XCAFPrs_Style myFaceStyle; //!< current face style
Quantity_ColorRGBA myFaceColor; //!< current face color
Standard_Boolean myHasFaceColor; //!< flag indicating that current face has assigned color
};
#endif // _RWMesh_FaceIterator_HeaderFile

View File

@ -0,0 +1,239 @@
// 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.
#include <RWMesh_MaterialMap.hxx>
#include <Message.hxx>
#include <Message_Messenger.hxx>
#include <OSD_Directory.hxx>
#include <OSD_File.hxx>
#include <OSD_Path.hxx>
#include <OSD_Protection.hxx>
#include <TDataStd_Name.hxx>
#include <TDF_Label.hxx>
// =======================================================================
// function : RWMesh_MaterialMap
// purpose :
// =======================================================================
RWMesh_MaterialMap::RWMesh_MaterialMap (const TCollection_AsciiString& theFile)
: myFileName (theFile),
myKeyPrefix ("mat_"),
myNbMaterials (0),
myIsFailed (false),
myMatNameAsKey (true)
{
TCollection_AsciiString aFileName, aFileExt;
OSD_Path::FolderAndFileFromPath (theFile, myFolder, aFileName);
OSD_Path::FileNameAndExtension (aFileName, myShortFileNameBase, aFileExt);
}
// =======================================================================
// function : ~RWMesh_MaterialMap
// purpose :
// =======================================================================
RWMesh_MaterialMap::~RWMesh_MaterialMap()
{
//
}
// =======================================================================
// function : AddMaterial
// purpose :
// =======================================================================
TCollection_AsciiString RWMesh_MaterialMap::AddMaterial (const XCAFPrs_Style& theStyle)
{
if (myStyles.IsBound1 (theStyle))
{
return myStyles.Find1 (theStyle);
}
TCollection_AsciiString aMatKey, aMatName, aMatNameSuffix;
int aCounter = 0;
int* aCounterPtr = &myNbMaterials;
if (myMatNameAsKey)
{
if (!theStyle.Material().IsNull()
&& !theStyle.Material()->IsEmpty())
{
aCounterPtr = &aCounter;
Handle(TDataStd_Name) aNodeName;
if (!theStyle.Material()->Label().IsNull()
&& theStyle.Material()->Label().FindAttribute (TDataStd_Name::GetID(), aNodeName))
{
aMatName = aNodeName->Get();
}
else
{
aMatName = "mat";
}
aMatNameSuffix = aMatName;
}
else
{
++myNbMaterials;
aMatNameSuffix = myKeyPrefix;
aMatName = aMatNameSuffix + myNbMaterials;
}
aMatKey = aMatName;
}
else
{
aMatKey = myNbMaterials++; // starts from 0
aMatNameSuffix = myKeyPrefix;
aMatName = aMatNameSuffix + aMatKey;
}
for (;; ++(*aCounterPtr))
{
if (myStyles.IsBound2 (aMatKey))
{
if (myMatNameAsKey)
{
aMatName = aMatNameSuffix + (*aCounterPtr);
aMatKey = aMatName;
}
else
{
aMatKey = *aCounterPtr;
aMatName = aMatNameSuffix + aMatKey;
}
continue;
}
break;
}
myStyles.Bind (theStyle, aMatKey);
DefineMaterial (theStyle, aMatKey, aMatName);
return aMatKey;
}
// =======================================================================
// function : copyFileTo
// purpose :
// =======================================================================
bool RWMesh_MaterialMap::copyFileTo (const TCollection_AsciiString& theFileSrc,
const TCollection_AsciiString& theFileDst)
{
if (theFileSrc.IsEmpty()
|| theFileDst.IsEmpty())
{
return false;
}
else if (theFileSrc == theFileDst)
{
return true;
}
try
{
OSD_Path aSrcPath (theFileSrc);
OSD_Path aDstPath (theFileDst);
OSD_File aFileSrc (aSrcPath);
if (!aFileSrc.Exists())
{
Message::DefaultMessenger()->Send (TCollection_AsciiString("Failed to copy file - source file '")
+ theFileSrc + "' does not exist\n", Message_Fail);
return false;
}
aFileSrc.Copy (aDstPath);
return !aFileSrc.Failed();
}
catch (Standard_Failure const& theException)
{
Message::DefaultMessenger()->Send (TCollection_AsciiString("Failed to copy file\n") +
theException.GetMessageString(), Message_Fail);
return false;
}
}
// =======================================================================
// function : CopyTexture
// purpose :
// =======================================================================
bool RWMesh_MaterialMap::CopyTexture (TCollection_AsciiString& theResTexture,
const Handle(Image_Texture)& theTexture,
const TCollection_AsciiString& theKey)
{
CreateTextureFolder();
TCollection_AsciiString aTexFileName;
TCollection_AsciiString aTextureSrc = theTexture->FilePath();
if (!aTextureSrc.IsEmpty())
{
TCollection_AsciiString aSrcTexFolder;
OSD_Path::FolderAndFileFromPath (aTextureSrc, aSrcTexFolder, aTexFileName);
const TCollection_AsciiString aResTexFile = myTexFolder + aTexFileName;
theResTexture = myTexFolderShort + aTexFileName;
return copyFileTo (aTextureSrc, aResTexFile);
}
TCollection_AsciiString anExt = theTexture->ProbeImageFileFormat();
if (anExt.IsEmpty())
{
anExt = "bin";
}
aTexFileName = theKey + "." + anExt;
const TCollection_AsciiString aResTexFile = myTexFolder + aTexFileName;
theResTexture = myTexFolderShort + aTexFileName;
return theTexture->WriteImage (aResTexFile);
}
// =======================================================================
// function : CreateTextureFolder
// purpose :
// =======================================================================
bool RWMesh_MaterialMap::CreateTextureFolder()
{
if (!myTexFolder.IsEmpty())
{
return true;
}
myTexFolderShort = myShortFileNameBase + "_textures/";
myTexFolder = myFolder + "/" + myTexFolderShort;
OSD_Path aTexFolderPath (myTexFolder);
OSD_Directory aTexDir (aTexFolderPath);
if (aTexDir.Exists())
{
return true;
}
OSD_Path aResFolderPath (myFolder);
OSD_Directory aResDir (aResFolderPath);
if (!aResDir.Exists())
{
return false;
}
const OSD_Protection aParentProt = aResDir.Protection();
OSD_Protection aProt = aParentProt;
if (aProt.User() == OSD_None)
{
aProt.SetUser (OSD_RWXD);
}
if (aProt.System() == OSD_None)
{
aProt.SetSystem (OSD_RWXD);
}
aTexDir.Build (aProt);
if (aTexDir.Failed())
{
// fallback to the same folder as output model file
myTexFolder = myFolder;
myTexFolderShort.Clear();
return true;
}
return true;
}

View File

@ -0,0 +1,101 @@
// 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 _RWMesh_MaterialMap_HeaderFile
#define _RWMesh_MaterialMap_HeaderFile
#include <NCollection_DoubleMap.hxx>
#include <NCollection_Map.hxx>
#include <XCAFPrs_Style.hxx>
//! Material manager.
//! Provides an interface for collecting all materials within the document before writing it into file,
//! and for copying associated image files (textures) into sub-folder near by exported model.
class RWMesh_MaterialMap
{
public:
//! Main constructor.
Standard_EXPORT RWMesh_MaterialMap (const TCollection_AsciiString& theFile);
//! Destructor.
Standard_EXPORT virtual ~RWMesh_MaterialMap();
//! 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; }
//! Find already registered material
TCollection_AsciiString FindMaterial (const XCAFPrs_Style& theStyle) const
{
if (myStyles.IsBound1 (theStyle))
{
return myStyles.Find1 (theStyle);
}
return TCollection_AsciiString();
}
//! Register material and return its name identifier.
Standard_EXPORT virtual TCollection_AsciiString AddMaterial (const XCAFPrs_Style& theStyle);
//! Create texture folder "modelName/textures"; for example:
//! MODEL: Path/ModelName.gltf
//! IMAGES: Path/ModelName/textures/
//! Warning! Output folder is NOT cleared.
Standard_EXPORT virtual bool CreateTextureFolder();
//! Copy and rename texture file to the new location.
//! @param theResTexture [out] result texture file path (relative to the model)
//! @param theTexture [in] original texture
//! @param theKey [in] material key
Standard_EXPORT virtual bool CopyTexture (TCollection_AsciiString& theResTexture,
const Handle(Image_Texture)& theTexture,
const TCollection_AsciiString& theKey);
//! Virtual method actually defining the material (e.g. export to the file).
virtual void DefineMaterial (const XCAFPrs_Style& theStyle,
const TCollection_AsciiString& theKey,
const TCollection_AsciiString& theName) = 0;
//! Return failed flag.
bool IsFailed() const { return myIsFailed; }
protected:
//! Copy file to another place.
Standard_EXPORT static bool copyFileTo (const TCollection_AsciiString& theFileSrc,
const TCollection_AsciiString& theFileDst);
protected:
TCollection_AsciiString myFolder; //!< output folder for glTF file
TCollection_AsciiString myTexFolder; //!< output folder for images (full path)
TCollection_AsciiString myTexFolderShort; //!< output folder for images (short path)
TCollection_AsciiString myFileName; //!< output glTF file path
TCollection_AsciiString myShortFileNameBase; //!< output glTF file name without extension
TCollection_AsciiString myKeyPrefix; //!< prefix for generated keys
NCollection_DoubleMap<XCAFPrs_Style, TCollection_AsciiString,
XCAFPrs_Style, TCollection_AsciiString>
myStyles; //!< map of processed styles
NCollection_Map<Handle(Image_Texture), Image_Texture>
myImageFailMap; //!< map of images failed to be copied
XCAFPrs_Style myDefaultStyle; //!< default material definition to be used for nodes with only color defined
Standard_Integer myNbMaterials; //!< number of registered materials
Standard_Boolean myIsFailed; //!< flag indicating failure
Standard_Boolean myMatNameAsKey; //!< flag indicating usage of material name as key
};
#endif // _RWMesh_MaterialMap_HeaderFile

View File

@ -20,4 +20,5 @@ TKSTL
TKVRML
TKLCAF
TKDCAF
TKXCAF
TKRWMesh

View File

@ -34,6 +34,23 @@ struct XCAFPrs_DocumentNode
Standard_Boolean IsAssembly; //!< flag indicating that this label is assembly
XCAFPrs_DocumentNode() : IsAssembly (Standard_False) {}
public: // Methods for hash map
//! Return hash code based on node string identifier.
static Standard_Integer HashCode (const XCAFPrs_DocumentNode& theNode,
const Standard_Integer theN)
{
return ::HashCode (theNode.Id, theN);
}
//! Return TRUE if two document nodes has the same string identifier.
static Standard_Boolean IsEqual (const XCAFPrs_DocumentNode& theNode1,
const XCAFPrs_DocumentNode& theNode2)
{
return theNode1.Id == theNode2.Id;
}
};
#endif // _XCAFPrs_DocumentNode_HeaderFile

View File

@ -43,6 +43,7 @@
#include <Quantity_HArray1OfColor.hxx>
#include <Quantity_NameOfColor.hxx>
#include <RWGltf_CafReader.hxx>
#include <RWGltf_CafWriter.hxx>
#include <RWStl.hxx>
#include <RWObj.hxx>
#include <RWObj_CafReader.hxx>
@ -69,6 +70,8 @@
#include <VrmlData_DataMapOfShapeAppearance.hxx>
#include <VrmlData_Scene.hxx>
#include <VrmlData_ShapeConvert.hxx>
#include <XCAFDoc_DocumentTool.hxx>
#include <XCAFDoc_ShapeTool.hxx>
#include <XSDRAW.hxx>
#include <XSDRAWIGES.hxx>
#include <XSDRAWSTEP.hxx>
@ -205,6 +208,117 @@ static Standard_Integer ReadGltf (Draw_Interpretor& theDI,
return 0;
}
//=============================================================================
//function : WriteGltf
//purpose : Writes glTF file
//=============================================================================
static Standard_Integer WriteGltf (Draw_Interpretor& theDI,
Standard_Integer theNbArgs,
const char** theArgVec)
{
TCollection_AsciiString aGltfFilePath;
Handle(TDocStd_Document) aDoc;
Handle(TDocStd_Application) anApp = DDocStd::GetApplication();
TColStd_IndexedDataMapOfStringString aFileInfo;
RWGltf_WriterTrsfFormat aTrsfFormat = RWGltf_WriterTrsfFormat_Compact;
bool toForceUVExport = false;
for (Standard_Integer anArgIter = 1; anArgIter < theNbArgs; ++anArgIter)
{
TCollection_AsciiString anArgCase (theArgVec[anArgIter]);
anArgCase.LowerCase();
if (anArgCase == "-comments"
&& anArgIter + 1 < theNbArgs)
{
aFileInfo.Add ("Comments", theArgVec[++anArgIter]);
}
else if (anArgCase == "-author"
&& anArgIter + 1 < theNbArgs)
{
aFileInfo.Add ("Author", theArgVec[++anArgIter]);
}
else if (anArgCase == "-forceuvexport"
|| anArgCase == "-forceuv")
{
toForceUVExport = true;
if (anArgIter + 1 < theNbArgs
&& ViewerTest::ParseOnOff (theArgVec[anArgIter + 1], toForceUVExport))
{
++anArgIter;
}
}
else if (anArgCase == "-trsfformat"
&& anArgIter + 1 < theNbArgs)
{
TCollection_AsciiString aTrsfStr (theArgVec[++anArgIter]);
aTrsfStr.LowerCase();
if (aTrsfStr == "compact")
{
aTrsfFormat = RWGltf_WriterTrsfFormat_Compact;
}
else if (aTrsfStr == "mat4")
{
aTrsfFormat = RWGltf_WriterTrsfFormat_Mat4;
}
else if (aTrsfStr == "trs")
{
aTrsfFormat = RWGltf_WriterTrsfFormat_TRS;
}
else
{
std::cout << "Syntax error at '" << anArgCase << "'\n";
return 1;
}
}
else if (aDoc.IsNull())
{
Standard_CString aNameVar = theArgVec[anArgIter];
DDocStd::GetDocument (aNameVar, aDoc, false);
if (aDoc.IsNull())
{
TopoDS_Shape aShape = DBRep::Get (aNameVar);
if (aShape.IsNull())
{
std::cout << "Syntax error: '" << aNameVar << "' is not a shape nor document\n";
return 1;
}
anApp->NewDocument (TCollection_ExtendedString ("BinXCAF"), aDoc);
Handle(XCAFDoc_ShapeTool) aShapeTool = XCAFDoc_DocumentTool::ShapeTool (aDoc->Main());
aShapeTool->AddShape (aShape);
}
}
else if (aGltfFilePath.IsEmpty())
{
aGltfFilePath = theArgVec[anArgIter];
}
else
{
std::cout << "Syntax error at '" << theArgVec[anArgIter] << "'\n";
return 1;
}
}
if (aGltfFilePath.IsEmpty())
{
std::cout << "Syntax error: wrong number of arguments\n";
return 1;
}
Handle(Draw_ProgressIndicator) aProgress = new Draw_ProgressIndicator (theDI, 1);
TCollection_AsciiString anExt = aGltfFilePath;
anExt.LowerCase();
const Standard_Real aSystemUnitFactor = UnitsMethods::GetCasCadeLengthUnit() * 0.001;
RWGltf_CafWriter aWriter (aGltfFilePath, anExt.EndsWith (".glb"));
aWriter.SetTransformationFormat (aTrsfFormat);
aWriter.SetForcedUVExport (toForceUVExport);
aWriter.ChangeCoordinateSystemConverter().SetInputLengthUnit (aSystemUnitFactor);
aWriter.ChangeCoordinateSystemConverter().SetInputCoordinateSystem (RWMesh_CoordinateSystem_Zup);
aWriter.Perform (aDoc, aFileInfo, aProgress);
return 0;
}
static Standard_Integer writestl
(Draw_Interpretor& di, Standard_Integer argc, const char** argv)
{
@ -1604,6 +1718,15 @@ void XSDRAWSTLVRML::InitCommands (Draw_Interpretor& theCommands)
"readgltf shape file"
"\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] [-comments Text] [-author Name] [-forceUVExport]"
"\n\t\t: Write XDE document into glTF file."
"\n\t\t: -trsfFormat preferred transformation format"
"\n\t\t: -forceUVExport always export UV coordinates",
__FILE__, WriteGltf, g);
theCommands.Add ("writegltf",
"writegltf shape file",
__FILE__, WriteGltf, g);
theCommands.Add ("writevrml", "shape file [version VRML#1.0/VRML#2.0 (1/2): 2 by default] [representation shaded/wireframe/both (0/1/2): 1 by default]",__FILE__,writevrml,g);
theCommands.Add ("writestl", "shape file [ascii/binary (0/1) : 1 by default] [InParallel (0/1) : 0 by default]",__FILE__,writestl,g);
theCommands.Add ("readstl",

View File

@ -1,3 +1,13 @@
if { [info exists occ_tmp_files] } {
foreach aTmpFileIter $occ_tmp_files {
if {[file exists "$aTmpFileIter"] == 1} {
puts "Deleting temporary file $aTmpFileIter"
file delete -force "$aTmpFileIter"
}
}
set occ_tmp_files ""
}
puts ""
puts "TEST COMPLETED"
puts ""

View File

@ -0,0 +1,13 @@
puts "========"
puts "0030953: Data Exchange - implement export of mesh data into glTF 2.0 format"
puts "Test case exporting BRep model into glb (binary glTF) file."
puts "========"
restore [locate_data_file Ball.brep] b
incmesh b 0.1
set aTmpGltf "${imagedir}/${casename}_tmp.glb"
writegltf b "$aTmpGltf"
ReadGltf D "$aTmpGltf"
XGetOneShape s D
checknbshapes s -face 17 -compound 3

View File

@ -0,0 +1,2 @@
pload XDE OCAF MODELING VISUALIZATION
catch { Close D }

View File

@ -0,0 +1,20 @@
vclear
vinit View1
XDisplay -dispMode 1 D
vaxo
vfit
vrenderparams -shadingModel PHONG
vlight -change 0 -intensity 2.5
vlight -change 1 -intensity 0.3
vcamera -orthographic
vdump ${imagedir}/${casename}.png
vdump ${imagedir}/${casename}_ortho_phong.png
vcamera -persp
vdump ${imagedir}/${casename}_persp_phong.png
vrenderparams -shadingModel PBR
vcamera -orthographic
vdump ${imagedir}/${casename}_ortho_pbr.png
vcamera -persp
vdump ${imagedir}/${casename}_persp_pbr.png

View File

@ -0,0 +1,20 @@
puts "========"
puts "0030953: Data Exchange - implement export of mesh data into glTF 2.0 format"
puts "Test case exporting glTF model into glTF file."
puts "========"
catch { Close D1 }
ReadGltf D1 [locate_data_file bug30691_DamagedHelmet.gltf]
set aTmpGltfBase "${imagedir}/${casename}_tmp"
set aTmpGltf "${aTmpGltfBase}.gltf"
lappend occ_tmp_files $aTmpGltf
lappend occ_tmp_files "${aTmpGltfBase}.bin"
lappend occ_tmp_files "${aTmpGltfBase}_textures"
WriteGltf D1 "$aTmpGltf"
ReadGltf D "$aTmpGltf"
XGetOneShape s D
checknbshapes s -face 1 -compound 0
checktrinfo s -tri 15452 -nod 14556

View File

@ -2,3 +2,4 @@
002 shape_write_stl
003 gltf_read
004 obj_read
005 gltf_write

View File

@ -18,18 +18,47 @@ foreach i [list 0 3] {
set aPrefix "g_";
set aColor "CCB11D"
}
set aColShapes {}
for { set m 0 } { $m <= $THE_UPPER } { incr m } {
set aRowShapes {}
for { set r 0 } { $r <= $THE_UPPER } { incr r } {
set aName ${aPrefix}m${m}r${r}
copy s $aName
lappend aRowShapes $aName
ttranslate $aName ${r} ${i} ${m}
set aLab [XAddShape D $aName]
SetName D $aLab $aName
}
set aName ${aPrefix}m${m}
compound {*}$aRowShapes $aName
lappend aColShapes $aName
}
set aName ${aPrefix}spheres
compound {*}$aColShapes $aName
set aLabName "Gray Spheres"
if { $i != 0 } { set aLabName "Golden Spheres" }
set aLabComp [XAddShape D $aName 0]
SetName D $aLabComp $aLabName
for { set m 0 } { $m <= $THE_UPPER } { incr m } {
set aMet [expr 100 * ${m}/$THE_UPPER]
set aName ${aPrefix}m${m}
XAddComponent D $aLabComp $aName
set aLabCompCol [XFindShape D $aName]
SetName D $aLabCompCol "${aPrefix}m${aMet}%"
SetName D {*}[XFindComponent D $aName] "${aPrefix}m${aMet}%"
for { set r 0 } { $r <= $THE_UPPER } { incr r } {
set aRoug [expr 100 * ${r}/$THE_UPPER]
set aName ${aPrefix}m${m}r${r}
XAddComponent D $aLabCompCol $aName
set aLab [XFindComponent D $aName]
SetName D {*}$aLab "${aPrefix}m${aMet}%_r${aRoug}%"
XAddVisMaterial D $aName -baseColor $aColor -metallic ${m}/$THE_UPPER -roughness ${r}/$THE_UPPER
XSetVisMaterial D $aLab $aName
XSetVisMaterial D {*}$aLab $aName
}
}
}
set aLab [XFindShape D s]
SetName D {*}$aLab "Sphere"
XGetAllVisMaterials D
# labels
@ -37,14 +66,21 @@ text2brep tm "Metal" -plane 0 -1 0 0 0 -1 -height 0.5 -pos -0.5 0 6.5 -hal
text2brep tnm "Non-metal" -plane 0 -1 0 0 0 -1 -height 0.5 -pos -0.5 0 -0.5 -halign right -valign top -font monospace
text2brep ts "Smooth" -plane 0 -1 0 1 0 0 -height 0.5 -pos -0.5 0 -0.5 -halign left -valign top -font monospace
text2brep tr "Rough" -plane 0 -1 0 1 0 0 -height 0.5 -pos 6.5 0 -0.5 -halign right -valign top -font monospace
set aLab [XAddShape D tm]
SetName D $aLab "Metal"
set aLab [XAddShape D tnm]
SetName D $aLab "Non-metal"
set aLab [XAddShape D ts]
SetName D $aLab "Smooth"
set aLab [XAddShape D tr]
SetName D $aLab "Rough"
compound tm tnm ts tr labs
set aLab [XAddShape D labs 0]
SetName D $aLab "Labels"
XAddComponent D $aLab tm
XAddComponent D $aLab tnm
XAddComponent D $aLab ts
XAddComponent D $aLab tr
SetName D {*}[XFindComponent D tm] "Metal"
SetName D [XFindShape D tm] "Metal"
SetName D {*}[XFindComponent D tnm] "Non-metal"
SetName D [XFindShape D tnm] "Non-metal"
SetName D {*}[XFindComponent D ts] "Smooth"
SetName D [XFindShape D ts] "Smooth"
SetName D {*}[XFindComponent D tr] "Rough"
SetName D [XFindShape D tr] "Rough"
vclear
vinit View1 -width 768 -height 768