1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-19 20:53:50 +00:00
openmw-tes3mp/components/detournavigator/recastmeshmanager.cpp

134 lines
4.9 KiB
C++
Raw Normal View History

2018-03-13 22:49:08 +00:00
#include "recastmeshmanager.hpp"
#include "recastmeshbuilder.hpp"
2018-03-13 22:49:08 +00:00
namespace DetourNavigator
{
2019-11-27 22:45:01 +00:00
RecastMeshManager::RecastMeshManager(const Settings& settings, const TileBounds& bounds, std::size_t generation)
: mSettings(settings)
, mGeneration(generation)
, mTileBounds(bounds)
{
}
2018-03-13 22:49:08 +00:00
bool RecastMeshManager::addObject(const ObjectId id, const CollisionShape& shape, const btTransform& transform,
2018-07-18 19:09:50 +00:00
const AreaType areaType)
2018-03-13 22:49:08 +00:00
{
const std::lock_guard lock(mMutex);
const auto object = mObjects.lower_bound(id);
if (object != mObjects.end() && object->first == id)
return false;
const auto iterator = mObjectsOrder.emplace(mObjectsOrder.end(),
OscillatingRecastMeshObject(RecastMeshObject(shape, transform, areaType), mRevision + 1));
mObjects.emplace_hint(object, id, iterator);
2019-11-27 22:45:01 +00:00
++mRevision;
return true;
2018-03-13 22:49:08 +00:00
}
2018-09-22 15:36:57 +00:00
bool RecastMeshManager::updateObject(const ObjectId id, const btTransform& transform, const AreaType areaType)
2018-05-26 14:44:25 +00:00
{
const std::lock_guard lock(mMutex);
2018-05-26 14:44:25 +00:00
const auto object = mObjects.find(id);
if (object == mObjects.end())
return false;
const std::size_t lastChangeRevision = mLastNavMeshReportedChange.has_value()
? mLastNavMeshReportedChange->mRevision : mRevision;
if (!object->second->update(transform, areaType, lastChangeRevision, mTileBounds))
2018-05-26 14:44:25 +00:00
return false;
2019-11-27 22:45:01 +00:00
++mRevision;
return true;
2018-05-26 14:44:25 +00:00
}
std::optional<RemovedRecastMeshObject> RecastMeshManager::removeObject(const ObjectId id)
2018-03-13 22:49:08 +00:00
{
const std::lock_guard lock(mMutex);
const auto object = mObjects.find(id);
if (object == mObjects.end())
return std::nullopt;
const RemovedRecastMeshObject result {object->second->getImpl().getShape(), object->second->getImpl().getTransform()};
mObjectsOrder.erase(object->second);
mObjects.erase(object);
2019-11-27 22:45:01 +00:00
++mRevision;
return result;
2018-03-13 22:49:08 +00:00
}
2018-07-20 19:11:34 +00:00
bool RecastMeshManager::addWater(const osg::Vec2i& cellPosition, const int cellSize,
const btTransform& transform)
{
const std::lock_guard lock(mMutex);
const auto iterator = mWaterOrder.emplace(mWaterOrder.end(), Water {cellSize, transform});
if (!mWater.emplace(cellPosition, iterator).second)
{
mWaterOrder.erase(iterator);
2018-07-20 19:11:34 +00:00
return false;
}
2019-11-27 22:45:01 +00:00
++mRevision;
2018-07-20 19:11:34 +00:00
return true;
}
std::optional<RecastMeshManager::Water> RecastMeshManager::removeWater(const osg::Vec2i& cellPosition)
2018-07-20 19:11:34 +00:00
{
const std::lock_guard lock(mMutex);
2018-07-20 19:11:34 +00:00
const auto water = mWater.find(cellPosition);
if (water == mWater.end())
return std::nullopt;
2019-11-27 22:45:01 +00:00
++mRevision;
const auto result = *water->second;
mWaterOrder.erase(water->second);
2018-07-20 19:11:34 +00:00
mWater.erase(water);
return result;
}
2018-03-13 22:49:08 +00:00
std::shared_ptr<RecastMesh> RecastMeshManager::getMesh()
{
RecastMeshBuilder builder(mSettings, mTileBounds);
using Object = std::tuple<
osg::ref_ptr<const osg::Object>,
std::reference_wrapper<const btCollisionShape>,
btTransform,
AreaType
>;
std::vector<Object> objects;
std::size_t revision;
{
const std::lock_guard lock(mMutex);
for (const auto& v : mWaterOrder)
builder.addWater(v.mCellSize, v.mTransform);
objects.reserve(mObjectsOrder.size());
for (const auto& object : mObjectsOrder)
{
const RecastMeshObject& impl = object.getImpl();
objects.emplace_back(impl.getHolder(), impl.getShape(), impl.getTransform(), impl.getAreaType());
}
revision = mRevision;
}
for (const auto& [holder, shape, transform, areaType] : objects)
builder.addObject(shape, transform, areaType);
return std::move(builder).create(mGeneration, revision);
2018-03-13 22:49:08 +00:00
}
2018-04-15 22:07:18 +00:00
bool RecastMeshManager::isEmpty() const
{
const std::lock_guard lock(mMutex);
2018-04-15 22:07:18 +00:00
return mObjects.empty();
}
void RecastMeshManager::reportNavMeshChange(Version recastMeshVersion, Version navMeshVersion)
{
if (recastMeshVersion.mGeneration != mGeneration)
return;
const std::lock_guard lock(mMutex);
if (mLastNavMeshReport.has_value() && navMeshVersion < mLastNavMeshReport->mNavMeshVersion)
return;
mLastNavMeshReport = {recastMeshVersion.mRevision, navMeshVersion};
if (!mLastNavMeshReportedChange.has_value()
|| mLastNavMeshReportedChange->mNavMeshVersion < mLastNavMeshReport->mNavMeshVersion)
mLastNavMeshReportedChange = mLastNavMeshReport;
}
Version RecastMeshManager::getVersion() const
{
const std::lock_guard lock(mMutex);
return Version {mGeneration, mRevision};
}
2018-03-13 22:49:08 +00:00
}