1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-04 18:06:22 +03:00

0033474: Data Exchange - Implement stream reading into RWMesh interface

Stream usage as parameter
This commit is contained in:
ichesnok 2023-09-12 11:58:55 +01:00 committed by dpasukhi
parent bd651bbbd9
commit 0435edfe54
11 changed files with 149 additions and 88 deletions

View File

@ -182,16 +182,15 @@ RWGltf_CafReader::RWGltf_CafReader()
// Function : performMesh // Function : performMesh
// Purpose : // Purpose :
//================================================================ //================================================================
Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& theFile, Standard_Boolean RWGltf_CafReader::performMesh (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) const Standard_Boolean theToProbe)
{ {
Message_ProgressScope aPSentry (theProgress, "Reading glTF", 2); Message_ProgressScope aPSentry(theProgress, "Reading glTF", 2);
aPSentry.Show(); aPSentry.Show();
const Handle(OSD_FileSystem)& aFileSystem = OSD_FileSystem::DefaultFileSystem(); if (!theStream.good())
std::shared_ptr<std::istream> aFile = aFileSystem->OpenIStream (theFile, std::ios::in | std::ios::binary);
if (aFile.get() == NULL || !aFile->good())
{ {
Message::SendFail (TCollection_AsciiString ("File '") + theFile + "' is not found"); Message::SendFail (TCollection_AsciiString ("File '") + theFile + "' is not found");
return false; return false;
@ -199,7 +198,7 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t
bool isBinaryFile = false; bool isBinaryFile = false;
char aGlbHeader[12] = {}; char aGlbHeader[12] = {};
aFile->read (aGlbHeader, sizeof (aGlbHeader)); theStream.read (aGlbHeader, sizeof (aGlbHeader));
int64_t aBinBodyOffset = 0; int64_t aBinBodyOffset = 0;
int64_t aBinBodyLen = 0; int64_t aBinBodyLen = 0;
int64_t aJsonBodyOffset = 0; int64_t aJsonBodyOffset = 0;
@ -218,7 +217,7 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t
} }
char aHeader1[8] = {}; char aHeader1[8] = {};
aFile->read (aHeader1, sizeof (aHeader1)); theStream.read (aHeader1, sizeof (aHeader1));
const uint32_t* aSceneLen = (const uint32_t* )(aHeader1 + 0); const uint32_t* aSceneLen = (const uint32_t* )(aHeader1 + 0);
const uint32_t* aSceneFormat = (const uint32_t* )(aHeader1 + 4); const uint32_t* aSceneFormat = (const uint32_t* )(aHeader1 + 4);
@ -240,16 +239,16 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t
Message::SendWarning (TCollection_AsciiString ("File '") + theFile + "' is written using unknown version " + int(*aVer)); Message::SendWarning (TCollection_AsciiString ("File '") + theFile + "' is written using unknown version " + int(*aVer));
} }
for (int aChunkIter = 0; !aFile->eof() && aChunkIter < 2; ++aChunkIter) for (int aChunkIter = 0; !theStream.eof() && aChunkIter < 2; ++aChunkIter)
{ {
char aChunkHeader2[8] = {}; char aChunkHeader2[8] = {};
if (int64_t (aFile->tellg()) + int64_t (sizeof (aChunkHeader2)) > int64_t (*aLen)) if (int64_t (theStream.tellg()) + int64_t (sizeof (aChunkHeader2)) > int64_t (*aLen))
{ {
break; break;
} }
aFile->read (aChunkHeader2, sizeof (aChunkHeader2)); theStream.read (aChunkHeader2, sizeof (aChunkHeader2));
if (!aFile->good()) if (!theStream.good())
{ {
Message::SendFail (TCollection_AsciiString ("File '") + theFile + "' is written using unsupported format"); Message::SendFail (TCollection_AsciiString ("File '") + theFile + "' is written using unsupported format");
return false; return false;
@ -259,26 +258,26 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t
const uint32_t* aChunkType = (const uint32_t* )(aChunkHeader2 + 4); const uint32_t* aChunkType = (const uint32_t* )(aChunkHeader2 + 4);
if (*aChunkType == 0x4E4F534A) if (*aChunkType == 0x4E4F534A)
{ {
aJsonBodyOffset = int64_t (aFile->tellg()); aJsonBodyOffset = int64_t (theStream.tellg());
aJsonBodyLen = int64_t (*aChunkLen); aJsonBodyLen = int64_t (*aChunkLen);
} }
else if (*aChunkType == 0x004E4942) else if (*aChunkType == 0x004E4942)
{ {
aBinBodyOffset = int64_t (aFile->tellg()); aBinBodyOffset = int64_t (theStream.tellg());
aBinBodyLen = int64_t (*aChunkLen); aBinBodyLen = int64_t (*aChunkLen);
} }
if (*aChunkLen != 0) if (*aChunkLen != 0)
{ {
aFile->seekg (*aChunkLen, std::ios_base::cur); theStream.seekg (*aChunkLen, std::ios_base::cur);
} }
} }
aFile->seekg ((std::streamoff )aJsonBodyOffset, std::ios_base::beg); theStream.seekg ((std::streamoff )aJsonBodyOffset, std::ios_base::beg);
} }
} }
else else
{ {
aFile->seekg (0, std::ios_base::beg); theStream.seekg (0, std::ios_base::beg);
} }
TCollection_AsciiString anErrPrefix = TCollection_AsciiString ("File '") + theFile + "' defines invalid glTF!\n"; TCollection_AsciiString anErrPrefix = TCollection_AsciiString ("File '") + theFile + "' defines invalid glTF!\n";
@ -303,10 +302,10 @@ Standard_Boolean RWGltf_CafReader::performMesh (const TCollection_AsciiString& t
#ifdef HAVE_RAPIDJSON #ifdef HAVE_RAPIDJSON
rapidjson::ParseResult aRes; rapidjson::ParseResult aRes;
rapidjson::IStreamWrapper aFileStream (*aFile); rapidjson::IStreamWrapper aFileStream (theStream);
if (isBinaryFile) if (isBinaryFile)
{ {
aRes = aDoc.ParseStream<rapidjson::kParseStopWhenDoneFlag, rapidjson::UTF8<>, rapidjson::IStreamWrapper> (aFileStream); aRes = aDoc.ParseStream<rapidjson::kParseStopWhenDoneFlag, rapidjson::UTF8<>, rapidjson::IStreamWrapper>(aFileStream);
} }
else else
{ {

View File

@ -83,7 +83,8 @@ public:
protected: protected:
//! Read the mesh from specified file. //! Read the mesh from specified file.
Standard_EXPORT virtual Standard_Boolean performMesh (const TCollection_AsciiString& theFile, Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) Standard_OVERRIDE; const Standard_Boolean theToProbe) Standard_OVERRIDE;

View File

@ -98,6 +98,20 @@ TopoDS_Shape RWMesh_CafReader::SingleShape() const
Standard_Boolean RWMesh_CafReader::perform (const TCollection_AsciiString& theFile, Standard_Boolean RWMesh_CafReader::perform (const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) const Standard_Boolean theToProbe)
{
std::ifstream aStream;
OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary);
return perform(aStream, theFile, theProgress, theToProbe);
}
// =======================================================================
// function : perform
// purpose :
// =======================================================================
Standard_Boolean RWMesh_CafReader::perform (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe)
{ {
Standard_Integer aNewRootsLower = 1; Standard_Integer aNewRootsLower = 1;
if (!myXdeDoc.IsNull()) if (!myXdeDoc.IsNull())
@ -109,7 +123,7 @@ Standard_Boolean RWMesh_CafReader::perform (const TCollection_AsciiString& theFi
OSD_Timer aLoadingTimer; OSD_Timer aLoadingTimer;
aLoadingTimer.Start(); aLoadingTimer.Start();
const Standard_Boolean isDone = performMesh (theFile, theProgress, theToProbe); const Standard_Boolean isDone = performMesh (theStream, theFile, theProgress, theToProbe);
if (theToProbe || theProgress.UserBreak()) if (theToProbe || theProgress.UserBreak())
{ {
return isDone; return isDone;

View File

@ -16,6 +16,7 @@
#define _RWMesh_CafReader_HeaderFile #define _RWMesh_CafReader_HeaderFile
#include <Message_ProgressRange.hxx> #include <Message_ProgressRange.hxx>
#include <OSD_OpenFile.hxx>
#include <RWMesh_CoordinateSystemConverter.hxx> #include <RWMesh_CoordinateSystemConverter.hxx>
#include <RWMesh_NodeAttributes.hxx> #include <RWMesh_NodeAttributes.hxx>
#include <TColStd_IndexedDataMapOfStringString.hxx> #include <TColStd_IndexedDataMapOfStringString.hxx>
@ -148,12 +149,22 @@ public:
public: public:
//! Read the data from specified file. //! Open stream and pass it to Perform method.
//! The Document instance should be set beforehand. //! The Document instance should be set beforehand.
bool Perform (const TCollection_AsciiString& theFile, bool Perform (const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress) const Message_ProgressRange& theProgress)
{ {
return perform (theFile, theProgress, Standard_False); std::ifstream aStream;
OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary);
return Perform(aStream, theProgress, theFile);
}
//! Read the data from specified file.
bool Perform (std::istream& theStream,
const Message_ProgressRange& theProgress,
const TCollection_AsciiString& theFile = "")
{
return perform(theStream, theFile, theProgress, Standard_False);
} }
//! Return extended status flags. //! Return extended status flags.
@ -171,19 +182,28 @@ public:
//! Return metadata map. //! Return metadata map.
const TColStd_IndexedDataMapOfStringString& Metadata() const { return myMetadata; } const TColStd_IndexedDataMapOfStringString& Metadata() const { return myMetadata; }
//! Read the header data from specified file without reading entire model. //! Open stream and pass it to ProbeHeader method.
//! The main purpose is collecting metadata and external references - for copying model into a new location, for example.
//! Can be NOT implemented (unsupported by format / reader).
Standard_Boolean ProbeHeader (const TCollection_AsciiString& theFile, Standard_Boolean ProbeHeader (const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress = Message_ProgressRange()) const Message_ProgressRange& theProgress = Message_ProgressRange())
{ {
return perform (theFile, theProgress, Standard_True); std::ifstream aStream;
OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary);
return ProbeHeader (aStream, theFile, theProgress);
}
//! Read the header data from specified file without reading entire model.
//! The main purpose is collecting metadata and external references - for copying model into a new location, for example.
//! Can be NOT implemented (unsupported by format / reader).
Standard_Boolean ProbeHeader (std::istream& theStream,
const TCollection_AsciiString& theFile = "",
const Message_ProgressRange& theProgress = Message_ProgressRange())
{
return perform(theStream, theFile, theProgress, Standard_True);
} }
protected: protected:
//! Read the data from specified file. //! Open stream and pass it to Perform method.
//! Default implementation calls performMesh() and fills XDE document from collected shapes.
//! @param theFile file to read //! @param theFile file to read
//! @param optional progress indicator //! @param optional progress indicator
//! @param theToProbe flag indicating that mesh data should be skipped and only basing information to be read //! @param theToProbe flag indicating that mesh data should be skipped and only basing information to be read
@ -191,8 +211,30 @@ protected:
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe); const Standard_Boolean theToProbe);
//! Read the mesh from specified file - interface to be implemented by sub-classes. //! Read the data from specified file.
//! Default implementation calls performMesh() and fills XDE document from collected shapes.
//! @param theStream input stream
//! @param theFile path of additional files
//! @param optional progress indicator
//! @param theToProbe flag indicating that mesh data should be skipped and only basing information to be read
Standard_EXPORT virtual Standard_Boolean perform (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe);
//! Read the mesh from specified file
Standard_EXPORT virtual Standard_Boolean performMesh (const TCollection_AsciiString& theFile, Standard_EXPORT virtual Standard_Boolean performMesh (const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe)
{
std::ifstream aStream;
OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary);
return performMesh(aStream, theFile, theProgress, theToProbe);
}
//! Read the mesh from specified file - interface to be implemented by sub-classes.
Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) = 0; const Standard_Boolean theToProbe) = 0;

View File

@ -94,7 +94,8 @@ Handle(RWObj_TriangulationReader) RWObj_CafReader::createReaderContext()
// Function : performMesh // Function : performMesh
// Purpose : // Purpose :
//================================================================ //================================================================
Standard_Boolean RWObj_CafReader::performMesh (const TCollection_AsciiString& theFile, Standard_Boolean RWObj_CafReader::performMesh (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) const Standard_Boolean theToProbe)
{ {
@ -107,11 +108,11 @@ Standard_Boolean RWObj_CafReader::performMesh (const TCollection_AsciiString& th
Standard_Boolean isDone = Standard_False; Standard_Boolean isDone = Standard_False;
if (theToProbe) if (theToProbe)
{ {
isDone = aCtx->Probe (theFile.ToCString(), theProgress); isDone = aCtx->Probe (theStream, theFile, theProgress);
} }
else else
{ {
isDone = aCtx->Read (theFile.ToCString(), theProgress); isDone = aCtx->Read (theStream, theFile, theProgress);
} }
if (!aCtx->FileComments().IsEmpty()) if (!aCtx->FileComments().IsEmpty())
{ {

View File

@ -35,7 +35,8 @@ public:
protected: protected:
//! Read the mesh from specified file. //! Read the mesh from specified file.
Standard_EXPORT virtual Standard_Boolean performMesh (const TCollection_AsciiString& theFile, Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) Standard_OVERRIDE; const Standard_Boolean theToProbe) Standard_OVERRIDE;

View File

@ -47,35 +47,6 @@ namespace
// The length of buffer to read (in bytes) // The length of buffer to read (in bytes)
static const size_t THE_BUFFER_SIZE = 4 * 1024; static const size_t THE_BUFFER_SIZE = 4 * 1024;
//! Simple wrapper.
struct RWObj_ReaderFile
{
FILE* File;
int64_t FileLen;
//! Constructor opening the file.
RWObj_ReaderFile (const TCollection_AsciiString& theFile)
: File (OSD_OpenFile (theFile.ToCString(), "rb")),
FileLen (0)
{
if (this->File != NULL)
{
// determine length of file
::fseek64 (this->File, 0, SEEK_END);
FileLen = ::ftell64 (this->File);
::fseek64 (this->File, 0, SEEK_SET);
}
}
//! Destructor closing the file.
~RWObj_ReaderFile()
{
if (File != NULL)
{
::fclose (File);
}
}
};
//! Return TRUE if given polygon has clockwise node order. //! Return TRUE if given polygon has clockwise node order.
static bool isClockwisePolygon (const Handle(BRepMesh_DataStructureOfDelaun)& theMesh, static bool isClockwisePolygon (const Handle(BRepMesh_DataStructureOfDelaun)& theMesh,
@ -115,7 +86,8 @@ RWObj_Reader::RWObj_Reader()
// Function : read // Function : read
// Purpose : // Purpose :
// ================================================================ // ================================================================
Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile, Standard_Boolean RWObj_Reader::read (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) const Standard_Boolean theToProbe)
{ {
@ -140,15 +112,16 @@ Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile,
myCurrElem.resize (1024, -1); myCurrElem.resize (1024, -1);
Standard_CLocaleSentry aLocaleSentry; Standard_CLocaleSentry aLocaleSentry;
RWObj_ReaderFile aFile (theFile); if (!theStream.good())
if (aFile.File == NULL)
{ {
Message::SendFail (TCollection_AsciiString ("Error: file '") + theFile + "' is not found"); Message::SendFail (TCollection_AsciiString ("Error: file '") + theFile + "' is not found");
return Standard_False; return Standard_False;
} }
// determine length of file // determine length of file
const int64_t aFileLen = aFile.FileLen; theStream.seekg(0, theStream.end);
const int64_t aFileLen = theStream.tellg();
theStream.seekg(0, theStream.beg);
if (aFileLen <= 0L) if (aFileLen <= 0L)
{ {
Message::SendFail (TCollection_AsciiString ("Error: file '") + theFile + "' is empty"); Message::SendFail (TCollection_AsciiString ("Error: file '") + theFile + "' is empty");
@ -158,12 +131,11 @@ Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile,
Standard_ReadLineBuffer aBuffer (THE_BUFFER_SIZE); Standard_ReadLineBuffer aBuffer (THE_BUFFER_SIZE);
aBuffer.SetMultilineMode (true); aBuffer.SetMultilineMode (true);
const Standard_Integer aNbMiBTotal = Standard_Integer(aFileLen / (1024 * 1024)); const Standard_Integer aNbMiBTotal = Standard_Integer(aFileLen / (1024 * 1024));
Standard_Integer aNbMiBPassed = 0; Standard_Integer aNbMiBPassed = 0;
Message_ProgressScope aPS (theProgress, "Reading text OBJ file", aNbMiBTotal); Message_ProgressScope aPS (theProgress, "Reading text OBJ file", aNbMiBTotal);
OSD_Timer aTimer; OSD_Timer aTimer;
aTimer.Start(); aTimer.Start();
bool isStart = true; bool isStart = true;
int64_t aPosition = 0; int64_t aPosition = 0;
size_t aLineLen = 0; size_t aLineLen = 0;
@ -171,7 +143,7 @@ Standard_Boolean RWObj_Reader::read (const TCollection_AsciiString& theFile,
const char* aLine = NULL; const char* aLine = NULL;
for (;;) for (;;)
{ {
aLine = aBuffer.ReadLine (aFile.File, aLineLen, aReadBytes); aLine = aBuffer.ReadLine (theStream, aLineLen, aReadBytes);
if (aLine == NULL) if (aLine == NULL)
{ {
break; break;

View File

@ -23,7 +23,7 @@
#include <NCollection_IndexedMap.hxx> #include <NCollection_IndexedMap.hxx>
#include <NCollection_Vector.hxx> #include <NCollection_Vector.hxx>
#include <NCollection_Shared.hxx> #include <NCollection_Shared.hxx>
#include <OSD_OpenFile.hxx>
#include <RWMesh_CoordinateSystemConverter.hxx> #include <RWMesh_CoordinateSystemConverter.hxx>
#include <RWObj_Material.hxx> #include <RWObj_Material.hxx>
#include <RWObj_SubMesh.hxx> #include <RWObj_SubMesh.hxx>
@ -48,17 +48,27 @@ public:
//! Empty constructor. //! Empty constructor.
Standard_EXPORT RWObj_Reader(); Standard_EXPORT RWObj_Reader();
//! Reads data from OBJ file. //! Open stream and pass it to Read method
//! Unicode paths can be given in UTF-8 encoding. //! Returns true if success, false on error.
//! Returns true if success, false on error or user break.
Standard_Boolean Read (const TCollection_AsciiString& theFile, Standard_Boolean Read (const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress) const Message_ProgressRange& theProgress)
{ {
return read (theFile, theProgress, Standard_False); std::ifstream aStream;
OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary);
return Read(aStream, theFile, theProgress);
} }
//! Probe data from OBJ file (comments, external references) without actually reading mesh data. //! Reads data from OBJ file.
//! Although mesh data will not be collected, the full file content will be parsed, due to OBJ format limitations. //! Unicode paths can be given in UTF-8 encoding.
//! Returns true if success, false on error or user break.
Standard_Boolean Read (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress)
{
return read(theStream, theFile, theProgress, Standard_False);
}
//! Open stream and pass it to Probe method.
//! @param theFile path to the file //! @param theFile path to the file
//! @param theProgress progress indicator //! @param theProgress progress indicator
//! @return TRUE if success, FALSE on error or user break. //! @return TRUE if success, FALSE on error or user break.
@ -66,7 +76,23 @@ public:
Standard_Boolean Probe (const TCollection_AsciiString& theFile, Standard_Boolean Probe (const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress) const Message_ProgressRange& theProgress)
{ {
return read (theFile, theProgress, Standard_True); std::ifstream aStream;
OSD_OpenStream(aStream, theFile, std::ios_base::in | std::ios_base::binary);
return Probe(aStream, theFile, theProgress);
}
//! Probe data from OBJ file (comments, external references) without actually reading mesh data.
//! Although mesh data will not be collected, the full file content will be parsed, due to OBJ format limitations.
//! @param theStream input stream
//! @param theFile path to the file
//! @param theProgress progress indicator
//! @return TRUE if success, FALSE on error or user break.
//! @sa FileComments(), ExternalFiles(), NbProbeNodes(), NbProbeElems().
Standard_Boolean Probe (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress)
{
return read(theStream, theFile, theProgress, Standard_True);
} }
//! Returns file comments (lines starting with # at the beginning of file). //! Returns file comments (lines starting with # at the beginning of file).
@ -107,7 +133,8 @@ protected:
//! Reads data from OBJ file. //! Reads data from OBJ file.
//! Unicode paths can be given in UTF-8 encoding. //! Unicode paths can be given in UTF-8 encoding.
//! Returns true if success, false on error or user break. //! Returns true if success, false on error or user break.
Standard_EXPORT Standard_Boolean read (const TCollection_AsciiString& theFile, Standard_EXPORT Standard_Boolean read (std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe); const Standard_Boolean theToProbe);
@ -355,7 +382,6 @@ protected:
RWObj_SubMesh myActiveSubMesh; //!< active sub-mesh definition RWObj_SubMesh myActiveSubMesh; //!< active sub-mesh definition
std::vector<Standard_Integer> myCurrElem; //!< indices for the current element std::vector<Standard_Integer> myCurrElem; //!< indices for the current element
}; };
#endif // _RWObj_Reader_HeaderFile #endif // _RWObj_Reader_HeaderFile

View File

@ -89,14 +89,13 @@ namespace
// function : performMesh // function : performMesh
// purpose : // purpose :
//======================================================================= //=======================================================================
bool VrmlAPI_CafReader::performMesh(const TCollection_AsciiString& theFile, bool VrmlAPI_CafReader::performMesh(std::istream& theStream,
const TCollection_AsciiString& theFile,
const Message_ProgressRange& theProgress, const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) const Standard_Boolean theToProbe)
{ {
(void)theProgress; (void)theProgress;
Handle(OSD_FileSystem) aFile = OSD_FileSystem::DefaultFileSystem(); if (!theStream.good())
std::shared_ptr<std::istream> aFileStream = aFile->OpenIStream(theFile, std::ios::in | std::ios::binary);
if (aFileStream.get() == nullptr || !aFileStream->good())
{ {
Message::SendFail() << "Error in VrmlAPI_CafReader: file '" << theFile << "' is not found"; Message::SendFail() << "Error in VrmlAPI_CafReader: file '" << theFile << "' is not found";
return false; return false;
@ -115,7 +114,7 @@ bool VrmlAPI_CafReader::performMesh(const TCollection_AsciiString& theFile,
VrmlData_Scene aScene; VrmlData_Scene aScene;
aScene.SetLinearScale(FileLengthUnit()); aScene.SetLinearScale(FileLengthUnit());
aScene.SetVrmlDir(aFolder); aScene.SetVrmlDir(aFolder);
aScene << *aFileStream; aScene << theStream;
VrmlData_DataMapOfShapeAppearance aShapeAppMap; VrmlData_DataMapOfShapeAppearance aShapeAppMap;
TopoDS_Shape aShape = aScene.GetShape(aShapeAppMap); TopoDS_Shape aShape = aScene.GetShape(aShapeAppMap);

View File

@ -24,13 +24,15 @@ class VrmlAPI_CafReader : public RWMesh_CafReader
protected: protected:
//! Read the mesh data from specified file. //! Read the mesh data from specified file.
//! @param theFile file to read //! @param theStream input stream
//! @param theFile path of additional files
//! @param theProgress progress indicator //! @param theProgress progress indicator
//! @param theToProbe flag for probing file without complete reading. Not supported. //! @param theToProbe flag for probing file without complete reading. Not supported.
//! @return false when theToProbe is set to true or reading has completed with error. //! @return false when theToProbe is set to true or reading has completed with error.
Standard_EXPORT virtual Standard_Boolean performMesh(const TCollection_AsciiString& theFile, Standard_EXPORT virtual Standard_Boolean performMesh (std::istream& theStream,
const Message_ProgressRange& theProgress, const TCollection_AsciiString& theFile,
const Standard_Boolean theToProbe) Standard_OVERRIDE; const Message_ProgressRange& theProgress,
const Standard_Boolean theToProbe) Standard_OVERRIDE;
}; };

View File

@ -170,6 +170,10 @@ Standard_OStream& operator << (Standard_OStream& theOutput,
void VrmlData_Scene::SetVrmlDir (const TCollection_ExtendedString& theDir) void VrmlData_Scene::SetVrmlDir (const TCollection_ExtendedString& theDir)
{ {
TCollection_ExtendedString& aDir = myVrmlDir.Append (theDir); TCollection_ExtendedString& aDir = myVrmlDir.Append (theDir);
if (aDir.IsEmpty())
{
return;
}
const Standard_ExtCharacter aTerminator = aDir.Value(aDir.Length()); const Standard_ExtCharacter aTerminator = aDir.Value(aDir.Length());
if (aTerminator != Standard_ExtCharacter('\\') && if (aTerminator != Standard_ExtCharacter('\\') &&
aTerminator != Standard_ExtCharacter('/')) aTerminator != Standard_ExtCharacter('/'))