1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-28 14:53:58 +00:00
OpenMW/components/detournavigator/recastmeshmanager.cpp

153 lines
5.3 KiB
C++
Raw Normal View History

2018-03-14 01:49:08 +03:00
#include "recastmeshmanager.hpp"
#include "recastmeshbuilder.hpp"
#include "settings.hpp"
#include "heightfieldshape.hpp"
#include <components/debug/debuglog.hpp>
#include <utility>
namespace
{
struct AddHeightfield
{
const DetourNavigator::Cell& mCell;
DetourNavigator::RecastMeshBuilder& mBuilder;
void operator()(const DetourNavigator::HeightfieldSurface& v)
{
mBuilder.addHeightfield(mCell.mSize, mCell.mShift, v.mHeights, v.mSize, v.mMinHeight, v.mMaxHeight);
}
void operator()(DetourNavigator::HeightfieldPlane v)
{
mBuilder.addHeightfield(mCell.mSize, mCell.mShift, v.mHeight);
}
};
}
2018-03-14 01:49:08 +03:00
namespace DetourNavigator
{
2019-11-27 23:45:01 +01:00
RecastMeshManager::RecastMeshManager(const Settings& settings, const TileBounds& bounds, std::size_t generation)
: mSettings(settings)
, mGeneration(generation)
, mTileBounds(bounds)
{
}
2018-03-14 01:49:08 +03:00
bool RecastMeshManager::addObject(const ObjectId id, const CollisionShape& shape, const btTransform& transform,
2018-07-18 22:09:50 +03:00
const AreaType areaType)
2018-03-14 01:49:08 +03:00
{
const auto object = mObjects.lower_bound(id);
if (object != mObjects.end() && object->first == id)
return false;
mObjects.emplace_hint(object, id,
OscillatingRecastMeshObject(RecastMeshObject(shape, transform, areaType), mRevision + 1));
2019-11-27 23:45:01 +01:00
++mRevision;
return true;
2018-03-14 01:49:08 +03:00
}
2018-09-22 18:36:57 +03:00
bool RecastMeshManager::updateObject(const ObjectId id, const btTransform& transform, const AreaType areaType)
2018-05-26 17:44:25 +03: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 17:44:25 +03:00
return false;
2019-11-27 23:45:01 +01:00
++mRevision;
return true;
2018-05-26 17:44:25 +03:00
}
std::optional<RemovedRecastMeshObject> RecastMeshManager::removeObject(const ObjectId id)
2018-03-14 01:49:08 +03:00
{
const auto object = mObjects.find(id);
if (object == mObjects.end())
return std::nullopt;
const RemovedRecastMeshObject result {object->second.getImpl().getShape(), object->second.getImpl().getTransform()};
mObjects.erase(object);
2019-11-27 23:45:01 +01:00
++mRevision;
return result;
2018-03-14 01:49:08 +03:00
}
bool RecastMeshManager::addWater(const osg::Vec2i& cellPosition, const int cellSize, const osg::Vec3f& shift)
2018-07-20 22:11:34 +03:00
{
2021-07-14 22:00:16 +02:00
if (!mWater.emplace(cellPosition, Cell {cellSize, shift}).second)
2018-07-20 22:11:34 +03:00
return false;
2019-11-27 23:45:01 +01:00
++mRevision;
2018-07-20 22:11:34 +03:00
return true;
}
2021-07-14 22:00:16 +02:00
std::optional<Cell> RecastMeshManager::removeWater(const osg::Vec2i& cellPosition)
2018-07-20 22:11:34 +03:00
{
const auto water = mWater.find(cellPosition);
if (water == mWater.end())
return std::nullopt;
2019-11-27 23:45:01 +01:00
++mRevision;
2021-07-14 22:00:16 +02:00
const Cell result = water->second;
2018-07-20 22:11:34 +03:00
mWater.erase(water);
return result;
}
bool RecastMeshManager::addHeightfield(const osg::Vec2i& cellPosition, int cellSize, const osg::Vec3f& shift,
const HeightfieldShape& shape)
{
if (!mHeightfields.emplace(cellPosition, Heightfield {Cell {cellSize, shift}, shape}).second)
return false;
++mRevision;
return true;
}
std::optional<Cell> RecastMeshManager::removeHeightfield(const osg::Vec2i& cellPosition)
{
const auto it = mHeightfields.find(cellPosition);
if (it == mHeightfields.end())
return std::nullopt;
++mRevision;
const auto result = std::make_optional(it->second.mCell);
mHeightfields.erase(it);
return result;
}
2018-03-14 01:49:08 +03:00
std::shared_ptr<RecastMesh> RecastMeshManager::getMesh()
{
TileBounds tileBounds = mTileBounds;
tileBounds.mMin /= mSettings.mRecastScaleFactor;
tileBounds.mMax /= mSettings.mRecastScaleFactor;
RecastMeshBuilder builder(tileBounds);
for (const auto& [k, v] : mWater)
2021-07-14 22:00:16 +02:00
builder.addWater(v.mSize, v.mShift);
for (const auto& [k, object] : mObjects)
{
const RecastMeshObject& v = object.getImpl();
builder.addObject(v.getShape(), v.getTransform(), v.getAreaType());
}
for (const auto& [cellPosition, v] : mHeightfields)
std::visit(AddHeightfield {v.mCell, builder}, v.mShape);
return std::move(builder).create(mGeneration, mRevision);
2018-03-14 01:49:08 +03:00
}
2018-04-16 01:07:18 +03:00
bool RecastMeshManager::isEmpty() const
{
return mObjects.empty() && mWater.empty() && mHeightfields.empty();
2018-04-16 01:07:18 +03:00
}
2021-06-25 21:52:02 +02:00
void RecastMeshManager::reportNavMeshChange(const Version& recastMeshVersion, const Version& navMeshVersion)
{
if (recastMeshVersion.mGeneration != mGeneration)
return;
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
{
return Version {mGeneration, mRevision};
}
2018-03-14 01:49:08 +03:00
}