2015-05-12 01:02:15 +00:00
|
|
|
#include "bulletshapemanager.hpp"
|
|
|
|
|
2021-11-10 19:24:17 +00:00
|
|
|
#include <cstring>
|
|
|
|
|
2016-03-13 23:32:32 +00:00
|
|
|
#include <osg/Drawable>
|
2015-11-16 23:18:27 +00:00
|
|
|
#include <osg/NodeVisitor>
|
2016-03-13 23:32:32 +00:00
|
|
|
#include <osg/Transform>
|
|
|
|
#include <osg/TriangleFunctor>
|
2015-11-16 23:18:27 +00:00
|
|
|
|
|
|
|
#include <BulletCollision/CollisionShapes/btTriangleMesh.h>
|
|
|
|
|
2021-11-10 19:24:17 +00:00
|
|
|
#include <components/misc/osguservalues.hpp>
|
2021-09-11 13:49:47 +00:00
|
|
|
#include <components/misc/pathhelpers.hpp>
|
2021-02-03 12:25:09 +00:00
|
|
|
#include <components/sceneutil/visitor.hpp>
|
2015-05-12 01:02:15 +00:00
|
|
|
#include <components/vfs/manager.hpp>
|
2023-05-31 21:11:03 +00:00
|
|
|
#include <components/vfs/pathutil.hpp>
|
2015-05-12 01:02:15 +00:00
|
|
|
|
|
|
|
#include <components/nifbullet/bulletnifloader.hpp>
|
|
|
|
|
2015-11-16 22:30:10 +00:00
|
|
|
#include "bulletshape.hpp"
|
2015-11-16 23:18:27 +00:00
|
|
|
#include "multiobjectcache.hpp"
|
2015-12-01 22:04:02 +00:00
|
|
|
#include "niffilemanager.hpp"
|
2016-02-05 22:59:37 +00:00
|
|
|
#include "objectcache.hpp"
|
2016-02-09 17:48:49 +00:00
|
|
|
#include "scenemanager.hpp"
|
2015-11-16 22:30:10 +00:00
|
|
|
|
|
|
|
namespace Resource
|
2015-05-12 01:02:15 +00:00
|
|
|
{
|
|
|
|
|
2015-11-16 23:18:27 +00:00
|
|
|
struct GetTriangleFunctor
|
|
|
|
{
|
|
|
|
GetTriangleFunctor()
|
2018-10-09 06:21:12 +00:00
|
|
|
: mTriMesh(nullptr)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2015-11-16 23:18:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void setTriMesh(btTriangleMesh* triMesh) { mTriMesh = triMesh; }
|
|
|
|
|
|
|
|
void setMatrix(const osg::Matrixf& matrix) { mMatrix = matrix; }
|
|
|
|
|
|
|
|
inline btVector3 toBullet(const osg::Vec3f& vec) { return btVector3(vec.x(), vec.y(), vec.z()); }
|
|
|
|
|
2021-10-13 14:12:47 +00:00
|
|
|
void inline operator()(const osg::Vec3& v1, const osg::Vec3& v2, const osg::Vec3& v3,
|
|
|
|
bool _temp = false) // Note: unused temp argument left here for OSG versions less than 3.5.6
|
2015-11-16 23:18:27 +00:00
|
|
|
{
|
|
|
|
if (mTriMesh)
|
|
|
|
mTriMesh->addTriangle(
|
|
|
|
toBullet(mMatrix.preMult(v1)), toBullet(mMatrix.preMult(v2)), toBullet(mMatrix.preMult(v3)));
|
|
|
|
}
|
|
|
|
|
|
|
|
btTriangleMesh* mTriMesh;
|
|
|
|
osg::Matrixf mMatrix;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Creates a BulletShape out of a Node hierarchy.
|
|
|
|
class NodeToShapeVisitor : public osg::NodeVisitor
|
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
public:
|
2015-11-16 23:18:27 +00:00
|
|
|
NodeToShapeVisitor()
|
|
|
|
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
|
2017-10-15 15:03:11 +00:00
|
|
|
, mTriangleMesh(nullptr)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
|
|
|
}
|
2015-11-16 23:18:27 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void apply(osg::Drawable& drawable) override
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2015-11-16 23:18:27 +00:00
|
|
|
if (!mTriangleMesh)
|
2017-10-15 15:03:11 +00:00
|
|
|
mTriangleMesh.reset(new btTriangleMesh);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-11-16 23:18:27 +00:00
|
|
|
osg::Matrixf worldMat = osg::computeLocalToWorld(getNodePath());
|
|
|
|
osg::TriangleFunctor<GetTriangleFunctor> functor;
|
2017-10-15 15:03:11 +00:00
|
|
|
functor.setTriMesh(mTriangleMesh.get());
|
2015-11-16 23:18:27 +00:00
|
|
|
functor.setMatrix(worldMat);
|
|
|
|
drawable.accept(functor);
|
|
|
|
}
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
osg::ref_ptr<BulletShape> getShape()
|
2015-11-16 23:18:27 +00:00
|
|
|
{
|
2023-03-28 20:54:26 +00:00
|
|
|
if (!mTriangleMesh || mTriangleMesh->getNumTriangles() == 0)
|
2015-11-17 00:51:21 +00:00
|
|
|
return osg::ref_ptr<BulletShape>();
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2017-10-15 15:03:11 +00:00
|
|
|
osg::ref_ptr<BulletShape> shape(new BulletShape);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2017-10-15 15:03:11 +00:00
|
|
|
auto triangleMeshShape = std::make_unique<TriangleMeshShape>(mTriangleMesh.release(), true);
|
|
|
|
btVector3 aabbMin = triangleMeshShape->getLocalAabbMin();
|
2015-11-16 23:18:27 +00:00
|
|
|
btVector3 aabbMax = triangleMeshShape->getLocalAabbMax();
|
2021-10-27 23:43:25 +00:00
|
|
|
shape->mCollisionBox.mExtents[0] = (aabbMax[0] - aabbMin[0]) / 2.0f;
|
|
|
|
shape->mCollisionBox.mExtents[1] = (aabbMax[1] - aabbMin[1]) / 2.0f;
|
|
|
|
shape->mCollisionBox.mExtents[2] = (aabbMax[2] - aabbMin[2]) / 2.0f;
|
2015-11-16 23:18:27 +00:00
|
|
|
shape->mCollisionBox.mCenter = osg::Vec3f(
|
|
|
|
(aabbMax[0] + aabbMin[0]) / 2.0f, (aabbMax[1] + aabbMin[1]) / 2.0f, (aabbMax[2] + aabbMin[2]) / 2.0f);
|
2017-10-15 15:03:11 +00:00
|
|
|
shape->mCollisionShape.reset(triangleMeshShape.release());
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-11-16 23:18:27 +00:00
|
|
|
return shape;
|
|
|
|
}
|
|
|
|
|
2015-11-17 00:51:21 +00:00
|
|
|
private:
|
2021-10-30 01:38:38 +00:00
|
|
|
std::unique_ptr<btTriangleMesh> mTriangleMesh;
|
2021-01-30 14:03:02 +00:00
|
|
|
};
|
|
|
|
|
2015-11-16 23:18:27 +00:00
|
|
|
BulletShapeManager::BulletShapeManager(
|
2023-09-09 17:29:26 +00:00
|
|
|
const VFS::Manager* vfs, SceneManager* sceneMgr, NifFileManager* nifFileManager, double expiryDelay)
|
|
|
|
: ResourceManager(vfs, expiryDelay)
|
2016-02-09 17:48:49 +00:00
|
|
|
, mInstanceCache(new MultiObjectCache)
|
2015-11-16 23:18:27 +00:00
|
|
|
, mSceneManager(sceneMgr)
|
2015-12-01 22:04:02 +00:00
|
|
|
, mNifFileManager(nifFileManager)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2015-11-16 23:18:27 +00:00
|
|
|
}
|
|
|
|
|
2015-12-01 22:04:02 +00:00
|
|
|
BulletShapeManager::~BulletShapeManager() {}
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2016-02-06 22:30:41 +00:00
|
|
|
osg::ref_ptr<const BulletShape> BulletShapeManager::getShape(const std::string& name)
|
2015-05-12 01:02:15 +00:00
|
|
|
{
|
2024-03-09 00:45:21 +00:00
|
|
|
const VFS::Path::Normalized normalized(name);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-11-16 23:18:27 +00:00
|
|
|
osg::ref_ptr<BulletShape> shape;
|
2018-07-08 19:22:34 +00:00
|
|
|
osg::ref_ptr<osg::Object> obj = mCache->getRefFromObjectCache(normalized);
|
|
|
|
if (obj)
|
|
|
|
shape = osg::ref_ptr<BulletShape>(static_cast<BulletShape*>(obj.get()));
|
2015-11-16 23:18:27 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (Misc::getFileExtension(normalized) == "nif")
|
2021-02-03 12:25:09 +00:00
|
|
|
{
|
2021-02-03 19:16:54 +00:00
|
|
|
NifBullet::BulletNifLoader loader;
|
2021-02-03 12:25:09 +00:00
|
|
|
shape = loader.load(*mNifFileManager->get(normalized));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-11-16 23:18:27 +00:00
|
|
|
// TODO: support .bullet shape files
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-11-16 23:18:27 +00:00
|
|
|
osg::ref_ptr<const osg::Node> constNode(mSceneManager->getTemplate(normalized));
|
|
|
|
osg::ref_ptr<osg::Node> node(const_cast<osg::Node*>(
|
|
|
|
constNode.get())); // const-trickery required because there is no const version of NodeVisitor
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2021-02-03 12:25:09 +00:00
|
|
|
// Check first if there's a custom collision node
|
2021-02-03 19:16:54 +00:00
|
|
|
unsigned int visitAllNodesMask = 0xffffffff;
|
2021-02-03 12:25:09 +00:00
|
|
|
SceneUtil::FindByNameVisitor nameFinder("Collision");
|
2021-02-03 19:16:54 +00:00
|
|
|
nameFinder.setTraversalMask(visitAllNodesMask);
|
|
|
|
nameFinder.setNodeMaskOverride(visitAllNodesMask);
|
2021-02-03 12:25:09 +00:00
|
|
|
node->accept(nameFinder);
|
|
|
|
if (nameFinder.mFoundNode)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2021-02-03 12:25:09 +00:00
|
|
|
NodeToShapeVisitor visitor;
|
2021-02-03 19:16:54 +00:00
|
|
|
visitor.setTraversalMask(visitAllNodesMask);
|
|
|
|
visitor.setNodeMaskOverride(visitAllNodesMask);
|
2021-02-03 12:25:09 +00:00
|
|
|
nameFinder.mFoundNode->accept(visitor);
|
|
|
|
shape = visitor.getShape();
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
|
|
|
|
2021-02-03 12:25:09 +00:00
|
|
|
// Generate a collision shape from the mesh
|
|
|
|
if (!shape)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2021-02-03 12:25:09 +00:00
|
|
|
NodeToShapeVisitor visitor;
|
|
|
|
node->accept(visitor);
|
2016-02-05 22:59:37 +00:00
|
|
|
shape = visitor.getShape();
|
2015-11-17 00:51:21 +00:00
|
|
|
if (!shape)
|
2021-02-03 12:25:09 +00:00
|
|
|
return osg::ref_ptr<BulletShape>();
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
|
|
|
|
2021-11-10 19:24:17 +00:00
|
|
|
if (shape != nullptr)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2021-11-10 19:24:17 +00:00
|
|
|
shape->mFileName = normalized;
|
2021-11-15 16:40:22 +00:00
|
|
|
constNode->getUserValue(Misc::OsgUserValues::sFileHash, shape->mFileHash);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2021-02-03 12:25:09 +00:00
|
|
|
}
|
2021-11-10 19:24:17 +00:00
|
|
|
|
|
|
|
mCache->addEntryToObjectCache(normalized, shape);
|
2015-11-16 23:18:27 +00:00
|
|
|
}
|
2016-02-05 22:59:37 +00:00
|
|
|
return shape;
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
2016-02-09 17:48:49 +00:00
|
|
|
osg::ref_ptr<BulletShapeInstance> BulletShapeManager::cacheInstance(const std::string& name)
|
|
|
|
{
|
2023-05-31 21:11:03 +00:00
|
|
|
const std::string normalized = VFS::Path::normalizeFilename(name);
|
2016-02-09 17:48:49 +00:00
|
|
|
|
|
|
|
osg::ref_ptr<BulletShapeInstance> instance = createInstance(normalized);
|
2017-10-15 15:01:18 +00:00
|
|
|
if (instance)
|
|
|
|
mInstanceCache->addEntryToObjectCache(normalized, instance.get());
|
2016-02-09 17:48:49 +00:00
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::ref_ptr<BulletShapeInstance> BulletShapeManager::getInstance(const std::string& name)
|
|
|
|
{
|
2023-05-31 21:11:03 +00:00
|
|
|
const std::string normalized = VFS::Path::normalizeFilename(name);
|
2016-02-09 17:48:49 +00:00
|
|
|
|
|
|
|
osg::ref_ptr<osg::Object> obj = mInstanceCache->takeFromObjectCache(normalized);
|
|
|
|
if (obj.get())
|
|
|
|
return static_cast<BulletShapeInstance*>(obj.get());
|
|
|
|
else
|
|
|
|
return createInstance(normalized);
|
|
|
|
}
|
|
|
|
|
2016-02-06 22:30:41 +00:00
|
|
|
osg::ref_ptr<BulletShapeInstance> BulletShapeManager::createInstance(const std::string& name)
|
|
|
|
{
|
|
|
|
osg::ref_ptr<const BulletShape> shape = getShape(name);
|
2016-02-05 22:59:37 +00:00
|
|
|
if (shape)
|
2021-10-30 01:16:21 +00:00
|
|
|
return makeInstance(std::move(shape));
|
|
|
|
return osg::ref_ptr<BulletShapeInstance>();
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
2016-02-09 17:48:49 +00:00
|
|
|
void BulletShapeManager::updateCache(double referenceTime)
|
|
|
|
{
|
|
|
|
ResourceManager::updateCache(referenceTime);
|
|
|
|
|
|
|
|
mInstanceCache->removeUnreferencedObjectsInCache();
|
|
|
|
}
|
|
|
|
|
2017-08-21 22:58:38 +00:00
|
|
|
void BulletShapeManager::clearCache()
|
|
|
|
{
|
|
|
|
ResourceManager::clearCache();
|
|
|
|
|
|
|
|
mInstanceCache->clear();
|
|
|
|
}
|
|
|
|
|
2017-03-07 03:02:06 +00:00
|
|
|
void BulletShapeManager::reportStats(unsigned int frameNumber, osg::Stats* stats) const
|
2017-02-22 01:18:18 +00:00
|
|
|
{
|
2023-12-21 23:23:49 +00:00
|
|
|
Resource::reportStats("Shape", frameNumber, mCache->getStats(), *stats);
|
|
|
|
Resource::reportStats("Shape Instance", frameNumber, mInstanceCache->getStats(), *stats);
|
2017-02-22 01:18:18 +00:00
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|