2021-06-29 01:49:21 +00:00
|
|
|
#include "generatenavmeshtile.hpp"
|
|
|
|
|
|
|
|
#include "dbrefgeometryobject.hpp"
|
|
|
|
#include "makenavmesh.hpp"
|
|
|
|
#include "offmeshconnectionsmanager.hpp"
|
|
|
|
#include "preparednavmeshdata.hpp"
|
|
|
|
#include "serialization.hpp"
|
|
|
|
#include "settings.hpp"
|
|
|
|
#include "tilecachedrecastmeshmanager.hpp"
|
|
|
|
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
|
|
|
|
#include <osg/Vec3f>
|
|
|
|
#include <osg/io_utils>
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <vector>
|
|
|
|
#include <optional>
|
|
|
|
#include <functional>
|
|
|
|
|
|
|
|
namespace DetourNavigator
|
|
|
|
{
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
struct Ignore
|
|
|
|
{
|
2022-02-18 20:35:09 +00:00
|
|
|
std::string_view mWorldspace;
|
|
|
|
const TilePosition& mTilePosition;
|
2021-06-29 01:49:21 +00:00
|
|
|
std::shared_ptr<NavMeshTileConsumer> mConsumer;
|
|
|
|
|
|
|
|
~Ignore() noexcept
|
|
|
|
{
|
|
|
|
if (mConsumer != nullptr)
|
2022-02-18 20:35:09 +00:00
|
|
|
mConsumer->ignore(mWorldspace, mTilePosition);
|
2021-06-29 01:49:21 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
GenerateNavMeshTile::GenerateNavMeshTile(std::string worldspace, const TilePosition& tilePosition,
|
|
|
|
RecastMeshProvider recastMeshProvider, const osg::Vec3f& agentHalfExtents,
|
|
|
|
const DetourNavigator::Settings& settings, std::weak_ptr<NavMeshTileConsumer> consumer)
|
|
|
|
: mWorldspace(std::move(worldspace))
|
|
|
|
, mTilePosition(tilePosition)
|
|
|
|
, mRecastMeshProvider(recastMeshProvider)
|
|
|
|
, mAgentHalfExtents(agentHalfExtents)
|
|
|
|
, mSettings(settings)
|
|
|
|
, mConsumer(std::move(consumer)) {}
|
|
|
|
|
|
|
|
void GenerateNavMeshTile::doWork()
|
|
|
|
{
|
|
|
|
impl();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GenerateNavMeshTile::impl() noexcept
|
|
|
|
{
|
|
|
|
const auto consumer = mConsumer.lock();
|
|
|
|
|
|
|
|
if (consumer == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2022-02-18 20:35:09 +00:00
|
|
|
Ignore ignore {mWorldspace, mTilePosition, consumer};
|
2021-06-29 01:49:21 +00:00
|
|
|
|
2021-07-09 20:51:42 +00:00
|
|
|
const std::shared_ptr<RecastMesh> recastMesh = mRecastMeshProvider.getMesh(mWorldspace, mTilePosition);
|
2021-06-29 01:49:21 +00:00
|
|
|
|
|
|
|
if (recastMesh == nullptr || isEmpty(*recastMesh))
|
|
|
|
return;
|
|
|
|
|
|
|
|
const std::vector<DbRefGeometryObject> objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
|
|
|
[&] (const MeshSource& v) { return consumer->resolveMeshSource(v); });
|
|
|
|
std::vector<std::byte> input = serialize(mSettings.mRecast, *recastMesh, objects);
|
|
|
|
const std::optional<NavMeshTileInfo> info = consumer->find(mWorldspace, mTilePosition, input);
|
|
|
|
|
|
|
|
if (info.has_value() && info->mVersion == mSettings.mNavMeshVersion)
|
2022-02-18 20:35:09 +00:00
|
|
|
{
|
|
|
|
consumer->identity(mWorldspace, mTilePosition, info->mTileId);
|
|
|
|
ignore.mConsumer = nullptr;
|
2021-06-29 01:49:21 +00:00
|
|
|
return;
|
2022-02-18 20:35:09 +00:00
|
|
|
}
|
2021-06-29 01:49:21 +00:00
|
|
|
|
|
|
|
const auto data = prepareNavMeshTileData(*recastMesh, mTilePosition, mAgentHalfExtents, mSettings.mRecast);
|
|
|
|
|
|
|
|
if (data == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (info.has_value())
|
2022-02-18 20:35:09 +00:00
|
|
|
consumer->update(mWorldspace, mTilePosition, info->mTileId, mSettings.mNavMeshVersion, *data);
|
2021-06-29 01:49:21 +00:00
|
|
|
else
|
|
|
|
consumer->insert(mWorldspace, mTilePosition, mSettings.mNavMeshVersion, input, *data);
|
|
|
|
|
|
|
|
ignore.mConsumer = nullptr;
|
|
|
|
}
|
|
|
|
catch (const std::exception& e)
|
|
|
|
{
|
|
|
|
Log(Debug::Warning) << "Failed to generate navmesh for worldspace \"" << mWorldspace
|
|
|
|
<< "\" tile " << mTilePosition << ": " << e.what();
|
2022-03-10 17:34:35 +00:00
|
|
|
consumer->cancel();
|
2021-06-29 01:49:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|