2017-11-20 21:36:17 +00:00
|
|
|
/*
|
|
|
|
Open Asset Import Library (assimp)
|
|
|
|
----------------------------------------------------------------------
|
|
|
|
|
2018-01-28 18:42:05 +00:00
|
|
|
Copyright (c) 2006-2018, assimp team
|
|
|
|
|
2017-11-20 21:36:17 +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.
|
|
|
|
|
|
|
|
----------------------------------------------------------------------
|
|
|
|
*/
|
2017-11-29 20:02:52 +00:00
|
|
|
#ifndef ASSIMP_BUILD_NO_EXPORT
|
|
|
|
#ifndef ASSIMP_BUILD_NO_3MF_EXPORTER
|
|
|
|
|
2017-11-20 21:36:17 +00:00
|
|
|
#include "D3MFExporter.h"
|
2017-11-24 17:59:37 +00:00
|
|
|
|
2017-11-20 21:36:17 +00:00
|
|
|
#include <assimp/scene.h>
|
|
|
|
#include <assimp/IOSystem.hpp>
|
2017-11-29 15:11:33 +00:00
|
|
|
#include <assimp/IOStream.hpp>
|
2017-11-20 21:36:17 +00:00
|
|
|
#include <assimp/Exporter.hpp>
|
|
|
|
#include <assimp/DefaultLogger.hpp>
|
|
|
|
|
2018-01-06 00:18:33 +00:00
|
|
|
#include <assimp/Exceptional.h>
|
2017-11-20 21:36:17 +00:00
|
|
|
#include "3MFXmlTags.h"
|
2017-11-24 17:59:37 +00:00
|
|
|
#include "D3MFOpcPackage.h"
|
2017-11-20 21:36:17 +00:00
|
|
|
|
2017-11-27 20:48:33 +00:00
|
|
|
#include <contrib/zip/src/zip.h>
|
|
|
|
|
2017-11-20 21:36:17 +00:00
|
|
|
namespace Assimp {
|
|
|
|
|
|
|
|
void ExportScene3MF( const char* pFile, IOSystem* pIOSystem, const aiScene* pScene, const ExportProperties* /*pProperties*/ ) {
|
2017-11-28 09:14:23 +00:00
|
|
|
if ( nullptr == pIOSystem ) {
|
|
|
|
throw DeadlyExportError( "Could not export 3MP archive: " + std::string( pFile ) );
|
|
|
|
}
|
2017-11-27 22:36:32 +00:00
|
|
|
D3MF::D3MFExporter myExporter( pFile, pScene );
|
2017-11-20 21:36:17 +00:00
|
|
|
if ( myExporter.validate() ) {
|
2017-12-06 20:41:48 +00:00
|
|
|
if ( pIOSystem->Exists( pFile ) ) {
|
|
|
|
if ( !pIOSystem->DeleteFile( pFile ) ) {
|
|
|
|
throw DeadlyExportError( "File exists, cannot override : " + std::string( pFile ) );
|
|
|
|
}
|
|
|
|
}
|
2017-11-23 21:47:18 +00:00
|
|
|
bool ok = myExporter.exportArchive(pFile);
|
2017-11-27 20:48:33 +00:00
|
|
|
if ( !ok ) {
|
|
|
|
throw DeadlyExportError( "Could not export 3MP archive: " + std::string( pFile ) );
|
|
|
|
}
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-21 17:41:16 +00:00
|
|
|
namespace D3MF {
|
|
|
|
|
2017-11-27 22:36:32 +00:00
|
|
|
D3MFExporter::D3MFExporter( const char* pFile, const aiScene* pScene )
|
|
|
|
: mArchiveName( pFile )
|
2017-11-27 20:48:33 +00:00
|
|
|
, m_zipArchive( nullptr )
|
2017-11-20 21:36:17 +00:00
|
|
|
, mScene( pScene )
|
2017-12-06 20:41:48 +00:00
|
|
|
, mModelOutput()
|
|
|
|
, mRelOutput()
|
|
|
|
, mContentOutput()
|
2017-11-24 17:59:37 +00:00
|
|
|
, mBuildItems()
|
|
|
|
, mRelations() {
|
2017-11-20 21:36:17 +00:00
|
|
|
// empty
|
|
|
|
}
|
|
|
|
|
|
|
|
D3MFExporter::~D3MFExporter() {
|
2017-11-24 17:59:37 +00:00
|
|
|
for ( size_t i = 0; i < mRelations.size(); ++i ) {
|
|
|
|
delete mRelations[ i ];
|
|
|
|
}
|
|
|
|
mRelations.clear();
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool D3MFExporter::validate() {
|
2017-11-24 17:59:37 +00:00
|
|
|
if ( mArchiveName.empty() ) {
|
2017-11-20 21:36:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( nullptr == mScene ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-11-23 21:47:18 +00:00
|
|
|
bool D3MFExporter::exportArchive( const char *file ) {
|
2017-11-21 20:34:25 +00:00
|
|
|
bool ok( true );
|
2017-11-27 20:48:33 +00:00
|
|
|
|
|
|
|
m_zipArchive = zip_open( file, ZIP_DEFAULT_COMPRESSION_LEVEL, 'w' );
|
|
|
|
if ( nullptr == m_zipArchive ) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-12-06 20:41:48 +00:00
|
|
|
ok |= exportContentTypes();
|
2017-11-21 20:34:25 +00:00
|
|
|
ok |= export3DModel();
|
2017-11-27 21:46:57 +00:00
|
|
|
ok |= exportRelations();
|
|
|
|
|
|
|
|
zip_close( m_zipArchive );
|
|
|
|
m_zipArchive = nullptr;
|
2017-11-21 20:34:25 +00:00
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2017-12-06 20:41:48 +00:00
|
|
|
|
|
|
|
bool D3MFExporter::exportContentTypes() {
|
|
|
|
mContentOutput.clear();
|
|
|
|
|
|
|
|
mContentOutput << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>";
|
|
|
|
mContentOutput << std::endl;
|
|
|
|
mContentOutput << "<Types xmlns = \"http://schemas.openxmlformats.org/package/2006/content-types\">";
|
|
|
|
mContentOutput << std::endl;
|
|
|
|
mContentOutput << "<Default Extension = \"rels\" ContentType = \"application/vnd.openxmlformats-package.relationships+xml\" />";
|
|
|
|
mContentOutput << std::endl;
|
|
|
|
mContentOutput << "<Default Extension = \"model\" ContentType = \"application/vnd.ms-package.3dmanufacturing-3dmodel+xml\" />";
|
|
|
|
mContentOutput << std::endl;
|
|
|
|
mContentOutput << "</Types>";
|
|
|
|
mContentOutput << std::endl;
|
|
|
|
exportContentTyp( XmlTag::CONTENT_TYPES_ARCHIVE );
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-11-21 20:34:25 +00:00
|
|
|
bool D3MFExporter::exportRelations() {
|
2017-11-27 21:46:57 +00:00
|
|
|
mRelOutput.clear();
|
2017-11-21 20:34:25 +00:00
|
|
|
|
2017-12-06 20:41:48 +00:00
|
|
|
mRelOutput << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>";
|
|
|
|
mRelOutput << std::endl;
|
|
|
|
mRelOutput << "<Relationships xmlns=\"http://schemas.openxmlformats.org/package/2006/relationships\">";
|
2017-11-23 21:47:18 +00:00
|
|
|
|
2017-11-24 17:59:37 +00:00
|
|
|
for ( size_t i = 0; i < mRelations.size(); ++i ) {
|
2017-12-06 20:41:48 +00:00
|
|
|
mRelOutput << "<Relationship Target=\"/" << mRelations[ i ]->target << "\" ";
|
|
|
|
mRelOutput << "Id=\"" << mRelations[i]->id << "\" ";
|
|
|
|
mRelOutput << "Type=\"" << mRelations[ i ]->type << "\" />";
|
2017-11-27 21:46:57 +00:00
|
|
|
mRelOutput << std::endl;
|
2017-11-24 17:59:37 +00:00
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
mRelOutput << "</Relationships>";
|
|
|
|
mRelOutput << std::endl;
|
2017-11-24 17:59:37 +00:00
|
|
|
|
|
|
|
writeRelInfoToFile( "_rels", ".rels" );
|
2017-11-27 21:46:57 +00:00
|
|
|
mRelOutput.flush();
|
2017-11-23 21:47:18 +00:00
|
|
|
|
2017-11-21 20:34:25 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool D3MFExporter::export3DModel() {
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput.clear();
|
2017-11-21 20:34:25 +00:00
|
|
|
|
2017-11-20 21:36:17 +00:00
|
|
|
writeHeader();
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<" << XmlTag::model << " " << XmlTag::model_unit << "=\"millimeter\""
|
2017-11-20 21:36:17 +00:00
|
|
|
<< "xmlns=\"http://schemas.microsoft.com/3dmanufacturing/core/2015/02\">"
|
2017-11-27 21:46:57 +00:00
|
|
|
<< std::endl;
|
|
|
|
mModelOutput << "<" << XmlTag::resources << ">";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
|
|
|
|
writeObjects();
|
|
|
|
|
2017-11-23 21:47:18 +00:00
|
|
|
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "</" << XmlTag::resources << ">";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
writeBuild();
|
|
|
|
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "</" << XmlTag::model << ">\n";
|
2017-11-20 22:53:53 +00:00
|
|
|
|
2017-11-24 17:59:37 +00:00
|
|
|
OpcPackageRelationship *info = new OpcPackageRelationship;
|
2017-12-06 20:41:48 +00:00
|
|
|
info->id = "rel0";
|
|
|
|
info->target = "/3D/3DModel.model";
|
|
|
|
info->type = XmlTag::PACKAGE_START_PART_RELATIONSHIP_TYPE;
|
2017-11-24 17:59:37 +00:00
|
|
|
mRelations.push_back( info );
|
|
|
|
|
2017-12-06 21:15:34 +00:00
|
|
|
writeModelToArchive( "3D", "3DModel.model" );
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput.flush();
|
2017-11-21 17:41:16 +00:00
|
|
|
|
2017-11-20 22:53:53 +00:00
|
|
|
return true;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void D3MFExporter::writeHeader() {
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<?xml version=\"1.0\" encoding=\"UTF - 8\"?>";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void D3MFExporter::writeObjects() {
|
|
|
|
if ( nullptr == mScene->mRootNode ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
aiNode *root = mScene->mRootNode;
|
|
|
|
for ( unsigned int i = 0; i < root->mNumChildren; ++i ) {
|
|
|
|
aiNode *currentNode( root->mChildren[ i ] );
|
|
|
|
if ( nullptr == currentNode ) {
|
|
|
|
continue;
|
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<" << XmlTag::object << " id=\"" << currentNode->mName.C_Str() << "\" type=\"model\">";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
for ( unsigned int j = 0; j < currentNode->mNumMeshes; ++j ) {
|
|
|
|
aiMesh *currentMesh = mScene->mMeshes[ currentNode->mMeshes[ j ] ];
|
|
|
|
if ( nullptr == currentMesh ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
writeMesh( currentMesh );
|
|
|
|
}
|
|
|
|
mBuildItems.push_back( i );
|
|
|
|
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "</" << XmlTag::object << ">";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void D3MFExporter::writeMesh( aiMesh *mesh ) {
|
|
|
|
if ( nullptr == mesh ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<" << XmlTag::mesh << ">" << std::endl;
|
|
|
|
mModelOutput << "<" << XmlTag::vertices << ">" << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
for ( unsigned int i = 0; i < mesh->mNumVertices; ++i ) {
|
|
|
|
writeVertex( mesh->mVertices[ i ] );
|
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "</" << XmlTag::vertices << ">" << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
|
|
|
|
writeFaces( mesh );
|
2017-12-07 16:50:18 +00:00
|
|
|
|
|
|
|
mModelOutput << "</" << XmlTag::mesh << ">" << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void D3MFExporter::writeVertex( const aiVector3D &pos ) {
|
2017-12-07 16:50:18 +00:00
|
|
|
mModelOutput << "<" << XmlTag::vertex << " x=\"" << pos.x << "\" y=\"" << pos.y << "\" z=\"" << pos.z << "\" />";
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void D3MFExporter::writeFaces( aiMesh *mesh ) {
|
|
|
|
if ( nullptr == mesh ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !mesh->HasFaces() ) {
|
|
|
|
return;
|
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<" << XmlTag::triangles << ">" << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
for ( unsigned int i = 0; i < mesh->mNumFaces; ++i ) {
|
|
|
|
aiFace ¤tFace = mesh->mFaces[ i ];
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<" << XmlTag::triangle << " v1=\"" << currentFace.mIndices[ 0 ] << "\" v2=\""
|
|
|
|
<< currentFace.mIndices[ 1 ] << "\" v3=\"" << currentFace.mIndices[ 2 ] << "\"/>";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "</" << XmlTag::triangles << ">";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void D3MFExporter::writeBuild() {
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<" << XmlTag::build << ">" << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
|
|
|
|
for ( size_t i = 0; i < mBuildItems.size(); ++i ) {
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "<" << XmlTag::item << " objectid=\"" << i + 1 << "\"/>";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
mModelOutput << "</" << XmlTag::build << ">";
|
|
|
|
mModelOutput << std::endl;
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
|
|
|
|
2017-12-06 20:41:48 +00:00
|
|
|
void D3MFExporter::exportContentTyp( const std::string &filename ) {
|
|
|
|
if ( nullptr == m_zipArchive ) {
|
|
|
|
throw DeadlyExportError( "3MF-Export: Zip archive not valid, nullptr." );
|
|
|
|
}
|
|
|
|
const std::string entry = filename;
|
|
|
|
zip_entry_open( m_zipArchive, entry.c_str() );
|
|
|
|
|
|
|
|
const std::string &exportTxt( mContentOutput.str() );
|
|
|
|
zip_entry_write( m_zipArchive, exportTxt.c_str(), exportTxt.size() );
|
|
|
|
|
|
|
|
zip_entry_close( m_zipArchive );
|
|
|
|
}
|
|
|
|
|
2017-11-24 17:59:37 +00:00
|
|
|
void D3MFExporter::writeModelToArchive( const std::string &folder, const std::string &modelName ) {
|
2017-11-28 08:08:16 +00:00
|
|
|
if ( nullptr == m_zipArchive ) {
|
|
|
|
throw DeadlyExportError( "3MF-Export: Zip archive not valid, nullptr." );
|
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
const std::string entry = folder + "/" + modelName;
|
2017-11-27 20:48:33 +00:00
|
|
|
zip_entry_open( m_zipArchive, entry.c_str() );
|
2017-11-24 17:59:37 +00:00
|
|
|
|
2017-11-27 21:46:57 +00:00
|
|
|
const std::string &exportTxt( mModelOutput.str() );
|
2017-11-27 20:48:33 +00:00
|
|
|
zip_entry_write( m_zipArchive, exportTxt.c_str(), exportTxt.size() );
|
2017-11-24 17:59:37 +00:00
|
|
|
|
2017-11-27 20:48:33 +00:00
|
|
|
zip_entry_close( m_zipArchive );
|
2017-11-24 17:59:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void D3MFExporter::writeRelInfoToFile( const std::string &folder, const std::string &relName ) {
|
2017-11-28 08:08:16 +00:00
|
|
|
if ( nullptr == m_zipArchive ) {
|
|
|
|
throw DeadlyExportError( "3MF-Export: Zip archive not valid, nullptr." );
|
|
|
|
}
|
2017-11-27 21:46:57 +00:00
|
|
|
const std::string entry = folder + "/" + relName;
|
2017-11-27 20:48:33 +00:00
|
|
|
zip_entry_open( m_zipArchive, entry.c_str() );
|
2017-11-27 21:46:57 +00:00
|
|
|
|
|
|
|
const std::string &exportTxt( mRelOutput.str() );
|
2017-11-27 20:48:33 +00:00
|
|
|
zip_entry_write( m_zipArchive, exportTxt.c_str(), exportTxt.size() );
|
2017-11-27 21:46:57 +00:00
|
|
|
|
2017-11-27 20:48:33 +00:00
|
|
|
zip_entry_close( m_zipArchive );
|
2017-11-20 21:36:17 +00:00
|
|
|
}
|
2017-11-24 17:59:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
} // Namespace D3MF
|
2017-11-20 21:36:17 +00:00
|
|
|
} // Namespace Assimp
|
2017-11-28 18:05:00 +00:00
|
|
|
|
2017-12-11 15:26:45 +00:00
|
|
|
#endif // ASSIMP_BUILD_NO_3MF_EXPORTER
|
2017-11-29 20:02:52 +00:00
|
|
|
#endif // ASSIMP_BUILD_NO_EXPORT
|