Tiled Runtime Nav-mesh generation for 3D worlds in Bevy. Based on Recast's Nav-mesh generation but in Rust.
Takes in Parry3d colliders that implement the OxidizedCollider
trait from entities with the NavMeshAffector
component and asynchronously generates tiles of navigation meshes based on NavMeshSettings
. OxidizedCollider
implementations for Bevy Rapier3D and Avian 3d are included in the oxidized_navigation_rapier
& oxidized_navigation_avian
crates respectively.
Nav-mesh generation:
- Choose which backend you're going to use (bevy_rapier3d, avian_3d, or custom parry3d based colliders) and use the relevant implementation crate.
- If you opted for custom parry3d colliders, implement the
OxidizedCollider
trait for your collider component that wraps aparry3d::shape::SharedShape
. This is already done forbevy_rapier3d
andavian_3d
. - Add
OxidizedNavigationPlugin
as a plugin. (eg. for avianOxidizedNavigationPlugin::<AvianCollider>::new(NavMeshSettings {...}
) - Attach a
NavMeshAffector
component and a collider that implements theOxidizedCollider
trait (already implemented forbevy_rapier3d
andavian_3d
) to any entity you want to affect the nav-mesh.
At this point nav-meshes will be automatically generated whenever the collider or GlobalTransform
of any entity with a NavMeshAffector
is changed.
// Add the plugin (for example for Avian)
app.add_plugins(OxidizedNavigationPlugin::<AvianCollider>::new(NavMeshSettings::from_agent_and_bounds(
0.5, 1.9, 250.0, -1.0,
)));
// Attach components to an entity
commands.spawn((
NavMeshAffector,
Collider::cuboid(25.0, 0.1, 25.0),
));
Querying the nav-mesh / Pathfinding:
- Your system needs to take in the
NavMesh
resource. - Get the underlying data from the nav-mesh using
NavMesh::get
. This data is wrapped in anRwLock
. - To access the data call
RwLock::read
. This will block until you get read acces on the lock. If a task is already writing to the lock it may take time. - Call
query::find_path
with theNavMeshTiles
returned from theRwLock
.
Also see the examples for how to run pathfinding in an async task which may be preferable.
I added the
OxidizedNavigationPlugin
to my app and now it won't compile.
You need to use OxidizedNavigationPlugin::<Collider>::new(NavMeshSettings {...}
, where Collider
is either RapierCollider
or AvianCollider
from the integration crates, or your own struct that implements the OxidizedCollider
trait. This is necessary to allow us to be generic over different Collider
components.
I don't want to use the Rapier3d or XPBD3d physics engines just to generate a navmesh. How do I create my own
parry3d
wrapper component?
You need to create a component that contains a parry3d SharedShape
, then implement the OxidizedCollider
trait. See the parry3d example for a basic example.
Can I use this with the builtin bevy shapes, or my own custom shapes?
Currently only parry3d
colliders are supported, or crates using parry3d
colliders. You'd have to write a function to convert your shapes/bevy shapes into parry3d
colliders.
My physics crate updated and now my nav-meshes won't generate.
This is due to how dependencies are handled, Oxidized Navigation depends on a specific parry3d version. The integration crates depend on specific versions of their physics crates.
If a new version of the integration crate hasn't released yet, you can implement the OxidizedCollider
trait yourself, see the parry3d example.
How do I draw the Nav-mesh for debugging?
Debug draw is available behind the debug_draw
feature & the OxidizedNavigationDebugDrawPlugin
, see usage in examples.
Crate Version | Bevy Version |
---|---|
0.13 | 0.15 |
Two integration crates are shipped for Oxidized Navigation for Avian3d (oxidized_navigation_avian
) & Rapier3d (oxidized_navigation_rapier
) respectively.
Before integration crates split
| Crate Version | Bevy Version | Bevy Rapier 3D Version | Bevy Xpbd 3D Version | Avian3D Version | Parry3d Version | |---------------|--------------|------------------------|----------------------|---------------------|-----------------| | 0.12.0 | 0.15 | 0.28 | unsupported | git-rev-52cbcec (1) | 0.17 | | 0.11.0 | 0.14 | 0.27 | 0.5 | unsupported | 0.15/0.16 | | 0.9.0 | 0.12 | 0.24 | 0.3 | unsupported | 0.13 | | 0.10.0 | 0.13 | 0.25 | 0.4 | unsupported | 0.13 | | 0.8.0 | 0.12 | 0.23 | 0.3 | unsupported | 0.13 | | 0.7.0 | 0.11 | 0.22 | 0.2 | unsupported | 0.13 | | 0.6.0 | 0.11 | 0.22 | unsupported | unsupported | unsupported | | 0.5.X | 0.10.X | 0.21 | unsupported | unsupported | unsupported | | 0.4.0 | 0.10.X | 0.21 | unsupported | unsupported | unsupported | | 0.3.0 | 0.10.0 | 0.21 | unsupported | unsupported | unsupported | | 0.2.0 | 0.9.X | 0.20 | unsupported | unsupported | unsupported | | 0.1.X | 0.9.X | 0.19 | unsupported | unsupported | unsupported |-
Allow creating nav-mesh from meshes (probably add an option to
NavMeshAffector
). -
Rebuild all tiles when
NavMeshSettings
are changed. -
Nav-mesh "layers" using different
NavMeshSettings
. -
Pathfinding ticket system (Call to pathfinding returns a ticket that one can check later, controlling async pathfinding like this allows us to limit the amount of parallel tasks & prioritize them)
-
Remove
create_nav_mesh_tile_from_poly_mesh
in favor of creating data in the right format from the start. -
Add local nav-mesh sub-grids that can be used for moving objects (platforms, ships, etc) without needing to regenerate it's interior every update.