@ -7,6 +7,7 @@
# include "navmeshtilescache.hpp"
# include "navmeshtilescache.hpp"
# include "offmeshconnection.hpp"
# include "offmeshconnection.hpp"
# include "preparednavmeshdata.hpp"
# include "preparednavmeshdata.hpp"
# include "recastcontext.hpp"
# include "recastmesh.hpp"
# include "recastmesh.hpp"
# include "recastmeshbuilder.hpp"
# include "recastmeshbuilder.hpp"
# include "recastparams.hpp"
# include "recastparams.hpp"
@ -14,6 +15,8 @@
# include "settingsutils.hpp"
# include "settingsutils.hpp"
# include "sharednavmesh.hpp"
# include "sharednavmesh.hpp"
# include "components/debug/debuglog.hpp"
# include <DetourNavMesh.h>
# include <DetourNavMesh.h>
# include <DetourNavMeshBuilder.h>
# include <DetourNavMeshBuilder.h>
# include <Recast.h>
# include <Recast.h>
@ -138,8 +141,8 @@ namespace DetourNavigator
return result ;
return result ;
}
}
void initHeightfield ( rc Context& context , const TilePosition & tilePosition , float minZ , float maxZ ,
[[nodiscard]] bool initHeightfield ( Recast Context& context , const TilePosition & tilePosition , float minZ ,
const RecastSettings & settings , rcHeightfield & solid )
float maxZ , const RecastSettings & settings , rcHeightfield & solid )
{
{
const int size = settings . mTileSize + settings . mBorderSize * 2 ;
const int size = settings . mTileSize + settings . mBorderSize * 2 ;
const int width = size ;
const int width = size ;
@ -150,14 +153,37 @@ namespace DetourNavigator
const osg : : Vec3f bmin ( shift . x ( ) - halfBoundsSize , minZ , shift . y ( ) - halfBoundsSize ) ;
const osg : : Vec3f bmin ( shift . x ( ) - halfBoundsSize , minZ , shift . y ( ) - halfBoundsSize ) ;
const osg : : Vec3f bmax ( shift . x ( ) + halfBoundsSize , maxZ , shift . y ( ) + halfBoundsSize ) ;
const osg : : Vec3f bmax ( shift . x ( ) + halfBoundsSize , maxZ , shift . y ( ) + halfBoundsSize ) ;
const auto result = rcCreateHeightfield (
if ( width < 0 )
& context , solid , width , height , bmin . ptr ( ) , bmax . ptr ( ) , settings . mCellSize , settings . mCellHeight ) ;
{
Log ( Debug : : Warning ) < < context . getPrefix ( ) < < " Invalid width to init heightfield: " < < width ;
return false ;
}
if ( height < 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( ) < < " Invalid height to init heightfield: " < < height ;
return false ;
}
if ( settings . mCellHeight < = 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid cell height to init heightfield: " < < settings . mCellHeight ;
return false ;
}
if ( settings . mCellSize < = 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid cell size to init heightfield: " < < settings . mCellSize ;
return false ;
}
if ( ! result )
return rcCreateHeightfield (
throw NavigatorException ( " Failed to create heightfield for navmesh " ) ;
& context , solid , width , height , bmin . ptr ( ) , bmax . ptr ( ) , settings . mCellSize , settings . mCellHeight ) ;
}
}
bool rasterizeTriangles ( rcContext & context , const Mesh & mesh , const RecastSettings & settings ,
[[nodiscard]] bool rasterizeTriangles ( Recast Context& context , const Mesh & mesh , const RecastSettings & settings ,
const RecastParams & params , rcHeightfield & solid )
const RecastParams & params , rcHeightfield & solid )
{
{
std : : vector < unsigned char > areas ( mesh . getAreaTypes ( ) . begin ( ) , mesh . getAreaTypes ( ) . end ( ) ) ;
std : : vector < unsigned char > areas ( mesh . getAreaTypes ( ) . begin ( ) , mesh . getAreaTypes ( ) . end ( ) ) ;
@ -178,7 +204,7 @@ namespace DetourNavigator
mesh . getIndices ( ) . data ( ) , areas . data ( ) , static_cast < int > ( areas . size ( ) ) , solid , params . mWalkableClimb ) ;
mesh . getIndices ( ) . data ( ) , areas . data ( ) , static_cast < int > ( areas . size ( ) ) , solid , params . mWalkableClimb ) ;
}
}
bool rasterizeTriangles ( rc Context& context , const Rectangle & rectangle , AreaType areaType ,
[[nodiscard]] bool rasterizeTriangles ( Recast Context& context , const Rectangle & rectangle , AreaType areaType ,
const RecastParams & params , rcHeightfield & solid )
const RecastParams & params , rcHeightfield & solid )
{
{
const std : : array vertices {
const std : : array vertices {
@ -199,9 +225,9 @@ namespace DetourNavigator
indices . data ( ) , areas . data ( ) , static_cast < int > ( areas . size ( ) ) , solid , params . mWalkableClimb ) ;
indices . data ( ) , areas . data ( ) , static_cast < int > ( areas . size ( ) ) , solid , params . mWalkableClimb ) ;
}
}
bool rasterizeTriangles ( rc Context& context , float agentHalfExtentsZ , const std : : vector < CellWater > & water ,
[[nodiscard]] bool rasterizeTriangles ( Recast Context& context , float agentHalfExtentsZ ,
const RecastSettings& settings , const RecastParams & params , const TileBounds & realTileBounds ,
const std: : vector < CellWater > & water , const RecastSettings& settings , const RecastParams & params ,
rcHeightfield & solid )
const TileBounds & realTileBounds , rcHeightfield & solid )
{
{
for ( const CellWater & cellWater : water )
for ( const CellWater & cellWater : water )
{
{
@ -219,7 +245,7 @@ namespace DetourNavigator
return true ;
return true ;
}
}
bool rasterizeTriangles ( rc Context& context , const TileBounds & realTileBounds ,
[[nodiscard]] bool rasterizeTriangles ( Recast Context& context , const TileBounds & realTileBounds ,
const std : : vector < FlatHeightfield > & heightfields , const RecastSettings & settings ,
const std : : vector < FlatHeightfield > & heightfields , const RecastSettings & settings ,
const RecastParams & params , rcHeightfield & solid )
const RecastParams & params , rcHeightfield & solid )
{
{
@ -237,7 +263,7 @@ namespace DetourNavigator
return true ;
return true ;
}
}
bool rasterizeTriangles ( rc Context& context , const std : : vector < Heightfield > & heightfields ,
[[nodiscard]] bool rasterizeTriangles ( Recast Context& context , const std : : vector < Heightfield > & heightfields ,
const RecastSettings & settings , const RecastParams & params , rcHeightfield & solid )
const RecastSettings & settings , const RecastParams & params , rcHeightfield & solid )
{
{
for ( const Heightfield & heightfield : heightfields )
for ( const Heightfield & heightfield : heightfields )
@ -249,9 +275,9 @@ namespace DetourNavigator
return true ;
return true ;
}
}
bool rasterizeTriangles ( rc Context& context , const TilePosition & tilePosition , float agentHalfExtentsZ ,
[[nodiscard]] bool rasterizeTriangles ( Recast Context& context , const TilePosition & tilePosition ,
const RecastMesh & recastMesh , const RecastSettings & settings , const RecastParams & params ,
float agentHalfExtentsZ , const RecastMesh & recastMesh , const RecastSettings & settings ,
rcHeightfield & solid )
const RecastParams & params , rcHeightfield & solid )
{
{
const TileBounds realTileBounds = makeRealTileBoundsWithBorder ( settings , tilePosition ) ;
const TileBounds realTileBounds = makeRealTileBoundsWithBorder ( settings , tilePosition ) ;
return rasterizeTriangles ( context , recastMesh . getMesh ( ) , settings , params , solid )
return rasterizeTriangles ( context , recastMesh . getMesh ( ) , settings , params , solid )
@ -262,66 +288,119 @@ namespace DetourNavigator
context , realTileBounds , recastMesh . getFlatHeightfields ( ) , settings , params , solid ) ;
context , realTileBounds , recastMesh . getFlatHeightfields ( ) , settings , params , solid ) ;
}
}
void buildCompactHeightfield ( rc Context& context , const int walkableHeight , const int walkableClimb ,
[[nodiscard]] bool buildCompactHeightfield ( Recast Context& context , const int walkableHeight ,
rcHeightfield & solid , rcCompactHeightfield & compact )
const int walkableClimb , rcHeightfield & solid , rcCompactHeightfield & compact )
{
{
const auto result = rcBuildCompactHeightfield ( & context , walkableHeight , walkableClimb , solid , compact ) ;
if ( walkableHeight < 3 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid walkableHeight to build compact heightfield: " < < walkableHeight ;
return false ;
}
if ( walkableClimb < 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid walkableClimb to build compact heightfield: " < < walkableClimb ;
return false ;
}
if ( ! result )
return rcBuildCompactHeightfield ( & context , walkableHeight , walkableClimb , solid , compact ) ;
throw NavigatorException ( " Failed to build compact heightfield for navmesh " ) ;
}
}
void erodeWalkableArea ( rcContext & context , int walkableRadius , rcCompactHeightfield & compact )
[[nodiscard]] bool erodeWalkableArea ( Recast Context& context , int walkableRadius , rcCompactHeightfield & compact )
{
{
const auto result = rcErodeWalkableArea ( & context , walkableRadius , compact ) ;
if ( walkableRadius < = 0 | | 255 < = walkableRadius )
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid walkableRadius to erode walkable area: " < < walkableRadius ;
return false ;
}
if ( ! result )
return rcErodeWalkableArea ( & context , walkableRadius , compact ) ;
throw NavigatorException ( " Failed to erode walkable area for navmesh " ) ;
}
}
void buildDistanceField ( rc Context& context , rcCompactHeightfield & compact )
[[nodiscard]] bool buildDistanceField ( Recast Context& context , rcCompactHeightfield & compact )
{
{
const auto result = rcBuildDistanceField ( & context , compact ) ;
return rcBuildDistanceField ( & context , compact ) ;
if ( ! result )
throw NavigatorException ( " Failed to build distance field for navmesh " ) ;
}
}
void buildRegions ( rc Context& context , rcCompactHeightfield & compact , const int borderSize ,
[[nodiscard]] bool buildRegions ( RecastContext & context , rcCompactHeightfield & compact , const int borderSize ,
const int minRegionArea , const int mergeRegionArea )
const int minRegionArea , const int mergeRegionArea )
{
{
const auto result = rcBuildRegions ( & context , compact , borderSize , minRegionArea , mergeRegionArea ) ;
if ( borderSize < 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( ) < < " Invalid borderSize to build regions: " < < borderSize ;
return false ;
}
if ( minRegionArea < 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid minRegionArea to build regions: " < < minRegionArea ;
return false ;
}
if ( ! result )
if ( mergeRegionArea < 0 )
throw NavigatorException ( " Failed to build distance field for navmesh " ) ;
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid mergeRegionArea to build regions: " < < mergeRegionArea ;
return false ;
}
return rcBuildRegions ( & context , compact , borderSize , minRegionArea , mergeRegionArea ) ;
}
}
void buildContours ( rcContext & context , rcCompactHeightfield & compact , const float maxError ,
[[nodiscard]] bool buildContours ( Recast Context& context , rcCompactHeightfield & compact , const float maxError ,
const int maxEdgeLen , rcContourSet & contourSet , const int buildFlags = RC_CONTOUR_TESS_WALL_EDGES )
const int maxEdgeLen , rcContourSet & contourSet , const int buildFlags = RC_CONTOUR_TESS_WALL_EDGES )
{
{
const auto result = rcBuildContours ( & context , compact , maxError , maxEdgeLen , contourSet , buildFlags ) ;
if ( maxError < 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( ) < < " Invalid maxError to build contours: " < < maxError ;
return false ;
}
if ( maxEdgeLen < 0 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( ) < < " Invalid maxEdgeLen to build contours: " < < maxEdgeLen ;
return false ;
}
if ( ! result )
return rcBuildContours ( & context , compact , maxError , maxEdgeLen , contourSet , buildFlags ) ;
throw NavigatorException ( " Failed to build contours for navmesh " ) ;
}
}
void buildPolyMesh (
[[nodiscard]] bool buildPolyMesh (
rcContext & context , rcContourSet & contourSet , const int maxVertsPerPoly , rcPolyMesh & polyMesh )
Recast Context& context , rcContourSet & contourSet , const int maxVertsPerPoly , rcPolyMesh & polyMesh )
{
{
const auto result = rcBuildPolyMesh ( & context , contourSet , maxVertsPerPoly , polyMesh ) ;
if ( maxVertsPerPoly < 3 )
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid maxVertsPerPoly to build poly mesh: " < < maxVertsPerPoly ;
return false ;
}
if ( ! result )
return rcBuildPolyMesh ( & context , contourSet , maxVertsPerPoly , polyMesh ) ;
throw NavigatorException ( " Failed to build poly mesh for navmesh " ) ;
}
}
void buildPolyMeshDetail ( rcContext & context , const rcPolyMesh & polyMesh , const rcCompactHeightfield & compact ,
[[nodiscard]] bool buildPolyMeshDetail ( RecastContext & context , const rcPolyMesh & polyMesh ,
const float sampleDist , const float sampleMaxError , rcPolyMeshDetail & polyMeshDetail )
const rcCompactHeightfield & compact , const float sampleDist , const float sampleMaxError ,
rcPolyMeshDetail & polyMeshDetail )
{
{
const auto result
if ( sampleDist < 0 )
= rcBuildPolyMeshDetail ( & context , polyMesh , compact , sampleDist , sampleMaxError , polyMeshDetail ) ;
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid sampleDist to build poly mesh detail: " < < sampleDist ;
return false ;
}
if ( ! result )
if ( sampleMaxError < 0 )
throw NavigatorException ( " Failed to build detail poly mesh for navmesh " ) ;
{
Log ( Debug : : Warning ) < < context . getPrefix ( )
< < " Invalid sampleMaxError to build poly mesh detail: " < < sampleMaxError ;
return false ;
}
return rcBuildPolyMeshDetail ( & context , polyMesh , compact , sampleDist , sampleMaxError , polyMeshDetail ) ;
}
}
void setPolyMeshFlags ( rcPolyMesh & polyMesh )
void setPolyMeshFlags ( rcPolyMesh & polyMesh )
@ -330,25 +409,36 @@ namespace DetourNavigator
polyMesh . flags [ i ] = getFlag ( static_cast < AreaType > ( polyMesh . areas [ i ] ) ) ;
polyMesh . flags [ i ] = getFlag ( static_cast < AreaType > ( polyMesh . areas [ i ] ) ) ;
}
}
bool fillPolyMesh ( rc Context& context , const RecastSettings & settings , const RecastParams & params ,
[[nodiscard]] bool fillPolyMesh ( Recast Context& context , const RecastSettings & settings ,
rcHeightfield & solid , rcPolyMesh & polyMesh , rcPolyMeshDetail & polyMeshDetail )
const RecastParams & params , rcHeightfield & solid , rcPolyMesh & polyMesh , rcPolyMeshDetail & polyMeshDetail )
{
{
rcCompactHeightfield compact ;
rcCompactHeightfield compact ;
buildCompactHeightfield ( context , params . mWalkableHeight , params . mWalkableClimb , solid , compact ) ;
if ( ! buildCompactHeightfield ( context , params . mWalkableHeight , params . mWalkableClimb , solid , compact ) )
return false ;
erodeWalkableArea ( context , params . mWalkableRadius , compact ) ;
if ( ! erodeWalkableArea ( context , params . mWalkableRadius , compact ) )
buildDistanceField ( context , compact ) ;
return false ;
buildRegions ( context , compact , settings . mBorderSize , settings . mRegionMinArea , settings . mRegionMergeArea ) ;
if ( ! buildDistanceField ( context , compact ) )
return false ;
if ( ! buildRegions (
context , compact , settings . mBorderSize , settings . mRegionMinArea , settings . mRegionMergeArea ) )
return false ;
rcContourSet contourSet ;
rcContourSet contourSet ;
buildContours ( context , compact , settings . mMaxSimplificationError , params . mMaxEdgeLen , contourSet ) ;
if ( ! buildContours ( context , compact , settings . mMaxSimplificationError , params . mMaxEdgeLen , contourSet ) )
return false ;
if ( contourSet . nconts = = 0 )
if ( contourSet . nconts = = 0 )
return false ;
return false ;
buildPolyMesh ( context , contourSet , settings . mMaxVertsPerPoly , polyMesh ) ;
if ( ! buildPolyMesh ( context , contourSet , settings . mMaxVertsPerPoly , polyMesh ) )
return false ;
buildPolyMeshDetail ( context , polyMesh , compact , params . mSampleDist , params . mSampleMaxError , polyMeshDetail ) ;
if ( ! buildPolyMeshDetail (
context , polyMesh , compact , params . mSampleDist , params . mSampleMaxError , polyMeshDetail ) )
return false ;
setPolyMeshFlags ( polyMesh ) ;
setPolyMeshFlags ( polyMesh ) ;
@ -410,13 +500,14 @@ namespace DetourNavigator
std : : unique_ptr < PreparedNavMeshData > prepareNavMeshTileData ( const RecastMesh & recastMesh ,
std : : unique_ptr < PreparedNavMeshData > prepareNavMeshTileData ( const RecastMesh & recastMesh ,
const TilePosition & tilePosition , const AgentBounds & agentBounds , const RecastSettings & settings )
const TilePosition & tilePosition , const AgentBounds & agentBounds , const RecastSettings & settings )
{
{
rcContext context ;
RecastContext context ( tilePosition , agentBounds ) ;
const auto [ minZ , maxZ ] = getBoundsByZ ( recastMesh , agentBounds . mHalfExtents . z ( ) , settings ) ;
const auto [ minZ , maxZ ] = getBoundsByZ ( recastMesh , agentBounds . mHalfExtents . z ( ) , settings ) ;
rcHeightfield solid ;
rcHeightfield solid ;
initHeightfield ( context , tilePosition , toNavMeshCoordinates ( settings , minZ ) ,
if ( ! initHeightfield ( context , tilePosition , toNavMeshCoordinates ( settings , minZ ) ,
toNavMeshCoordinates ( settings , maxZ ) , settings , solid ) ;
toNavMeshCoordinates ( settings , maxZ ) , settings , solid ) )
return nullptr ;
const RecastParams params = makeRecastParams ( settings , agentBounds ) ;
const RecastParams params = makeRecastParams ( settings , agentBounds ) ;