123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256 |
- /*
- Open Asset Import Library (assimp)
- ----------------------------------------------------------------------
- Copyright (c) 2006-2025, assimp team
- 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.
- ----------------------------------------------------------------------
- */
- #ifndef ASSIMP_BUILD_NO_3MF_IMPORTER
- #include "D3MFOpcPackage.h"
- #include <assimp/Exceptional.h>
- #include <assimp/XmlParser.h>
- #include <assimp/ZipArchiveIOSystem.h>
- #include <assimp/ai_assert.h>
- #include <assimp/DefaultLogger.hpp>
- #include <assimp/IOStream.hpp>
- #include <assimp/IOSystem.hpp>
- #include <assimp/texture.h>
- #include "3MFXmlTags.h"
- #include <algorithm>
- #include <cassert>
- #include <cstdlib>
- #include <map>
- #include <vector>
- namespace Assimp {
- namespace D3MF {
- // ------------------------------------------------------------------------------------------------
- using OpcPackageRelationshipPtr = std::shared_ptr<OpcPackageRelationship>;
- class OpcPackageRelationshipReader {
- public:
- OpcPackageRelationshipReader(XmlParser &parser) :
- mRelations() {
- XmlNode root = parser.getRootNode();
- ParseRootNode(root);
- }
- void ParseRootNode(XmlNode &node) {
- ParseAttributes(node);
- for (XmlNode currentNode = node.first_child(); currentNode; currentNode = currentNode.next_sibling()) {
- std::string name = currentNode.name();
- if (name == "Relationships") {
- ParseRelationsNode(currentNode);
- }
- }
- }
- void ParseAttributes(XmlNode & /*node*/) {
- // empty
- }
- bool validateRels(OpcPackageRelationshipPtr &relPtr) {
- if (relPtr->id.empty() || relPtr->type.empty() || relPtr->target.empty()) {
- return false;
- }
- return true;
- }
- void ParseRelationsNode(XmlNode &node) {
- if (node.empty()) {
- return;
- }
- for (XmlNode currentNode = node.first_child(); currentNode; currentNode = currentNode.next_sibling()) {
- const std::string name = currentNode.name();
- if (name == "Relationship") {
- OpcPackageRelationshipPtr relPtr(new OpcPackageRelationship());
- relPtr->id = currentNode.attribute(XmlTag::RELS_ATTRIB_ID).as_string();
- relPtr->type = currentNode.attribute(XmlTag::RELS_ATTRIB_TYPE).as_string();
- relPtr->target = currentNode.attribute(XmlTag::RELS_ATTRIB_TARGET).as_string();
- if (validateRels(relPtr)) {
- mRelations.push_back(relPtr);
- }
- }
- }
- }
- std::vector<OpcPackageRelationshipPtr> mRelations;
- };
- static bool IsEmbeddedTexture( const std::string &filename ) {
- const std::string extension = BaseImporter::GetExtension(filename);
- if (extension == "jpg" || extension == "png" || extension == "jpeg") {
- std::string::size_type pos = filename.find("thumbnail");
- if (pos != std::string::npos) {
- return false;
- }
- return true;
- }
- return false;
- }
- // ------------------------------------------------------------------------------------------------
- D3MFOpcPackage::D3MFOpcPackage(IOSystem *pIOHandler, const std::string &rFile) :
- mRootStream(nullptr),
- mZipArchive() {
- mZipArchive = new ZipArchiveIOSystem(pIOHandler, rFile);
- if (!mZipArchive->isOpen()) {
- throw DeadlyImportError("Failed to open file ", rFile, ".");
- }
- std::vector<std::string> fileList;
- mZipArchive->getFileList(fileList);
- for (auto &file : fileList) {
- if (file == D3MF::XmlTag::ROOT_RELATIONSHIPS_ARCHIVE) {
- if (!mZipArchive->Exists(file.c_str())) {
- continue;
- }
- IOStream *fileStream = mZipArchive->Open(file.c_str());
- if (nullptr == fileStream) {
- ASSIMP_LOG_ERROR("Filestream is nullptr.");
- continue;
- }
- std::string rootFile = ReadPackageRootRelationship(fileStream);
- if (!rootFile.empty() && rootFile[0] == '/') {
- rootFile = rootFile.substr(1);
- if (rootFile[0] == '/') {
- // deal with zip-bug
- rootFile = rootFile.substr(1);
- }
- }
- ASSIMP_LOG_VERBOSE_DEBUG(rootFile);
- mZipArchive->Close(fileStream);
- mRootStream = mZipArchive->Open(rootFile.c_str());
- ai_assert(mRootStream != nullptr);
- if (nullptr == mRootStream) {
- throw DeadlyImportError("Cannot open root-file in archive : " + rootFile);
- }
- } else if (file == D3MF::XmlTag::CONTENT_TYPES_ARCHIVE) {
- ASSIMP_LOG_WARN("Ignored file of unsupported type CONTENT_TYPES_ARCHIVES", file);
- } else if (IsEmbeddedTexture(file)) {
- IOStream *fileStream = mZipArchive->Open(file.c_str());
- LoadEmbeddedTextures(fileStream, file);
- mZipArchive->Close(fileStream);
- } else {
- ASSIMP_LOG_WARN("Ignored file of unknown type: ", file);
- }
- }
- }
- D3MFOpcPackage::~D3MFOpcPackage() {
- mZipArchive->Close(mRootStream);
- delete mZipArchive;
- }
- IOStream *D3MFOpcPackage::RootStream() const {
- return mRootStream;
- }
- const std::vector<aiTexture *> &D3MFOpcPackage::GetEmbeddedTextures() const {
- return mEmbeddedTextures;
- }
- static const char *const ModelRef = "3D/3dmodel.model";
- bool D3MFOpcPackage::validate() {
- if (nullptr == mRootStream || nullptr == mZipArchive) {
- return false;
- }
- return mZipArchive->Exists(ModelRef);
- }
- std::string D3MFOpcPackage::ReadPackageRootRelationship(IOStream *stream) {
- XmlParser xmlParser;
- if (!xmlParser.parse(stream)) {
- return std::string();
- }
- OpcPackageRelationshipReader reader(xmlParser);
- auto itr = std::find_if(reader.mRelations.begin(), reader.mRelations.end(), [](const OpcPackageRelationshipPtr &rel) {
- return rel->type == XmlTag::PACKAGE_START_PART_RELATIONSHIP_TYPE;
- });
- if (itr == reader.mRelations.end()) {
- throw DeadlyImportError("Cannot find ", XmlTag::PACKAGE_START_PART_RELATIONSHIP_TYPE);
- }
- return (*itr)->target;
- }
- void D3MFOpcPackage::LoadEmbeddedTextures(IOStream *fileStream, const std::string &filename) {
- if (nullptr == fileStream) {
- return;
- }
- const size_t size = fileStream->FileSize();
- if (0 == size) {
- return;
- }
- unsigned char *data = new unsigned char[size];
- fileStream->Read(data, 1, size);
- aiTexture *texture = new aiTexture;
- std::string embName = "*" + filename;
- texture->mFilename.Set(embName.c_str());
- texture->mWidth = static_cast<unsigned int>(size);
- texture->mHeight = 0;
- texture->achFormatHint[0] = 'p';
- texture->achFormatHint[1] = 'n';
- texture->achFormatHint[2] = 'g';
- texture->achFormatHint[3] = '\0';
- texture->pcData = (aiTexel*) data;
- mEmbeddedTextures.emplace_back(texture);
- }
- } // Namespace D3MF
- } // Namespace Assimp
- #endif //ASSIMP_BUILD_NO_3MF_IMPORTER
|