mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 08:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			182 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			182 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifndef OPENMW_BULLET_SHAPE_LOADER_H_
 | |
| #define OPENMW_BULLET_SHAPE_LOADER_H_
 | |
| 
 | |
| #include <OgreResource.h>
 | |
| #include <OgreResourceManager.h>
 | |
| #include <btBulletCollisionCommon.h>
 | |
| #include <OgreVector3.h>
 | |
| 
 | |
| namespace OEngine {
 | |
| namespace Physic
 | |
| {
 | |
| 
 | |
| /**
 | |
| *Define a new resource which describe a Shape usable by bullet.See BulletShapeManager for how to get/use them.
 | |
| */
 | |
| class BulletShape : public Ogre::Resource
 | |
| {
 | |
| protected:
 | |
|     void loadImpl();
 | |
|     void unloadImpl();
 | |
|     size_t calculateSize() const;
 | |
| 
 | |
|     void deleteShape(btCollisionShape* shape);
 | |
| 
 | |
| public:
 | |
| 
 | |
|     BulletShape(Ogre::ResourceManager *creator, const Ogre::String &name,
 | |
|         Ogre::ResourceHandle handle, const Ogre::String &group, bool isManual = false,
 | |
|         Ogre::ManualResourceLoader *loader = 0);
 | |
| 
 | |
|     virtual ~BulletShape();
 | |
| 
 | |
|     // Stores animated collision shapes. If any collision nodes in the NIF are animated, then mCollisionShape
 | |
|     // will be a btCompoundShape (which consists of one or more child shapes).
 | |
|     // In this map, for each animated collision shape,
 | |
|     // we store the bone name mapped to the child index of the shape in the btCompoundShape.
 | |
|     std::map<std::string, int> mAnimatedShapes;
 | |
| 
 | |
|     std::map<std::string, int> mAnimatedRaycastingShapes;
 | |
| 
 | |
|     btCollisionShape* mCollisionShape;
 | |
|     btCollisionShape* mRaycastingShape;
 | |
| 
 | |
|     // Whether or not a NiRootCollisionNode was present in the .nif. If there is none, the collision behaviour
 | |
|     // depends on object type, so we need to expose this variable.
 | |
|     bool mHasCollisionNode;
 | |
| 
 | |
|     Ogre::Vector3 mBoxTranslation;
 | |
|     Ogre::Quaternion mBoxRotation;
 | |
|     //this flag indicate if the shape is used for collision or if it's for raycasting only.
 | |
|     bool mCollide;
 | |
| };
 | |
| 
 | |
| /**
 | |
| *
 | |
| */
 | |
| 
 | |
| #if (OGRE_VERSION < ((1 << 16) | (9 << 8) | 0))
 | |
| class BulletShapePtr : public Ogre::SharedPtr<BulletShape>
 | |
| {
 | |
| public:
 | |
|     BulletShapePtr() : Ogre::SharedPtr<BulletShape>() {}
 | |
|     explicit BulletShapePtr(BulletShape *rep) : Ogre::SharedPtr<BulletShape>(rep) {}
 | |
|     BulletShapePtr(const BulletShapePtr &r) : Ogre::SharedPtr<BulletShape>(r) {}
 | |
|     BulletShapePtr(const Ogre::ResourcePtr &r) : Ogre::SharedPtr<BulletShape>()
 | |
|     {
 | |
|         if( r.isNull() )
 | |
|             return;
 | |
|         // lock & copy other mutex pointer
 | |
|         OGRE_LOCK_MUTEX(*r.OGRE_AUTO_MUTEX_NAME)
 | |
|             OGRE_COPY_AUTO_SHARED_MUTEX(r.OGRE_AUTO_MUTEX_NAME)
 | |
|             pRep = static_cast<BulletShape*>(r.getPointer());
 | |
|         pUseCount = r.useCountPointer();
 | |
|         useFreeMethod = r.freeMethod();
 | |
|         if (pUseCount)
 | |
|         {
 | |
|             ++(*pUseCount);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /// Operator used to convert a ResourcePtr to a BulletShapePtr
 | |
|     BulletShapePtr& operator=(const Ogre::ResourcePtr& r)
 | |
|     {
 | |
|         if(pRep == static_cast<BulletShape*>(r.getPointer()))
 | |
|             return *this;
 | |
|         release();
 | |
|         if( r.isNull() )
 | |
|             return *this; // resource ptr is null, so the call to release above has done all we need to do.
 | |
|         // lock & copy other mutex pointer
 | |
|         OGRE_LOCK_MUTEX(*r.OGRE_AUTO_MUTEX_NAME)
 | |
|             OGRE_COPY_AUTO_SHARED_MUTEX(r.OGRE_AUTO_MUTEX_NAME)
 | |
|             pRep = static_cast<BulletShape*>(r.getPointer());
 | |
|         pUseCount = r.useCountPointer();
 | |
|         useFreeMethod = r.freeMethod();
 | |
|         if (pUseCount)
 | |
|         {
 | |
|             ++(*pUseCount);
 | |
|         }
 | |
|         return *this;
 | |
|     }
 | |
| };
 | |
| #else
 | |
| typedef Ogre::SharedPtr<BulletShape> BulletShapePtr;
 | |
| #endif
 | |
| 
 | |
| /**
 | |
| *Hold any BulletShape that was created by the ManualBulletShapeLoader.
 | |
| *
 | |
| *To get a bulletShape, you must load it first.
 | |
| *First, create a manualBulletShapeLoader. Then call ManualBulletShapeManager->load(). This create an "empty" resource.
 | |
| *Then use BulletShapeManager->load(). This will fill the resource with the required info.
 | |
| *To get the resource,use BulletShapeManager::getByName.
 | |
| *When you use the resource no more, just use BulletShapeManager->unload(). It won't completly delete the resource, but it will
 | |
| *"empty" it.This allow a better management of memory: when you are leaving a cell, just unload every useless shape.
 | |
| *
 | |
| *Alternatively, you can call BulletShape->load() in order to actually load the resource.
 | |
| *When you are finished with it, just call BulletShape->unload().
 | |
| *
 | |
| *IMO: prefere the first methode, i am not completly sure about the 2nd.
 | |
| *
 | |
| *Important Note: i have no idea of what happen if you try to load two time the same resource without unloading.
 | |
| *It won't crash, but it might lead to memory leaks(I don't know how Ogre handle this). So don't do it!
 | |
| */
 | |
| class BulletShapeManager : public Ogre::ResourceManager
 | |
| {
 | |
| protected:
 | |
| 
 | |
|     // must implement this from ResourceManager's interface
 | |
|     Ogre::Resource *createImpl(const Ogre::String &name, Ogre::ResourceHandle handle,
 | |
|         const Ogre::String &group, bool isManual, Ogre::ManualResourceLoader *loader,
 | |
|         const Ogre::NameValuePairList *createParams);
 | |
| 
 | |
|     static BulletShapeManager *sThis;
 | |
| 
 | |
| private:
 | |
|     /** \brief Explicit private copy constructor. This is a forbidden operation.*/
 | |
|     BulletShapeManager(const BulletShapeManager &);
 | |
| 
 | |
|     /** \brief Private operator= . This is a forbidden operation. */
 | |
|     BulletShapeManager& operator=(const BulletShapeManager &);
 | |
| 
 | |
| 
 | |
| public:
 | |
| 
 | |
|     BulletShapeManager();
 | |
|     virtual ~BulletShapeManager();
 | |
| 
 | |
| 
 | |
| #if (OGRE_VERSION >= ((1 << 16) | (9 << 8) | 0))
 | |
|     /// Get a resource by name
 | |
|     /// @see ResourceManager::getByName
 | |
|     BulletShapePtr getByName(const Ogre::String& name, const Ogre::String& groupName = Ogre::ResourceGroupManager::AUTODETECT_RESOURCE_GROUP_NAME);
 | |
| 
 | |
|     /// Create a new shape
 | |
|     /// @see ResourceManager::createResource
 | |
|     BulletShapePtr create (const Ogre::String& name, const Ogre::String& group,
 | |
|                         bool isManual = false, Ogre::ManualResourceLoader* loader = 0,
 | |
|                         const Ogre::NameValuePairList* createParams = 0);
 | |
| #endif
 | |
| 
 | |
|     virtual BulletShapePtr load(const Ogre::String &name, const Ogre::String &group);
 | |
| 
 | |
|     static BulletShapeManager &getSingleton();
 | |
|     static BulletShapeManager *getSingletonPtr();
 | |
| };
 | |
| 
 | |
| class BulletShapeLoader : public Ogre::ManualResourceLoader
 | |
| {
 | |
| public:
 | |
| 
 | |
|     BulletShapeLoader(){};
 | |
|     virtual ~BulletShapeLoader() {}
 | |
| 
 | |
|     virtual void loadResource(Ogre::Resource *resource);
 | |
| 
 | |
|     virtual void load(const std::string &name,const std::string &group);
 | |
| };
 | |
| 
 | |
| }
 | |
| }
 | |
| 
 | |
| #endif
 |