2023-01-05 13:25:55 +01:00
/**************************************************************************/
/* navigation_agent_3d.cpp */
/**************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/**************************************************************************/
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
/* */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the */
/* "Software"), to deal in the Software without restriction, including */
/* without limitation the rights to use, copy, modify, merge, publish, */
/* distribute, sublicense, and/or sell copies of the Software, and to */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions: */
/* */
/* The above copyright notice and this permission notice shall be */
/* included in all copies or substantial portions of the Software. */
/* */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
/**************************************************************************/
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
# include "navigation_agent_3d.h"
2020-01-10 12:22:34 +01:00
2023-02-13 13:18:13 +01:00
# include "scene/3d/navigation_link_3d.h"
2020-03-27 19:21:27 +01:00
# include "servers/navigation_server_3d.h"
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
void NavigationAgent3D : : _bind_methods ( ) {
2021-04-19 23:24:30 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_rid " ) , & NavigationAgent3D : : get_rid ) ;
2022-05-20 22:55:27 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_avoidance_enabled " , " enabled " ) , & NavigationAgent3D : : set_avoidance_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " get_avoidance_enabled " ) , & NavigationAgent3D : : get_avoidance_enabled ) ;
2022-06-18 19:11:49 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_path_desired_distance " , " desired_distance " ) , & NavigationAgent3D : : set_path_desired_distance ) ;
ClassDB : : bind_method ( D_METHOD ( " get_path_desired_distance " ) , & NavigationAgent3D : : get_path_desired_distance ) ;
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_target_desired_distance " , " desired_distance " ) , & NavigationAgent3D : : set_target_desired_distance ) ;
ClassDB : : bind_method ( D_METHOD ( " get_target_desired_distance " ) , & NavigationAgent3D : : get_target_desired_distance ) ;
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_radius " , " radius " ) , & NavigationAgent3D : : set_radius ) ;
ClassDB : : bind_method ( D_METHOD ( " get_radius " ) , & NavigationAgent3D : : get_radius ) ;
2020-01-10 12:22:34 +01:00
2023-01-10 07:14:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_height " , " height " ) , & NavigationAgent3D : : set_height ) ;
ClassDB : : bind_method ( D_METHOD ( " get_height " ) , & NavigationAgent3D : : get_height ) ;
2020-01-10 12:22:34 +01:00
2023-01-10 07:14:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_path_height_offset " , " path_height_offset " ) , & NavigationAgent3D : : set_path_height_offset ) ;
ClassDB : : bind_method ( D_METHOD ( " get_path_height_offset " ) , & NavigationAgent3D : : get_path_height_offset ) ;
ClassDB : : bind_method ( D_METHOD ( " set_use_3d_avoidance " , " enabled " ) , & NavigationAgent3D : : set_use_3d_avoidance ) ;
ClassDB : : bind_method ( D_METHOD ( " get_use_3d_avoidance " ) , & NavigationAgent3D : : get_use_3d_avoidance ) ;
2020-01-10 12:22:34 +01:00
2022-08-13 12:26:13 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_neighbor_distance " , " neighbor_distance " ) , & NavigationAgent3D : : set_neighbor_distance ) ;
ClassDB : : bind_method ( D_METHOD ( " get_neighbor_distance " ) , & NavigationAgent3D : : get_neighbor_distance ) ;
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_max_neighbors " , " max_neighbors " ) , & NavigationAgent3D : : set_max_neighbors ) ;
ClassDB : : bind_method ( D_METHOD ( " get_max_neighbors " ) , & NavigationAgent3D : : get_max_neighbors ) ;
2020-01-10 12:22:34 +01:00
2023-01-10 07:14:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_time_horizon_agents " , " time_horizon " ) , & NavigationAgent3D : : set_time_horizon_agents ) ;
ClassDB : : bind_method ( D_METHOD ( " get_time_horizon_agents " ) , & NavigationAgent3D : : get_time_horizon_agents ) ;
ClassDB : : bind_method ( D_METHOD ( " set_time_horizon_obstacles " , " time_horizon " ) , & NavigationAgent3D : : set_time_horizon_obstacles ) ;
ClassDB : : bind_method ( D_METHOD ( " get_time_horizon_obstacles " ) , & NavigationAgent3D : : get_time_horizon_obstacles ) ;
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_max_speed " , " max_speed " ) , & NavigationAgent3D : : set_max_speed ) ;
ClassDB : : bind_method ( D_METHOD ( " get_max_speed " ) , & NavigationAgent3D : : get_max_speed ) ;
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_path_max_distance " , " max_speed " ) , & NavigationAgent3D : : set_path_max_distance ) ;
ClassDB : : bind_method ( D_METHOD ( " get_path_max_distance " ) , & NavigationAgent3D : : get_path_max_distance ) ;
2020-01-10 12:22:34 +01:00
2022-06-14 23:34:43 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_navigation_layers " , " navigation_layers " ) , & NavigationAgent3D : : set_navigation_layers ) ;
ClassDB : : bind_method ( D_METHOD ( " get_navigation_layers " ) , & NavigationAgent3D : : get_navigation_layers ) ;
2022-03-19 21:47:02 +01:00
2022-06-19 13:23:13 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_navigation_layer_value " , " layer_number " , " value " ) , & NavigationAgent3D : : set_navigation_layer_value ) ;
ClassDB : : bind_method ( D_METHOD ( " get_navigation_layer_value " , " layer_number " ) , & NavigationAgent3D : : get_navigation_layer_value ) ;
2023-03-25 01:00:09 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_pathfinding_algorithm " , " pathfinding_algorithm " ) , & NavigationAgent3D : : set_pathfinding_algorithm ) ;
ClassDB : : bind_method ( D_METHOD ( " get_pathfinding_algorithm " ) , & NavigationAgent3D : : get_pathfinding_algorithm ) ;
ClassDB : : bind_method ( D_METHOD ( " set_path_postprocessing " , " path_postprocessing " ) , & NavigationAgent3D : : set_path_postprocessing ) ;
ClassDB : : bind_method ( D_METHOD ( " get_path_postprocessing " ) , & NavigationAgent3D : : get_path_postprocessing ) ;
2022-10-06 01:24:45 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_path_metadata_flags " , " flags " ) , & NavigationAgent3D : : set_path_metadata_flags ) ;
ClassDB : : bind_method ( D_METHOD ( " get_path_metadata_flags " ) , & NavigationAgent3D : : get_path_metadata_flags ) ;
2022-06-14 20:39:15 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_navigation_map " , " navigation_map " ) , & NavigationAgent3D : : set_navigation_map ) ;
ClassDB : : bind_method ( D_METHOD ( " get_navigation_map " ) , & NavigationAgent3D : : get_navigation_map ) ;
2022-12-06 23:32:11 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_target_position " , " position " ) , & NavigationAgent3D : : set_target_position ) ;
ClassDB : : bind_method ( D_METHOD ( " get_target_position " ) , & NavigationAgent3D : : get_target_position ) ;
2022-09-07 04:03:55 +02:00
2022-12-06 23:32:11 +01:00
ClassDB : : bind_method ( D_METHOD ( " get_next_path_position " ) , & NavigationAgent3D : : get_next_path_position ) ;
2023-01-10 07:14:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_velocity_forced " , " velocity " ) , & NavigationAgent3D : : set_velocity_forced ) ;
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_velocity " , " velocity " ) , & NavigationAgent3D : : set_velocity ) ;
2023-01-10 07:14:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " get_velocity " ) , & NavigationAgent3D : : get_velocity ) ;
ClassDB : : bind_method ( D_METHOD ( " distance_to_target " ) , & NavigationAgent3D : : distance_to_target ) ;
2022-10-06 01:24:45 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_current_navigation_result " ) , & NavigationAgent3D : : get_current_navigation_result ) ;
2023-01-10 07:14:16 +01:00
2022-12-11 18:02:35 +01:00
ClassDB : : bind_method ( D_METHOD ( " get_current_navigation_path " ) , & NavigationAgent3D : : get_current_navigation_path ) ;
ClassDB : : bind_method ( D_METHOD ( " get_current_navigation_path_index " ) , & NavigationAgent3D : : get_current_navigation_path_index ) ;
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " is_target_reached " ) , & NavigationAgent3D : : is_target_reached ) ;
ClassDB : : bind_method ( D_METHOD ( " is_target_reachable " ) , & NavigationAgent3D : : is_target_reachable ) ;
ClassDB : : bind_method ( D_METHOD ( " is_navigation_finished " ) , & NavigationAgent3D : : is_navigation_finished ) ;
2022-12-06 23:32:11 +01:00
ClassDB : : bind_method ( D_METHOD ( " get_final_position " ) , & NavigationAgent3D : : get_final_position ) ;
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " _avoidance_done " , " new_velocity " ) , & NavigationAgent3D : : _avoidance_done ) ;
2020-01-10 12:22:34 +01:00
2023-01-10 07:14:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_avoidance_layers " , " layers " ) , & NavigationAgent3D : : set_avoidance_layers ) ;
ClassDB : : bind_method ( D_METHOD ( " get_avoidance_layers " ) , & NavigationAgent3D : : get_avoidance_layers ) ;
ClassDB : : bind_method ( D_METHOD ( " set_avoidance_mask " , " mask " ) , & NavigationAgent3D : : set_avoidance_mask ) ;
ClassDB : : bind_method ( D_METHOD ( " get_avoidance_mask " ) , & NavigationAgent3D : : get_avoidance_mask ) ;
ClassDB : : bind_method ( D_METHOD ( " set_avoidance_layer_value " , " layer_number " , " value " ) , & NavigationAgent3D : : set_avoidance_layer_value ) ;
ClassDB : : bind_method ( D_METHOD ( " get_avoidance_layer_value " , " layer_number " ) , & NavigationAgent3D : : get_avoidance_layer_value ) ;
ClassDB : : bind_method ( D_METHOD ( " set_avoidance_mask_value " , " mask_number " , " value " ) , & NavigationAgent3D : : set_avoidance_mask_value ) ;
ClassDB : : bind_method ( D_METHOD ( " get_avoidance_mask_value " , " mask_number " ) , & NavigationAgent3D : : get_avoidance_mask_value ) ;
ClassDB : : bind_method ( D_METHOD ( " set_avoidance_priority " , " priority " ) , & NavigationAgent3D : : set_avoidance_priority ) ;
ClassDB : : bind_method ( D_METHOD ( " get_avoidance_priority " ) , & NavigationAgent3D : : get_avoidance_priority ) ;
2022-06-29 18:42:51 +02:00
ADD_GROUP ( " Pathfinding " , " " ) ;
2022-12-06 23:32:11 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : VECTOR3 , " target_position " , PROPERTY_HINT_NONE , " " , PROPERTY_USAGE_NO_EDITOR ) , " set_target_position " , " get_target_position " ) ;
2023-02-16 12:33:26 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " path_desired_distance " , PROPERTY_HINT_RANGE , " 0.1,100,0.01,or_greater,suffix:m " ) , " set_path_desired_distance " , " get_path_desired_distance " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " target_desired_distance " , PROPERTY_HINT_RANGE , " 0.1,100,0.01,or_greater,suffix:m " ) , " set_target_desired_distance " , " get_target_desired_distance " ) ;
2023-01-10 07:14:16 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " path_height_offset " , PROPERTY_HINT_RANGE , " -100.0,100,0.01,or_greater,suffix:m " ) , " set_path_height_offset " , " get_path_height_offset " ) ;
2023-02-16 12:33:26 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " path_max_distance " , PROPERTY_HINT_RANGE , " 0.01,100,0.1,or_greater,suffix:m " ) , " set_path_max_distance " , " get_path_max_distance " ) ;
2022-06-29 18:42:51 +02:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " navigation_layers " , PROPERTY_HINT_LAYERS_3D_NAVIGATION ) , " set_navigation_layers " , " get_navigation_layers " ) ;
2023-03-25 01:00:09 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " pathfinding_algorithm " , PROPERTY_HINT_ENUM , " AStar " ) , " set_pathfinding_algorithm " , " get_pathfinding_algorithm " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " path_postprocessing " , PROPERTY_HINT_ENUM , " Corridorfunnel,Edgecentered " ) , " set_path_postprocessing " , " get_path_postprocessing " ) ;
2022-10-06 01:24:45 +02:00
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " path_metadata_flags " , PROPERTY_HINT_FLAGS , " Include Types,Include RIDs,Include Owners " ) , " set_path_metadata_flags " , " get_path_metadata_flags " ) ;
2022-06-29 18:42:51 +02:00
ADD_GROUP ( " Avoidance " , " " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " avoidance_enabled " ) , " set_avoidance_enabled " , " get_avoidance_enabled " ) ;
2023-01-10 07:14:16 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : VECTOR3 , " velocity " , PROPERTY_HINT_NONE , " " , PROPERTY_USAGE_NO_EDITOR ) , " set_velocity " , " get_velocity " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " height " , PROPERTY_HINT_RANGE , " 0.01,100,0.01,or_greater,suffix:m " ) , " set_height " , " get_height " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " radius " , PROPERTY_HINT_RANGE , " 0.01,100,0.01,or_greater,suffix:m " ) , " set_radius " , " get_radius " ) ;
2023-02-16 12:33:26 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " neighbor_distance " , PROPERTY_HINT_RANGE , " 0.1,10000,0.01,or_greater,suffix:m " ) , " set_neighbor_distance " , " get_neighbor_distance " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " max_neighbors " , PROPERTY_HINT_RANGE , " 1,10000,1,or_greater " ) , " set_max_neighbors " , " get_max_neighbors " ) ;
2023-01-10 07:14:16 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " time_horizon_agents " , PROPERTY_HINT_RANGE , " 0.0,10,0.01,or_greater,suffix:s " ) , " set_time_horizon_agents " , " get_time_horizon_agents " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " time_horizon_obstacles " , PROPERTY_HINT_RANGE , " 0.0,10,0.01,or_greater,suffix:s " ) , " set_time_horizon_obstacles " , " get_time_horizon_obstacles " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " max_speed " , PROPERTY_HINT_RANGE , " 0.01,10000,0.01,or_greater,suffix:m/s " ) , " set_max_speed " , " get_max_speed " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " use_3d_avoidance " ) , " set_use_3d_avoidance " , " get_use_3d_avoidance " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " avoidance_layers " , PROPERTY_HINT_LAYERS_AVOIDANCE ) , " set_avoidance_layers " , " get_avoidance_layers " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : INT , " avoidance_mask " , PROPERTY_HINT_LAYERS_AVOIDANCE ) , " set_avoidance_mask " , " get_avoidance_mask " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " avoidance_priority " , PROPERTY_HINT_RANGE , " 0.0,1.0,0.01 " ) , " set_avoidance_priority " , " get_avoidance_priority " ) ;
2020-01-10 12:22:34 +01:00
ADD_SIGNAL ( MethodInfo ( " path_changed " ) ) ;
ADD_SIGNAL ( MethodInfo ( " target_reached " ) ) ;
2022-10-06 01:24:45 +02:00
ADD_SIGNAL ( MethodInfo ( " waypoint_reached " , PropertyInfo ( Variant : : DICTIONARY , " details " ) ) ) ;
ADD_SIGNAL ( MethodInfo ( " link_reached " , PropertyInfo ( Variant : : DICTIONARY , " details " ) ) ) ;
2020-01-10 12:22:34 +01:00
ADD_SIGNAL ( MethodInfo ( " navigation_finished " ) ) ;
ADD_SIGNAL ( MethodInfo ( " velocity_computed " , PropertyInfo ( Variant : : VECTOR3 , " safe_velocity " ) ) ) ;
2023-01-16 17:45:03 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_debug_enabled " , " enabled " ) , & NavigationAgent3D : : set_debug_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " get_debug_enabled " ) , & NavigationAgent3D : : get_debug_enabled ) ;
ClassDB : : bind_method ( D_METHOD ( " set_debug_use_custom " , " enabled " ) , & NavigationAgent3D : : set_debug_use_custom ) ;
ClassDB : : bind_method ( D_METHOD ( " get_debug_use_custom " ) , & NavigationAgent3D : : get_debug_use_custom ) ;
ClassDB : : bind_method ( D_METHOD ( " set_debug_path_custom_color " , " color " ) , & NavigationAgent3D : : set_debug_path_custom_color ) ;
ClassDB : : bind_method ( D_METHOD ( " get_debug_path_custom_color " ) , & NavigationAgent3D : : get_debug_path_custom_color ) ;
ClassDB : : bind_method ( D_METHOD ( " set_debug_path_custom_point_size " , " point_size " ) , & NavigationAgent3D : : set_debug_path_custom_point_size ) ;
ClassDB : : bind_method ( D_METHOD ( " get_debug_path_custom_point_size " ) , & NavigationAgent3D : : get_debug_path_custom_point_size ) ;
2023-02-11 09:34:08 +01:00
ADD_GROUP ( " Debug " , " debug_ " ) ;
2023-01-16 17:45:03 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " debug_enabled " ) , " set_debug_enabled " , " get_debug_enabled " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " debug_use_custom " ) , " set_debug_use_custom " , " get_debug_use_custom " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : COLOR , " debug_path_custom_color " ) , " set_debug_path_custom_color " , " get_debug_path_custom_color " ) ;
2023-03-16 09:15:34 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " debug_path_custom_point_size " , PROPERTY_HINT_RANGE , " 0,50,0.01,or_greater,suffix:px " ) , " set_debug_path_custom_point_size " , " get_debug_path_custom_point_size " ) ;
2020-01-10 12:22:34 +01:00
}
2023-01-10 07:14:16 +01:00
# ifndef DISABLE_DEPRECATED
// Compatibility with Godot 4.0 beta 10 or below.
// Functions in block below all renamed or replaced in 4.0 beta 1X avoidance rework.
bool NavigationAgent3D : : _set ( const StringName & p_name , const Variant & p_value ) {
if ( p_name = = " time_horizon " ) {
set_time_horizon_agents ( p_value ) ;
return true ;
}
if ( p_name = = " target_location " ) {
set_target_position ( p_value ) ;
return true ;
}
if ( p_name = = " agent_height_offset " ) {
set_path_height_offset ( p_value ) ;
return true ;
}
return false ;
}
bool NavigationAgent3D : : _get ( const StringName & p_name , Variant & r_ret ) const {
if ( p_name = = " time_horizon " ) {
r_ret = get_time_horizon_agents ( ) ;
return true ;
}
if ( p_name = = " target_location " ) {
r_ret = get_target_position ( ) ;
return true ;
}
if ( p_name = = " agent_height_offset " ) {
r_ret = get_path_height_offset ( ) ;
return true ;
}
return false ;
}
# endif // DISABLE_DEPRECATED
2020-03-26 22:49:16 +01:00
void NavigationAgent3D : : _notification ( int p_what ) {
2020-01-10 12:22:34 +01:00
switch ( p_what ) {
2022-06-01 06:45:12 +02:00
case NOTIFICATION_POST_ENTER_TREE : {
// need to use POST_ENTER_TREE cause with normal ENTER_TREE not all required Nodes are ready.
2023-02-01 10:49:05 +01:00
// cannot use READY as ready does not get called if Node is re-added to SceneTree
2022-06-01 06:45:12 +02:00
set_agent_parent ( get_parent ( ) ) ;
2020-01-10 12:22:34 +01:00
set_physics_process_internal ( true ) ;
2023-01-16 17:45:03 +01:00
2023-07-02 11:32:14 +02:00
if ( agent_parent & & avoidance_enabled ) {
2023-01-10 07:14:16 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_position ( agent , agent_parent - > get_global_transform ( ) . origin ) ;
}
2023-01-16 17:45:03 +01:00
# ifdef DEBUG_ENABLED
if ( NavigationServer3D : : get_singleton ( ) - > get_debug_enabled ( ) ) {
debug_path_dirty = true ;
}
# endif // DEBUG_ENABLED
2023-01-10 07:14:16 +01:00
2020-01-10 12:22:34 +01:00
} break ;
2022-02-15 18:06:48 +01:00
2022-06-01 06:45:12 +02:00
case NOTIFICATION_PARENTED : {
if ( is_inside_tree ( ) & & ( get_parent ( ) ! = agent_parent ) ) {
// only react to PARENTED notifications when already inside_tree and parent changed, e.g. users switch nodes around
// PARENTED notification fires also when Node is added in scripts to a parent
// this would spam transforms fails and world fails while Node is outside SceneTree
// when node gets reparented when joining the tree POST_ENTER_TREE takes care of this
set_agent_parent ( get_parent ( ) ) ;
set_physics_process_internal ( true ) ;
}
} break ;
case NOTIFICATION_UNPARENTED : {
// if agent has no parent no point in processing it until reparented
set_agent_parent ( nullptr ) ;
set_physics_process_internal ( false ) ;
} break ;
2020-01-10 12:22:34 +01:00
case NOTIFICATION_EXIT_TREE : {
2022-06-01 06:45:12 +02:00
set_agent_parent ( nullptr ) ;
2020-01-10 12:22:34 +01:00
set_physics_process_internal ( false ) ;
2023-01-16 17:45:03 +01:00
# ifdef DEBUG_ENABLED
if ( debug_path_instance . is_valid ( ) ) {
RS : : get_singleton ( ) - > instance_set_visible ( debug_path_instance , false ) ;
}
# endif // DEBUG_ENABLED
2020-01-10 12:22:34 +01:00
} break ;
2022-02-15 18:06:48 +01:00
2022-05-17 20:08:39 +02:00
case NOTIFICATION_PAUSED : {
2023-06-15 15:35:53 +02:00
if ( agent_parent ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_paused ( get_rid ( ) , ! agent_parent - > can_process ( ) ) ;
2022-05-17 20:08:39 +02:00
}
} break ;
case NOTIFICATION_UNPAUSED : {
2023-06-15 15:35:53 +02:00
if ( agent_parent ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_paused ( get_rid ( ) , ! agent_parent - > can_process ( ) ) ;
2022-05-17 20:08:39 +02:00
}
} break ;
2020-01-10 12:22:34 +01:00
case NOTIFICATION_INTERNAL_PHYSICS_PROCESS : {
2023-06-30 00:11:26 +02:00
if ( agent_parent & & avoidance_enabled ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_position ( agent , agent_parent - > get_global_position ( ) ) ;
}
2022-12-06 22:30:34 +01:00
if ( agent_parent & & target_position_submitted ) {
2023-01-10 07:14:16 +01:00
if ( velocity_submitted ) {
velocity_submitted = false ;
if ( avoidance_enabled ) {
if ( ! use_3d_avoidance ) {
stored_y_velocity = velocity . y ;
velocity . y = 0.0 ;
}
NavigationServer3D : : get_singleton ( ) - > agent_set_velocity ( agent , velocity ) ;
}
}
if ( velocity_forced_submitted ) {
velocity_forced_submitted = false ;
if ( avoidance_enabled ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_velocity_forced ( agent , velocity_forced ) ;
}
2022-06-01 06:45:12 +02:00
}
2021-04-27 12:31:50 +02:00
_check_distance_to_target ( ) ;
2020-01-10 12:22:34 +01:00
}
2023-01-16 17:45:03 +01:00
# ifdef DEBUG_ENABLED
if ( debug_path_dirty ) {
_update_debug_path ( ) ;
}
# endif // DEBUG_ENABLED
2020-01-10 12:22:34 +01:00
} break ;
}
}
2020-05-12 17:01:17 +02:00
NavigationAgent3D : : NavigationAgent3D ( ) {
2020-03-27 19:21:27 +01:00
agent = NavigationServer3D : : get_singleton ( ) - > agent_create ( ) ;
2023-01-10 07:14:16 +01:00
2023-02-02 08:38:28 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_neighbor_distance ( agent , neighbor_distance ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_max_neighbors ( agent , max_neighbors ) ;
2023-01-10 07:14:16 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_time_horizon_agents ( agent , time_horizon_agents ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_time_horizon_obstacles ( agent , time_horizon_obstacles ) ;
2023-02-02 08:38:28 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_radius ( agent , radius ) ;
2023-01-10 07:14:16 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_height ( agent , height ) ;
2023-02-02 08:38:28 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_max_speed ( agent , max_speed ) ;
2022-09-20 07:56:54 +02:00
// Preallocate query and result objects to improve performance.
navigation_query = Ref < NavigationPathQueryParameters3D > ( ) ;
navigation_query . instantiate ( ) ;
navigation_result = Ref < NavigationPathQueryResult3D > ( ) ;
navigation_result . instantiate ( ) ;
2023-01-16 17:45:03 +01:00
2023-01-10 07:14:16 +01:00
set_avoidance_layers ( avoidance_layers ) ;
set_avoidance_mask ( avoidance_mask ) ;
set_avoidance_priority ( avoidance_priority ) ;
set_use_3d_avoidance ( use_3d_avoidance ) ;
set_avoidance_enabled ( avoidance_enabled ) ;
2023-01-16 17:45:03 +01:00
# ifdef DEBUG_ENABLED
NavigationServer3D : : get_singleton ( ) - > connect ( SNAME ( " navigation_debug_changed " ) , callable_mp ( this , & NavigationAgent3D : : _navigation_debug_changed ) ) ;
# endif // DEBUG_ENABLED
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
NavigationAgent3D : : ~ NavigationAgent3D ( ) {
2022-12-12 18:42:37 +01:00
ERR_FAIL_NULL ( NavigationServer3D : : get_singleton ( ) ) ;
2020-03-27 19:21:27 +01:00
NavigationServer3D : : get_singleton ( ) - > free ( agent ) ;
2020-01-10 12:22:34 +01:00
agent = RID ( ) ; // Pointless
2023-01-16 17:45:03 +01:00
# ifdef DEBUG_ENABLED
NavigationServer3D : : get_singleton ( ) - > disconnect ( SNAME ( " navigation_debug_changed " ) , callable_mp ( this , & NavigationAgent3D : : _navigation_debug_changed ) ) ;
ERR_FAIL_NULL ( RenderingServer : : get_singleton ( ) ) ;
if ( debug_path_instance . is_valid ( ) ) {
RenderingServer : : get_singleton ( ) - > free ( debug_path_instance ) ;
}
if ( debug_path_mesh . is_valid ( ) ) {
RenderingServer : : get_singleton ( ) - > free ( debug_path_mesh - > get_rid ( ) ) ;
}
# endif // DEBUG_ENABLED
2020-01-10 12:22:34 +01:00
}
2022-05-20 22:55:27 +02:00
void NavigationAgent3D : : set_avoidance_enabled ( bool p_enabled ) {
2023-02-02 08:38:28 +01:00
if ( avoidance_enabled = = p_enabled ) {
return ;
}
2022-05-20 22:55:27 +02:00
avoidance_enabled = p_enabled ;
2023-02-02 08:38:28 +01:00
2022-05-20 22:55:27 +02:00
if ( avoidance_enabled ) {
2023-01-10 07:14:16 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_enabled ( agent , true ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_callback ( agent , callable_mp ( this , & NavigationAgent3D : : _avoidance_done ) ) ;
2022-05-20 22:55:27 +02:00
} else {
2023-01-10 07:14:16 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_enabled ( agent , false ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_callback ( agent , Callable ( ) ) ;
2022-05-20 22:55:27 +02:00
}
}
bool NavigationAgent3D : : get_avoidance_enabled ( ) const {
return avoidance_enabled ;
}
2022-06-01 06:45:12 +02:00
void NavigationAgent3D : : set_agent_parent ( Node * p_agent_parent ) {
2023-02-02 08:38:28 +01:00
if ( agent_parent = = p_agent_parent ) {
return ;
}
2022-06-01 06:45:12 +02:00
// remove agent from any avoidance map before changing parent or there will be leftovers on the RVO map
2023-01-10 07:14:16 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_callback ( agent , Callable ( ) ) ;
2023-01-28 05:15:10 +01:00
2022-06-01 06:45:12 +02:00
if ( Object : : cast_to < Node3D > ( p_agent_parent ) ! = nullptr ) {
// place agent on navigation map first or else the RVO agent callback creation fails silently later
agent_parent = Object : : cast_to < Node3D > ( p_agent_parent ) ;
2022-06-14 20:39:15 +02:00
if ( map_override . is_valid ( ) ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , map_override ) ;
} else {
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , agent_parent - > get_world_3d ( ) - > get_navigation_map ( ) ) ;
}
2023-01-28 05:15:10 +01:00
2022-06-01 06:45:12 +02:00
// create new avoidance callback if enabled
2023-02-02 08:38:28 +01:00
if ( avoidance_enabled ) {
2023-01-10 07:14:16 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_callback ( agent , callable_mp ( this , & NavigationAgent3D : : _avoidance_done ) ) ;
2023-02-02 08:38:28 +01:00
}
2022-06-01 06:45:12 +02:00
} else {
agent_parent = nullptr ;
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , RID ( ) ) ;
}
}
2022-06-14 23:34:43 +02:00
void NavigationAgent3D : : set_navigation_layers ( uint32_t p_navigation_layers ) {
2023-02-02 08:38:28 +01:00
if ( navigation_layers = = p_navigation_layers ) {
return ;
2022-03-19 21:47:02 +01:00
}
2023-02-02 08:38:28 +01:00
navigation_layers = p_navigation_layers ;
_request_repath ( ) ;
2022-03-19 21:47:02 +01:00
}
2022-06-14 23:34:43 +02:00
uint32_t NavigationAgent3D : : get_navigation_layers ( ) const {
return navigation_layers ;
2022-03-19 21:47:02 +01:00
}
2022-06-19 13:23:13 +02:00
void NavigationAgent3D : : set_navigation_layer_value ( int p_layer_number , bool p_value ) {
ERR_FAIL_COND_MSG ( p_layer_number < 1 , " Navigation layer number must be between 1 and 32 inclusive. " ) ;
ERR_FAIL_COND_MSG ( p_layer_number > 32 , " Navigation layer number must be between 1 and 32 inclusive. " ) ;
uint32_t _navigation_layers = get_navigation_layers ( ) ;
if ( p_value ) {
_navigation_layers | = 1 < < ( p_layer_number - 1 ) ;
} else {
_navigation_layers & = ~ ( 1 < < ( p_layer_number - 1 ) ) ;
}
set_navigation_layers ( _navigation_layers ) ;
}
bool NavigationAgent3D : : get_navigation_layer_value ( int p_layer_number ) const {
ERR_FAIL_COND_V_MSG ( p_layer_number < 1 , false , " Navigation layer number must be between 1 and 32 inclusive. " ) ;
ERR_FAIL_COND_V_MSG ( p_layer_number > 32 , false , " Navigation layer number must be between 1 and 32 inclusive. " ) ;
return get_navigation_layers ( ) & ( 1 < < ( p_layer_number - 1 ) ) ;
}
2023-03-25 01:00:09 +01:00
void NavigationAgent3D : : set_pathfinding_algorithm ( const NavigationPathQueryParameters3D : : PathfindingAlgorithm p_pathfinding_algorithm ) {
if ( pathfinding_algorithm = = p_pathfinding_algorithm ) {
return ;
}
pathfinding_algorithm = p_pathfinding_algorithm ;
navigation_query - > set_pathfinding_algorithm ( pathfinding_algorithm ) ;
}
void NavigationAgent3D : : set_path_postprocessing ( const NavigationPathQueryParameters3D : : PathPostProcessing p_path_postprocessing ) {
if ( path_postprocessing = = p_path_postprocessing ) {
return ;
}
path_postprocessing = p_path_postprocessing ;
navigation_query - > set_path_postprocessing ( path_postprocessing ) ;
}
2022-10-06 01:24:45 +02:00
void NavigationAgent3D : : set_path_metadata_flags ( BitField < NavigationPathQueryParameters3D : : PathMetadataFlags > p_path_metadata_flags ) {
if ( path_metadata_flags = = p_path_metadata_flags ) {
return ;
}
path_metadata_flags = p_path_metadata_flags ;
}
2022-06-14 20:39:15 +02:00
void NavigationAgent3D : : set_navigation_map ( RID p_navigation_map ) {
2023-02-02 08:38:28 +01:00
if ( map_override = = p_navigation_map ) {
return ;
}
2022-06-14 20:39:15 +02:00
map_override = p_navigation_map ;
2023-02-02 08:38:28 +01:00
2022-06-14 20:39:15 +02:00
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( agent , map_override ) ;
_request_repath ( ) ;
}
RID NavigationAgent3D : : get_navigation_map ( ) const {
if ( map_override . is_valid ( ) ) {
return map_override ;
} else if ( agent_parent ! = nullptr ) {
return agent_parent - > get_world_3d ( ) - > get_navigation_map ( ) ;
}
return RID ( ) ;
}
2023-02-02 08:38:28 +01:00
void NavigationAgent3D : : set_path_desired_distance ( real_t p_path_desired_distance ) {
if ( Math : : is_equal_approx ( path_desired_distance , p_path_desired_distance ) ) {
return ;
}
path_desired_distance = p_path_desired_distance ;
2022-06-18 19:11:49 +02:00
}
2023-02-02 08:38:28 +01:00
void NavigationAgent3D : : set_target_desired_distance ( real_t p_target_desired_distance ) {
if ( Math : : is_equal_approx ( target_desired_distance , p_target_desired_distance ) ) {
return ;
}
target_desired_distance = p_target_desired_distance ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
void NavigationAgent3D : : set_radius ( real_t p_radius ) {
2023-01-10 07:14:16 +01:00
ERR_FAIL_COND_MSG ( p_radius < 0.0 , " Radius must be positive. " ) ;
2023-02-02 08:38:28 +01:00
if ( Math : : is_equal_approx ( radius , p_radius ) ) {
return ;
}
2020-01-10 12:22:34 +01:00
radius = p_radius ;
2023-02-02 08:38:28 +01:00
2020-03-27 19:21:27 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_radius ( agent , radius ) ;
2020-01-10 12:22:34 +01:00
}
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_height ( real_t p_height ) {
ERR_FAIL_COND_MSG ( p_height < 0.0 , " Height must be positive. " ) ;
if ( Math : : is_equal_approx ( height , p_height ) ) {
2023-02-02 08:38:28 +01:00
return ;
}
2023-01-10 07:14:16 +01:00
height = p_height ;
NavigationServer3D : : get_singleton ( ) - > agent_set_height ( agent , height ) ;
2020-01-10 12:22:34 +01:00
}
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_path_height_offset ( real_t p_path_height_offset ) {
path_height_offset = p_path_height_offset ;
}
2023-02-02 08:38:28 +01:00
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_use_3d_avoidance ( bool p_use_3d_avoidance ) {
use_3d_avoidance = p_use_3d_avoidance ;
NavigationServer3D : : get_singleton ( ) - > agent_set_use_3d_avoidance ( agent , use_3d_avoidance ) ;
notify_property_list_changed ( ) ;
2020-01-10 12:22:34 +01:00
}
2022-08-13 12:26:13 +02:00
void NavigationAgent3D : : set_neighbor_distance ( real_t p_distance ) {
2023-02-02 08:38:28 +01:00
if ( Math : : is_equal_approx ( neighbor_distance , p_distance ) ) {
return ;
}
2022-08-13 12:26:13 +02:00
neighbor_distance = p_distance ;
2023-02-02 08:38:28 +01:00
2022-08-13 12:26:13 +02:00
NavigationServer3D : : get_singleton ( ) - > agent_set_neighbor_distance ( agent , neighbor_distance ) ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
void NavigationAgent3D : : set_max_neighbors ( int p_count ) {
2023-02-02 08:38:28 +01:00
if ( max_neighbors = = p_count ) {
return ;
}
2020-01-10 12:22:34 +01:00
max_neighbors = p_count ;
2023-02-02 08:38:28 +01:00
2020-03-27 19:21:27 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_max_neighbors ( agent , max_neighbors ) ;
2020-01-10 12:22:34 +01:00
}
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_time_horizon_agents ( real_t p_time_horizon ) {
ERR_FAIL_COND_MSG ( p_time_horizon < 0.0 , " Time horizion must be positive. " ) ;
if ( Math : : is_equal_approx ( time_horizon_agents , p_time_horizon ) ) {
2023-02-02 08:38:28 +01:00
return ;
}
2023-01-10 07:14:16 +01:00
time_horizon_agents = p_time_horizon ;
NavigationServer3D : : get_singleton ( ) - > agent_set_time_horizon_agents ( agent , time_horizon_agents ) ;
}
2023-02-02 08:38:28 +01:00
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_time_horizon_obstacles ( real_t p_time_horizon ) {
ERR_FAIL_COND_MSG ( p_time_horizon < 0.0 , " Time horizion must be positive. " ) ;
if ( Math : : is_equal_approx ( time_horizon_obstacles , p_time_horizon ) ) {
return ;
}
time_horizon_obstacles = p_time_horizon ;
NavigationServer3D : : get_singleton ( ) - > agent_set_time_horizon_obstacles ( agent , time_horizon_obstacles ) ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
void NavigationAgent3D : : set_max_speed ( real_t p_max_speed ) {
2023-01-10 07:14:16 +01:00
ERR_FAIL_COND_MSG ( p_max_speed < 0.0 , " Max speed must be positive. " ) ;
2023-02-02 08:38:28 +01:00
if ( Math : : is_equal_approx ( max_speed , p_max_speed ) ) {
return ;
}
2020-01-10 12:22:34 +01:00
max_speed = p_max_speed ;
2023-02-02 08:38:28 +01:00
2020-03-27 19:21:27 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_max_speed ( agent , max_speed ) ;
2020-01-10 12:22:34 +01:00
}
2023-02-02 08:38:28 +01:00
void NavigationAgent3D : : set_path_max_distance ( real_t p_path_max_distance ) {
if ( Math : : is_equal_approx ( path_max_distance , p_path_max_distance ) ) {
return ;
}
path_max_distance = p_path_max_distance ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
real_t NavigationAgent3D : : get_path_max_distance ( ) {
2020-01-10 12:22:34 +01:00
return path_max_distance ;
}
2022-12-06 23:32:11 +01:00
void NavigationAgent3D : : set_target_position ( Vector3 p_position ) {
2023-02-09 08:18:53 +01:00
// Intentionally not checking for equality of the parameter, as we want to update the path even if the target position is the same in case the world changed.
// Revisit later when the navigation server can update the path without requesting a new path.
2023-02-02 08:38:28 +01:00
2022-12-06 23:32:11 +01:00
target_position = p_position ;
2022-12-06 22:30:34 +01:00
target_position_submitted = true ;
2023-02-02 08:38:28 +01:00
2022-03-19 21:47:02 +01:00
_request_repath ( ) ;
2020-01-10 12:22:34 +01:00
}
2022-12-06 23:32:11 +01:00
Vector3 NavigationAgent3D : : get_target_position ( ) const {
return target_position ;
2020-01-10 12:22:34 +01:00
}
2022-12-06 23:32:11 +01:00
Vector3 NavigationAgent3D : : get_next_path_position ( ) {
2020-01-10 12:22:34 +01:00
update_navigation ( ) ;
2022-09-20 07:56:54 +02:00
const Vector < Vector3 > & navigation_path = navigation_result - > get_path ( ) ;
2020-01-10 12:22:34 +01:00
if ( navigation_path . size ( ) = = 0 ) {
2023-09-09 17:52:40 +02:00
ERR_FAIL_NULL_V_MSG ( agent_parent , Vector3 ( ) , " The agent has no parent. " ) ;
2023-02-02 08:38:28 +01:00
return agent_parent - > get_global_position ( ) ;
2020-01-10 12:22:34 +01:00
} else {
2023-01-10 07:14:16 +01:00
return navigation_path [ navigation_path_index ] - Vector3 ( 0 , path_height_offset , 0 ) ;
2020-01-10 12:22:34 +01:00
}
}
2020-03-26 22:49:16 +01:00
real_t NavigationAgent3D : : distance_to_target ( ) const {
2023-09-09 17:52:40 +02:00
ERR_FAIL_NULL_V_MSG ( agent_parent , 0.0 , " The agent has no parent. " ) ;
2023-02-02 08:38:28 +01:00
return agent_parent - > get_global_position ( ) . distance_to ( target_position ) ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
bool NavigationAgent3D : : is_target_reached ( ) const {
2020-01-10 12:22:34 +01:00
return target_reached ;
}
2020-03-26 22:49:16 +01:00
bool NavigationAgent3D : : is_target_reachable ( ) {
2022-12-06 23:32:11 +01:00
return target_desired_distance > = get_final_position ( ) . distance_to ( target_position ) ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
bool NavigationAgent3D : : is_navigation_finished ( ) {
2020-01-10 12:22:34 +01:00
update_navigation ( ) ;
return navigation_finished ;
}
2022-12-06 23:32:11 +01:00
Vector3 NavigationAgent3D : : get_final_position ( ) {
2020-01-10 12:22:34 +01:00
update_navigation ( ) ;
2022-09-20 07:56:54 +02:00
const Vector < Vector3 > & navigation_path = navigation_result - > get_path ( ) ;
2020-01-10 12:22:34 +01:00
if ( navigation_path . size ( ) = = 0 ) {
return Vector3 ( ) ;
}
return navigation_path [ navigation_path . size ( ) - 1 ] ;
}
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_velocity_forced ( Vector3 p_velocity ) {
2023-02-09 08:18:53 +01:00
// Intentionally not checking for equality of the parameter.
// We need to always submit the velocity to the navigation server, even when it is the same, in order to run avoidance every frame.
// Revisit later when the navigation server can update avoidance without users resubmitting the velocity.
2023-02-02 08:38:28 +01:00
2023-01-10 07:14:16 +01:00
velocity_forced = p_velocity ;
velocity_forced_submitted = true ;
}
2023-02-02 08:38:28 +01:00
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_velocity ( const Vector3 p_velocity ) {
velocity = p_velocity ;
velocity_submitted = true ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
void NavigationAgent3D : : _avoidance_done ( Vector3 p_new_velocity ) {
2023-01-10 07:14:16 +01:00
safe_velocity = p_new_velocity ;
if ( ! use_3d_avoidance ) {
safe_velocity . y = stored_y_velocity ;
2020-01-10 12:22:34 +01:00
}
2023-01-10 07:14:16 +01:00
emit_signal ( SNAME ( " velocity_computed " ) , safe_velocity ) ;
2020-01-10 12:22:34 +01:00
}
2022-09-19 17:43:15 +02:00
PackedStringArray NavigationAgent3D : : get_configuration_warnings ( ) const {
PackedStringArray warnings = Node : : get_configuration_warnings ( ) ;
2020-05-14 22:59:27 +02:00
2020-03-26 22:49:16 +01:00
if ( ! Object : : cast_to < Node3D > ( get_parent ( ) ) ) {
2022-05-20 22:55:27 +02:00
warnings . push_back ( RTR ( " The NavigationAgent3D can be used only under a Node3D inheriting parent node. " ) ) ;
2020-01-10 12:22:34 +01:00
}
2020-10-29 11:01:28 +01:00
return warnings ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
void NavigationAgent3D : : update_navigation ( ) {
2020-05-14 16:41:43 +02:00
if ( agent_parent = = nullptr ) {
2020-05-10 12:56:01 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2021-03-08 09:47:18 +01:00
if ( ! agent_parent - > is_inside_tree ( ) ) {
2020-05-10 12:56:01 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2022-12-06 22:30:34 +01:00
if ( ! target_position_submitted ) {
return ;
}
2020-01-10 12:22:34 +01:00
update_frame_id = Engine : : get_singleton ( ) - > get_physics_frames ( ) ;
2023-02-02 08:38:28 +01:00
Vector3 origin = agent_parent - > get_global_position ( ) ;
2020-01-10 12:22:34 +01:00
bool reload_path = false ;
2020-03-27 19:21:27 +01:00
if ( NavigationServer3D : : get_singleton ( ) - > agent_is_map_changed ( agent ) ) {
2020-01-10 12:22:34 +01:00
reload_path = true ;
2022-09-20 07:56:54 +02:00
} else if ( navigation_result - > get_path ( ) . size ( ) = = 0 ) {
2020-01-10 12:22:34 +01:00
reload_path = true ;
} else {
// Check if too far from the navigation path
2022-12-11 18:02:35 +01:00
if ( navigation_path_index > 0 ) {
2022-09-20 07:56:54 +02:00
const Vector < Vector3 > & navigation_path = navigation_result - > get_path ( ) ;
2020-01-10 12:22:34 +01:00
Vector3 segment [ 2 ] ;
2022-12-11 18:02:35 +01:00
segment [ 0 ] = navigation_path [ navigation_path_index - 1 ] ;
segment [ 1 ] = navigation_path [ navigation_path_index ] ;
2023-01-10 07:14:16 +01:00
segment [ 0 ] . y - = path_height_offset ;
segment [ 1 ] . y - = path_height_offset ;
2022-09-20 07:56:54 +02:00
Vector3 p = Geometry3D : : get_closest_point_to_segment ( origin , segment ) ;
if ( origin . distance_to ( p ) > = path_max_distance ) {
2020-01-10 12:22:34 +01:00
// To faraway, reload path
reload_path = true ;
}
}
}
if ( reload_path ) {
2022-09-20 07:56:54 +02:00
navigation_query - > set_start_position ( origin ) ;
2022-12-06 23:32:11 +01:00
navigation_query - > set_target_position ( target_position ) ;
2022-09-20 07:56:54 +02:00
navigation_query - > set_navigation_layers ( navigation_layers ) ;
2022-10-06 01:24:45 +02:00
navigation_query - > set_metadata_flags ( path_metadata_flags ) ;
2022-09-20 07:56:54 +02:00
2022-06-14 20:39:15 +02:00
if ( map_override . is_valid ( ) ) {
2022-09-20 07:56:54 +02:00
navigation_query - > set_map ( map_override ) ;
2022-06-14 20:39:15 +02:00
} else {
2022-09-20 07:56:54 +02:00
navigation_query - > set_map ( agent_parent - > get_world_3d ( ) - > get_navigation_map ( ) ) ;
2022-06-14 20:39:15 +02:00
}
2022-09-20 07:56:54 +02:00
NavigationServer3D : : get_singleton ( ) - > query_path ( navigation_query , navigation_result ) ;
2023-01-16 17:45:03 +01:00
# ifdef DEBUG_ENABLED
debug_path_dirty = true ;
# endif // DEBUG_ENABLED
2020-01-10 12:22:34 +01:00
navigation_finished = false ;
2022-12-11 18:02:35 +01:00
navigation_path_index = 0 ;
2021-07-17 23:22:52 +02:00
emit_signal ( SNAME ( " path_changed " ) ) ;
2020-01-10 12:22:34 +01:00
}
2022-09-20 07:56:54 +02:00
if ( navigation_result - > get_path ( ) . size ( ) = = 0 ) {
2020-01-10 12:22:34 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2020-01-10 12:22:34 +01:00
// Check if we can advance the navigation path
if ( navigation_finished = = false ) {
2022-12-06 23:32:11 +01:00
// Advances to the next far away position.
2022-09-20 07:56:54 +02:00
const Vector < Vector3 > & navigation_path = navigation_result - > get_path ( ) ;
2022-10-06 01:24:45 +02:00
const Vector < int32_t > & navigation_path_types = navigation_result - > get_path_types ( ) ;
const TypedArray < RID > & navigation_path_rids = navigation_result - > get_path_rids ( ) ;
const Vector < int64_t > & navigation_path_owners = navigation_result - > get_path_owner_ids ( ) ;
2023-01-10 07:14:16 +01:00
while ( origin . distance_to ( navigation_path [ navigation_path_index ] - Vector3 ( 0 , path_height_offset , 0 ) ) < path_desired_distance ) {
2022-10-06 01:24:45 +02:00
Dictionary details ;
const Vector3 waypoint = navigation_path [ navigation_path_index ] ;
2022-12-06 23:32:11 +01:00
details [ SNAME ( " position " ) ] = waypoint ;
2022-10-06 01:24:45 +02:00
int waypoint_type = - 1 ;
if ( path_metadata_flags . has_flag ( NavigationPathQueryParameters3D : : PathMetadataFlags : : PATH_METADATA_INCLUDE_TYPES ) ) {
const NavigationPathQueryResult3D : : PathSegmentType type = NavigationPathQueryResult3D : : PathSegmentType ( navigation_path_types [ navigation_path_index ] ) ;
details [ SNAME ( " type " ) ] = type ;
waypoint_type = type ;
}
if ( path_metadata_flags . has_flag ( NavigationPathQueryParameters3D : : PathMetadataFlags : : PATH_METADATA_INCLUDE_RIDS ) ) {
details [ SNAME ( " rid " ) ] = navigation_path_rids [ navigation_path_index ] ;
}
if ( path_metadata_flags . has_flag ( NavigationPathQueryParameters3D : : PathMetadataFlags : : PATH_METADATA_INCLUDE_OWNERS ) ) {
const ObjectID waypoint_owner_id = ObjectID ( navigation_path_owners [ navigation_path_index ] ) ;
// Get a reference to the owning object.
Object * owner = nullptr ;
if ( waypoint_owner_id . is_valid ( ) ) {
owner = ObjectDB : : get_instance ( waypoint_owner_id ) ;
}
details [ SNAME ( " owner " ) ] = owner ;
2023-02-13 13:18:13 +01:00
if ( waypoint_type = = NavigationPathQueryResult3D : : PATH_SEGMENT_TYPE_LINK ) {
const NavigationLink3D * navlink = Object : : cast_to < NavigationLink3D > ( owner ) ;
if ( navlink ) {
Vector3 link_global_start_position = navlink - > get_global_start_position ( ) ;
Vector3 link_global_end_position = navlink - > get_global_end_position ( ) ;
if ( waypoint . distance_to ( link_global_start_position ) < waypoint . distance_to ( link_global_end_position ) ) {
details [ SNAME ( " link_entry_position " ) ] = link_global_start_position ;
details [ SNAME ( " link_exit_position " ) ] = link_global_end_position ;
} else {
details [ SNAME ( " link_entry_position " ) ] = link_global_end_position ;
details [ SNAME ( " link_exit_position " ) ] = link_global_start_position ;
}
}
}
2022-10-06 01:24:45 +02:00
}
// Emit a signal for the waypoint
emit_signal ( SNAME ( " waypoint_reached " ) , details ) ;
// Emit a signal if we've reached a navigation link
if ( waypoint_type = = NavigationPathQueryResult3D : : PATH_SEGMENT_TYPE_LINK ) {
emit_signal ( SNAME ( " link_reached " ) , details ) ;
}
// Move to the next waypoint on the list
2022-12-11 18:02:35 +01:00
navigation_path_index + = 1 ;
2022-10-06 01:24:45 +02:00
// Check to see if we've finished our route
2022-12-11 18:02:35 +01:00
if ( navigation_path_index = = navigation_path . size ( ) ) {
2021-04-27 12:31:50 +02:00
_check_distance_to_target ( ) ;
2022-12-11 18:02:35 +01:00
navigation_path_index - = 1 ;
2020-01-10 12:22:34 +01:00
navigation_finished = true ;
2022-12-06 22:30:34 +01:00
target_position_submitted = false ;
2023-06-29 21:09:26 +02:00
if ( avoidance_enabled ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_position ( agent , agent_parent - > get_global_transform ( ) . origin ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_velocity ( agent , Vector3 ( 0.0 , 0.0 , 0.0 ) ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_velocity_forced ( agent , Vector3 ( 0.0 , 0.0 , 0.0 ) ) ;
}
2021-07-17 23:22:52 +02:00
emit_signal ( SNAME ( " navigation_finished " ) ) ;
2020-01-10 12:22:34 +01:00
break ;
}
}
}
}
2021-04-27 12:31:50 +02:00
2022-03-19 21:47:02 +01:00
void NavigationAgent3D : : _request_repath ( ) {
2022-09-20 07:56:54 +02:00
navigation_result - > reset ( ) ;
2022-03-19 21:47:02 +01:00
target_reached = false ;
navigation_finished = false ;
update_frame_id = 0 ;
}
2021-04-27 12:31:50 +02:00
void NavigationAgent3D : : _check_distance_to_target ( ) {
if ( ! target_reached ) {
if ( distance_to_target ( ) < target_desired_distance ) {
target_reached = true ;
2022-10-27 10:17:10 +02:00
emit_signal ( SNAME ( " target_reached " ) ) ;
2021-04-27 12:31:50 +02:00
}
}
}
2023-01-16 17:45:03 +01:00
2023-01-10 07:14:16 +01:00
void NavigationAgent3D : : set_avoidance_layers ( uint32_t p_layers ) {
avoidance_layers = p_layers ;
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_layers ( get_rid ( ) , avoidance_layers ) ;
}
uint32_t NavigationAgent3D : : get_avoidance_layers ( ) const {
return avoidance_layers ;
}
void NavigationAgent3D : : set_avoidance_mask ( uint32_t p_mask ) {
avoidance_mask = p_mask ;
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_mask ( get_rid ( ) , avoidance_mask ) ;
}
uint32_t NavigationAgent3D : : get_avoidance_mask ( ) const {
return avoidance_mask ;
}
void NavigationAgent3D : : set_avoidance_layer_value ( int p_layer_number , bool p_value ) {
ERR_FAIL_COND_MSG ( p_layer_number < 1 , " Avoidance layer number must be between 1 and 32 inclusive. " ) ;
ERR_FAIL_COND_MSG ( p_layer_number > 32 , " Avoidance layer number must be between 1 and 32 inclusive. " ) ;
uint32_t avoidance_layers_new = get_avoidance_layers ( ) ;
if ( p_value ) {
avoidance_layers_new | = 1 < < ( p_layer_number - 1 ) ;
} else {
avoidance_layers_new & = ~ ( 1 < < ( p_layer_number - 1 ) ) ;
}
set_avoidance_layers ( avoidance_layers_new ) ;
}
bool NavigationAgent3D : : get_avoidance_layer_value ( int p_layer_number ) const {
ERR_FAIL_COND_V_MSG ( p_layer_number < 1 , false , " Avoidance layer number must be between 1 and 32 inclusive. " ) ;
ERR_FAIL_COND_V_MSG ( p_layer_number > 32 , false , " Avoidance layer number must be between 1 and 32 inclusive. " ) ;
return get_avoidance_layers ( ) & ( 1 < < ( p_layer_number - 1 ) ) ;
}
void NavigationAgent3D : : set_avoidance_mask_value ( int p_mask_number , bool p_value ) {
ERR_FAIL_COND_MSG ( p_mask_number < 1 , " Avoidance mask number must be between 1 and 32 inclusive. " ) ;
ERR_FAIL_COND_MSG ( p_mask_number > 32 , " Avoidance mask number must be between 1 and 32 inclusive. " ) ;
uint32_t mask = get_avoidance_mask ( ) ;
if ( p_value ) {
mask | = 1 < < ( p_mask_number - 1 ) ;
} else {
mask & = ~ ( 1 < < ( p_mask_number - 1 ) ) ;
}
set_avoidance_mask ( mask ) ;
}
bool NavigationAgent3D : : get_avoidance_mask_value ( int p_mask_number ) const {
ERR_FAIL_COND_V_MSG ( p_mask_number < 1 , false , " Avoidance mask number must be between 1 and 32 inclusive. " ) ;
ERR_FAIL_COND_V_MSG ( p_mask_number > 32 , false , " Avoidance mask number must be between 1 and 32 inclusive. " ) ;
return get_avoidance_mask ( ) & ( 1 < < ( p_mask_number - 1 ) ) ;
}
void NavigationAgent3D : : set_avoidance_priority ( real_t p_priority ) {
ERR_FAIL_COND_MSG ( p_priority < 0.0 , " Avoidance priority must be between 0.0 and 1.0 inclusive. " ) ;
ERR_FAIL_COND_MSG ( p_priority > 1.0 , " Avoidance priority must be between 0.0 and 1.0 inclusive. " ) ;
avoidance_priority = p_priority ;
NavigationServer3D : : get_singleton ( ) - > agent_set_avoidance_priority ( get_rid ( ) , p_priority ) ;
}
real_t NavigationAgent3D : : get_avoidance_priority ( ) const {
return avoidance_priority ;
}
2023-01-16 17:45:03 +01:00
////////DEBUG////////////////////////////////////////////////////////////
void NavigationAgent3D : : set_debug_enabled ( bool p_enabled ) {
2023-02-08 20:37:57 +01:00
# ifdef DEBUG_ENABLED
2023-02-02 08:38:28 +01:00
if ( debug_enabled = = p_enabled ) {
return ;
}
2023-01-16 17:45:03 +01:00
debug_enabled = p_enabled ;
debug_path_dirty = true ;
2023-02-08 20:37:57 +01:00
# endif // DEBUG_ENABLED
2023-01-16 17:45:03 +01:00
}
bool NavigationAgent3D : : get_debug_enabled ( ) const {
return debug_enabled ;
}
void NavigationAgent3D : : set_debug_use_custom ( bool p_enabled ) {
2023-02-08 20:37:57 +01:00
# ifdef DEBUG_ENABLED
2023-02-02 08:38:28 +01:00
if ( debug_use_custom = = p_enabled ) {
return ;
}
2023-01-16 17:45:03 +01:00
debug_use_custom = p_enabled ;
debug_path_dirty = true ;
2023-02-08 20:37:57 +01:00
# endif // DEBUG_ENABLED
2023-01-16 17:45:03 +01:00
}
bool NavigationAgent3D : : get_debug_use_custom ( ) const {
return debug_use_custom ;
}
void NavigationAgent3D : : set_debug_path_custom_color ( Color p_color ) {
2023-02-08 20:37:57 +01:00
# ifdef DEBUG_ENABLED
2023-02-02 08:38:28 +01:00
if ( debug_path_custom_color = = p_color ) {
return ;
}
2023-01-16 17:45:03 +01:00
debug_path_custom_color = p_color ;
debug_path_dirty = true ;
2023-02-08 20:37:57 +01:00
# endif // DEBUG_ENABLED
2023-01-16 17:45:03 +01:00
}
Color NavigationAgent3D : : get_debug_path_custom_color ( ) const {
return debug_path_custom_color ;
}
void NavigationAgent3D : : set_debug_path_custom_point_size ( float p_point_size ) {
2023-02-08 20:37:57 +01:00
# ifdef DEBUG_ENABLED
2023-02-02 08:38:28 +01:00
if ( Math : : is_equal_approx ( debug_path_custom_point_size , p_point_size ) ) {
return ;
}
2023-03-16 09:15:34 +01:00
debug_path_custom_point_size = MAX ( 0.0 , p_point_size ) ;
2023-01-16 17:45:03 +01:00
debug_path_dirty = true ;
2023-02-08 20:37:57 +01:00
# endif // DEBUG_ENABLED
2023-01-16 17:45:03 +01:00
}
float NavigationAgent3D : : get_debug_path_custom_point_size ( ) const {
return debug_path_custom_point_size ;
}
2023-02-08 20:37:57 +01:00
# ifdef DEBUG_ENABLED
2023-01-16 17:45:03 +01:00
void NavigationAgent3D : : _navigation_debug_changed ( ) {
debug_path_dirty = true ;
}
void NavigationAgent3D : : _update_debug_path ( ) {
if ( ! debug_path_dirty ) {
return ;
}
debug_path_dirty = false ;
if ( ! debug_path_instance . is_valid ( ) ) {
debug_path_instance = RenderingServer : : get_singleton ( ) - > instance_create ( ) ;
}
if ( ! debug_path_mesh . is_valid ( ) ) {
debug_path_mesh = Ref < ArrayMesh > ( memnew ( ArrayMesh ) ) ;
}
debug_path_mesh - > clear_surfaces ( ) ;
if ( ! ( debug_enabled & & NavigationServer3D : : get_singleton ( ) - > get_debug_navigation_enable_agent_paths ( ) ) ) {
return ;
}
if ( ! ( agent_parent & & agent_parent - > is_inside_tree ( ) ) ) {
return ;
}
const Vector < Vector3 > & navigation_path = navigation_result - > get_path ( ) ;
if ( navigation_path . size ( ) < = 1 ) {
return ;
}
Vector < Vector3 > debug_path_lines_vertex_array ;
for ( int i = 0 ; i < navigation_path . size ( ) - 1 ; i + + ) {
debug_path_lines_vertex_array . push_back ( navigation_path [ i ] ) ;
debug_path_lines_vertex_array . push_back ( navigation_path [ i + 1 ] ) ;
}
Array debug_path_lines_mesh_array ;
debug_path_lines_mesh_array . resize ( Mesh : : ARRAY_MAX ) ;
debug_path_lines_mesh_array [ Mesh : : ARRAY_VERTEX ] = debug_path_lines_vertex_array ;
debug_path_mesh - > add_surface_from_arrays ( Mesh : : PRIMITIVE_LINES , debug_path_lines_mesh_array ) ;
Ref < StandardMaterial3D > debug_agent_path_line_material = NavigationServer3D : : get_singleton ( ) - > get_debug_navigation_agent_path_line_material ( ) ;
if ( debug_use_custom ) {
if ( ! debug_agent_path_line_custom_material . is_valid ( ) ) {
debug_agent_path_line_custom_material = debug_agent_path_line_material - > duplicate ( ) ;
}
debug_agent_path_line_custom_material - > set_albedo ( debug_path_custom_color ) ;
debug_path_mesh - > surface_set_material ( 0 , debug_agent_path_line_custom_material ) ;
} else {
debug_path_mesh - > surface_set_material ( 0 , debug_agent_path_line_material ) ;
}
2023-03-16 09:15:34 +01:00
if ( debug_path_custom_point_size > 0.0 ) {
Vector < Vector3 > debug_path_points_vertex_array ;
2023-01-16 17:45:03 +01:00
2023-03-16 09:15:34 +01:00
for ( int i = 0 ; i < navigation_path . size ( ) ; i + + ) {
debug_path_points_vertex_array . push_back ( navigation_path [ i ] ) ;
}
2023-01-16 17:45:03 +01:00
2023-03-16 09:15:34 +01:00
Array debug_path_points_mesh_array ;
debug_path_points_mesh_array . resize ( Mesh : : ARRAY_MAX ) ;
debug_path_points_mesh_array [ Mesh : : ARRAY_VERTEX ] = debug_path_points_vertex_array ;
2023-01-16 17:45:03 +01:00
2023-03-16 09:15:34 +01:00
debug_path_mesh - > add_surface_from_arrays ( Mesh : : PRIMITIVE_POINTS , debug_path_points_mesh_array ) ;
2023-01-16 17:45:03 +01:00
2023-03-16 09:15:34 +01:00
Ref < StandardMaterial3D > debug_agent_path_point_material = NavigationServer3D : : get_singleton ( ) - > get_debug_navigation_agent_path_point_material ( ) ;
if ( debug_use_custom ) {
if ( ! debug_agent_path_point_custom_material . is_valid ( ) ) {
debug_agent_path_point_custom_material = debug_agent_path_point_material - > duplicate ( ) ;
}
debug_agent_path_point_custom_material - > set_albedo ( debug_path_custom_color ) ;
debug_agent_path_point_custom_material - > set_point_size ( debug_path_custom_point_size ) ;
debug_path_mesh - > surface_set_material ( 1 , debug_agent_path_point_custom_material ) ;
} else {
debug_path_mesh - > surface_set_material ( 1 , debug_agent_path_point_material ) ;
2023-01-16 17:45:03 +01:00
}
}
RS : : get_singleton ( ) - > instance_set_base ( debug_path_instance , debug_path_mesh - > get_rid ( ) ) ;
RS : : get_singleton ( ) - > instance_set_scenario ( debug_path_instance , agent_parent - > get_world_3d ( ) - > get_scenario ( ) ) ;
RS : : get_singleton ( ) - > instance_set_visible ( debug_path_instance , agent_parent - > is_visible_in_tree ( ) ) ;
}
# endif // DEBUG_ENABLED