2023-01-05 13:25:55 +01:00
/**************************************************************************/
/* navigation_obstacle_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_obstacle_3d.h"
2020-01-10 12:22:34 +01:00
2020-03-26 22:49:16 +01:00
# include "scene/3d/collision_shape_3d.h"
# include "scene/3d/physics_body_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 NavigationObstacle3D : : _bind_methods ( ) {
2022-05-14 23:33:09 +02:00
ClassDB : : bind_method ( D_METHOD ( " get_rid " ) , & NavigationObstacle3D : : get_rid ) ;
2022-08-14 16:23:27 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_navigation_map " , " navigation_map " ) , & NavigationObstacle3D : : set_navigation_map ) ;
ClassDB : : bind_method ( D_METHOD ( " get_navigation_map " ) , & NavigationObstacle3D : : get_navigation_map ) ;
2021-10-21 22:25:06 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_estimate_radius " , " estimate_radius " ) , & NavigationObstacle3D : : set_estimate_radius ) ;
ClassDB : : bind_method ( D_METHOD ( " is_radius_estimated " ) , & NavigationObstacle3D : : is_radius_estimated ) ;
ClassDB : : bind_method ( D_METHOD ( " set_radius " , " radius " ) , & NavigationObstacle3D : : set_radius ) ;
ClassDB : : bind_method ( D_METHOD ( " get_radius " ) , & NavigationObstacle3D : : 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,100,0.01 " ) , " set_radius " , " get_radius " ) ;
}
void NavigationObstacle3D : : _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
}
2020-03-26 22:49:16 +01:00
void NavigationObstacle3D : : _notification ( int p_what ) {
2020-01-10 12:22:34 +01:00
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_node3d ) ) {
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_node3d & & ! parent_node3d - > can_process ( ) ) {
map_before_pause = NavigationServer3D : : get_singleton ( ) - > agent_get_map ( get_rid ( ) ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , RID ( ) ) ;
} else if ( parent_node3d & & parent_node3d - > can_process ( ) & & ! ( map_before_pause = = RID ( ) ) ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , map_before_pause ) ;
map_before_pause = RID ( ) ;
}
} break ;
case NOTIFICATION_UNPAUSED : {
if ( parent_node3d & & ! parent_node3d - > can_process ( ) ) {
map_before_pause = NavigationServer3D : : get_singleton ( ) - > agent_get_map ( get_rid ( ) ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , RID ( ) ) ;
} else if ( parent_node3d & & parent_node3d - > can_process ( ) & & ! ( map_before_pause = = RID ( ) ) ) {
NavigationServer3D : : 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_node3d & & parent_node3d - > is_inside_tree ( ) ) {
2021-03-08 09:47:18 +01:00
NavigationServer3D : : get_singleton ( ) - > agent_set_position ( agent , parent_node3d - > get_global_transform ( ) . origin ) ;
PhysicsBody3D * rigid = Object : : cast_to < PhysicsBody3D > ( get_parent ( ) ) ;
if ( rigid ) {
Vector3 v = rigid - > get_linear_velocity ( ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_velocity ( agent , v ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_target_velocity ( agent , v ) ;
}
2020-01-10 12:22:34 +01:00
}
} break ;
}
}
2020-05-12 17:01:17 +02:00
NavigationObstacle3D : : NavigationObstacle3D ( ) {
2020-03-27 19:21:27 +01:00
agent = NavigationServer3D : : get_singleton ( ) - > agent_create ( ) ;
2022-01-07 19:55:22 +01:00
initialize_agent ( ) ;
2020-01-10 12:22:34 +01:00
}
2020-03-26 22:49:16 +01:00
NavigationObstacle3D : : ~ NavigationObstacle3D ( ) {
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
}
2022-09-19 17:43:15 +02:00
PackedStringArray NavigationObstacle3D : : get_configuration_warnings ( ) const {
PackedStringArray warnings = Node : : get_configuration_warnings ( ) ;
2020-05-14 22:59:27 +02:00
2020-10-29 11:01:28 +01:00
if ( ! Object : : cast_to < Node3D > ( get_parent ( ) ) ) {
2022-05-14 23:33:09 +02:00
warnings . push_back ( RTR ( " The NavigationObstacle3D only serves to provide collision avoidance to a Node3D inheriting parent object. " ) ) ;
}
if ( Object : : cast_to < StaticBody3D > ( get_parent ( ) ) ) {
2022-08-25 19:35:52 +02:00
warnings . push_back ( RTR ( " The NavigationObstacle3D is intended for constantly moving bodies like CharacterBody3D or RigidBody3D 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 (re)baked to a NavigationMesh so agents can not only avoid them but also move along those objects outline at high detail " ) ) ;
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
}
2021-10-21 22:25:06 +02:00
void NavigationObstacle3D : : initialize_agent ( ) {
2022-08-13 12:26:13 +02:00
NavigationServer3D : : get_singleton ( ) - > agent_set_neighbor_distance ( agent , 0.0 ) ;
2021-10-21 22:25:06 +02:00
NavigationServer3D : : get_singleton ( ) - > agent_set_max_neighbors ( agent , 0 ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_time_horizon ( agent , 0.0 ) ;
NavigationServer3D : : get_singleton ( ) - > agent_set_max_speed ( agent , 0.0 ) ;
}
void NavigationObstacle3D : : reevaluate_agent_radius ( ) {
if ( ! estimate_radius ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_radius ( agent , radius ) ;
2022-01-07 19:55:22 +01:00
} else if ( parent_node3d & & parent_node3d - > is_inside_tree ( ) ) {
2021-10-21 22:25:06 +02:00
NavigationServer3D : : get_singleton ( ) - > agent_set_radius ( agent , estimate_agent_radius ( ) ) ;
}
}
real_t NavigationObstacle3D : : estimate_agent_radius ( ) const {
2022-05-15 20:29:23 +02:00
if ( parent_node3d & & parent_node3d - > 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_node3d - > get_child_count ( ) ; i + + ) {
// For each collision shape
CollisionShape3D * cs = Object : : cast_to < CollisionShape3D > ( parent_node3d - > 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 ( ) . origin . length ( ) ;
if ( cs - > get_shape ( ) . is_valid ( ) ) {
// and add the enclosing shape radius
r + = cs - > get_shape ( ) - > get_enclosing_radius ( ) ;
}
Vector3 s = cs - > get_global_transform ( ) . basis . get_scale ( ) ;
r * = MAX ( s . x , MAX ( s . y , s . z ) ) ;
// 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 CollisionShape3D of the NavigationObstacle3D parent node was not inside the SceneTree when estimating the obstacle radius. "
" \n Move the NavigationObstacle3D to a child position below any CollisionShape3D node of the parent node so the CollisionShape3D is already inside the SceneTree. " ) ;
2020-01-10 12:22:34 +01:00
}
}
2021-03-08 09:47:18 +01:00
Vector3 s = parent_node3d - > get_global_transform ( ) . basis . get_scale ( ) ;
2022-09-29 11:53:28 +02:00
max_radius * = MAX ( s . x , MAX ( s . y , s . z ) ) ;
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 NavigationObstacle3D : : set_agent_parent ( Node * p_agent_parent ) {
2023-02-02 08:38:28 +01:00
if ( parent_node3d = = p_agent_parent ) {
return ;
}
2022-08-14 16:23:27 +02:00
if ( Object : : cast_to < Node3D > ( p_agent_parent ) ! = nullptr ) {
parent_node3d = Object : : cast_to < Node3D > ( p_agent_parent ) ;
if ( map_override . is_valid ( ) ) {
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , map_override ) ;
} else {
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , parent_node3d - > get_world_3d ( ) - > 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.
NavigationServer3D : : get_singleton ( ) - > agent_set_callback ( get_rid ( ) , callable_mp ( this , & NavigationObstacle3D : : _avoidance_done ) ) ;
2022-08-14 16:23:27 +02:00
reevaluate_agent_radius ( ) ;
} else {
parent_node3d = nullptr ;
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( get_rid ( ) , RID ( ) ) ;
2023-04-06 16:32:49 +02:00
NavigationServer3D : : get_singleton ( ) - > agent_set_callback ( agent , Callable ( ) ) ;
2022-08-14 16:23:27 +02:00
}
}
2023-04-06 16:32:49 +02:00
void NavigationObstacle3D : : _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 NavigationObstacle3D : : 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
NavigationServer3D : : get_singleton ( ) - > agent_set_map ( agent , map_override ) ;
}
RID NavigationObstacle3D : : get_navigation_map ( ) const {
if ( map_override . is_valid ( ) ) {
return map_override ;
} else if ( parent_node3d ! = nullptr ) {
return parent_node3d - > get_world_3d ( ) - > get_navigation_map ( ) ;
}
return RID ( ) ;
}
2021-10-21 22:25:06 +02:00
void NavigationObstacle3D : : 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 NavigationObstacle3D : : 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
}