/* OpenMW - The completely unofficial reimplementation of Morrowind Copyright (C) 2008-2010 Nicolay Korslund Email: < korslund@gmail.com > WWW: http://openmw.sourceforge.net/ This file (ogre_nif_loader.cpp) is part of the OpenMW package. OpenMW is distributed as free software: you can redistribute it and/or modify it under the terms of the GNU General Public License version 3, as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License version 3 along with this program. If not, see http://www.gnu.org/licenses/ . */ #include "bullet_nif_loader.hpp" #include #include #include #include "../nif/nif_file.hpp" #include "../nif/node.hpp" #include "../nif/data.hpp" #include "../nif/property.hpp" #include "../nif/controller.hpp" #include "../nif/extra.hpp" #include #include #include // For warning messages #include // float infinity #include typedef unsigned char ubyte; using namespace std; using namespace Ogre; using namespace Nif; using namespace Mangle::VFS; BulletShape::BulletShape(Ogre::ResourceManager* creator, const Ogre::String &name, Ogre::ResourceHandle handle, const Ogre::String &group, bool isManual, Ogre::ManualResourceLoader *loader) : Ogre::Resource(creator, name, handle, group, isManual, loader) { /* If you were storing a pointer to an object, then you would set that pointer to NULL here. */ /* For consistency with StringInterface, but we don't add any parameters here That's because the Resource implementation of StringInterface is to list all the options that need to be set before loading, of which we have none as such. Full details can be set through scripts. */ Shape = NULL; collide = true; createParamDictionary("BulletShape"); } BulletShape::~BulletShape() { } // farm out to BulletShapeLoader void BulletShape::loadImpl() { mLoader->loadResource(this); } void BulletShape::deleteShape(btCollisionShape* mShape) { if(mShape!=NULL) { if(mShape->isCompound()) { btCompoundShape* ms = static_cast(Shape); btCompoundShapeChild* child = ms->getChildList(); int a = ms->getNumChildShapes(); for(int i=0; i getChildShape(i)); } } delete mShape; } mShape = NULL; } void BulletShape::unloadImpl() { deleteShape(Shape); } //TODO:change this? size_t BulletShape::calculateSize() const { return 1; } //============================================================================================================= template<> BulletShapeManager *Ogre::Singleton::ms_Singleton = 0; BulletShapeManager *BulletShapeManager::getSingletonPtr() { return ms_Singleton; } BulletShapeManager &BulletShapeManager::getSingleton() { assert(ms_Singleton); return(*ms_Singleton); } BulletShapeManager::BulletShapeManager() { mResourceType = "BulletShape"; // low, because it will likely reference other resources mLoadOrder = 30.0f; // this is how we register the ResourceManager with OGRE Ogre::ResourceGroupManager::getSingleton()._registerResourceManager(mResourceType, this); } BulletShapeManager::~BulletShapeManager() { // and this is how we unregister it Ogre::ResourceGroupManager::getSingleton()._unregisterResourceManager(mResourceType); } BulletShapePtr BulletShapeManager::load(const Ogre::String &name, const Ogre::String &group) { BulletShapePtr textf = getByName(name); if (textf.isNull()) textf = create(name, group); textf->load(); return textf; } Ogre::Resource *BulletShapeManager::createImpl(const Ogre::String &name, Ogre::ResourceHandle handle, const Ogre::String &group, bool isManual, Ogre::ManualResourceLoader *loader, const Ogre::NameValuePairList *createParams) { BulletShape* res = new BulletShape(this, name, handle, group, isManual, loader); //if(isManual) //{ //loader->loadResource(res); //} return res; } //==================================================================================================== Ogre::Matrix3 ManualBulletShapeLoader::getMatrix(Nif::Transformation* tr) { Ogre::Matrix3 rot(tr->rotation.v[0].array[0],tr->rotation.v[0].array[1],tr->rotation.v[0].array[2], tr->rotation.v[1].array[0],tr->rotation.v[1].array[1],tr->rotation.v[1].array[2], tr->rotation.v[2].array[0],tr->rotation.v[2].array[1],tr->rotation.v[2].array[2]); return rot; } Ogre::Vector3 ManualBulletShapeLoader::getVector(Nif::Transformation* tr) { Ogre::Vector3 vect3(tr->pos.array[0],tr->pos.array[1],tr->pos.array[2]); return vect3; } btQuaternion ManualBulletShapeLoader::getbtQuat(Ogre::Matrix3 m) { Ogre::Quaternion oquat(m); btQuaternion quat; quat.setW(oquat.w); quat.setX(oquat.x); quat.setY(oquat.y); quat.setZ(oquat.z); return quat; } btVector3 ManualBulletShapeLoader::getbtVector(Nif::Vector v) { btVector3 a(v.array[0],v.array[1],v.array[2]); return a; } void ManualBulletShapeLoader::loadResource(Ogre::Resource *resource) { cShape = static_cast(resource); resourceName = cShape->getName(); cShape->collide = false; currentShape = new btCompoundShape(); cShape->Shape = currentShape; if (!vfs) vfs = new OgreVFS(resourceGroup); if (!vfs->isFile(resourceName)) { warn("File not found."); return; } // Load the NIF. TODO: Wrap this in a try-catch block once we're out // of the early stages of development. Right now we WANT to catch // every error as early and intrusively as possible, as it's most // likely a sign of incomplete code rather than faulty input. Nif::NIFFile nif(vfs->open(resourceName), resourceName); if (nif.numRecords() < 1) { warn("Found no records in NIF."); return; } // The first record is assumed to be the root node Nif::Record *r = nif.getRecord(0); assert(r != NULL); Nif::Node *node = dynamic_cast(r); if (node == NULL) { warn("First record in file was not a node, but a " + r->recName.toString() + ". Skipping file."); return; } //do a first pass handleNode(node,0,Ogre::Matrix3::IDENTITY,Ogre::Vector3::ZERO,1,false,false); //if collide = false, then it does a second pass which create a shape for raycasting. if(cShape->collide == false) { handleNode(node,0,Ogre::Matrix3::IDENTITY,Ogre::Vector3::ZERO,1,false,true); } } void ManualBulletShapeLoader::handleNode(Nif::Node *node, int flags, Ogre::Matrix3 parentRot,Ogre::Vector3 parentPos,float parentScale,bool isCollisionNode,bool raycastingOnly) { // Accumulate the flags from all the child nodes. This works for all // the flags we currently use, at least. flags |= node->flags; // Check for extra data Nif::Extra *e = node; while (!e->extra.empty()) { // Get the next extra data in the list e = e->extra.getPtr(); assert(e != NULL); if (e->recType == Nif::RC_NiStringExtraData) { // String markers may contain important information // affecting the entire subtree of this node Nif::NiStringExtraData *sd = (Nif::NiStringExtraData*)e; if (sd->string == "NCO" && !raycastingOnly) { // No collision. Use an internal flag setting to mark this. // We ignor this node! flags |= 0x800; return; } else if (sd->string == "MRK" && !raycastingOnly) // Marker objects. These are only visible in the // editor. Until and unless we add an editor component to // the engine, just skip this entire node. return; } } //transfo of parents node + curent node Ogre::Matrix3 finalRot; Ogre::Vector3 finalPos; float finalScale; Nif::Transformation &final = *((Nif::Transformation*)node->trafo); Ogre::Vector3 nodePos = getVector(&final); Ogre::Matrix3 nodeRot = getMatrix(&final); finalPos = nodePos + parentPos; finalRot = parentRot*nodeRot; finalScale = final.scale*parentScale; // For NiNodes, loop through children if (node->recType == Nif::RC_NiNode) { Nif::NodeList &list = ((Nif::NiNode*)node)->children; int n = list.length(); for (int i=0; irecType == Nif::RC_NiTriShape && isCollisionNode) { cShape->collide = true; handleNiTriShape(dynamic_cast(node), flags,finalRot,finalPos,finalScale,raycastingOnly); } else if(node->recType == Nif::RC_RootCollisionNode) { Nif::NodeList &list = ((Nif::NiNode*)node)->children; int n = list.length(); for (int i=0; i