2023-01-05 13:25:55 +01:00
/**************************************************************************/
/* navigation_obstacle_2d.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
# include "navigation_obstacle_2d.h"
# include "scene/2d/collision_shape_2d.h"
2022-05-14 23:33:09 +02:00
# include "scene/2d/physics_body_2d.h"
2022-02-12 02:46:22 +01:00
# include "scene/resources/world_2d.h"
2020-03-27 19:21:27 +01:00
# include "servers/navigation_server_2d.h"
2020-01-10 12:22:34 +01:00
void NavigationObstacle2D : : _bind_methods ( ) {
2022-05-14 23:33:09 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_rid " ) , & NavigationObstacle2D : : get_rid ) ;
2022-08-14 16:23:27 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_navigation_map " , " navigation_map " ) , & NavigationObstacle2D : : set_navigation_map ) ;
ClassDB : : bind_method ( D_METHOD ( " get_navigation_map " ) , & NavigationObstacle2D : : get_navigation_map ) ;
2021-10-21 22:25:06 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_estimate_radius " , " estimate_radius " ) , & NavigationObstacle2D : : set_estimate_radius ) ;
ClassDB : : bind_method ( D_METHOD ( " is_radius_estimated " ) , & NavigationObstacle2D : : is_radius_estimated ) ;
ClassDB : : bind_method ( D_METHOD ( " set_radius " , " radius " ) , & NavigationObstacle2D : : set_radius ) ;
ClassDB : : bind_method ( D_METHOD ( " get_radius " ) , & NavigationObstacle2D : : get_radius ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " estimate_radius " ) , " set_estimate_radius " , " is_radius_estimated " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " radius " , PROPERTY_HINT_RANGE , " 0.01,500,0.01 " ) , " set_radius " , " get_radius " ) ;
}
void NavigationObstacle2D : : _validate_property ( PropertyInfo & p_property ) const {
if ( p_property . name = = " radius " ) {
if ( estimate_radius ) {
2021-11-03 23:06:17 +01:00
p_property . usage = PROPERTY_USAGE_NO_EDITOR ;
2021-10-21 22:25:06 +02:00
}
}
2020-01-10 12:22:34 +01:00
}
void NavigationObstacle2D : : _notification ( int p_what ) {
switch ( p_what ) {
2022-08-14 16:23:27 +02:00
case NOTIFICATION_POST_ENTER_TREE : {
set_agent_parent ( get_parent ( ) ) ;
2020-01-10 12:22:34 +01:00
set_physics_process_internal ( true ) ;
} break ;
2022-02-15 18:06:48 +01:00
2020-01-10 12:22:34 +01:00
case NOTIFICATION_EXIT_TREE : {
2022-08-14 16:23:27 +02:00
set_agent_parent ( nullptr ) ;
2020-01-10 12:22:34 +01:00
set_physics_process_internal ( false ) ;
} break ;
2022-02-15 18:06:48 +01:00
2021-03-08 09:47:18 +01:00
case NOTIFICATION_PARENTED : {
2022-08-14 16:23:27 +02:00
if ( is_inside_tree ( ) & & ( get_parent ( ) ! = parent_node2d ) ) {
set_agent_parent ( get_parent ( ) ) ;
set_physics_process_internal ( true ) ;
}
2021-03-08 09:47:18 +01:00
} break ;
2022-02-15 18:06:48 +01:00
2021-03-08 09:47:18 +01:00
case NOTIFICATION_UNPARENTED : {
2022-08-14 16:23:27 +02:00
set_agent_parent ( nullptr ) ;
set_physics_process_internal ( false ) ;
2021-03-08 09:47:18 +01:00
} break ;
2022-02-15 18:06:48 +01:00
2022-05-17 20:08:39 +02:00
case NOTIFICATION_PAUSED : {
if ( parent_node2d & & ! parent_node2d - > can_process ( ) ) {
map_before_pause = NavigationServer2D : : get_singleton ( ) - > agent_get_map ( get_rid ( ) ) ;
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , RID ( ) ) ;
} else if ( parent_node2d & & parent_node2d - > can_process ( ) & & ! ( map_before_pause = = RID ( ) ) ) {
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , map_before_pause ) ;
map_before_pause = RID ( ) ;
}
} break ;
case NOTIFICATION_UNPAUSED : {
if ( parent_node2d & & ! parent_node2d - > can_process ( ) ) {
map_before_pause = NavigationServer2D : : get_singleton ( ) - > agent_get_map ( get_rid ( ) ) ;
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , RID ( ) ) ;
} else if ( parent_node2d & & parent_node2d - > can_process ( ) & & ! ( map_before_pause = = RID ( ) ) ) {
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , map_before_pause ) ;
map_before_pause = RID ( ) ;
}
} break ;
2020-01-10 12:22:34 +01:00
case NOTIFICATION_INTERNAL_PHYSICS_PROCESS : {
2022-05-15 20:29:23 +02:00
if ( parent_node2d & & parent_node2d - > is_inside_tree ( ) ) {
2021-08-14 10:16:57 +02:00
NavigationServer2D : : get_singleton ( ) - > agent_set_position ( agent , parent_node2d - > get_global_position ( ) ) ;
2020-01-10 12:22:34 +01:00
}
} break ;
}
}
2020-05-12 17:01:17 +02:00
NavigationObstacle2D : : NavigationObstacle2D ( ) {
2020-03-27 19:21:27 +01:00
agent = NavigationServer2D : : get_singleton ( ) - > agent_create ( ) ;
2022-01-07 19:55:22 +01:00
initialize_agent ( ) ;
2020-01-10 12:22:34 +01:00
}
NavigationObstacle2D : : ~ NavigationObstacle2D ( ) {
2022-12-12 18:42:37 +01:00
ERR_FAIL_NULL ( NavigationServer2D : : get_singleton ( ) ) ;
2020-03-27 19:21:27 +01:00
NavigationServer2D : : get_singleton ( ) - > free ( agent ) ;
2020-01-10 12:22:34 +01:00
agent = RID ( ) ; // Pointless
}
2022-09-19 17:43:15 +02:00
PackedStringArray NavigationObstacle2D : : get_configuration_warnings ( ) const {
PackedStringArray warnings = Node : : get_configuration_warnings ( ) ;
2020-05-14 22:59:27 +02:00
2020-01-10 12:22:34 +01:00
if ( ! Object : : cast_to < Node2D > ( get_parent ( ) ) ) {
2022-03-28 15:24:14 +02:00
warnings . push_back ( RTR ( " The NavigationObstacle2D only serves to provide collision avoidance to a Node2D object. " ) ) ;
2020-01-10 12:22:34 +01:00
}
2022-05-14 23:33:09 +02:00
if ( Object : : cast_to < StaticBody2D > ( get_parent ( ) ) ) {
2022-08-25 19:35:52 +02:00
warnings . push_back ( RTR ( " The NavigationObstacle2D is intended for constantly moving bodies like CharacterBody2D or RigidBody2D as it creates only an RVO avoidance radius and does not follow scene geometry exactly. "
2022-05-14 23:33:09 +02:00
" \n Not constantly moving or complete static objects should be captured with a refreshed NavigationPolygon so agents can not only avoid them but also move along those objects outline at high detail " ) ) ;
}
2020-10-29 11:01:28 +01:00
return warnings ;
2020-01-10 12:22:34 +01:00
}
2021-10-21 22:25:06 +02:00
void NavigationObstacle2D : : initialize_agent ( ) {
2022-08-13 12:26:13 +02:00
NavigationServer2D : : get_singleton ( ) - > agent_set_neighbor_distance ( agent , 0.0 ) ;
2021-10-21 22:25:06 +02:00
NavigationServer2D : : get_singleton ( ) - > agent_set_max_neighbors ( agent , 0 ) ;
NavigationServer2D : : get_singleton ( ) - > agent_set_time_horizon ( agent , 0.0 ) ;
NavigationServer2D : : get_singleton ( ) - > agent_set_max_speed ( agent , 0.0 ) ;
}
void NavigationObstacle2D : : reevaluate_agent_radius ( ) {
if ( ! estimate_radius ) {
NavigationServer2D : : get_singleton ( ) - > agent_set_radius ( agent , radius ) ;
2022-01-07 19:55:22 +01:00
} else if ( parent_node2d & & parent_node2d - > is_inside_tree ( ) ) {
2021-10-21 22:25:06 +02:00
NavigationServer2D : : get_singleton ( ) - > agent_set_radius ( agent , estimate_agent_radius ( ) ) ;
}
}
real_t NavigationObstacle2D : : estimate_agent_radius ( ) const {
2022-05-15 20:29:23 +02:00
if ( parent_node2d & & parent_node2d - > is_inside_tree ( ) ) {
2021-03-08 09:47:18 +01:00
// Estimate the radius of this physics body
2022-09-29 11:53:28 +02:00
real_t max_radius = 0.0 ;
2021-03-08 09:47:18 +01:00
for ( int i ( 0 ) ; i < parent_node2d - > get_child_count ( ) ; i + + ) {
// For each collision shape
CollisionShape2D * cs = Object : : cast_to < CollisionShape2D > ( parent_node2d - > get_child ( i ) ) ;
2022-05-15 20:29:23 +02:00
if ( cs & & cs - > is_inside_tree ( ) ) {
2021-03-08 09:47:18 +01:00
// Take the distance between the Body center to the shape center
real_t r = cs - > get_transform ( ) . get_origin ( ) . length ( ) ;
if ( cs - > get_shape ( ) . is_valid ( ) ) {
// and add the enclosing shape radius
r + = cs - > get_shape ( ) - > get_enclosing_radius ( ) ;
}
2021-08-14 10:16:57 +02:00
Size2 s = cs - > get_global_scale ( ) ;
2021-03-08 09:47:18 +01:00
r * = MAX ( s . x , s . y ) ;
// Takes the biggest radius
2022-09-29 11:53:28 +02:00
max_radius = MAX ( max_radius , r ) ;
2022-05-15 20:29:23 +02:00
} else if ( cs & & ! cs - > is_inside_tree ( ) ) {
WARN_PRINT ( " A CollisionShape2D of the NavigationObstacle2D parent node was not inside the SceneTree when estimating the obstacle radius. "
" \n Move the NavigationObstacle2D to a child position below any CollisionShape2D node of the parent node so the CollisionShape2D is already inside the SceneTree. " ) ;
2020-01-10 12:22:34 +01:00
}
}
2021-08-14 10:16:57 +02:00
Vector2 s = parent_node2d - > get_global_scale ( ) ;
2022-09-29 11:53:28 +02:00
max_radius * = MAX ( s . x , s . y ) ;
2020-01-10 12:22:34 +01:00
2022-09-29 11:53:28 +02:00
if ( max_radius > 0.0 ) {
return max_radius ;
2021-03-08 09:47:18 +01:00
}
}
2021-10-21 22:25:06 +02:00
return 1.0 ; // Never a 0 radius
}
2022-08-14 16:23:27 +02:00
void NavigationObstacle2D : : set_agent_parent ( Node * p_agent_parent ) {
2023-02-02 08:38:28 +01:00
if ( parent_node2d = = p_agent_parent ) {
return ;
}
2022-08-14 16:23:27 +02:00
if ( Object : : cast_to < Node2D > ( p_agent_parent ) ! = nullptr ) {
parent_node2d = Object : : cast_to < Node2D > ( p_agent_parent ) ;
if ( map_override . is_valid ( ) ) {
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , map_override ) ;
} else {
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , parent_node2d - > get_world_2d ( ) - > get_navigation_map ( ) ) ;
}
2023-04-06 16:32:49 +02:00
// Need to register Callback as obstacle requires a valid Callback to be added to avoidance simulation.
NavigationServer2D : : get_singleton ( ) - > agent_set_callback ( get_rid ( ) , callable_mp ( this , & NavigationObstacle2D : : _avoidance_done ) ) ;
2022-08-14 16:23:27 +02:00
reevaluate_agent_radius ( ) ;
} else {
parent_node2d = nullptr ;
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , RID ( ) ) ;
2023-04-06 16:32:49 +02:00
NavigationServer2D : : get_singleton ( ) - > agent_set_callback ( agent , Callable ( ) ) ;
2022-08-14 16:23:27 +02:00
}
}
2023-04-06 16:32:49 +02:00
void NavigationObstacle2D : : _avoidance_done ( Vector3 p_new_velocity ) {
// Dummy function as obstacle requires a valid Callback to be added to avoidance simulation.
}
2022-08-14 16:23:27 +02:00
void NavigationObstacle2D : : set_navigation_map ( RID p_navigation_map ) {
2023-02-02 08:38:28 +01:00
if ( map_override = = p_navigation_map ) {
return ;
}
2022-08-14 16:23:27 +02:00
map_override = p_navigation_map ;
2023-02-02 08:38:28 +01:00
2022-08-14 16:23:27 +02:00
NavigationServer2D : : get_singleton ( ) - > agent_set_map ( agent , map_override ) ;
}
RID NavigationObstacle2D : : get_navigation_map ( ) const {
if ( map_override . is_valid ( ) ) {
return map_override ;
} else if ( parent_node2d ! = nullptr ) {
return parent_node2d - > get_world_2d ( ) - > get_navigation_map ( ) ;
}
return RID ( ) ;
}
2021-10-21 22:25:06 +02:00
void NavigationObstacle2D : : set_estimate_radius ( bool p_estimate_radius ) {
2023-02-02 08:38:28 +01:00
if ( estimate_radius = = p_estimate_radius ) {
return ;
}
2021-10-21 22:25:06 +02:00
estimate_radius = p_estimate_radius ;
2023-02-02 08:38:28 +01:00
2021-10-21 22:25:06 +02:00
notify_property_list_changed ( ) ;
reevaluate_agent_radius ( ) ;
}
void NavigationObstacle2D : : set_radius ( real_t p_radius ) {
ERR_FAIL_COND_MSG ( p_radius < = 0.0 , " Radius must be greater than 0. " ) ;
2023-02-02 08:38:28 +01:00
if ( Math : : is_equal_approx ( radius , p_radius ) ) {
return ;
}
2021-10-21 22:25:06 +02:00
radius = p_radius ;
2023-02-02 08:38:28 +01:00
2021-10-21 22:25:06 +02:00
reevaluate_agent_radius ( ) ;
2020-01-10 12:22:34 +01:00
}