2016-10-04 10:45:28 +00:00
/*
2016-10-03 17:24:14 +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-03 17:24:14 +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 .
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*/
/// \file AMFImporter.cpp
2016-09-28 14:49:15 +00:00
/// \brief AMF-format files importer for Assimp: main algorithm implementation.
/// \date 2016
/// \author smal.root@gmail.com
# ifndef ASSIMP_BUILD_NO_AMF_IMPORTER
// Header files, Assimp.
# include "AMFImporter.hpp"
# include "AMFImporter_Macro.hpp"
2017-02-28 02:31:56 +00:00
# include <assimp/DefaultIOSystem.h>
2020-01-27 21:10:42 +00:00
# include <assimp/fast_atof.h>
2016-09-28 14:49:15 +00:00
2016-09-29 20:48:41 +00:00
# include <memory>
2016-09-28 14:49:15 +00:00
2020-01-23 20:16:10 +00:00
namespace Assimp {
2016-09-28 14:49:15 +00:00
/// \var aiImporterDesc AMFImporter::Description
2020-01-27 21:10:42 +00:00
/// Constant which hold importer description
2016-09-28 14:49:15 +00:00
const aiImporterDesc AMFImporter : : Description = {
" Additive manufacturing file format(AMF) Importer " ,
" smalcom " ,
" " ,
" See documentation in source code. Chapter: Limitations. " ,
aiImporterFlags_SupportTextFlavour | aiImporterFlags_LimitedSupport | aiImporterFlags_Experimental ,
0 ,
0 ,
0 ,
0 ,
" amf "
} ;
2020-01-23 20:16:10 +00:00
void AMFImporter : : Clear ( ) {
2016-09-29 15:04:31 +00:00
mNodeElement_Cur = nullptr ;
2016-09-28 14:49:15 +00:00
mUnit . clear ( ) ;
mMaterial_Converted . clear ( ) ;
mTexture_Converted . clear ( ) ;
// Delete all elements
2020-01-27 21:10:42 +00:00
if ( ! mNodeElement_List . empty ( ) ) {
for ( CAMFImporter_NodeElement * ne : mNodeElement_List ) {
delete ne ;
}
2016-09-28 14:49:15 +00:00
mNodeElement_List . clear ( ) ;
}
}
2020-01-23 20:16:10 +00:00
AMFImporter : : ~ AMFImporter ( ) {
2020-02-05 21:51:39 +00:00
if ( mXmlParser ! = nullptr ) {
delete mXmlParser ;
mXmlParser = nullptr ;
2020-01-23 20:16:10 +00:00
}
2016-09-28 14:49:15 +00:00
// Clear() is accounting if data already is deleted. So, just check again if all data is deleted.
Clear ( ) ;
}
/*********************************************************************************************************************************************/
/************************************************************ Functions: find set ************************************************************/
/*********************************************************************************************************************************************/
2020-01-27 21:10:42 +00:00
bool AMFImporter : : Find_NodeElement ( const std : : string & pID , const CAMFImporter_NodeElement : : EType pType , CAMFImporter_NodeElement * * pNodeElement ) const {
for ( CAMFImporter_NodeElement * ne : mNodeElement_List ) {
if ( ( ne - > ID = = pID ) & & ( ne - > Type = = pType ) ) {
2020-02-05 21:51:39 +00:00
if ( pNodeElement ! = nullptr ) {
* pNodeElement = ne ;
}
2016-09-28 14:49:15 +00:00
return true ;
}
2020-01-27 21:10:42 +00:00
} // for(CAMFImporter_NodeElement* ne: mNodeElement_List)
2016-09-28 14:49:15 +00:00
return false ;
}
2020-01-27 21:10:42 +00:00
bool AMFImporter : : Find_ConvertedNode ( const std : : string & id , std : : list < aiNode * > & nodeList , aiNode * * pNode ) const {
aiString node_name ( id . c_str ( ) ) ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
for ( aiNode * node : nodeList ) {
if ( node - > mName = = node_name ) {
2020-01-23 20:16:10 +00:00
if ( pNode ! = nullptr ) {
* pNode = node ;
}
2016-09-28 14:49:15 +00:00
return true ;
}
2020-01-27 21:10:42 +00:00
} // for(aiNode* node: pNodeList)
2016-09-28 14:49:15 +00:00
return false ;
}
2020-01-27 21:10:42 +00:00
bool AMFImporter : : Find_ConvertedMaterial ( const std : : string & id , const SPP_Material * * pConvertedMaterial ) const {
for ( const SPP_Material & mat : mMaterial_Converted ) {
if ( mat . ID = = id ) {
2020-01-23 20:16:10 +00:00
if ( pConvertedMaterial ! = nullptr ) {
* pConvertedMaterial = & mat ;
}
2016-09-28 14:49:15 +00:00
return true ;
}
2020-01-27 21:10:42 +00:00
} // for(const SPP_Material& mat: mMaterial_Converted)
2016-09-28 14:49:15 +00:00
return false ;
}
/*********************************************************************************************************************************************/
/************************************************************ Functions: throw set ***********************************************************/
/*********************************************************************************************************************************************/
2020-01-27 21:10:42 +00:00
void AMFImporter : : Throw_CloseNotFound ( const std : : string & pNode ) {
2016-09-28 14:49:15 +00:00
throw DeadlyImportError ( " Close tag for node < " + pNode + " > not found. Seems file is corrupt. " ) ;
}
2020-01-23 20:25:25 +00:00
void AMFImporter : : Throw_IncorrectAttr ( const std : : string & nodeName , const std : : string & pAttrName ) {
throw DeadlyImportError ( " Node < " + nodeName + " > has incorrect attribute \" " + pAttrName + " \" . " ) ;
2016-09-28 14:49:15 +00:00
}
2020-01-23 20:25:25 +00:00
void AMFImporter : : Throw_IncorrectAttrValue ( const std : : string & nodeName , const std : : string & pAttrName ) {
2020-02-03 20:19:03 +00:00
throw DeadlyImportError ( " Attribute \" " + pAttrName + " \" in node < " + nodeName + " > has incorrect value. " ) ;
2016-09-28 14:49:15 +00:00
}
2020-02-03 20:19:03 +00:00
void AMFImporter : : Throw_MoreThanOnceDefined ( const std : : string & nodeType , const std : : string & nodeName , const std : : string & pDescription ) {
throw DeadlyImportError ( " \" " + nodeType + " \" node can be used only once in " + nodeName + " . Description: " + pDescription ) ;
2016-09-28 14:49:15 +00:00
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : Throw_ID_NotFound ( const std : : string & pID ) const {
2016-09-29 20:38:24 +00:00
throw DeadlyImportError ( " Not found node with name \" " + pID + " \" . " ) ;
2016-09-28 14:49:15 +00:00
}
/*********************************************************************************************************************************************/
/************************************************************* Functions: XML set ************************************************************/
/*********************************************************************************************************************************************/
2020-02-05 21:51:39 +00:00
void AMFImporter : : XML_CheckNode_MustHaveChildren ( XmlNode & node ) {
if ( node . children ( ) . begin ( ) = = node . children ( ) . end ( ) ) {
throw DeadlyImportError ( std : : string ( " Node < " ) + std : : string ( node . name ( ) ) + " > must have children. " ) ;
2020-02-03 20:19:03 +00:00
}
2016-09-28 14:49:15 +00:00
}
2020-02-03 20:19:03 +00:00
/*void AMFImporter::XML_CheckNode_SkipUnsupported(XmlNode *node, const std::string &pParentNodeName) {
2020-01-27 21:10:42 +00:00
static const size_t Uns_Skip_Len = 3 ;
const char * Uns_Skip [ Uns_Skip_Len ] = { " composite " , " edge " , " normal " } ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
static bool skipped_before [ Uns_Skip_Len ] = { false , false , false } ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
std : : string nn ( mReader - > getNodeName ( ) ) ;
bool found = false ;
bool close_found = false ;
size_t sk_idx ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
for ( sk_idx = 0 ; sk_idx < Uns_Skip_Len ; sk_idx + + ) {
if ( nn ! = Uns_Skip [ sk_idx ] ) continue ;
2016-09-28 14:49:15 +00:00
found = true ;
2020-01-27 21:10:42 +00:00
if ( mReader - > isEmptyElement ( ) ) {
2016-09-28 14:49:15 +00:00
close_found = true ;
goto casu_cres ;
}
2020-01-27 21:10:42 +00:00
while ( mReader - > read ( ) ) {
if ( ( mReader - > getNodeType ( ) = = irr : : io : : EXN_ELEMENT_END ) & & ( nn = = mReader - > getNodeName ( ) ) ) {
2016-09-28 14:49:15 +00:00
close_found = true ;
goto casu_cres ;
}
}
2020-01-27 21:10:42 +00:00
} // for(sk_idx = 0; sk_idx < Uns_Skip_Len; sk_idx++)
2016-09-28 14:49:15 +00:00
casu_cres :
2020-01-27 21:10:42 +00:00
if ( ! found ) throw DeadlyImportError ( " Unknown node \" " + nn + " \" in " + pParentNodeName + " . " ) ;
if ( ! close_found ) Throw_CloseNotFound ( nn ) ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
if ( ! skipped_before [ sk_idx ] ) {
2016-09-28 14:49:15 +00:00
skipped_before [ sk_idx ] = true ;
2020-01-27 21:10:42 +00:00
ASSIMP_LOG_WARN_F ( " Skipping node \" " , nn , " \" in " , pParentNodeName , " . " ) ;
2016-09-28 14:49:15 +00:00
}
}
2020-02-03 20:19:03 +00:00
*/
2020-02-05 21:51:39 +00:00
bool AMFImporter : : XML_SearchNode ( const std : : string & nodeName ) {
XmlNode * root = mXmlParser - > getRootNode ( ) ;
if ( nullptr = = root ) {
return false ;
}
2016-09-28 14:49:15 +00:00
2020-02-05 21:51:39 +00:00
find_node_by_name_predicate predicate ( nodeName ) ;
XmlNode node = root - > find_node ( predicate ) ;
if ( node . empty ( ) ) {
return false ;
}
return true ;
2016-09-28 14:49:15 +00:00
}
2020-02-05 21:51:39 +00:00
bool AMFImporter : : XML_ReadNode_GetAttrVal_AsBool ( const int pAttrIdx ) {
std : : string val ( mXmlParser - > getAttributeValue ( pAttrIdx ) ) ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
if ( ( val = = " false " ) | | ( val = = " 0 " ) )
2016-09-28 14:49:15 +00:00
return false ;
2020-01-27 21:10:42 +00:00
else if ( ( val = = " true " ) | | ( val = = " 1 " ) )
2016-09-28 14:49:15 +00:00
return true ;
else
throw DeadlyImportError ( " Bool attribute value can contain \" false \" / \" 0 \" or \" true \" / \" 1 \" not the \" " + val + " \" " ) ;
}
2020-01-27 21:10:42 +00:00
float AMFImporter : : XML_ReadNode_GetAttrVal_AsFloat ( const int pAttrIdx ) {
std : : string val ;
float tvalf ;
2016-09-28 14:49:15 +00:00
2020-02-05 21:51:39 +00:00
ParseHelper_FixTruncatedFloatString ( mXmlParser - > getAttributeValue ( pAttrIdx ) , val ) ;
2016-09-28 14:49:15 +00:00
fast_atoreal_move ( val . c_str ( ) , tvalf , false ) ;
return tvalf ;
}
2020-01-27 21:10:42 +00:00
uint32_t AMFImporter : : XML_ReadNode_GetAttrVal_AsU32 ( const int pAttrIdx ) {
2020-02-05 21:51:39 +00:00
return strtoul10 ( mXmlParser - > getAttributeValue ( pAttrIdx ) ) ;
2016-09-28 14:49:15 +00:00
}
2020-01-27 21:10:42 +00:00
float AMFImporter : : XML_ReadNode_GetVal_AsFloat ( ) {
std : : string val ;
float tvalf ;
2016-09-28 14:49:15 +00:00
2020-02-05 21:51:39 +00:00
if ( ! mXmlParser - > read ( ) ) throw DeadlyImportError ( " XML_ReadNode_GetVal_AsFloat. No data, seems file is corrupt. " ) ;
if ( mXmlParser - > getNodeType ( ) ! = irr : : io : : EXN_TEXT ) throw DeadlyImportError ( " XML_ReadNode_GetVal_AsFloat. Invalid type of XML element, seems file is corrupt. " ) ;
2016-09-28 14:49:15 +00:00
2020-02-05 21:51:39 +00:00
ParseHelper_FixTruncatedFloatString ( mXmlParser - > getNodeData ( ) , val ) ;
2016-09-28 14:49:15 +00:00
fast_atoreal_move ( val . c_str ( ) , tvalf , false ) ;
return tvalf ;
}
2020-01-27 21:10:42 +00:00
uint32_t AMFImporter : : XML_ReadNode_GetVal_AsU32 ( ) {
2020-02-05 21:51:39 +00:00
if ( ! mXmlParser - > read ( ) ) throw DeadlyImportError ( " XML_ReadNode_GetVal_AsU32. No data, seems file is corrupt. " ) ;
if ( mXmlParser - > getNodeType ( ) ! = irr : : io : : EXN_TEXT ) throw DeadlyImportError ( " XML_ReadNode_GetVal_AsU32. Invalid type of XML element, seems file is corrupt. " ) ;
2016-09-28 14:49:15 +00:00
2020-02-05 21:51:39 +00:00
return strtoul10 ( mXmlParser - > getNodeData ( ) ) ;
2016-09-28 14:49:15 +00:00
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : XML_ReadNode_GetVal_AsString ( std : : string & pValue ) {
2020-02-05 21:51:39 +00:00
if ( ! mXmlParser - > read ( ) ) throw DeadlyImportError ( " XML_ReadNode_GetVal_AsString. No data, seems file is corrupt. " ) ;
if ( mXmlParser - > getNodeType ( ) ! = irr : : io : : EXN_TEXT )
2016-09-28 14:49:15 +00:00
throw DeadlyImportError ( " XML_ReadNode_GetVal_AsString. Invalid type of XML element, seems file is corrupt. " ) ;
2020-02-05 21:51:39 +00:00
pValue = mXmlParser - > getNodeData ( ) ;
2016-09-28 14:49:15 +00:00
}
/*********************************************************************************************************************************************/
/************************************************************ Functions: parse set ***********************************************************/
/*********************************************************************************************************************************************/
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseHelper_Node_Enter ( CAMFImporter_NodeElement * pNode ) {
mNodeElement_Cur - > Child . push_back ( pNode ) ; // add new element to current element child list.
mNodeElement_Cur = pNode ; // switch current element to new one.
2016-09-28 14:49:15 +00:00
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseHelper_Node_Exit ( ) {
2016-09-28 14:49:15 +00:00
// check if we can walk up.
2020-01-27 21:10:42 +00:00
if ( mNodeElement_Cur ! = nullptr ) mNodeElement_Cur = mNodeElement_Cur - > Parent ;
2016-09-28 14:49:15 +00:00
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseHelper_FixTruncatedFloatString ( const char * pInStr , std : : string & pOutString ) {
size_t instr_len ;
2016-09-28 14:49:15 +00:00
pOutString . clear ( ) ;
instr_len = strlen ( pInStr ) ;
2020-01-27 21:10:42 +00:00
if ( ! instr_len ) return ;
2016-09-28 14:49:15 +00:00
pOutString . reserve ( instr_len * 3 / 2 ) ;
// check and correct floats in format ".x". Must be "x.y".
2020-01-27 21:10:42 +00:00
if ( pInStr [ 0 ] = = ' . ' ) pOutString . push_back ( ' 0 ' ) ;
2016-09-28 14:49:15 +00:00
pOutString . push_back ( pInStr [ 0 ] ) ;
2020-01-27 21:10:42 +00:00
for ( size_t ci = 1 ; ci < instr_len ; ci + + ) {
if ( ( pInStr [ ci ] = = ' . ' ) & & ( ( pInStr [ ci - 1 ] = = ' ' ) | | ( pInStr [ ci - 1 ] = = ' - ' ) | | ( pInStr [ ci - 1 ] = = ' + ' ) | | ( pInStr [ ci - 1 ] = = ' \t ' ) ) ) {
2016-09-28 14:49:15 +00:00
pOutString . push_back ( ' 0 ' ) ;
pOutString . push_back ( ' . ' ) ;
2020-01-27 21:10:42 +00:00
} else {
2016-09-28 14:49:15 +00:00
pOutString . push_back ( pInStr [ ci ] ) ;
}
}
}
2020-01-27 21:10:42 +00:00
static bool ParseHelper_Decode_Base64_IsBase64 ( const char pChar ) {
2016-09-28 14:49:15 +00:00
return ( isalnum ( pChar ) | | ( pChar = = ' + ' ) | | ( pChar = = ' / ' ) ) ;
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseHelper_Decode_Base64 ( const std : : string & pInputBase64 , std : : vector < uint8_t > & pOutputData ) const {
// With help from
// René Nyffenegger http://www.adp-gmbh.ch/cpp/common/base64.html
const std : : string base64_chars = " ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/ " ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
uint8_t tidx = 0 ;
uint8_t arr4 [ 4 ] , arr3 [ 3 ] ;
2016-09-28 14:49:15 +00:00
// check input data
2020-01-27 21:10:42 +00:00
if ( pInputBase64 . size ( ) % 4 ) throw DeadlyImportError ( " Base64-encoded data must have size multiply of four. " ) ;
2016-09-28 14:49:15 +00:00
// prepare output place
pOutputData . clear ( ) ;
pOutputData . reserve ( pInputBase64 . size ( ) / 4 * 3 ) ;
2020-01-27 21:10:42 +00:00
for ( size_t in_len = pInputBase64 . size ( ) , in_idx = 0 ; ( in_len > 0 ) & & ( pInputBase64 [ in_idx ] ! = ' = ' ) ; in_len - - ) {
if ( ParseHelper_Decode_Base64_IsBase64 ( pInputBase64 [ in_idx ] ) ) {
2016-09-28 14:49:15 +00:00
arr4 [ tidx + + ] = pInputBase64 [ in_idx + + ] ;
2020-01-27 21:10:42 +00:00
if ( tidx = = 4 ) {
for ( tidx = 0 ; tidx < 4 ; tidx + + )
arr4 [ tidx ] = ( uint8_t ) base64_chars . find ( arr4 [ tidx ] ) ;
2016-09-28 14:49:15 +00:00
arr3 [ 0 ] = ( arr4 [ 0 ] < < 2 ) + ( ( arr4 [ 1 ] & 0x30 ) > > 4 ) ;
arr3 [ 1 ] = ( ( arr4 [ 1 ] & 0x0F ) < < 4 ) + ( ( arr4 [ 2 ] & 0x3C ) > > 2 ) ;
arr3 [ 2 ] = ( ( arr4 [ 2 ] & 0x03 ) < < 6 ) + arr4 [ 3 ] ;
2020-01-27 21:10:42 +00:00
for ( tidx = 0 ; tidx < 3 ; tidx + + )
pOutputData . push_back ( arr3 [ tidx ] ) ;
2016-09-28 14:49:15 +00:00
tidx = 0 ;
2020-01-27 21:10:42 +00:00
} // if(tidx == 4)
} // if(ParseHelper_Decode_Base64_IsBase64(pInputBase64[in_idx]))
else {
2016-09-28 14:49:15 +00:00
in_idx + + ;
2020-01-27 21:10:42 +00:00
} // if(ParseHelper_Decode_Base64_IsBase64(pInputBase64[in_idx])) else
2016-09-28 14:49:15 +00:00
}
2020-01-27 21:10:42 +00:00
if ( tidx ) {
for ( uint8_t i = tidx ; i < 4 ; i + + )
arr4 [ i ] = 0 ;
for ( uint8_t i = 0 ; i < 4 ; i + + )
arr4 [ i ] = ( uint8_t ) ( base64_chars . find ( arr4 [ i ] ) ) ;
2016-09-28 14:49:15 +00:00
arr3 [ 0 ] = ( arr4 [ 0 ] < < 2 ) + ( ( arr4 [ 1 ] & 0x30 ) > > 4 ) ;
arr3 [ 1 ] = ( ( arr4 [ 1 ] & 0x0F ) < < 4 ) + ( ( arr4 [ 2 ] & 0x3C ) > > 2 ) ;
arr3 [ 2 ] = ( ( arr4 [ 2 ] & 0x03 ) < < 6 ) + arr4 [ 3 ] ;
2020-01-27 21:10:42 +00:00
for ( uint8_t i = 0 ; i < ( tidx - 1 ) ; i + + )
pOutputData . push_back ( arr3 [ i ] ) ;
2016-09-28 14:49:15 +00:00
}
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseFile ( const std : : string & pFile , IOSystem * pIOHandler ) {
std : : unique_ptr < IOStream > file ( pIOHandler - > Open ( pFile , " rb " ) ) ;
2016-09-28 14:49:15 +00:00
// Check whether we can read from the file
2020-01-27 21:10:42 +00:00
if ( file . get ( ) = = nullptr ) {
throw DeadlyImportError ( " Failed to open AMF file " + pFile + " . " ) ;
}
2016-09-28 14:49:15 +00:00
2020-02-05 21:51:39 +00:00
mXmlParser = new XmlParser ;
XmlNode * root = mXmlParser - > parse ( file . get ( ) ) ;
2020-01-27 21:10:42 +00:00
if ( nullptr = = root ) {
2020-01-23 20:16:10 +00:00
throw DeadlyImportError ( " Failed to create XML reader for file " + pFile + " . " ) ;
}
2016-09-28 14:49:15 +00:00
// start reading
// search for root tag <amf>
2020-01-27 21:10:42 +00:00
if ( ! root - > getNode ( ) - > find_child ( " amf " ) ) {
2016-09-28 14:49:15 +00:00
throw DeadlyImportError ( " Root node \" amf \" not found. " ) ;
2020-01-27 21:10:42 +00:00
}
ParseNode_Root ( root ) ;
2016-09-28 14:49:15 +00:00
2020-02-05 21:51:39 +00:00
delete mXmlParser ;
mXmlParser = nullptr ;
2016-09-28 14:49:15 +00:00
}
// <amf
// unit="" - The units to be used. May be "inch", "millimeter", "meter", "feet", or "micron".
// version="" - Version of file format.
// >
// </amf>
// Root XML element.
// Multi elements - No.
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseNode_Root ( XmlNode * root ) {
std : : string unit , version ;
CAMFImporter_NodeElement * ne ( nullptr ) ;
2016-09-28 14:49:15 +00:00
// Read attributes for node <amf>.
2020-01-27 21:10:42 +00:00
pugi : : xml_node * node ( root - > getNode ( ) ) ;
for ( pugi : : xml_attribute_iterator ait = node - > attributes_begin ( ) ; ait ! = node - > attributes_end ( ) ; + + ait ) {
if ( ait - > name ( ) = = " unit " ) {
unit = ait - > as_string ( ) ;
} else if ( ait - > name ( ) = = " version " ) {
version = ait - > as_string ( ) ;
}
}
/*MACRO_ATTRREAD_LOOPBEG;
MACRO_ATTRREAD_CHECK_RET ( " unit " , unit , mReader - > getAttributeValue ) ;
MACRO_ATTRREAD_CHECK_RET ( " version " , version , mReader - > getAttributeValue ) ;
MACRO_ATTRREAD_LOOPEND_WSKIP ; */
2016-09-28 14:49:15 +00:00
// Check attributes
2020-01-27 21:10:42 +00:00
if ( ! mUnit . empty ( ) ) {
if ( ( mUnit ! = " inch " ) & & ( mUnit ! = " millimeter " ) & & ( mUnit ! = " meter " ) & & ( mUnit ! = " feet " ) & & ( mUnit ! = " micron " ) ) {
Throw_IncorrectAttrValue ( " unit " ) ;
}
2016-09-28 14:49:15 +00:00
}
// create root node element.
2016-09-29 15:04:31 +00:00
ne = new CAMFImporter_NodeElement_Root ( nullptr ) ;
2020-01-27 21:10:42 +00:00
2020-01-27 21:18:48 +00:00
// set first "current" element
mNodeElement_Cur = ne ;
2020-01-27 21:10:42 +00:00
2020-01-27 21:18:48 +00:00
// and assign attributes values
2020-01-27 21:10:42 +00:00
( ( CAMFImporter_NodeElement_Root * ) ne ) - > Unit = unit ;
( ( CAMFImporter_NodeElement_Root * ) ne ) - > Version = version ;
2016-09-28 14:49:15 +00:00
// Check for child nodes
2020-01-27 21:10:42 +00:00
for ( pugi : : xml_node child : node - > children ( ) ) {
if ( child . name ( ) = = " object " ) {
2020-01-27 21:18:48 +00:00
ParseNode_Object ( & child ) ;
2020-01-27 21:10:42 +00:00
} else if ( child . name ( ) = = " material " ) {
ParseNode_Material ( ) ;
} else if ( child . name ( ) = = " texture " ) {
ParseNode_Texture ( ) ;
} else if ( child . name ( ) = = " constellation " ) {
ParseNode_Constellation ( ) ;
} else if ( child . name ( ) = = " metadata " ) {
ParseNode_Metadata ( ) ;
}
}
/*if (!mReader->isEmptyElement()) {
2016-09-28 14:49:15 +00:00
MACRO_NODECHECK_LOOPBEGIN ( " amf " ) ;
2020-01-27 21:10:42 +00:00
if ( XML_CheckNode_NameEqual ( " object " ) ) {
ParseNode_Object ( ) ;
continue ;
}
if ( XML_CheckNode_NameEqual ( " material " ) ) {
ParseNode_Material ( ) ;
continue ;
}
if ( XML_CheckNode_NameEqual ( " texture " ) ) {
ParseNode_Texture ( ) ;
continue ;
}
if ( XML_CheckNode_NameEqual ( " constellation " ) ) {
ParseNode_Constellation ( ) ;
continue ;
}
if ( XML_CheckNode_NameEqual ( " metadata " ) ) {
ParseNode_Metadata ( ) ;
continue ;
}
2016-09-28 14:49:15 +00:00
MACRO_NODECHECK_LOOPEND ( " amf " ) ;
2020-01-27 21:10:42 +00:00
mNodeElement_Cur = ne ; // force restore "current" element
} // if(!mReader->isEmptyElement())*/
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
mNodeElement_List . push_back ( ne ) ; // add to node element list because its a new object in graph.
2016-09-28 14:49:15 +00:00
}
// <constellation
// id="" - The Object ID of the new constellation being defined.
// >
// </constellation>
// A collection of objects or constellations with specific relative locations.
// Multi elements - Yes.
// Parent element - <amf>.
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseNode_Constellation ( ) {
std : : string id ;
CAMFImporter_NodeElement * ne ( nullptr ) ;
2016-09-28 14:49:15 +00:00
// Read attributes for node <constellation>.
MACRO_ATTRREAD_LOOPBEG ;
2020-02-05 21:51:39 +00:00
MACRO_ATTRREAD_CHECK_RET ( " id " , id , mXmlParser - > getAttributeValue ) ;
2016-09-28 14:49:15 +00:00
MACRO_ATTRREAD_LOOPEND ;
// create and if needed - define new grouping object.
ne = new CAMFImporter_NodeElement_Constellation ( mNodeElement_Cur ) ;
2020-01-27 21:10:42 +00:00
CAMFImporter_NodeElement_Constellation & als = * ( ( CAMFImporter_NodeElement_Constellation * ) ne ) ; // alias for convenience
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
if ( ! id . empty ( ) ) als . ID = id ;
2016-09-28 14:49:15 +00:00
// Check for child nodes
2020-02-05 21:51:39 +00:00
if ( ! mXmlParser - > isEmptyElement ( ) ) {
2016-09-28 14:49:15 +00:00
ParseHelper_Node_Enter ( ne ) ;
MACRO_NODECHECK_LOOPBEGIN ( " constellation " ) ;
2020-01-27 21:10:42 +00:00
if ( XML_CheckNode_NameEqual ( " instance " ) ) {
ParseNode_Instance ( ) ;
continue ;
}
if ( XML_CheckNode_NameEqual ( " metadata " ) ) {
ParseNode_Metadata ( ) ;
continue ;
}
2016-09-28 14:49:15 +00:00
MACRO_NODECHECK_LOOPEND ( " constellation " ) ;
ParseHelper_Node_Exit ( ) ;
2020-01-27 21:10:42 +00:00
} // if(!mReader->isEmptyElement())
else {
mNodeElement_Cur - > Child . push_back ( ne ) ; // Add element to child list of current element
} // if(!mReader->isEmptyElement()) else
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
mNodeElement_List . push_back ( ne ) ; // and to node element list because its a new object in graph.
2016-09-28 14:49:15 +00:00
}
// <instance
// objectid="" - The Object ID of the new constellation being defined.
// >
// </instance>
// A collection of objects or constellations with specific relative locations.
// Multi elements - Yes.
// Parent element - <amf>.
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseNode_Instance ( ) {
std : : string objectid ;
CAMFImporter_NodeElement * ne ( nullptr ) ;
2016-09-28 14:49:15 +00:00
// Read attributes for node <constellation>.
MACRO_ATTRREAD_LOOPBEG ;
2020-02-05 21:51:39 +00:00
MACRO_ATTRREAD_CHECK_RET ( " objectid " , objectid , mXmlParser - > getAttributeValue ) ;
2016-09-28 14:49:15 +00:00
MACRO_ATTRREAD_LOOPEND ;
// used object id must be defined, check that.
2020-01-27 21:10:42 +00:00
if ( objectid . empty ( ) ) throw DeadlyImportError ( " \" objectid \" in <instance> must be defined. " ) ;
2016-09-28 14:49:15 +00:00
// create and define new grouping object.
ne = new CAMFImporter_NodeElement_Instance ( mNodeElement_Cur ) ;
2020-01-27 21:10:42 +00:00
CAMFImporter_NodeElement_Instance & als = * ( ( CAMFImporter_NodeElement_Instance * ) ne ) ; // alias for convenience
2016-09-28 14:49:15 +00:00
als . ObjectID = objectid ;
// Check for child nodes
2020-02-05 21:51:39 +00:00
if ( ! mXmlParser - > isEmptyElement ( ) ) {
2016-09-28 14:49:15 +00:00
bool read_flag [ 6 ] = { false , false , false , false , false , false } ;
als . Delta . Set ( 0 , 0 , 0 ) ;
als . Rotation . Set ( 0 , 0 , 0 ) ;
ParseHelper_Node_Enter ( ne ) ;
MACRO_NODECHECK_LOOPBEGIN ( " instance " ) ;
2020-01-27 21:10:42 +00:00
MACRO_NODECHECK_READCOMP_F ( " deltax " , read_flag [ 0 ] , als . Delta . x ) ;
MACRO_NODECHECK_READCOMP_F ( " deltay " , read_flag [ 1 ] , als . Delta . y ) ;
MACRO_NODECHECK_READCOMP_F ( " deltaz " , read_flag [ 2 ] , als . Delta . z ) ;
MACRO_NODECHECK_READCOMP_F ( " rx " , read_flag [ 3 ] , als . Rotation . x ) ;
MACRO_NODECHECK_READCOMP_F ( " ry " , read_flag [ 4 ] , als . Rotation . y ) ;
MACRO_NODECHECK_READCOMP_F ( " rz " , read_flag [ 5 ] , als . Rotation . z ) ;
2016-09-28 14:49:15 +00:00
MACRO_NODECHECK_LOOPEND ( " instance " ) ;
ParseHelper_Node_Exit ( ) ;
// also convert degrees to radians.
als . Rotation . x = AI_MATH_PI_F * als . Rotation . x / 180.0f ;
als . Rotation . y = AI_MATH_PI_F * als . Rotation . y / 180.0f ;
als . Rotation . z = AI_MATH_PI_F * als . Rotation . z / 180.0f ;
2020-01-27 21:10:42 +00:00
} // if(!mReader->isEmptyElement())
else {
mNodeElement_Cur - > Child . push_back ( ne ) ; // Add element to child list of current element
} // if(!mReader->isEmptyElement()) else
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
mNodeElement_List . push_back ( ne ) ; // and to node element list because its a new object in graph.
2016-09-28 14:49:15 +00:00
}
// <object
// id="" - A unique ObjectID for the new object being defined.
// >
// </object>
// An object definition.
// Multi elements - Yes.
// Parent element - <amf>.
2020-01-27 21:18:48 +00:00
void AMFImporter : : ParseNode_Object ( XmlNode * nodeInst ) {
2020-01-27 21:10:42 +00:00
std : : string id ;
CAMFImporter_NodeElement * ne ( nullptr ) ;
2020-01-27 21:18:48 +00:00
pugi : : xml_node * node = nodeInst - > getNode ( ) ;
for ( pugi : : xml_attribute_iterator ait = node - > attributes_begin ( ) ; ait ! = node - > attributes_end ( ) ; + + ait ) {
if ( ait - > name ( ) = = " id " ) {
id = ait - > as_string ( ) ;
2020-02-03 20:19:03 +00:00
}
2020-01-27 21:18:48 +00:00
}
2016-09-28 14:49:15 +00:00
// Read attributes for node <object>.
2020-01-27 21:18:48 +00:00
/*MACRO_ATTRREAD_LOOPBEG;
2020-01-27 21:10:42 +00:00
MACRO_ATTRREAD_CHECK_RET ( " id " , id , mReader - > getAttributeValue ) ;
2020-01-27 21:18:48 +00:00
MACRO_ATTRREAD_LOOPEND ; */
2016-09-28 14:49:15 +00:00
// create and if needed - define new geometry object.
ne = new CAMFImporter_NodeElement_Object ( mNodeElement_Cur ) ;
2020-01-27 21:10:42 +00:00
CAMFImporter_NodeElement_Object & als = * ( ( CAMFImporter_NodeElement_Object * ) ne ) ; // alias for convenience
2016-09-28 14:49:15 +00:00
2020-01-27 21:18:48 +00:00
if ( ! id . empty ( ) ) {
als . ID = id ;
}
2016-09-28 14:49:15 +00:00
// Check for child nodes
2020-01-27 21:18:48 +00:00
2020-02-03 20:19:03 +00:00
for ( pugi : : xml_node_iterator it = node - > children ( ) . begin ( ) ; it ! = node - > children - > end ( ) ; + + it ) {
2020-01-27 21:18:48 +00:00
bool col_read = false ;
if ( it - > name ( ) = = " mesh " ) {
2020-02-03 20:19:03 +00:00
ParseNode_Mesh ( * it ) ;
} else if ( it - > name ( ) = = " metadata " ) {
ParseNode_Metadata ( * it ) ;
}
2020-01-27 21:18:48 +00:00
}
2020-02-05 21:51:39 +00:00
if ( ! mXmlParser - > isEmptyElement ( ) ) {
2016-09-28 14:49:15 +00:00
bool col_read = false ;
ParseHelper_Node_Enter ( ne ) ;
MACRO_NODECHECK_LOOPBEGIN ( " object " ) ;
2020-01-27 21:10:42 +00:00
if ( XML_CheckNode_NameEqual ( " color " ) ) {
// Check if color already defined for object.
if ( col_read ) Throw_MoreThanOnceDefined ( " color " , " Only one color can be defined for <object>. " ) ;
// read data and set flag about it
ParseNode_Color ( ) ;
col_read = true ;
continue ;
}
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
if ( XML_CheckNode_NameEqual ( " mesh " ) ) {
ParseNode_Mesh ( ) ;
continue ;
}
if ( XML_CheckNode_NameEqual ( " metadata " ) ) {
ParseNode_Metadata ( ) ;
continue ;
}
2016-09-28 14:49:15 +00:00
MACRO_NODECHECK_LOOPEND ( " object " ) ;
ParseHelper_Node_Exit ( ) ;
2020-01-27 21:10:42 +00:00
} // if(!mReader->isEmptyElement())
else {
mNodeElement_Cur - > Child . push_back ( ne ) ; // Add element to child list of current element
} // if(!mReader->isEmptyElement()) else
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
mNodeElement_List . push_back ( ne ) ; // and to node element list because its a new object in graph.
2016-09-28 14:49:15 +00:00
}
// <metadata
// type="" - The type of the attribute.
// >
// </metadata>
// Specify additional information about an entity.
// Multi elements - Yes.
// Parent element - <amf>, <object>, <volume>, <material>, <vertex>.
//
// Reserved types are:
// "Name" - The alphanumeric label of the entity, to be used by the interpreter if interacting with the user.
// "Description" - A description of the content of the entity
// "URL" - A link to an external resource relating to the entity
// "Author" - Specifies the name(s) of the author(s) of the entity
// "Company" - Specifying the company generating the entity
// "CAD" - specifies the name of the originating CAD software and version
// "Revision" - specifies the revision of the entity
// "Tolerance" - specifies the desired manufacturing tolerance of the entity in entity's unit system
// "Volume" - specifies the total volume of the entity, in the entity's unit system, to be used for verification (object and volume only)
2020-01-27 21:10:42 +00:00
void AMFImporter : : ParseNode_Metadata ( ) {
std : : string type , value ;
CAMFImporter_NodeElement * ne ( nullptr ) ;
2016-09-28 14:49:15 +00:00
// read attribute
MACRO_ATTRREAD_LOOPBEG ;
2020-02-05 21:51:39 +00:00
MACRO_ATTRREAD_CHECK_RET ( " type " , type , mXmlParser - > getAttributeValue ) ;
2016-09-28 14:49:15 +00:00
MACRO_ATTRREAD_LOOPEND ;
// and value of node.
2020-02-05 21:51:39 +00:00
value = mXmlParser - > getNodeData ( ) ;
2016-09-28 14:49:15 +00:00
// Create node element and assign read data.
ne = new CAMFImporter_NodeElement_Metadata ( mNodeElement_Cur ) ;
2020-01-27 21:10:42 +00:00
( ( CAMFImporter_NodeElement_Metadata * ) ne ) - > Type = type ;
( ( CAMFImporter_NodeElement_Metadata * ) ne ) - > Value = value ;
mNodeElement_Cur - > Child . push_back ( ne ) ; // Add element to child list of current element
mNodeElement_List . push_back ( ne ) ; // and to node element list because its a new object in graph.
2016-09-28 14:49:15 +00:00
}
/*********************************************************************************************************************************************/
/******************************************************** Functions: BaseImporter set ********************************************************/
/*********************************************************************************************************************************************/
2020-01-27 21:10:42 +00:00
bool AMFImporter : : CanRead ( const std : : string & pFile , IOSystem * pIOHandler , bool pCheckSig ) const {
const std : : string extension = GetExtension ( pFile ) ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
if ( extension = = " amf " ) {
return true ;
}
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
if ( ! extension . length ( ) | | pCheckSig ) {
const char * tokens [ ] = { " <amf " } ;
2016-09-28 14:49:15 +00:00
2020-01-27 21:10:42 +00:00
return SearchFileHeaderForToken ( pIOHandler , pFile , tokens , 1 ) ;
2016-09-28 14:49:15 +00:00
}
return false ;
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : GetExtensionList ( std : : set < std : : string > & pExtensionList ) {
2016-09-28 14:49:15 +00:00
pExtensionList . insert ( " amf " ) ;
}
2020-01-27 21:10:42 +00:00
const aiImporterDesc * AMFImporter : : GetInfo ( ) const {
2016-09-28 14:49:15 +00:00
return & Description ;
}
2020-01-27 21:10:42 +00:00
void AMFImporter : : InternReadFile ( const std : : string & pFile , aiScene * pScene , IOSystem * pIOHandler ) {
Clear ( ) ; // delete old graph.
2016-09-28 14:49:15 +00:00
ParseFile ( pFile , pIOHandler ) ;
Postprocess_BuildScene ( pScene ) ;
// scene graph is ready, exit.
}
2020-01-27 21:10:42 +00:00
} // namespace Assimp
2016-09-28 14:49:15 +00:00
# endif // !ASSIMP_BUILD_NO_AMF_IMPORTER