navigation_introduction_2d.rst 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. .. _doc_navigation_overview_2d:
  2. 2D Navigation Overview
  3. ======================
  4. Godot provides multiple objects, classes and servers to facilitate grid-based or mesh-based navigation and pathfinding for 2D and 3D games.
  5. The following section provides a quick overview over all available navigation related objects in Godot for 2D scenes and their primary use.
  6. Godot provides the following objects and classes for 2D navigation:
  7. - :ref:`Astar2D<class_Astar2D>`
  8. ``Astar2D`` objects provide an option to find the shortest path in a graph of weighted **points**.
  9. The AStar2D class is best suited for cellbased 2D gameplay that does not require actors to reach any possible position within an area but only predefined, distinct positions.
  10. - :ref:`NavigationServer2D<class_NavigationServer2D>`
  11. ``NavigationServer2D`` provides a powerful server API to find the shortest path between two positions on a area defined by a navigation mesh.
  12. The NavigationServer is best suited for 2D realtime gameplay that does require actors to reach any possible position within an navmesh defined area.
  13. 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.
  14. The NavigationServer holds different navigation maps that each consist of regions that hold navigation mesh data.
  15. Agents can be placed on a map for avoidance calculation.
  16. RIDs are used to reference the internal maps, regions and agents when communicating with the server.
  17. The following NavigationServer RID types are available.
  18. - NavMap RID
  19. Reference to a specific navigation map that holds regions and agents.
  20. The map will attempt to join changed navigation meshes of regions by proximity.
  21. The map will synchronise regions and agents each physics frame.
  22. - NavRegion RID
  23. Reference to a specific navigation region that can hold navigation mesh data.
  24. The region can be enabled / disabled or the use restricted with a navigationlayer bitmask.
  25. - NavAgent RID
  26. Reference to a specific avoidance agent with a radius value use solely in avoidance.
  27. The following SceneTree Nodes are available as helpers to work with the NavigationServer2D API.
  28. - :ref:`NavigationRegion2D<class_NavigationRegion2D>` Node
  29. A Node that holds a NavigationPolygon resource that defines a navigation mesh for the NavigationServer2D.
  30. The region can be enabled / disabled.
  31. The use in pathfinding can be further restricted through the navigationlayers bitmask.
  32. Regions can join their navigation meshes by proximity for a combined navigation mesh.
  33. - :ref:`NavigationAgent2D<class_NavigationAgent2D>` Node
  34. An optional helper Node to facilitate common NavigationServer2D API calls for pathfinding and avoidance
  35. for a Node2D inheriting parent Node.
  36. - :ref:`NavigationObstacle2D<class_NavigationObstacle2D>` Node
  37. A Node that acts as an agent with avoidance radius, to work it needs to be added under a Node2D
  38. inheriting parent Node. Obstacles are intended as a last resort option for constantly moving objects
  39. that cannot be re(baked) to a navigation mesh efficiently. This node also only works if RVO processing
  40. is being used.
  41. The 2D navigationm eshes are defined with the following resources:
  42. - :ref:`NavigationPolygon<class_NavigationPolygon>` Resource
  43. A resource that holds 2D navigation mesh data and provides polygon drawtools to define navigation areas inside the Editor as well as at runtime.
  44. - The NavigationRegion2D Node uses this resource to define its navigation area.
  45. - The NavigationServer2D uses this resource to update navmesh of individual regions.
  46. - The TileSet Editor creates and uses this resource internally when defining tile navigation areas.
  47. .. seealso::
  48. You can see how 2D navigation works in action using the
  49. `2D Navigation Polygon <https://github.com/godotengine/godot-demo-projects/tree/master/2d/navigation>`__
  50. and `Grid-based Navigation with AStarGrid2D <https://github.com/godotengine/godot-demo-projects/tree/master/2d/navigation_astar>`__
  51. demo projects.
  52. Setup for 2D scene
  53. ------------------
  54. The following steps show the basic setup for a minimum viable navigation in 2D that uses the
  55. NavigationServer2D and a NavigationAgent2D for path movement.
  56. #. Add a NavigationRegion2D Node to the scene.
  57. #. Click on the region node and add a new NavigationPolygon Resource to the region node.
  58. .. image:: img/nav_2d_min_setup_step1.png
  59. #. Define the moveable navigation area with the NavigationPolygon draw tool.
  60. .. image:: img/nav_2d_min_setup_step2.png
  61. .. note::
  62. The navigation mesh defines the area where an actor can stand and move with its center.
  63. Leave enough margin between the navpolygon edges and collision objects to not get path
  64. following actors repeatedly stuck on collision.
  65. #. Add a CharacterBody2D node in the scene with a basic collision shape and a sprite or mesh
  66. for visuals.
  67. #. Add a NavigationAgent2D node below the character node.
  68. .. image:: img/nav_2d_min_setup_step3.webp
  69. #. Add the following script to the CharacterBody2D node. We make sure to set a movement target
  70. after the scene has fully loaded and the NavigationServer had time to sync.
  71. .. tabs::
  72. .. code-tab:: gdscript GDScript
  73. extends CharacterBody2D
  74. var movement_speed: float = 200.0
  75. var movement_target_position: Vector2 = Vector2(60.0,180.0)
  76. @onready var navigation_agent: NavigationAgent2D = $NavigationAgent2D
  77. func _ready():
  78. # These values need to be adjusted for the actor's speed
  79. # and the navigation layout.
  80. navigation_agent.path_desired_distance = 4.0
  81. navigation_agent.target_desired_distance = 4.0
  82. # Make sure to not await during _ready.
  83. call_deferred("actor_setup")
  84. func actor_setup():
  85. # Wait for the first physics frame so the NavigationServer can sync.
  86. await get_tree().physics_frame
  87. # Now that the navigation map is no longer empty, set the movement target.
  88. set_movement_target(movement_target_position)
  89. func set_movement_target(movement_target: Vector2):
  90. navigation_agent.target_position = movement_target
  91. func _physics_process(delta):
  92. if navigation_agent.is_navigation_finished():
  93. return
  94. var current_agent_position: Vector2 = global_transform.origin
  95. var next_path_position: Vector2 = navigation_agent.get_next_path_position()
  96. var new_velocity: Vector2 = next_path_position - current_agent_position
  97. new_velocity = new_velocity.normalized()
  98. new_velocity = new_velocity * movement_speed
  99. set_velocity(new_velocity)
  100. move_and_slide()
  101. .. code-tab:: csharp C#
  102. using Godot;
  103. public partial class MyCharacterBody2D : CharacterBody2D
  104. {
  105. private NavigationAgent2D _navigationAgent;
  106. private float _movementSpeed = 200.0f;
  107. private Vector2 _movementTargetPosition = new Vector2(70.0f, 226.0f);
  108. public Vector2 MovementTarget
  109. {
  110. get { return _navigationAgent.TargetPosition; }
  111. set { _navigationAgent.TargetPosition = value; }
  112. }
  113. public override void _Ready()
  114. {
  115. base._Ready();
  116. _navigationAgent = GetNode<NavigationAgent2D>("NavigationAgent2D");
  117. // These values need to be adjusted for the actor's speed
  118. // and the navigation layout.
  119. _navigationAgent.PathDesiredDistance = 4.0f;
  120. _navigationAgent.TargetDesiredDistance = 4.0f;
  121. // Make sure to not await during _Ready.
  122. Callable.From(ActorSetup).CallDeferred();
  123. }
  124. public override void _PhysicsProcess(double delta)
  125. {
  126. base._PhysicsProcess(delta);
  127. if (_navigationAgent.IsNavigationFinished())
  128. {
  129. return;
  130. }
  131. Vector2 currentAgentPosition = GlobalTransform.Origin;
  132. Vector2 nextPathPosition = _navigationAgent.GetNextPathPosition();
  133. Vector2 newVelocity = (nextPathPosition - currentAgentPosition).Normalized();
  134. newVelocity *= _movementSpeed;
  135. Velocity = newVelocity;
  136. MoveAndSlide();
  137. }
  138. private async void ActorSetup()
  139. {
  140. // Wait for the first physics frame so the NavigationServer can sync.
  141. await ToSignal(GetTree(), SceneTree.SignalName.PhysicsFrame);
  142. // Now that the navigation map is no longer empty, set the movement target.
  143. MovementTarget = _movementTargetPosition;
  144. }
  145. }
  146. .. note::
  147. On the first frame the NavigationServer map has not synchronised region data and any path query
  148. will return empty. Await one frame to pause scripts until the NavigationServer had time to sync.