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

229 lines
8.1 KiB
C++
Raw Normal View History

2019-02-16 12:27:02 +00:00
#include "navigatorimpl.hpp"
2018-03-13 22:49:08 +00:00
#include "debug.hpp"
#include "settingsutils.hpp"
#include <components/esm/loadpgrd.hpp>
#include <components/misc/coordinateconverter.hpp>
2018-03-13 22:49:08 +00:00
namespace DetourNavigator
{
2019-02-16 12:27:02 +00:00
NavigatorImpl::NavigatorImpl(const Settings& settings)
2018-03-13 22:49:08 +00:00
: mSettings(settings)
, mNavMeshManager(mSettings)
, mUpdatesEnabled(true)
2018-03-13 22:49:08 +00:00
{
}
2019-02-16 12:27:02 +00:00
void NavigatorImpl::addAgent(const osg::Vec3f& agentHalfExtents)
2018-03-13 22:49:08 +00:00
{
2021-03-07 19:58:09 +00:00
if(agentHalfExtents.length2() <= 0)
return;
2018-03-13 22:49:08 +00:00
++mAgents[agentHalfExtents];
mNavMeshManager.addAgent(agentHalfExtents);
2018-03-13 22:49:08 +00:00
}
2019-02-16 12:27:02 +00:00
void NavigatorImpl::removeAgent(const osg::Vec3f& agentHalfExtents)
2018-03-13 22:49:08 +00:00
{
const auto it = mAgents.find(agentHalfExtents);
if (it == mAgents.end())
2018-03-13 22:49:08 +00:00
return;
if (it->second > 0)
--it->second;
2018-03-13 22:49:08 +00:00
}
bool NavigatorImpl::addObject(const ObjectId id, const osg::ref_ptr<const osg::Object>& holder,
const btHeightfieldTerrainShape& shape, const btTransform& transform)
2018-04-02 21:04:19 +00:00
{
const CollisionShape collisionShape {holder, shape};
return mNavMeshManager.addObject(id, collisionShape, transform, AreaType_ground);
2018-07-12 08:44:11 +00:00
}
2019-02-16 12:27:02 +00:00
bool NavigatorImpl::addObject(const ObjectId id, const ObjectShapes& shapes, const btTransform& transform)
2018-07-12 08:44:11 +00:00
{
const CollisionShape collisionShape {shapes.mShapeInstance, *shapes.mShapeInstance->getCollisionShape()};
bool result = mNavMeshManager.addObject(id, collisionShape, transform, AreaType_ground);
if (const btCollisionShape* const avoidShape = shapes.mShapeInstance->getAvoidCollisionShape())
2018-07-12 08:44:11 +00:00
{
const ObjectId avoidId(avoidShape);
const CollisionShape collisionShape {shapes.mShapeInstance, *avoidShape};
if (mNavMeshManager.addObject(avoidId, collisionShape, transform, AreaType_null))
2018-07-12 08:44:11 +00:00
{
updateAvoidShapeId(id, avoidId);
result = true;
}
}
return result;
2018-04-02 21:04:19 +00:00
}
2019-02-16 12:27:02 +00:00
bool NavigatorImpl::addObject(const ObjectId id, const DoorShapes& shapes, const btTransform& transform)
{
if (addObject(id, static_cast<const ObjectShapes&>(shapes), transform))
{
const osg::Vec3f start = toNavMeshCoordinates(mSettings, shapes.mConnectionStart);
const osg::Vec3f end = toNavMeshCoordinates(mSettings, shapes.mConnectionEnd);
mNavMeshManager.addOffMeshConnection(id, start, end, AreaType_door);
mNavMeshManager.addOffMeshConnection(id, end, start, AreaType_door);
return true;
}
return false;
}
2019-02-16 12:27:02 +00:00
bool NavigatorImpl::updateObject(const ObjectId id, const ObjectShapes& shapes, const btTransform& transform)
2018-07-12 08:44:11 +00:00
{
const CollisionShape collisionShape {shapes.mShapeInstance, *shapes.mShapeInstance->getCollisionShape()};
bool result = mNavMeshManager.updateObject(id, collisionShape, transform, AreaType_ground);
if (const btCollisionShape* const avoidShape = shapes.mShapeInstance->getAvoidCollisionShape())
2018-07-12 08:44:11 +00:00
{
const ObjectId avoidId(avoidShape);
const CollisionShape collisionShape {shapes.mShapeInstance, *avoidShape};
if (mNavMeshManager.updateObject(avoidId, collisionShape, transform, AreaType_null))
2018-07-12 08:44:11 +00:00
{
updateAvoidShapeId(id, avoidId);
result = true;
}
}
return result;
2018-05-26 14:44:25 +00:00
}
2019-02-16 12:27:02 +00:00
bool NavigatorImpl::updateObject(const ObjectId id, const DoorShapes& shapes, const btTransform& transform)
{
return updateObject(id, static_cast<const ObjectShapes&>(shapes), transform);
}
2019-02-16 12:27:02 +00:00
bool NavigatorImpl::removeObject(const ObjectId id)
2018-03-13 22:49:08 +00:00
{
2018-07-12 08:44:11 +00:00
bool result = mNavMeshManager.removeObject(id);
const auto avoid = mAvoidIds.find(id);
2018-07-20 19:11:34 +00:00
if (avoid != mAvoidIds.end())
result = mNavMeshManager.removeObject(avoid->second) || result;
const auto water = mWaterIds.find(id);
if (water != mWaterIds.end())
result = mNavMeshManager.removeObject(water->second) || result;
mNavMeshManager.removeOffMeshConnections(id);
2018-07-20 19:11:34 +00:00
return result;
}
2019-02-16 12:27:02 +00:00
bool NavigatorImpl::addWater(const osg::Vec2i& cellPosition, const int cellSize, const btScalar level,
2018-07-20 19:11:34 +00:00
const btTransform& transform)
{
return mNavMeshManager.addWater(cellPosition, cellSize,
btTransform(transform.getBasis(), btVector3(transform.getOrigin().x(), transform.getOrigin().y(), level)));
}
2019-02-16 12:27:02 +00:00
bool NavigatorImpl::removeWater(const osg::Vec2i& cellPosition)
2018-07-20 19:11:34 +00:00
{
return mNavMeshManager.removeWater(cellPosition);
2018-03-13 22:49:08 +00:00
}
void NavigatorImpl::addPathgrid(const ESM::Cell& cell, const ESM::Pathgrid& pathgrid)
{
Misc::CoordinateConverter converter(&cell);
for (auto edge : pathgrid.mEdges)
{
const auto src = Misc::Convert::makeOsgVec3f(converter.toWorldPoint(pathgrid.mPoints[edge.mV0]));
const auto dst = Misc::Convert::makeOsgVec3f(converter.toWorldPoint(pathgrid.mPoints[edge.mV1]));
mNavMeshManager.addOffMeshConnection(
ObjectId(&pathgrid),
toNavMeshCoordinates(mSettings, src),
toNavMeshCoordinates(mSettings, dst),
AreaType_pathgrid
);
}
}
void NavigatorImpl::removePathgrid(const ESM::Pathgrid& pathgrid)
{
mNavMeshManager.removeOffMeshConnections(ObjectId(&pathgrid));
}
2019-02-16 12:27:02 +00:00
void NavigatorImpl::update(const osg::Vec3f& playerPosition)
2018-03-13 22:49:08 +00:00
{
if (!mUpdatesEnabled)
return;
removeUnusedNavMeshes();
2018-03-13 22:49:08 +00:00
for (const auto& v : mAgents)
mNavMeshManager.update(playerPosition, v.first);
2018-03-13 22:49:08 +00:00
}
void NavigatorImpl::updatePlayerPosition(const osg::Vec3f& playerPosition)
{
const TilePosition tilePosition = getTilePosition(mSettings, toNavMeshCoordinates(mSettings, playerPosition));
if (mLastPlayerPosition.has_value() && *mLastPlayerPosition == tilePosition)
return;
update(playerPosition);
mLastPlayerPosition = tilePosition;
}
void NavigatorImpl::setUpdatesEnabled(bool enabled)
{
mUpdatesEnabled = enabled;
}
void NavigatorImpl::wait(Loading::Listener& listener, WaitConditionType waitConditionType)
2018-03-13 22:49:08 +00:00
{
mNavMeshManager.wait(listener, waitConditionType);
2018-03-13 22:49:08 +00:00
}
2019-02-16 12:27:02 +00:00
SharedNavMeshCacheItem NavigatorImpl::getNavMesh(const osg::Vec3f& agentHalfExtents) const
{
return mNavMeshManager.getNavMesh(agentHalfExtents);
}
2019-02-16 12:27:02 +00:00
std::map<osg::Vec3f, SharedNavMeshCacheItem> NavigatorImpl::getNavMeshes() const
{
return mNavMeshManager.getNavMeshes();
}
const Settings& NavigatorImpl::getSettings() const
{
return mSettings;
}
2018-07-12 08:44:11 +00:00
2019-03-17 17:18:53 +00:00
void NavigatorImpl::reportStats(unsigned int frameNumber, osg::Stats& stats) const
{
mNavMeshManager.reportStats(frameNumber, stats);
}
2019-11-27 22:45:01 +00:00
RecastMeshTiles NavigatorImpl::getRecastMeshTiles()
{
return mNavMeshManager.getRecastMeshTiles();
}
2019-02-16 12:27:02 +00:00
void NavigatorImpl::updateAvoidShapeId(const ObjectId id, const ObjectId avoidId)
2018-07-20 19:11:34 +00:00
{
updateId(id, avoidId, mWaterIds);
}
2019-02-16 12:27:02 +00:00
void NavigatorImpl::updateWaterShapeId(const ObjectId id, const ObjectId waterId)
2018-07-20 19:11:34 +00:00
{
updateId(id, waterId, mWaterIds);
}
2019-02-16 12:27:02 +00:00
void NavigatorImpl::updateId(const ObjectId id, const ObjectId updateId, std::unordered_map<ObjectId, ObjectId>& ids)
2018-07-12 08:44:11 +00:00
{
2018-07-20 19:11:34 +00:00
auto inserted = ids.insert(std::make_pair(id, updateId));
2018-07-12 08:44:11 +00:00
if (!inserted.second)
{
mNavMeshManager.removeObject(inserted.first->second);
2018-09-22 15:36:57 +00:00
inserted.first->second = updateId;
2018-07-12 08:44:11 +00:00
}
}
void NavigatorImpl::removeUnusedNavMeshes()
{
for (auto it = mAgents.begin(); it != mAgents.end();)
{
if (it->second == 0 && mNavMeshManager.reset(it->first))
it = mAgents.erase(it);
else
++it;
}
}
float NavigatorImpl::getMaxNavmeshAreaRealRadius() const
{
const auto& settings = getSettings();
return getRealTileSize(settings) * getMaxNavmeshAreaRadius(settings);
}
2018-03-13 22:49:08 +00:00
}