123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204 |
- /*
- Open Asset Import Library (assimp)
- ----------------------------------------------------------------------
- Copyright (c) 2006-2020, 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/ZipArchiveIOSystem.h>
- #include <assimp/ai_assert.h>
- #include <assimp/DefaultLogger.hpp>
- #include <assimp/IOStream.hpp>
- #include <assimp/IOSystem.hpp>
- #include "3MFXmlTags.h"
- #include <algorithm>
- #include <cassert>
- #include <cstdlib>
- #include <map>
- #include <memory>
- #include <vector>
- namespace Assimp {
- namespace D3MF {
- // ------------------------------------------------------------------------------------------------
- typedef std::shared_ptr<OpcPackageRelationship> OpcPackageRelationshipPtr;
- class OpcPackageRelationshipReader {
- public:
- OpcPackageRelationshipReader(XmlReader *xmlReader) {
- while (xmlReader->read()) {
- if (xmlReader->getNodeType() == irr::io::EXN_ELEMENT &&
- xmlReader->getNodeName() == XmlTag::RELS_RELATIONSHIP_CONTAINER) {
- ParseRootNode(xmlReader);
- }
- }
- }
- void ParseRootNode(XmlReader *xmlReader) {
- ParseAttributes(xmlReader);
- while (xmlReader->read()) {
- if (xmlReader->getNodeType() == irr::io::EXN_ELEMENT &&
- xmlReader->getNodeName() == XmlTag::RELS_RELATIONSHIP_NODE) {
- ParseChildNode(xmlReader);
- }
- }
- }
- void ParseAttributes(XmlReader *) {
- // empty
- }
- bool validateRels(OpcPackageRelationshipPtr &relPtr) {
- if (relPtr->id.empty() || relPtr->type.empty() || relPtr->target.empty()) {
- return false;
- }
- return true;
- }
- void ParseChildNode(XmlReader *xmlReader) {
- OpcPackageRelationshipPtr relPtr(new OpcPackageRelationship());
- relPtr->id = xmlReader->getAttributeValueSafe(XmlTag::RELS_ATTRIB_ID.c_str());
- relPtr->type = xmlReader->getAttributeValueSafe(XmlTag::RELS_ATTRIB_TYPE.c_str());
- relPtr->target = xmlReader->getAttributeValueSafe(XmlTag::RELS_ATTRIB_TARGET.c_str());
- if (validateRels(relPtr)) {
- m_relationShips.push_back(relPtr);
- }
- }
- std::vector<OpcPackageRelationshipPtr> m_relationShips;
- };
- // ------------------------------------------------------------------------------------------------
- D3MFOpcPackage::D3MFOpcPackage(IOSystem *pIOHandler, const std::string &rFile) :
- mRootStream(nullptr), mZipArchive() {
- mZipArchive.reset(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) {
- ai_assert(fileStream != nullptr);
- continue;
- }
- std::string rootFile = ReadPackageRootRelationship(fileStream);
- if (rootFile.size() > 0 && 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 DeadlyExportError("Cannot open root-file in archive : " + rootFile);
- }
- } else if (file == D3MF::XmlTag::CONTENT_TYPES_ARCHIVE) {
- ASSIMP_LOG_WARN_F("Ignored file of unsupported type CONTENT_TYPES_ARCHIVES", file);
- } else {
- ASSIMP_LOG_WARN_F("Ignored file of unknown type: ", file);
- }
- }
- }
- D3MFOpcPackage::~D3MFOpcPackage() {
- mZipArchive->Close(mRootStream);
- }
- IOStream *D3MFOpcPackage::RootStream() const {
- return mRootStream;
- }
- static const std::string ModelRef = "3D/3dmodel.model";
- bool D3MFOpcPackage::validate() {
- if (nullptr == mRootStream || nullptr == mZipArchive) {
- return false;
- }
- return mZipArchive->Exists(ModelRef.c_str());
- }
- std::string D3MFOpcPackage::ReadPackageRootRelationship(IOStream *stream) {
- std::unique_ptr<CIrrXML_IOStreamReader> xmlStream(new CIrrXML_IOStreamReader(stream));
- std::unique_ptr<XmlReader> xml(irr::io::createIrrXMLReader(xmlStream.get()));
- OpcPackageRelationshipReader reader(xml.get());
- auto itr = std::find_if(reader.m_relationShips.begin(), reader.m_relationShips.end(), [](const OpcPackageRelationshipPtr &rel) {
- return rel->type == XmlTag::PACKAGE_START_PART_RELATIONSHIP_TYPE;
- });
- if (itr == reader.m_relationShips.end()) {
- throw DeadlyImportError("Cannot find ", XmlTag::PACKAGE_START_PART_RELATIONSHIP_TYPE);
- }
- return (*itr)->target;
- }
- } // Namespace D3MF
- } // Namespace Assimp
- #endif //ASSIMP_BUILD_NO_3MF_IMPORTER
|