2023-09-05 12:57:28 +02:00
|
|
|
.. _doc_navigation_overview_3d:
|
|
|
|
|
|
|
|
3D Navigation Overview
|
|
|
|
======================
|
|
|
|
|
2024-03-16 20:56:52 +01:00
|
|
|
Pandemonium provides multiple objects, classes and servers to facilitate grid-based or mesh-based navigation and pathfinding for 2D and 3D games.
|
|
|
|
The following section provides a quick overview over all available navigation related objects in Pandemonium for 3D scenes and their primary use.
|
2023-09-05 12:57:28 +02:00
|
|
|
|
2024-03-16 20:56:52 +01:00
|
|
|
Pandemonium provides the following objects and classes for 3D navigation:
|
2023-09-05 12:57:28 +02:00
|
|
|
|
|
|
|
- :ref:`Astar3D<class_Astar3D>`
|
|
|
|
``Astar3D`` objects provide an option to find the shortest path in a graph of weighted **points**.
|
|
|
|
|
|
|
|
The AStar3D class is best suited for cellbased 3D gameplay that does not require actors to reach any
|
|
|
|
possible position within an area but only predefined, distinct positions.
|
|
|
|
|
|
|
|
- :ref:`NavigationServer3D<class_NavigationServer3D>`
|
|
|
|
``NavigationServer3D`` provides a powerful server API to find the shortest path between two positions
|
|
|
|
on a area defined by a navigation mesh.
|
|
|
|
|
|
|
|
The NavigationServer is best suited for 3D realtime gameplay that does require actors to reach any
|
|
|
|
possible position within an navmesh defined area. Meshbased navigation scales well with large gameworlds
|
|
|
|
as a large area can often be defined with a single polygon when it would require many, many grid cells.
|
|
|
|
|
|
|
|
The NavigationServer holds different navigation maps that each consist of regions that hold navigation mesh
|
|
|
|
data. Agents can be placed on a map for avoidance calculation. RIDs are used to reference the internal maps,
|
|
|
|
regions and agents when communicating with the server.
|
|
|
|
|
|
|
|
The following NavigationServer RID types are available.
|
|
|
|
- NavMap RID
|
|
|
|
Reference to a specific navigation map that holds regions and agents.
|
|
|
|
The map will attempt to join changed navigation meshes of regions by proximity.
|
|
|
|
The map will synchronize regions and agents each physics frame.
|
|
|
|
- NavRegion RID
|
|
|
|
Reference to a specific navigation region that can hold navigation mesh data.
|
|
|
|
The region can be enabled / disabled or the use restricted with a navigationlayer bitmask.
|
|
|
|
- NavLink RID
|
|
|
|
Reference to a specific navigation link that connects two navigation mesh positions over arbitrary distances.
|
|
|
|
- NavAgent RID
|
|
|
|
Reference to a specific avoidance agent with a radius value use solely in avoidance.
|
|
|
|
|
|
|
|
The following SceneTree Nodes are available as helpers to work with the NavigationServer3D API.
|
|
|
|
|
|
|
|
- :ref:`NavigationRegion3D<class_NavigationRegion3D>` Node
|
|
|
|
A Node that holds a Navigation Mesh resource that defines a navigation mesh for the NavigationServer3D.
|
|
|
|
|
|
|
|
- The region can be enabled / disabled.
|
|
|
|
- The use in pathfinding can be further restricted through the navigationlayers bitmask.
|
|
|
|
- Regions can join their navigation meshes by proximity for a combined navigation mesh.
|
|
|
|
|
|
|
|
- :ref:`NavigationLink3D<class_NavigationLink3D>` Node
|
|
|
|
A Node that connects two positions on navigation mesh over arbitrary distances for pathfinding.
|
|
|
|
|
|
|
|
- The link can be enabled / disabled.
|
|
|
|
- The link can be made one-way or bidirectional.
|
|
|
|
- The use in pathfinding can be further restricted through the navigationlayers bitmask.
|
|
|
|
|
|
|
|
Links tell the pathfinding that a connection exists and at what cost. The actual agent handling and movement needs to happen in custom scripts.
|
|
|
|
|
|
|
|
- :ref:`NavigationAgent3D<class_NavigationAgent3D>` Node
|
|
|
|
An optional helper Node to facilitate common NavigationServer3D API calls for pathfinding and avoidance for
|
|
|
|
a Node3D inheriting parent Node.
|
|
|
|
|
|
|
|
- :ref:`NavigationObstacle3D<class_NavigationObstacle3D>` Node
|
|
|
|
A Node that acts as an agent with avoidance radius, to work it needs to be added under a Node3D
|
|
|
|
inheriting parent Node. Obstacles are intended as a last resort option for constantly moving objects
|
|
|
|
that cannot be re(baked) to a navigation mesh efficiently. This node also only works if RVO processing
|
|
|
|
is being used.
|
|
|
|
|
|
|
|
The 3D navigation meshes are defined with the following resources:
|
|
|
|
|
|
|
|
- :ref:`NavigationMesh<class_NavigationMesh>` Resource
|
|
|
|
A resource that holds 3D navigation mesh data and provides 3D geometry baking options to define navigation
|
|
|
|
areas inside the Editor as well as at runtime.
|
|
|
|
|
|
|
|
- The NavigationRegion3D Node uses this resource to define its navigation area.
|
|
|
|
- The NavigationServer3D uses this resource to update navmesh of individual regions.
|
|
|
|
- The GridMap Editor uses this resource when specific navigation meshes are defined for each gridcell.
|
|
|
|
|
|
|
|
.. seealso::
|
|
|
|
|
|
|
|
You can see how 3D navigation works in action using the
|
2024-03-16 21:04:42 +01:00
|
|
|
`3D Navigation demo project <https://github.com/Relintai/pandemonium_engine-demo-projects/tree/master/3d/navigation>`__.
|
2023-09-05 12:57:28 +02:00
|
|
|
|
|
|
|
Setup for 3D scene
|
|
|
|
------------------
|
|
|
|
|
|
|
|
The following steps show how to setup a minimum viable navigation in 3D that uses the NavigationServer3D and
|
|
|
|
a NavigationAgent3D for path movement.
|
|
|
|
|
|
|
|
#. Add a NavigationRegion3D Node to the scene.
|
|
|
|
|
|
|
|
#. Click on the region node and add a new :ref:`NavigationMesh<class_NavigationMesh>` Resource to
|
|
|
|
the region node.
|
|
|
|
|
|
|
|
.. image:: img/nav_3d_min_setup_step1.png
|
|
|
|
|
|
|
|
#. Add a new MeshInstance3D node as a child of the region node.
|
|
|
|
|
|
|
|
#. Select the MeshInstance3D node and add a new PlaneMesh and increase the xy size to 10.
|
|
|
|
|
|
|
|
#. Select the region node again and press the "Bake Navmesh" button on the top bar.
|
|
|
|
|
|
|
|
.. image:: img/nav_3d_min_setup_step2.png
|
|
|
|
|
|
|
|
#. Now a transparent navigation mesh appeared that hovers some distance on top the planemesh.
|
|
|
|
|
|
|
|
.. image:: img/nav_3d_min_setup_step3.png
|
|
|
|
|
|
|
|
#. Add a CharacterBody3D node in the scene with a basic collision shape and some mesh for visuals.
|
|
|
|
|
|
|
|
#. Add a NavigationAgent3D node below the character node.
|
|
|
|
|
|
|
|
.. image:: img/nav_3d_min_setup_step4.webp
|
|
|
|
|
|
|
|
#. Add a script to the CharacterBody3D node with the following content. We make sure to set a
|
|
|
|
movement target after the scene has fully loaded and the NavigationServer had time to sync.
|
|
|
|
Also, add a Camera3D and some light and environment to see something.
|
|
|
|
|
|
|
|
.. tabs::
|
|
|
|
.. code-tab:: gdscript GDScript
|
|
|
|
|
|
|
|
extends CharacterBody3D
|
|
|
|
|
|
|
|
var movement_speed: float = 2.0
|
|
|
|
var movement_target_position: Vector3 = Vector3(-3.0,0.0,2.0)
|
|
|
|
|
|
|
|
@onready var navigation_agent: NavigationAgent3D = $NavigationAgent3D
|
|
|
|
|
|
|
|
func _ready():
|
|
|
|
# These values need to be adjusted for the actor's speed
|
|
|
|
# and the navigation layout.
|
|
|
|
navigation_agent.path_desired_distance = 0.5
|
|
|
|
navigation_agent.target_desired_distance = 0.5
|
|
|
|
|
|
|
|
# Make sure to not await during _ready.
|
|
|
|
call_deferred("actor_setup")
|
|
|
|
|
|
|
|
func actor_setup():
|
|
|
|
# Wait for the first physics frame so the NavigationServer can sync.
|
|
|
|
await get_tree().physics_frame
|
|
|
|
|
|
|
|
# Now that the navigation map is no longer empty, set the movement target.
|
|
|
|
set_movement_target(movement_target_position)
|
|
|
|
|
|
|
|
func set_movement_target(movement_target: Vector3):
|
|
|
|
navigation_agent.set_target_position(movement_target)
|
|
|
|
|
|
|
|
func _physics_process(delta):
|
|
|
|
if navigation_agent.is_navigation_finished():
|
|
|
|
return
|
|
|
|
|
|
|
|
var current_agent_position: Vector3 = global_position
|
|
|
|
var next_path_position: Vector3 = navigation_agent.get_next_path_position()
|
|
|
|
|
|
|
|
var new_velocity: Vector3 = next_path_position - current_agent_position
|
|
|
|
new_velocity = new_velocity.normalized()
|
|
|
|
new_velocity = new_velocity * movement_speed
|
|
|
|
|
|
|
|
velocity = new_velocity
|
|
|
|
move_and_slide()
|
|
|
|
|
|
|
|
.. code-tab:: csharp C#
|
|
|
|
|
2024-03-16 20:56:52 +01:00
|
|
|
using Pandemonium;
|
2023-09-05 12:57:28 +02:00
|
|
|
|
|
|
|
public partial class MyCharacterBody3D : CharacterBody3D
|
|
|
|
{
|
|
|
|
private NavigationAgent3D _navigationAgent;
|
|
|
|
|
|
|
|
private float _movementSpeed = 2.0f;
|
|
|
|
private Vector3 _movementTargetPosition = new Vector3(-3.0f, 0.0f, 2.0f);
|
|
|
|
|
|
|
|
public Vector3 MovementTarget
|
|
|
|
{
|
|
|
|
get { return _navigationAgent.TargetPosition; }
|
|
|
|
set { _navigationAgent.TargetPosition = value; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void _Ready()
|
|
|
|
{
|
|
|
|
base._Ready();
|
|
|
|
|
|
|
|
_navigationAgent = GetNode<NavigationAgent3D>("NavigationAgent3D");
|
|
|
|
|
|
|
|
// These values need to be adjusted for the actor's speed
|
|
|
|
// and the navigation layout.
|
|
|
|
_navigationAgent.PathDesiredDistance = 0.5f;
|
|
|
|
_navigationAgent.TargetDesiredDistance = 0.5f;
|
|
|
|
|
|
|
|
// Make sure to not await during _Ready.
|
|
|
|
Callable.From(ActorSetup).CallDeferred();
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void _PhysicsProcess(double delta)
|
|
|
|
{
|
|
|
|
base._PhysicsProcess(delta);
|
|
|
|
|
|
|
|
if (_navigationAgent.IsNavigationFinished())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector3 currentAgentPosition = GlobalTransform.Origin;
|
|
|
|
Vector3 nextPathPosition = _navigationAgent.GetNextPathPosition();
|
|
|
|
|
|
|
|
Vector3 newVelocity = (nextPathPosition - currentAgentPosition).Normalized();
|
|
|
|
newVelocity *= _movementSpeed;
|
|
|
|
|
|
|
|
Velocity = newVelocity;
|
|
|
|
|
|
|
|
MoveAndSlide();
|
|
|
|
}
|
|
|
|
|
|
|
|
private async void ActorSetup()
|
|
|
|
{
|
|
|
|
// Wait for the first physics frame so the NavigationServer can sync.
|
|
|
|
await ToSignal(GetTree(), SceneTree.SignalName.PhysicsFrame);
|
|
|
|
|
|
|
|
// Now that the navigation map is no longer empty, set the movement target.
|
|
|
|
MovementTarget = _movementTargetPosition;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
On the first frame the NavigationServer map has not synchronized region data and any path query
|
|
|
|
will return empty. Await one frame to pause scripts until the NavigationServer had time to sync.
|