2018-04-16 19:57:35 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHCACHEITEM_H
|
|
|
|
#define OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHCACHEITEM_H
|
|
|
|
|
|
|
|
#include "sharednavmesh.hpp"
|
2018-09-30 22:33:25 +00:00
|
|
|
#include "tileposition.hpp"
|
|
|
|
#include "navmeshtilescache.hpp"
|
2019-03-10 12:48:12 +00:00
|
|
|
#include "dtstatus.hpp"
|
2021-07-13 22:03:10 +00:00
|
|
|
#include "navmeshdata.hpp"
|
2021-11-01 19:21:27 +00:00
|
|
|
#include "version.hpp"
|
2018-04-16 19:57:35 +00:00
|
|
|
|
2018-09-30 22:33:25 +00:00
|
|
|
#include <components/misc/guarded.hpp>
|
|
|
|
|
|
|
|
#include <map>
|
2021-08-07 09:48:37 +00:00
|
|
|
#include <ostream>
|
2021-08-05 22:05:09 +00:00
|
|
|
#include <set>
|
2018-04-16 19:57:35 +00:00
|
|
|
|
2021-07-13 22:03:10 +00:00
|
|
|
struct dtMeshTile;
|
|
|
|
|
2018-04-16 19:57:35 +00:00
|
|
|
namespace DetourNavigator
|
|
|
|
{
|
2019-03-10 12:48:12 +00:00
|
|
|
enum class UpdateNavMeshStatus : unsigned
|
|
|
|
{
|
|
|
|
ignored = 0,
|
|
|
|
removed = 1 << 0,
|
|
|
|
added = 1 << 1,
|
|
|
|
replaced = removed | added,
|
|
|
|
failed = 1 << 2,
|
|
|
|
lost = removed | failed,
|
2020-10-07 22:04:51 +00:00
|
|
|
cached = 1 << 3,
|
|
|
|
unchanged = replaced | cached,
|
|
|
|
restored = added | cached,
|
2019-03-10 12:48:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
inline bool isSuccess(UpdateNavMeshStatus value)
|
|
|
|
{
|
|
|
|
return (static_cast<unsigned>(value) & static_cast<unsigned>(UpdateNavMeshStatus::failed)) == 0;
|
|
|
|
}
|
|
|
|
|
2021-08-07 09:48:37 +00:00
|
|
|
inline std::ostream& operator <<(std::ostream& stream, UpdateNavMeshStatus value)
|
|
|
|
{
|
|
|
|
switch (value)
|
|
|
|
{
|
|
|
|
case UpdateNavMeshStatus::ignored:
|
|
|
|
return stream << "ignore";
|
|
|
|
case UpdateNavMeshStatus::removed:
|
|
|
|
return stream << "removed";
|
|
|
|
case UpdateNavMeshStatus::added:
|
|
|
|
return stream << "add";
|
|
|
|
case UpdateNavMeshStatus::replaced:
|
|
|
|
return stream << "replaced";
|
|
|
|
case UpdateNavMeshStatus::failed:
|
|
|
|
return stream << "failed";
|
|
|
|
case UpdateNavMeshStatus::lost:
|
|
|
|
return stream << "lost";
|
|
|
|
case UpdateNavMeshStatus::cached:
|
|
|
|
return stream << "cached";
|
|
|
|
case UpdateNavMeshStatus::unchanged:
|
|
|
|
return stream << "unchanged";
|
|
|
|
case UpdateNavMeshStatus::restored:
|
|
|
|
return stream << "restored";
|
|
|
|
}
|
|
|
|
return stream << "unknown(" << static_cast<unsigned>(value) << ")";
|
|
|
|
}
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
class UpdateNavMeshStatusBuilder
|
2018-04-16 19:57:35 +00:00
|
|
|
{
|
2018-09-30 22:33:25 +00:00
|
|
|
public:
|
2019-03-10 12:48:12 +00:00
|
|
|
UpdateNavMeshStatusBuilder() = default;
|
|
|
|
|
2020-10-07 22:04:51 +00:00
|
|
|
explicit UpdateNavMeshStatusBuilder(UpdateNavMeshStatus value)
|
|
|
|
: mResult(value) {}
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
UpdateNavMeshStatusBuilder removed(bool value)
|
2018-09-30 22:33:25 +00:00
|
|
|
{
|
2019-03-10 12:48:12 +00:00
|
|
|
if (value)
|
|
|
|
set(UpdateNavMeshStatus::removed);
|
|
|
|
else
|
|
|
|
unset(UpdateNavMeshStatus::removed);
|
|
|
|
return *this;
|
2018-09-30 22:33:25 +00:00
|
|
|
}
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
UpdateNavMeshStatusBuilder added(bool value)
|
2018-09-30 22:33:25 +00:00
|
|
|
{
|
2019-03-10 12:48:12 +00:00
|
|
|
if (value)
|
|
|
|
set(UpdateNavMeshStatus::added);
|
|
|
|
else
|
|
|
|
unset(UpdateNavMeshStatus::added);
|
|
|
|
return *this;
|
2018-09-30 22:33:25 +00:00
|
|
|
}
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
UpdateNavMeshStatusBuilder failed(bool value)
|
2018-09-30 22:33:25 +00:00
|
|
|
{
|
2019-03-10 12:48:12 +00:00
|
|
|
if (value)
|
|
|
|
set(UpdateNavMeshStatus::failed);
|
|
|
|
else
|
|
|
|
unset(UpdateNavMeshStatus::failed);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2020-10-07 22:04:51 +00:00
|
|
|
UpdateNavMeshStatusBuilder cached(bool value)
|
|
|
|
{
|
|
|
|
if (value)
|
|
|
|
set(UpdateNavMeshStatus::cached);
|
|
|
|
else
|
|
|
|
unset(UpdateNavMeshStatus::cached);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
UpdateNavMeshStatus getResult() const
|
|
|
|
{
|
|
|
|
return mResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
UpdateNavMeshStatus mResult = UpdateNavMeshStatus::ignored;
|
|
|
|
|
|
|
|
void set(UpdateNavMeshStatus value)
|
|
|
|
{
|
|
|
|
mResult = static_cast<UpdateNavMeshStatus>(static_cast<unsigned>(mResult) | static_cast<unsigned>(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
void unset(UpdateNavMeshStatus value)
|
|
|
|
{
|
|
|
|
mResult = static_cast<UpdateNavMeshStatus>(static_cast<unsigned>(mResult) & ~static_cast<unsigned>(value));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-11-01 19:05:34 +00:00
|
|
|
const dtMeshTile* getTile(const dtNavMesh& navMesh, const TilePosition& position);
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
class NavMeshCacheItem
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
NavMeshCacheItem(const NavMeshPtr& impl, std::size_t generation)
|
2021-11-01 19:21:27 +00:00
|
|
|
: mImpl(impl)
|
|
|
|
, mVersion {generation, 0}
|
2019-03-10 12:48:12 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
const dtNavMesh& getImpl() const
|
|
|
|
{
|
|
|
|
return *mImpl;
|
2018-09-30 22:33:25 +00:00
|
|
|
}
|
|
|
|
|
2021-11-01 19:21:27 +00:00
|
|
|
const Version& getVersion() const { return mVersion; }
|
2018-09-30 22:33:25 +00:00
|
|
|
|
2021-07-13 22:03:10 +00:00
|
|
|
UpdateNavMeshStatus updateTile(const TilePosition& position, NavMeshTilesCache::Value&& cached,
|
|
|
|
NavMeshData&& navMeshData);
|
2019-03-10 12:48:12 +00:00
|
|
|
|
2021-07-13 22:03:10 +00:00
|
|
|
UpdateNavMeshStatus removeTile(const TilePosition& position);
|
2019-03-10 12:48:12 +00:00
|
|
|
|
2021-08-05 22:05:09 +00:00
|
|
|
UpdateNavMeshStatus markAsEmpty(const TilePosition& position);
|
|
|
|
|
|
|
|
bool isEmptyTile(const TilePosition& position) const;
|
|
|
|
|
2021-11-01 19:05:34 +00:00
|
|
|
template <class Function>
|
|
|
|
void forEachUsedTile(Function&& function) const
|
|
|
|
{
|
|
|
|
for (const auto& [position, tile] : mUsedTiles)
|
|
|
|
if (const dtMeshTile* meshTile = getTile(*mImpl, position))
|
|
|
|
function(position, tile.mVersion, *meshTile);
|
|
|
|
}
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
private:
|
2021-11-01 19:05:34 +00:00
|
|
|
struct Tile
|
|
|
|
{
|
|
|
|
Version mVersion;
|
|
|
|
NavMeshTilesCache::Value mCached;
|
|
|
|
NavMeshData mData;
|
|
|
|
};
|
|
|
|
|
2019-03-10 12:48:12 +00:00
|
|
|
NavMeshPtr mImpl;
|
2021-11-01 19:21:27 +00:00
|
|
|
Version mVersion;
|
2021-11-01 19:05:34 +00:00
|
|
|
std::map<TilePosition, Tile> mUsedTiles;
|
2021-08-05 22:05:09 +00:00
|
|
|
std::set<TilePosition> mEmptyTiles;
|
2018-04-16 19:57:35 +00:00
|
|
|
};
|
2018-09-30 22:33:25 +00:00
|
|
|
|
2019-03-10 13:49:56 +00:00
|
|
|
using GuardedNavMeshCacheItem = Misc::ScopeGuarded<NavMeshCacheItem>;
|
|
|
|
using SharedNavMeshCacheItem = std::shared_ptr<GuardedNavMeshCacheItem>;
|
2018-04-16 19:57:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|