diff --git a/apps/openmw/mwworld/scene.cpp b/apps/openmw/mwworld/scene.cpp index 3aca0af325..73dc632525 100644 --- a/apps/openmw/mwworld/scene.cpp +++ b/apps/openmw/mwworld/scene.cpp @@ -3,6 +3,7 @@ #include #include +#include #include #include @@ -83,12 +84,9 @@ namespace if (const auto object = physics.getObject(ptr)) { - if (const auto concaveShape = dynamic_cast(object->getShapeInstance()->mCollisionShape)) - { - const auto navigator = MWBase::Environment::get().getWorld()->getNavigator(); - navigator->addObject(reinterpret_cast(object), *concaveShape, - object->getCollisionObject()->getWorldTransform()); - } + const auto navigator = MWBase::Environment::get().getWorld()->getNavigator(); + navigator->addObject(reinterpret_cast(object), *object->getCollisionObject()->getCollisionShape(), + object->getCollisionObject()->getWorldTransform()); } else if (const auto actor = physics.getActor(ptr)) { diff --git a/components/detournavigator/cachedrecastmeshmanager.cpp b/components/detournavigator/cachedrecastmeshmanager.cpp index 0fcaf116e0..60306ecc68 100644 --- a/components/detournavigator/cachedrecastmeshmanager.cpp +++ b/components/detournavigator/cachedrecastmeshmanager.cpp @@ -5,7 +5,14 @@ namespace DetourNavigator { CachedRecastMeshManager::CachedRecastMeshManager(const Settings& settings) : mImpl(settings) + {} + + bool CachedRecastMeshManager::addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform) { + if (!mImpl.addObject(id, shape, transform)) + return false; + mCached.reset(); + return true; } boost::optional CachedRecastMeshManager::removeObject(std::size_t id) @@ -21,5 +28,5 @@ namespace DetourNavigator if (!mCached) mCached = mImpl.getMesh(); return mCached; -} + } } diff --git a/components/detournavigator/cachedrecastmeshmanager.hpp b/components/detournavigator/cachedrecastmeshmanager.hpp index 4be6fb1343..6705a26f19 100644 --- a/components/detournavigator/cachedrecastmeshmanager.hpp +++ b/components/detournavigator/cachedrecastmeshmanager.hpp @@ -12,14 +12,7 @@ namespace DetourNavigator public: CachedRecastMeshManager(const Settings& settings); - template - bool addObject(std::size_t id, const T& shape, const btTransform& transform) - { - if (!mImpl.addObject(id, shape, transform)) - return false; - mCached.reset(); - return true; - } + bool addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform); boost::optional removeObject(std::size_t id); diff --git a/components/detournavigator/navigator.cpp b/components/detournavigator/navigator.cpp index c3ec41e962..fe1b931af3 100644 --- a/components/detournavigator/navigator.cpp +++ b/components/detournavigator/navigator.cpp @@ -25,6 +25,11 @@ namespace DetourNavigator mNavMeshManager.reset(agentHalfExtents); } + bool Navigator::addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform) + { + return mNavMeshManager.addObject(id, shape, transform); + } + bool Navigator::removeObject(std::size_t id) { return mNavMeshManager.removeObject(id); diff --git a/components/detournavigator/navigator.hpp b/components/detournavigator/navigator.hpp index e48d630af0..69db211b17 100644 --- a/components/detournavigator/navigator.hpp +++ b/components/detournavigator/navigator.hpp @@ -17,11 +17,7 @@ namespace DetourNavigator void removeAgent(const osg::Vec3f& agentHalfExtents); - template - bool addObject(std::size_t id, const T& shape, const btTransform& transform) - { - return mNavMeshManager.addObject(id, shape, transform); - } + bool addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform); bool removeObject(std::size_t id); diff --git a/components/detournavigator/navmeshmanager.cpp b/components/detournavigator/navmeshmanager.cpp index 89d5e0809d..e2aa148795 100644 --- a/components/detournavigator/navmeshmanager.cpp +++ b/components/detournavigator/navmeshmanager.cpp @@ -19,6 +19,15 @@ namespace DetourNavigator , mAsyncNavMeshUpdater(settings) {} + bool NavMeshManager::addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform) + { + if (!mRecastMeshManager.addObject(id, shape, transform)) + return false; + ++mRevision; + addChangedTiles(shape, transform); + return true; + } + bool NavMeshManager::removeObject(std::size_t id) { const auto object = mRecastMeshManager.removeObject(id); diff --git a/components/detournavigator/navmeshmanager.hpp b/components/detournavigator/navmeshmanager.hpp index 2736140f19..f3285a8db8 100644 --- a/components/detournavigator/navmeshmanager.hpp +++ b/components/detournavigator/navmeshmanager.hpp @@ -23,15 +23,7 @@ namespace DetourNavigator public: NavMeshManager(const Settings& settings); - template - bool addObject(std::size_t id, const T& shape, const btTransform& transform) - { - if (!mRecastMeshManager.addObject(id, shape, transform)) - return false; - ++mRevision; - addChangedTiles(shape, transform); - return true; - } + bool addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform); bool removeObject(std::size_t id); diff --git a/components/detournavigator/recastmeshbuilder.cpp b/components/detournavigator/recastmeshbuilder.cpp index ba9b8a3de8..eb44dbcd52 100644 --- a/components/detournavigator/recastmeshbuilder.cpp +++ b/components/detournavigator/recastmeshbuilder.cpp @@ -1,12 +1,15 @@ #include "recastmeshbuilder.hpp" #include "chunkytrimesh.hpp" +#include "debug.hpp" #include "settings.hpp" #include "settingsutils.hpp" +#include "exceptions.hpp" #include -#include +#include #include +#include namespace { @@ -22,7 +25,25 @@ namespace DetourNavigator RecastMeshBuilder::RecastMeshBuilder(const Settings& settings) : mSettings(settings) + {} + + void RecastMeshBuilder::addObject(const btCollisionShape& shape, const btTransform& transform) + { + if (shape.isCompound()) + return addObject(static_cast(shape), transform); + else if (shape.getShapeType() == TERRAIN_SHAPE_PROXYTYPE) + return addObject(static_cast(shape), transform); + else if (shape.isConcave()) + return addObject(static_cast(shape), transform); + std::ostringstream message; + message << "Unsupported shape type: " << BroadphaseNativeTypes(shape.getShapeType()); + throw InvalidArgument(message.str()); + } + + void RecastMeshBuilder::addObject(const btCompoundShape& shape, const btTransform& transform) { + for (int i = 0, num = shape.getNumChildShapes(); i < num; ++i) + addObject(*shape.getChildShape(i), transform * shape.getChildTransform(i)); } void RecastMeshBuilder::addObject(const btConcaveShape& shape, const btTransform& transform) diff --git a/components/detournavigator/recastmeshbuilder.hpp b/components/detournavigator/recastmeshbuilder.hpp index 3236b63982..5010a95e0d 100644 --- a/components/detournavigator/recastmeshbuilder.hpp +++ b/components/detournavigator/recastmeshbuilder.hpp @@ -3,6 +3,8 @@ #include "recastmesh.hpp" +class btCollisionShape; +class btCompoundShape; class btConcaveShape; class btHeightfieldTerrainShape; class btTransform; @@ -16,6 +18,10 @@ namespace DetourNavigator public: RecastMeshBuilder(const Settings& settings); + void addObject(const btCollisionShape& shape, const btTransform& transform); + + void addObject(const btCompoundShape& shape, const btTransform& transform); + void addObject(const btConcaveShape& shape, const btTransform& transform); void addObject(const btHeightfieldTerrainShape& shape, const btTransform& transform); diff --git a/components/detournavigator/recastmeshmanager.cpp b/components/detournavigator/recastmeshmanager.cpp index 8568c6e148..1c32ac1e77 100644 --- a/components/detournavigator/recastmeshmanager.cpp +++ b/components/detournavigator/recastmeshmanager.cpp @@ -9,15 +9,7 @@ namespace DetourNavigator , mMeshBuilder(settings) {} - bool RecastMeshManager::addObject(std::size_t id, const btHeightfieldTerrainShape& shape, const btTransform& transform) - { - if (!mObjects.insert(std::make_pair(id, Object {&shape, transform})).second) - return false; - mShouldRebuild = true; - return true; - } - - bool RecastMeshManager::addObject(std::size_t id, const btConcaveShape& shape, const btTransform& transform) + bool RecastMeshManager::addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform) { if (!mObjects.insert(std::make_pair(id, Object {&shape, transform})).second) return false; diff --git a/components/detournavigator/recastmeshmanager.hpp b/components/detournavigator/recastmeshmanager.hpp index b4a57ffe0b..429430707b 100644 --- a/components/detournavigator/recastmeshmanager.hpp +++ b/components/detournavigator/recastmeshmanager.hpp @@ -24,9 +24,7 @@ namespace DetourNavigator RecastMeshManager(const Settings& settings); - bool addObject(std::size_t id, const btHeightfieldTerrainShape& shape, const btTransform& transform); - - bool addObject(std::size_t id, const btConcaveShape& shape, const btTransform& transform); + bool addObject(std::size_t id, const btCollisionShape& shape, const btTransform& transform); boost::optional removeObject(std::size_t id);