2016-10-04 09:45:08 +00:00
|
|
|
/*
|
|
|
|
Open Asset Import Library (assimp)
|
|
|
|
----------------------------------------------------------------------
|
|
|
|
|
2020-01-20 13:53:12 +00:00
|
|
|
Copyright (c) 2006-2020, assimp team
|
2018-01-28 18:42:05 +00:00
|
|
|
|
2016-10-04 09:45:08 +00:00
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use of this software in source and binary forms,
|
|
|
|
with or without modification, are permitted provided that the
|
|
|
|
following conditions are met:
|
|
|
|
|
|
|
|
* Redistributions of source code must retain the above
|
|
|
|
copyright notice, this list of conditions and the
|
|
|
|
following disclaimer.
|
|
|
|
|
|
|
|
* Redistributions in binary form must reproduce the above
|
|
|
|
copyright notice, this list of conditions and the
|
|
|
|
following disclaimer in the documentation and/or other
|
|
|
|
materials provided with the distribution.
|
|
|
|
|
|
|
|
* Neither the name of the assimp team, nor the names of its
|
|
|
|
contributors may be used to endorse or promote products
|
|
|
|
derived from this software without specific prior
|
|
|
|
written permission of the assimp team.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
----------------------------------------------------------------------
|
|
|
|
*/
|
2016-10-02 00:08:34 +00:00
|
|
|
/// \file X3DImporter.cpp
|
|
|
|
/// \brief X3D-format files importer for Assimp: main algorithm implementation.
|
|
|
|
/// \date 2015-2016
|
|
|
|
/// \author smal.root@gmail.com
|
2016-09-29 15:50:24 +00:00
|
|
|
|
|
|
|
#ifndef ASSIMP_BUILD_NO_X3D_IMPORTER
|
|
|
|
|
|
|
|
#include "X3DImporter.hpp"
|
2018-01-06 00:18:33 +00:00
|
|
|
#include <assimp/StringUtils.h>
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2016-10-02 00:08:34 +00:00
|
|
|
// Header files, Assimp.
|
2017-02-28 02:31:56 +00:00
|
|
|
#include <assimp/DefaultIOSystem.h>
|
2018-01-06 00:18:33 +00:00
|
|
|
#include <assimp/fast_atof.h>
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2016-10-02 00:08:34 +00:00
|
|
|
// Header files, stdlib.
|
2020-07-26 08:17:21 +00:00
|
|
|
#include <iterator>
|
2016-10-02 00:08:34 +00:00
|
|
|
#include <memory>
|
2016-09-29 15:50:24 +00:00
|
|
|
#include <string>
|
|
|
|
|
2016-10-15 19:25:16 +00:00
|
|
|
namespace Assimp {
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2016-10-15 19:25:16 +00:00
|
|
|
/// Constant which holds the importer description
|
2016-09-29 15:50:24 +00:00
|
|
|
const aiImporterDesc X3DImporter::Description = {
|
2020-07-26 08:17:21 +00:00
|
|
|
"Extensible 3D(X3D) Importer",
|
|
|
|
"smalcom",
|
|
|
|
"",
|
|
|
|
"See documentation in source code. Chapter: Limitations.",
|
|
|
|
aiImporterFlags_SupportTextFlavour | aiImporterFlags_SupportBinaryFlavour | aiImporterFlags_LimitedSupport | aiImporterFlags_Experimental,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
"x3d x3db"
|
2016-09-29 15:50:24 +00:00
|
|
|
};
|
|
|
|
|
2019-07-30 05:33:42 +00:00
|
|
|
struct WordIterator {
|
|
|
|
using iterator_category = std::input_iterator_tag;
|
2020-07-26 08:17:21 +00:00
|
|
|
using value_type = const char *;
|
2019-07-30 05:33:42 +00:00
|
|
|
using difference_type = ptrdiff_t;
|
2020-07-26 08:17:21 +00:00
|
|
|
using pointer = value_type *;
|
|
|
|
using reference = value_type &;
|
2019-07-30 05:33:42 +00:00
|
|
|
|
2017-07-12 17:57:10 +00:00
|
|
|
static const char *whitespace;
|
2020-07-26 08:17:21 +00:00
|
|
|
const char *mStart, *mEnd;
|
|
|
|
|
|
|
|
WordIterator(const char *start, const char *end) :
|
|
|
|
mStart(start),
|
|
|
|
mEnd(end) {
|
|
|
|
mStart = start + ::strspn(start, whitespace);
|
|
|
|
if (mStart >= mEnd) {
|
|
|
|
mStart = 0;
|
2017-07-12 17:57:10 +00:00
|
|
|
}
|
|
|
|
}
|
2020-07-26 08:17:21 +00:00
|
|
|
WordIterator() :
|
|
|
|
mStart(0),
|
|
|
|
mEnd(0) {}
|
|
|
|
WordIterator(const WordIterator &other) :
|
|
|
|
mStart(other.mStart),
|
|
|
|
mEnd(other.mEnd) {}
|
2017-07-12 18:48:49 +00:00
|
|
|
WordIterator &operator=(const WordIterator &other) {
|
2020-07-26 08:17:21 +00:00
|
|
|
mStart = other.mStart;
|
|
|
|
mEnd = other.mEnd;
|
2017-07-12 18:48:49 +00:00
|
|
|
return *this;
|
|
|
|
}
|
2020-07-26 08:17:21 +00:00
|
|
|
|
|
|
|
bool operator==(const WordIterator &other) const { return mStart == other.mStart; }
|
|
|
|
|
|
|
|
bool operator!=(const WordIterator &other) const { return mStart != other.mStart; }
|
|
|
|
|
2017-07-12 17:57:10 +00:00
|
|
|
WordIterator &operator++() {
|
2020-07-26 08:17:21 +00:00
|
|
|
mStart += strcspn(mStart, whitespace);
|
|
|
|
mStart += strspn(mStart, whitespace);
|
|
|
|
if (mStart >= mEnd) {
|
|
|
|
mStart = 0;
|
2017-07-12 17:57:10 +00:00
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
2017-07-12 18:48:49 +00:00
|
|
|
WordIterator operator++(int) {
|
|
|
|
WordIterator result(*this);
|
|
|
|
++(*this);
|
|
|
|
return result;
|
|
|
|
}
|
2020-07-26 08:17:21 +00:00
|
|
|
const char *operator*() const { return mStart; }
|
2017-07-12 17:57:10 +00:00
|
|
|
};
|
|
|
|
|
2020-08-18 18:44:06 +00:00
|
|
|
const char *WordIterator::whitespace = ", \t\r\n";
|
2017-06-22 13:05:23 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
X3DImporter::X3DImporter() :
|
2020-08-18 14:54:29 +00:00
|
|
|
mNodeElementCur(nullptr),
|
|
|
|
mXmlParser(nullptr) {
|
2017-03-29 19:08:40 +00:00
|
|
|
// empty
|
|
|
|
}
|
|
|
|
|
|
|
|
X3DImporter::~X3DImporter() {
|
|
|
|
// Clear() is accounting if data already is deleted. So, just check again if all data is deleted.
|
|
|
|
Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void X3DImporter::Clear() {
|
2020-07-26 08:17:21 +00:00
|
|
|
mNodeElementCur = nullptr;
|
|
|
|
// Delete all elements
|
|
|
|
if (!NodeElement_List.empty()) {
|
|
|
|
for (std::list<X3DNodeElementBase *>::iterator it = NodeElement_List.begin(); it != NodeElement_List.end(); ++it) {
|
2017-03-29 19:08:40 +00:00
|
|
|
delete *it;
|
|
|
|
}
|
2020-07-26 08:17:21 +00:00
|
|
|
NodeElement_List.clear();
|
|
|
|
}
|
2016-09-29 15:50:24 +00:00
|
|
|
}
|
|
|
|
|
2020-08-27 15:05:09 +00:00
|
|
|
void X3DImporter::ParseFile(const std::string &file, IOSystem *pIOHandler) {
|
|
|
|
ai_assert(nullptr != pIOHandler);
|
|
|
|
|
|
|
|
static const std::string mode = "rb";
|
|
|
|
std::unique_ptr<IOStream> fileStream(pIOHandler->Open(file, mode));
|
|
|
|
if (!fileStream.get()) {
|
|
|
|
throw DeadlyImportError("Failed to open file " + file + ".");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-29 15:50:24 +00:00
|
|
|
/*********************************************************************************************************************************************/
|
|
|
|
/************************************************************ Functions: find set ************************************************************/
|
|
|
|
/*********************************************************************************************************************************************/
|
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
bool X3DImporter::CanRead(const std::string &pFile, IOSystem *pIOHandler, bool pCheckSig) const {
|
2016-10-04 09:45:08 +00:00
|
|
|
const std::string extension = GetExtension(pFile);
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2020-08-27 15:05:09 +00:00
|
|
|
if ((extension == "x3d") || (extension == "x3db")) {
|
|
|
|
return true;
|
|
|
|
}
|
2017-06-22 14:29:26 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
if (!extension.length() || pCheckSig) {
|
|
|
|
const char *tokens[] = { "DOCTYPE X3D PUBLIC", "http://www.web3d.org/specifications/x3d" };
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
return SearchFileHeaderForToken(pIOHandler, pFile, tokens, 2);
|
|
|
|
}
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void X3DImporter::GetExtensionList(std::set<std::string> &pExtensionList) {
|
|
|
|
pExtensionList.insert("x3d");
|
|
|
|
pExtensionList.insert("x3db");
|
|
|
|
}
|
|
|
|
|
|
|
|
const aiImporterDesc *X3DImporter::GetInfo() const {
|
|
|
|
return &Description;
|
|
|
|
}
|
|
|
|
|
|
|
|
void X3DImporter::InternReadFile(const std::string &pFile, aiScene *pScene, IOSystem *pIOHandler) {
|
|
|
|
mpIOHandler = pIOHandler;
|
|
|
|
|
|
|
|
Clear(); // delete old graph.
|
|
|
|
std::string::size_type slashPos = pFile.find_last_of("\\/");
|
|
|
|
pIOHandler->PushDirectory(slashPos == std::string::npos ? std::string() : pFile.substr(0, slashPos + 1));
|
|
|
|
ParseFile(pFile, pIOHandler);
|
|
|
|
pIOHandler->PopDirectory();
|
2020-09-02 19:49:40 +00:00
|
|
|
if (NodeElement_List.empty())
|
|
|
|
return;
|
2020-07-26 08:17:21 +00:00
|
|
|
//
|
|
|
|
// Assimp use static arrays of objects for fast speed of rendering. That's good, but need some additional operations/
|
|
|
|
// We know that geometry objects(meshes) are stored in <Shape>, also in <Shape>-><Appearance> materials(in Assimp logical view)
|
|
|
|
// are stored. So at first we need to count how meshes and materials are stored in scene graph.
|
|
|
|
//
|
|
|
|
// at first creating root node for aiScene.
|
|
|
|
pScene->mRootNode = new aiNode;
|
|
|
|
pScene->mRootNode->mParent = nullptr;
|
|
|
|
pScene->mFlags |= AI_SCENE_FLAGS_ALLOW_SHARED;
|
|
|
|
//search for root node element
|
|
|
|
mNodeElementCur = NodeElement_List.front();
|
|
|
|
while (mNodeElementCur->Parent != nullptr)
|
|
|
|
mNodeElementCur = mNodeElementCur->Parent;
|
|
|
|
|
|
|
|
{ // fill aiScene with objects.
|
|
|
|
std::list<aiMesh *> mesh_list;
|
|
|
|
std::list<aiMaterial *> mat_list;
|
|
|
|
std::list<aiLight *> light_list;
|
|
|
|
|
|
|
|
// create nodes tree
|
2020-08-18 14:54:29 +00:00
|
|
|
// Postprocess_BuildNode(*mNodeElementCur, *pScene->mRootNode, mesh_list, mat_list, light_list);
|
2020-07-26 08:17:21 +00:00
|
|
|
// copy needed data to scene
|
|
|
|
if (!mesh_list.empty()) {
|
|
|
|
std::list<aiMesh *>::const_iterator it = mesh_list.begin();
|
|
|
|
|
|
|
|
pScene->mNumMeshes = static_cast<unsigned int>(mesh_list.size());
|
|
|
|
pScene->mMeshes = new aiMesh *[pScene->mNumMeshes];
|
|
|
|
for (size_t i = 0; i < pScene->mNumMeshes; i++)
|
|
|
|
pScene->mMeshes[i] = *it++;
|
|
|
|
}
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
if (!mat_list.empty()) {
|
|
|
|
std::list<aiMaterial *>::const_iterator it = mat_list.begin();
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
pScene->mNumMaterials = static_cast<unsigned int>(mat_list.size());
|
|
|
|
pScene->mMaterials = new aiMaterial *[pScene->mNumMaterials];
|
|
|
|
for (size_t i = 0; i < pScene->mNumMaterials; i++)
|
|
|
|
pScene->mMaterials[i] = *it++;
|
|
|
|
}
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
if (!light_list.empty()) {
|
|
|
|
std::list<aiLight *>::const_iterator it = light_list.begin();
|
2016-09-29 15:50:24 +00:00
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
pScene->mNumLights = static_cast<unsigned int>(light_list.size());
|
|
|
|
pScene->mLights = new aiLight *[pScene->mNumLights];
|
|
|
|
for (size_t i = 0; i < pScene->mNumLights; i++)
|
|
|
|
pScene->mLights[i] = *it++;
|
|
|
|
}
|
2020-08-18 14:54:29 +00:00
|
|
|
}
|
2016-09-29 15:50:24 +00:00
|
|
|
}
|
|
|
|
|
2020-07-26 08:17:21 +00:00
|
|
|
} // namespace Assimp
|
2016-09-29 15:50:24 +00:00
|
|
|
|
|
|
|
#endif // !ASSIMP_BUILD_NO_X3D_IMPORTER
|