| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206 | /*Open Asset Import Library (assimp)----------------------------------------------------------------------Copyright (c) 2006-2019, assimp teamAll rights reserved.Redistribution and use of this software in source and binary forms,with or without modification, are permitted provided that thefollowing 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 NOTLIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FORA PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHTOWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOTLIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANYTHEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USEOF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.----------------------------------------------------------------------*/#ifndef ASSIMP_BUILD_NO_OGRE_IMPORTER#include "OgreStructs.h"#include <assimp/TinyFormatter.h>#include <assimp/scene.h>#include <assimp/DefaultLogger.hpp>#include <assimp/Exceptional.h>namespace Assimp{namespace Ogre{// VertexElementVertexElement::VertexElement() :    index(0),    source(0),    offset(0),    type(VET_FLOAT1),    semantic(VES_POSITION){}size_t VertexElement::Size() const{    return TypeSize(type);}size_t VertexElement::ComponentCount() const{    return ComponentCount(type);}size_t VertexElement::ComponentCount(Type type){    switch(type)    {        case VET_COLOUR:        case VET_COLOUR_ABGR:        case VET_COLOUR_ARGB:        case VET_FLOAT1:        case VET_DOUBLE1:        case VET_SHORT1:        case VET_USHORT1:        case VET_INT1:        case VET_UINT1:            return 1;        case VET_FLOAT2:        case VET_DOUBLE2:        case VET_SHORT2:        case VET_USHORT2:        case VET_INT2:        case VET_UINT2:            return 2;        case VET_FLOAT3:        case VET_DOUBLE3:        case VET_SHORT3:        case VET_USHORT3:        case VET_INT3:        case VET_UINT3:            return 3;        case VET_FLOAT4:        case VET_DOUBLE4:        case VET_SHORT4:        case VET_USHORT4:        case VET_INT4:        case VET_UINT4:        case VET_UBYTE4:            return 4;    }    return 0;}size_t VertexElement::TypeSize(Type type){    switch(type)    {        case VET_COLOUR:        case VET_COLOUR_ABGR:        case VET_COLOUR_ARGB:            return sizeof(unsigned int);        case VET_FLOAT1:            return sizeof(float);        case VET_FLOAT2:            return sizeof(float)*2;        case VET_FLOAT3:            return sizeof(float)*3;        case VET_FLOAT4:            return sizeof(float)*4;        case VET_DOUBLE1:            return sizeof(double);        case VET_DOUBLE2:            return sizeof(double)*2;        case VET_DOUBLE3:            return sizeof(double)*3;        case VET_DOUBLE4:            return sizeof(double)*4;        case VET_SHORT1:            return sizeof(short);        case VET_SHORT2:            return sizeof(short)*2;        case VET_SHORT3:            return sizeof(short)*3;        case VET_SHORT4:            return sizeof(short)*4;        case VET_USHORT1:            return sizeof(unsigned short);        case VET_USHORT2:            return sizeof(unsigned short)*2;        case VET_USHORT3:            return sizeof(unsigned short)*3;        case VET_USHORT4:            return sizeof(unsigned short)*4;        case VET_INT1:            return sizeof(int);        case VET_INT2:            return sizeof(int)*2;        case VET_INT3:            return sizeof(int)*3;        case VET_INT4:            return sizeof(int)*4;        case VET_UINT1:            return sizeof(unsigned int);        case VET_UINT2:            return sizeof(unsigned int)*2;        case VET_UINT3:            return sizeof(unsigned int)*3;        case VET_UINT4:            return sizeof(unsigned int)*4;        case VET_UBYTE4:            return sizeof(unsigned char)*4;    }    return 0;}std::string VertexElement::TypeToString(){    return TypeToString(type);}std::string VertexElement::TypeToString(Type type){    switch(type)    {        case VET_COLOUR:        return "COLOUR";        case VET_COLOUR_ABGR:   return "COLOUR_ABGR";        case VET_COLOUR_ARGB:   return "COLOUR_ARGB";        case VET_FLOAT1:        return "FLOAT1";        case VET_FLOAT2:        return "FLOAT2";        case VET_FLOAT3:        return "FLOAT3";        case VET_FLOAT4:        return "FLOAT4";        case VET_DOUBLE1:       return "DOUBLE1";        case VET_DOUBLE2:       return "DOUBLE2";        case VET_DOUBLE3:       return "DOUBLE3";        case VET_DOUBLE4:       return "DOUBLE4";        case VET_SHORT1:        return "SHORT1";        case VET_SHORT2:        return "SHORT2";        case VET_SHORT3:        return "SHORT3";        case VET_SHORT4:        return "SHORT4";        case VET_USHORT1:       return "USHORT1";        case VET_USHORT2:       return "USHORT2";        case VET_USHORT3:       return "USHORT3";        case VET_USHORT4:       return "USHORT4";        case VET_INT1:          return "INT1";        case VET_INT2:          return "INT2";        case VET_INT3:          return "INT3";        case VET_INT4:          return "INT4";        case VET_UINT1:         return "UINT1";        case VET_UINT2:         return "UINT2";        case VET_UINT3:         return "UINT3";        case VET_UINT4:         return "UINT4";        case VET_UBYTE4:        return "UBYTE4";    }    return "Uknown_VertexElement::Type";}std::string VertexElement::SemanticToString(){    return SemanticToString(semantic);}std::string VertexElement::SemanticToString(Semantic semantic){    switch(semantic)    {        case VES_POSITION:              return "POSITION";        case VES_BLEND_WEIGHTS:         return "BLEND_WEIGHTS";        case VES_BLEND_INDICES:         return "BLEND_INDICES";        case VES_NORMAL:                return "NORMAL";        case VES_DIFFUSE:               return "DIFFUSE";        case VES_SPECULAR:              return "SPECULAR";        case VES_TEXTURE_COORDINATES:   return "TEXTURE_COORDINATES";        case VES_BINORMAL:              return "BINORMAL";        case VES_TANGENT:               return "TANGENT";    }    return "Uknown_VertexElement::Semantic";}// IVertexDataIVertexData::IVertexData() :    count(0){}bool IVertexData::HasBoneAssignments() const{    return !boneAssignments.empty();}void IVertexData::AddVertexMapping(uint32_t oldIndex, uint32_t newIndex){    BoneAssignmentsForVertex(oldIndex, newIndex, boneAssignmentsMap[newIndex]);    vertexIndexMapping[oldIndex].push_back(newIndex);}void IVertexData::BoneAssignmentsForVertex(uint32_t currentIndex, uint32_t newIndex, VertexBoneAssignmentList &dest) const{    for (const auto &boneAssign : boneAssignments)    {        if (boneAssign.vertexIndex == currentIndex)        {            VertexBoneAssignment a = boneAssign;            a.vertexIndex = newIndex;            dest.push_back(a);        }    }}AssimpVertexBoneWeightList IVertexData::AssimpBoneWeights(size_t vertices){    AssimpVertexBoneWeightList weights;    for(size_t vi=0; vi<vertices; ++vi)    {        VertexBoneAssignmentList &vertexWeights = boneAssignmentsMap[static_cast<unsigned int>(vi)];        for (VertexBoneAssignmentList::const_iterator iter=vertexWeights.begin(), end=vertexWeights.end();            iter!=end; ++iter)        {            std::vector<aiVertexWeight> &boneWeights = weights[iter->boneIndex];            boneWeights.push_back(aiVertexWeight(static_cast<unsigned int>(vi), iter->weight));        }    }    return weights;}std::set<uint16_t> IVertexData::ReferencedBonesByWeights() const{    std::set<uint16_t> referenced;    for (const auto &boneAssign : boneAssignments)    {        referenced.insert(boneAssign.boneIndex);    }    return referenced;}// VertexDataVertexData::VertexData(){}VertexData::~VertexData(){    Reset();}void VertexData::Reset(){    // Releases shared ptr memory streams.    vertexBindings.clear();    vertexElements.clear();}uint32_t VertexData::VertexSize(uint16_t source) const{    uint32_t size = 0;    for(const auto &element : vertexElements)    {        if (element.source == source)            size += static_cast<uint32_t>(element.Size());    }    return size;}MemoryStream *VertexData::VertexBuffer(uint16_t source){    if (vertexBindings.find(source) != vertexBindings.end())        return vertexBindings[source].get();    return 0;}VertexElement *VertexData::GetVertexElement(VertexElement::Semantic semantic, uint16_t index){    for(auto & element : vertexElements)    {        if (element.semantic == semantic && element.index == index)            return &element;    }    return 0;}// VertexDataXmlVertexDataXml::VertexDataXml(){}bool VertexDataXml::HasPositions() const{    return !positions.empty();}bool VertexDataXml::HasNormals() const{    return !normals.empty();}bool VertexDataXml::HasTangents() const{    return !tangents.empty();}bool VertexDataXml::HasUvs() const{    return !uvs.empty();}size_t VertexDataXml::NumUvs() const{    return uvs.size();}// IndexDataIndexData::IndexData() :    count(0),    faceCount(0),    is32bit(false){}IndexData::~IndexData(){    Reset();}void IndexData::Reset(){    // Release shared ptr memory stream.    buffer.reset();}size_t IndexData::IndexSize() const{    return (is32bit ? sizeof(uint32_t) : sizeof(uint16_t));}size_t IndexData::FaceSize() const{    return IndexSize() * 3;}// MeshMesh::Mesh()     : hasSkeletalAnimations(false)    , skeleton(NULL)    , sharedVertexData(NULL)    , subMeshes()    , animations()    , poses(){}Mesh::~Mesh(){    Reset();}void Mesh::Reset(){    OGRE_SAFE_DELETE(skeleton)    OGRE_SAFE_DELETE(sharedVertexData)    for(auto &mesh : subMeshes) {        OGRE_SAFE_DELETE(mesh)    }    subMeshes.clear();    for(auto &anim : animations) {        OGRE_SAFE_DELETE(anim)    }    animations.clear();    for(auto &pose : poses) {        OGRE_SAFE_DELETE(pose)    }    poses.clear();}size_t Mesh::NumSubMeshes() const{    return subMeshes.size();}SubMesh *Mesh::GetSubMesh( size_t index ) const{    for ( size_t i = 0; i < subMeshes.size(); ++i ) {        if ( subMeshes[ i ]->index == index ) {            return subMeshes[ i ];        }    }    return 0;}void Mesh::ConvertToAssimpScene(aiScene* dest){    if ( nullptr == dest ) {        return;    }    // Setup    dest->mNumMeshes = static_cast<unsigned int>(NumSubMeshes());    dest->mMeshes = new aiMesh*[dest->mNumMeshes];    // Create root node    dest->mRootNode = new aiNode();    dest->mRootNode->mNumMeshes = dest->mNumMeshes;    dest->mRootNode->mMeshes = new unsigned int[dest->mRootNode->mNumMeshes];    // Export meshes    for(size_t i=0; i<dest->mNumMeshes; ++i) {        dest->mMeshes[i] = subMeshes[i]->ConvertToAssimpMesh(this);        dest->mRootNode->mMeshes[i] = static_cast<unsigned int>(i);    }    // Export skeleton    if (skeleton)    {        // Bones        if (!skeleton->bones.empty())        {            BoneList rootBones = skeleton->RootBones();            dest->mRootNode->mNumChildren = static_cast<unsigned int>(rootBones.size());            dest->mRootNode->mChildren = new aiNode*[dest->mRootNode->mNumChildren];            for(size_t i=0, len=rootBones.size(); i<len; ++i)            {                dest->mRootNode->mChildren[i] = rootBones[i]->ConvertToAssimpNode(skeleton, dest->mRootNode);            }        }        // Animations        if (!skeleton->animations.empty())        {            dest->mNumAnimations = static_cast<unsigned int>(skeleton->animations.size());            dest->mAnimations = new aiAnimation*[dest->mNumAnimations];            for(size_t i=0, len=skeleton->animations.size(); i<len; ++i)            {                dest->mAnimations[i] = skeleton->animations[i]->ConvertToAssimpAnimation();            }        }    }}// ISubMeshISubMesh::ISubMesh() :    index(0),    materialIndex(-1),    usesSharedVertexData(false),    operationType(OT_POINT_LIST){}// SubMeshSubMesh::SubMesh() :    vertexData(0),    indexData(new IndexData()){}SubMesh::~SubMesh(){    Reset();}void SubMesh::Reset(){    OGRE_SAFE_DELETE(vertexData)    OGRE_SAFE_DELETE(indexData)}aiMesh *SubMesh::ConvertToAssimpMesh(Mesh *parent){    if (operationType != OT_TRIANGLE_LIST) {        throw DeadlyImportError(Formatter::format() << "Only mesh operation type OT_TRIANGLE_LIST is supported. Found " << operationType);    }    aiMesh *dest = new aiMesh();    dest->mPrimitiveTypes = aiPrimitiveType_TRIANGLE;    if (!name.empty())        dest->mName = name;    // Material index    if (materialIndex != -1)        dest->mMaterialIndex = materialIndex;    // Pick source vertex data from shader geometry or from internal geometry.    VertexData *src = (!usesSharedVertexData ? vertexData : parent->sharedVertexData);    VertexElement *positionsElement = src->GetVertexElement(VertexElement::VES_POSITION);    VertexElement *normalsElement   = src->GetVertexElement(VertexElement::VES_NORMAL);    VertexElement *uv1Element       = src->GetVertexElement(VertexElement::VES_TEXTURE_COORDINATES, 0);    VertexElement *uv2Element       = src->GetVertexElement(VertexElement::VES_TEXTURE_COORDINATES, 1);    // Sanity checks    if (!positionsElement) {        throw DeadlyImportError("Failed to import Ogre VertexElement::VES_POSITION. Mesh does not have vertex positions!");    } else if (positionsElement->type != VertexElement::VET_FLOAT3) {        throw DeadlyImportError("Ogre Mesh position vertex element type != VertexElement::VET_FLOAT3. This is not supported.");    } else if (normalsElement && normalsElement->type != VertexElement::VET_FLOAT3) {        throw DeadlyImportError("Ogre Mesh normal vertex element type != VertexElement::VET_FLOAT3. This is not supported.");    }    // Faces    dest->mNumFaces = indexData->faceCount;    dest->mFaces = new aiFace[dest->mNumFaces];    // Assimp required unique vertices, we need to convert from Ogres shared indexing.    size_t uniqueVertexCount = dest->mNumFaces * 3;    dest->mNumVertices = static_cast<unsigned int>(uniqueVertexCount);    dest->mVertices = new aiVector3D[dest->mNumVertices];    // Source streams    MemoryStream *positions      = src->VertexBuffer(positionsElement->source);    MemoryStream *normals        = (normalsElement ? src->VertexBuffer(normalsElement->source) : 0);    MemoryStream *uv1            = (uv1Element ? src->VertexBuffer(uv1Element->source) : 0);    MemoryStream *uv2            = (uv2Element ? src->VertexBuffer(uv2Element->source) : 0);    // Element size    const size_t sizePosition    = positionsElement->Size();    const size_t sizeNormal      = (normalsElement ? normalsElement->Size() : 0);    const size_t sizeUv1         = (uv1Element ? uv1Element->Size() : 0);    const size_t sizeUv2         = (uv2Element ? uv2Element->Size() : 0);    // Vertex width    const size_t vWidthPosition  = src->VertexSize(positionsElement->source);    const size_t vWidthNormal    = (normalsElement ? src->VertexSize(normalsElement->source) : 0);    const size_t vWidthUv1       = (uv1Element ? src->VertexSize(uv1Element->source) : 0);    const size_t vWidthUv2       = (uv2Element ? src->VertexSize(uv2Element->source) : 0);    bool boneAssignments = src->HasBoneAssignments();    // Prepare normals    if (normals)        dest->mNormals = new aiVector3D[dest->mNumVertices];    // Prepare UVs, ignoring incompatible UVs.    if (uv1)    {        if (uv1Element->type == VertexElement::VET_FLOAT2 || uv1Element->type == VertexElement::VET_FLOAT3)        {            dest->mNumUVComponents[0] = static_cast<unsigned int>(uv1Element->ComponentCount());            dest->mTextureCoords[0] = new aiVector3D[dest->mNumVertices];        }        else        {            ASSIMP_LOG_WARN(Formatter::format() << "Ogre imported UV0 type " << uv1Element->TypeToString() << " is not compatible with Assimp. Ignoring UV.");            uv1 = 0;        }    }    if (uv2)    {        if (uv2Element->type == VertexElement::VET_FLOAT2 || uv2Element->type == VertexElement::VET_FLOAT3)        {            dest->mNumUVComponents[1] = static_cast<unsigned int>(uv2Element->ComponentCount());            dest->mTextureCoords[1] = new aiVector3D[dest->mNumVertices];        }        else        {            ASSIMP_LOG_WARN(Formatter::format() << "Ogre imported UV0 type " << uv2Element->TypeToString() << " is not compatible with Assimp. Ignoring UV.");            uv2 = 0;        }    }    aiVector3D *uv1Dest = (uv1 ? dest->mTextureCoords[0] : 0);    aiVector3D *uv2Dest = (uv2 ? dest->mTextureCoords[1] : 0);    MemoryStream *faces = indexData->buffer.get();    for (size_t fi=0, isize=indexData->IndexSize(), fsize=indexData->FaceSize();         fi<dest->mNumFaces; ++fi)    {        // Source Ogre face        aiFace ogreFace;        ogreFace.mNumIndices = 3;        ogreFace.mIndices = new unsigned int[3];        faces->Seek(fi * fsize, aiOrigin_SET);        if (indexData->is32bit)        {            faces->Read(&ogreFace.mIndices[0], isize, 3);        }        else        {            uint16_t iout = 0;            for (size_t ii=0; ii<3; ++ii)            {                faces->Read(&iout, isize, 1);                ogreFace.mIndices[ii] = static_cast<unsigned int>(iout);            }        }        // Destination Assimp face        aiFace &face = dest->mFaces[fi];        face.mNumIndices = 3;        face.mIndices = new unsigned int[3];        const size_t pos = fi * 3;        for (size_t v=0; v<3; ++v)        {            const size_t newIndex = pos + v;            // Write face index            face.mIndices[v] = static_cast<unsigned int>(newIndex);            // Ogres vertex index to ref into the source buffers.            const size_t ogreVertexIndex = ogreFace.mIndices[v];            src->AddVertexMapping(static_cast<uint32_t>(ogreVertexIndex), static_cast<uint32_t>(newIndex));            // Position            positions->Seek((vWidthPosition * ogreVertexIndex) + positionsElement->offset, aiOrigin_SET);            positions->Read(&dest->mVertices[newIndex], sizePosition, 1);            // Normal            if (normals)            {                normals->Seek((vWidthNormal * ogreVertexIndex) + normalsElement->offset, aiOrigin_SET);                normals->Read(&dest->mNormals[newIndex], sizeNormal, 1);            }            // UV0            if (uv1 && uv1Dest)            {                uv1->Seek((vWidthUv1 * ogreVertexIndex) + uv1Element->offset, aiOrigin_SET);                uv1->Read(&uv1Dest[newIndex], sizeUv1, 1);                uv1Dest[newIndex].y = (uv1Dest[newIndex].y * -1) + 1; // Flip UV from Ogre to Assimp form            }            // UV1            if (uv2 && uv2Dest)            {                uv2->Seek((vWidthUv2 * ogreVertexIndex) + uv2Element->offset, aiOrigin_SET);                uv2->Read(&uv2Dest[newIndex], sizeUv2, 1);                uv2Dest[newIndex].y = (uv2Dest[newIndex].y * -1) + 1; // Flip UV from Ogre to Assimp form            }        }    }    // Bones and bone weights    if (parent->skeleton && boneAssignments)    {        AssimpVertexBoneWeightList weights = src->AssimpBoneWeights(dest->mNumVertices);        std::set<uint16_t> referencedBones = src->ReferencedBonesByWeights();        dest->mNumBones = static_cast<unsigned int>(referencedBones.size());        dest->mBones = new aiBone*[dest->mNumBones];        size_t assimpBoneIndex = 0;        for(std::set<uint16_t>::const_iterator rbIter=referencedBones.begin(), rbEnd=referencedBones.end(); rbIter != rbEnd; ++rbIter, ++assimpBoneIndex)        {            Bone *bone = parent->skeleton->BoneById((*rbIter));            dest->mBones[assimpBoneIndex] = bone->ConvertToAssimpBone(parent->skeleton, weights[bone->id]);        }    }    return dest;}// MeshXmlMeshXml::MeshXml() :    skeleton(0),    sharedVertexData(0){}MeshXml::~MeshXml(){    Reset();}void MeshXml::Reset(){    OGRE_SAFE_DELETE(skeleton)    OGRE_SAFE_DELETE(sharedVertexData)    for(auto &mesh : subMeshes) {        OGRE_SAFE_DELETE(mesh)    }    subMeshes.clear();}size_t MeshXml::NumSubMeshes() const{    return subMeshes.size();}SubMeshXml *MeshXml::GetSubMesh(uint16_t index) const{    for(size_t i=0; i<subMeshes.size(); ++i)        if (subMeshes[i]->index == index)            return subMeshes[i];    return 0;}void MeshXml::ConvertToAssimpScene(aiScene* dest){    // Setup    dest->mNumMeshes = static_cast<unsigned int>(NumSubMeshes());    dest->mMeshes = new aiMesh*[dest->mNumMeshes];    // Create root node    dest->mRootNode = new aiNode();    dest->mRootNode->mNumMeshes = dest->mNumMeshes;    dest->mRootNode->mMeshes = new unsigned int[dest->mRootNode->mNumMeshes];    // Export meshes    for(size_t i=0; i<dest->mNumMeshes; ++i)    {        dest->mMeshes[i] = subMeshes[i]->ConvertToAssimpMesh(this);        dest->mRootNode->mMeshes[i] = static_cast<unsigned int>(i);    }    // Export skeleton    if (skeleton)    {        // Bones        if (!skeleton->bones.empty())        {            BoneList rootBones = skeleton->RootBones();            dest->mRootNode->mNumChildren = static_cast<unsigned int>(rootBones.size());            dest->mRootNode->mChildren = new aiNode*[dest->mRootNode->mNumChildren];            for(size_t i=0, len=rootBones.size(); i<len; ++i)            {                dest->mRootNode->mChildren[i] = rootBones[i]->ConvertToAssimpNode(skeleton, dest->mRootNode);            }        }        // Animations        if (!skeleton->animations.empty())        {            dest->mNumAnimations = static_cast<unsigned int>(skeleton->animations.size());            dest->mAnimations = new aiAnimation*[dest->mNumAnimations];            for(size_t i=0, len=skeleton->animations.size(); i<len; ++i)            {                dest->mAnimations[i] = skeleton->animations[i]->ConvertToAssimpAnimation();            }        }    }}// SubMeshXmlSubMeshXml::SubMeshXml() :    indexData(new IndexDataXml()),    vertexData(0){}SubMeshXml::~SubMeshXml(){    Reset();}void SubMeshXml::Reset(){    OGRE_SAFE_DELETE(indexData)    OGRE_SAFE_DELETE(vertexData)}aiMesh *SubMeshXml::ConvertToAssimpMesh(MeshXml *parent){    aiMesh *dest = new aiMesh();    dest->mPrimitiveTypes = aiPrimitiveType_TRIANGLE;    if (!name.empty())        dest->mName = name;    // Material index    if (materialIndex != -1)        dest->mMaterialIndex = materialIndex;    // Faces    dest->mNumFaces = indexData->faceCount;    dest->mFaces = new aiFace[dest->mNumFaces];    // Assimp required unique vertices, we need to convert from Ogres shared indexing.    size_t uniqueVertexCount = dest->mNumFaces * 3;    dest->mNumVertices = static_cast<unsigned int>(uniqueVertexCount);    dest->mVertices = new aiVector3D[dest->mNumVertices];    VertexDataXml *src = (!usesSharedVertexData ? vertexData : parent->sharedVertexData);    bool boneAssignments = src->HasBoneAssignments();    bool normals = src->HasNormals();    size_t uvs = src->NumUvs();    // Prepare normals    if (normals)        dest->mNormals = new aiVector3D[dest->mNumVertices];    // Prepare UVs    for(size_t uvi=0; uvi<uvs; ++uvi)    {        dest->mNumUVComponents[uvi] = 2;        dest->mTextureCoords[uvi] = new aiVector3D[dest->mNumVertices];    }    for (size_t fi=0; fi<dest->mNumFaces; ++fi)    {        // Source Ogre face        aiFace &ogreFace = indexData->faces[fi];        // Destination Assimp face        aiFace &face = dest->mFaces[fi];        face.mNumIndices = 3;        face.mIndices = new unsigned int[3];        const size_t pos = fi * 3;        for (size_t v=0; v<3; ++v)        {            const size_t newIndex = pos + v;            // Write face index            face.mIndices[v] = static_cast<unsigned int>(newIndex);            // Ogres vertex index to ref into the source buffers.            const size_t ogreVertexIndex = ogreFace.mIndices[v];            src->AddVertexMapping(static_cast<uint32_t>(ogreVertexIndex), static_cast<uint32_t>(newIndex));            // Position            dest->mVertices[newIndex] = src->positions[ogreVertexIndex];            // Normal            if (normals)                dest->mNormals[newIndex] = src->normals[ogreVertexIndex];            // UVs            for(size_t uvi=0; uvi<uvs; ++uvi)            {                aiVector3D *uvDest = dest->mTextureCoords[uvi];                std::vector<aiVector3D> &uvSrc = src->uvs[uvi];                uvDest[newIndex] = uvSrc[ogreVertexIndex];            }        }    }    // Bones and bone weights    if (parent->skeleton && boneAssignments)    {        AssimpVertexBoneWeightList weights = src->AssimpBoneWeights(dest->mNumVertices);        std::set<uint16_t> referencedBones = src->ReferencedBonesByWeights();        dest->mNumBones = static_cast<unsigned int>(referencedBones.size());        dest->mBones = new aiBone*[dest->mNumBones];        size_t assimpBoneIndex = 0;        for(std::set<uint16_t>::const_iterator rbIter=referencedBones.begin(), rbEnd=referencedBones.end(); rbIter != rbEnd; ++rbIter, ++assimpBoneIndex)        {            Bone *bone = parent->skeleton->BoneById((*rbIter));            dest->mBones[assimpBoneIndex] = bone->ConvertToAssimpBone(parent->skeleton, weights[bone->id]);        }    }    return dest;}// AnimationAnimation::Animation(Skeleton *parent) :    parentMesh(NULL),    parentSkeleton(parent),    length(0.0f),    baseTime(-1.0f){}Animation::Animation(Mesh *parent) :    parentMesh(parent),    parentSkeleton(0),    length(0.0f),    baseTime(-1.0f){}VertexData *Animation::AssociatedVertexData(VertexAnimationTrack *track) const{    if (!parentMesh)        return 0;    bool sharedGeom = (track->target == 0);    if (sharedGeom)        return parentMesh->sharedVertexData;    else        return parentMesh->GetSubMesh(track->target-1)->vertexData;}aiAnimation *Animation::ConvertToAssimpAnimation(){    aiAnimation *anim = new aiAnimation();    anim->mName = name;    anim->mDuration = static_cast<double>(length);    anim->mTicksPerSecond = 1.0;    // Tracks    if (!tracks.empty())    {        anim->mNumChannels = static_cast<unsigned int>(tracks.size());        anim->mChannels = new aiNodeAnim*[anim->mNumChannels];        for(size_t i=0, len=tracks.size(); i<len; ++i)        {            anim->mChannels[i] = tracks[i].ConvertToAssimpAnimationNode(parentSkeleton);        }    }    return anim;}// SkeletonSkeleton::Skeleton() :    bones(),    animations(),    blendMode(ANIMBLEND_AVERAGE){}Skeleton::~Skeleton(){    Reset();}void Skeleton::Reset(){    for(auto &bone : bones) {        OGRE_SAFE_DELETE(bone)    }    bones.clear();    for(auto &anim : animations) {        OGRE_SAFE_DELETE(anim)    }    animations.clear();}BoneList Skeleton::RootBones() const{    BoneList rootBones;    for(BoneList::const_iterator iter = bones.begin(); iter != bones.end(); ++iter)    {        if (!(*iter)->IsParented())            rootBones.push_back((*iter));    }    return rootBones;}size_t Skeleton::NumRootBones() const{    size_t num = 0;    for(BoneList::const_iterator iter = bones.begin(); iter != bones.end(); ++iter)    {        if (!(*iter)->IsParented())            num++;    }    return num;}Bone *Skeleton::BoneByName(const std::string &name) const{    for(BoneList::const_iterator iter = bones.begin(); iter != bones.end(); ++iter)    {        if ((*iter)->name == name)            return (*iter);    }    return 0;}Bone *Skeleton::BoneById(uint16_t id) const{    for(BoneList::const_iterator iter = bones.begin(); iter != bones.end(); ++iter)    {        if ((*iter)->id == id)            return (*iter);    }    return 0;}// BoneBone::Bone() :    id(0),    parent(0),    parentId(-1),    scale(1.0f, 1.0f, 1.0f){}bool Bone::IsParented() const{    return (parentId != -1 && parent != 0);}uint16_t Bone::ParentId() const{    return static_cast<uint16_t>(parentId);}void Bone::AddChild(Bone *bone){    if (!bone)        return;    if (bone->IsParented())        throw DeadlyImportError("Attaching child Bone that is already parented: " + bone->name);    bone->parent = this;    bone->parentId = id;    children.push_back(bone->id);}void Bone::CalculateWorldMatrixAndDefaultPose(Skeleton *skeleton){    if (!IsParented())        worldMatrix = aiMatrix4x4(scale, rotation, position).Inverse();    else        worldMatrix = aiMatrix4x4(scale, rotation, position).Inverse() * parent->worldMatrix;    defaultPose = aiMatrix4x4(scale, rotation, position);    // Recursively for all children now that the parent matrix has been calculated.    for (auto boneId : children)    {        Bone *child = skeleton->BoneById(boneId);        if (!child) {            throw DeadlyImportError(Formatter::format() << "CalculateWorldMatrixAndDefaultPose: Failed to find child bone " << boneId << " for parent " << id << " " << name);        }        child->CalculateWorldMatrixAndDefaultPose(skeleton);    }}aiNode *Bone::ConvertToAssimpNode(Skeleton *skeleton, aiNode *parentNode){    // Bone node    aiNode* node = new aiNode(name);    node->mParent = parentNode;    node->mTransformation = defaultPose;    // Children    if (!children.empty())    {        node->mNumChildren = static_cast<unsigned int>(children.size());        node->mChildren = new aiNode*[node->mNumChildren];        for(size_t i=0, len=children.size(); i<len; ++i)        {            Bone *child = skeleton->BoneById(children[i]);            if (!child) {                throw DeadlyImportError(Formatter::format() << "ConvertToAssimpNode: Failed to find child bone " << children[i] << " for parent " << id << " " << name);            }            node->mChildren[i] = child->ConvertToAssimpNode(skeleton, node);        }    }    return node;}aiBone *Bone::ConvertToAssimpBone(Skeleton * /*parent*/, const std::vector<aiVertexWeight> &boneWeights){    aiBone *bone = new aiBone();    bone->mName = name;    bone->mOffsetMatrix = worldMatrix;    if (!boneWeights.empty())    {        bone->mNumWeights = static_cast<unsigned int>(boneWeights.size());        bone->mWeights = new aiVertexWeight[boneWeights.size()];        memcpy(bone->mWeights, &boneWeights[0], boneWeights.size() * sizeof(aiVertexWeight));    }    return bone;}// VertexAnimationTrackVertexAnimationTrack::VertexAnimationTrack() :    type(VAT_NONE),    target(0){}aiNodeAnim *VertexAnimationTrack::ConvertToAssimpAnimationNode(Skeleton *skeleton){    if (boneName.empty() || type != VAT_TRANSFORM) {        throw DeadlyImportError("VertexAnimationTrack::ConvertToAssimpAnimationNode: Cannot convert track that has no target bone name or is not type of VAT_TRANSFORM");    }    aiNodeAnim *nodeAnim = new aiNodeAnim();    nodeAnim->mNodeName = boneName;    Bone *bone = skeleton->BoneByName(boneName);    if (!bone) {        throw DeadlyImportError("VertexAnimationTrack::ConvertToAssimpAnimationNode: Failed to find bone " + boneName + " from parent Skeleton");    }    // Keyframes    size_t numKeyframes = transformKeyFrames.size();    nodeAnim->mPositionKeys = new aiVectorKey[numKeyframes];    nodeAnim->mRotationKeys = new aiQuatKey[numKeyframes];    nodeAnim->mScalingKeys = new aiVectorKey[numKeyframes];    nodeAnim->mNumPositionKeys = static_cast<unsigned int>(numKeyframes);    nodeAnim->mNumRotationKeys = static_cast<unsigned int>(numKeyframes);    nodeAnim->mNumScalingKeys  = static_cast<unsigned int>(numKeyframes);    for(size_t kfi=0; kfi<numKeyframes; ++kfi)    {        TransformKeyFrame &kfSource = transformKeyFrames[kfi];        // Calculate the complete transformation from world space to bone space        aiVector3D pos; aiQuaternion rot; aiVector3D scale;        aiMatrix4x4 finalTransform = bone->defaultPose * kfSource.Transform();        finalTransform.Decompose(scale, rot, pos);        double t = static_cast<double>(kfSource.timePos);        nodeAnim->mPositionKeys[kfi].mTime = t;        nodeAnim->mRotationKeys[kfi].mTime = t;        nodeAnim->mScalingKeys[kfi].mTime = t;        nodeAnim->mPositionKeys[kfi].mValue = pos;        nodeAnim->mRotationKeys[kfi].mValue = rot;        nodeAnim->mScalingKeys[kfi].mValue = scale;    }    return nodeAnim;}// TransformKeyFrameTransformKeyFrame::TransformKeyFrame() :    timePos(0.0f),    scale(1.0f, 1.0f, 1.0f){}aiMatrix4x4 TransformKeyFrame::Transform(){    return aiMatrix4x4(scale, rotation, position);}} // Ogre} // Assimp#endif // ASSIMP_BUILD_NO_OGRE_IMPORTER
 |