2022-03-15 13:29:32 +01:00
/*************************************************************************/
/* nav_map.cpp */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* */
/* 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. */
/*************************************************************************/
# include "nav_map.h"
2023-06-10 20:58:49 +02:00
# include "core/config/project_settings.h"
2022-03-15 13:29:32 +01:00
# include "core/os/threaded_array_processor.h"
2023-06-10 20:58:49 +02:00
# include "nav_agent.h"
2023-06-08 17:16:11 +02:00
# include "nav_link.h"
2023-06-10 20:58:49 +02:00
# include "nav_obstacle.h"
2022-03-15 13:29:32 +01:00
# include "nav_region.h"
2022-03-16 21:09:41 +01:00
# include "scene/resources/mesh.h"
2023-12-10 14:23:24 +01:00
# include "scene/resources/navigation/navigation_mesh.h"
2023-06-10 20:58:49 +02:00
# include <Obstacle2d.h>
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
# define THREE_POINTS_CROSS_PRODUCT(m_a, m_b, m_c) (((m_c) - (m_a)).cross((m_b) - (m_a)))
2022-03-15 13:29:32 +01:00
2023-06-09 19:08:38 +02:00
// Helper macro
# define APPEND_METADATA(poly) \
if ( r_path_types ) { \
r_path_types - > push_back ( poly - > owner - > get_type ( ) ) ; \
} \
if ( r_path_rids ) { \
r_path_rids - > push_back ( poly - > owner - > get_self ( ) ) ; \
} \
if ( r_path_owners ) { \
r_path_owners - > push_back ( poly - > owner - > get_owner_id ( ) ) ; \
}
2022-03-15 13:29:32 +01:00
void NavMap : : set_up ( Vector3 p_up ) {
2023-06-09 16:43:14 +02:00
if ( up = = p_up ) {
return ;
}
2022-03-15 13:29:32 +01:00
up = p_up ;
regenerate_polygons = true ;
}
2023-06-09 08:53:00 +02:00
void NavMap : : set_cell_size ( real_t p_cell_size ) {
2023-06-09 16:43:14 +02:00
if ( cell_size = = p_cell_size ) {
return ;
}
2022-03-15 13:29:32 +01:00
cell_size = p_cell_size ;
regenerate_polygons = true ;
}
2023-06-09 08:53:00 +02:00
void NavMap : : set_cell_height ( real_t p_cell_height ) {
2023-06-09 16:43:14 +02:00
if ( cell_height = = p_cell_height ) {
return ;
}
2022-03-15 13:29:32 +01:00
cell_height = p_cell_height ;
regenerate_polygons = true ;
}
2023-06-09 20:54:05 +02:00
void NavMap : : set_use_edge_connections ( bool p_enabled ) {
if ( use_edge_connections = = p_enabled ) {
return ;
}
use_edge_connections = p_enabled ;
regenerate_links = true ;
}
2023-06-09 08:53:00 +02:00
void NavMap : : set_edge_connection_margin ( real_t p_edge_connection_margin ) {
2023-06-09 16:43:14 +02:00
if ( edge_connection_margin = = p_edge_connection_margin ) {
return ;
}
2022-03-15 13:29:32 +01:00
edge_connection_margin = p_edge_connection_margin ;
regenerate_links = true ;
}
2023-06-09 08:53:00 +02:00
void NavMap : : set_link_connection_radius ( real_t p_link_connection_radius ) {
2023-06-09 16:43:14 +02:00
if ( link_connection_radius = = p_link_connection_radius ) {
return ;
}
2023-06-08 17:16:11 +02:00
link_connection_radius = p_link_connection_radius ;
regenerate_links = true ;
}
2022-03-15 13:29:32 +01:00
gd : : PointKey NavMap : : get_point_key ( const Vector3 & p_pos ) const {
const int x = static_cast < int > ( Math : : round ( p_pos . x / cell_size ) ) ;
const int y = static_cast < int > ( Math : : round ( p_pos . y / cell_height ) ) ;
const int z = static_cast < int > ( Math : : round ( p_pos . z / cell_size ) ) ;
gd : : PointKey p ;
p . key = 0 ;
p . x = x ;
p . y = y ;
p . z = z ;
return p ;
}
2023-06-09 19:08:38 +02:00
Vector < Vector3 > NavMap : : get_path ( Vector3 p_origin , Vector3 p_destination , bool p_optimize , uint32_t p_navigation_layers , Vector < int32_t > * r_path_types , Array * r_path_rids , Vector < uint64_t > * r_path_owners ) const {
2023-09-02 13:05:19 +02:00
ERR_FAIL_COND_V_MSG ( map_update_id = = 0 , Vector < Vector3 > ( ) , " NavigationServer map query failed because it was made before first map synchronization. " ) ;
2023-06-09 19:08:38 +02:00
// Clear metadata outputs.
if ( r_path_types ) {
r_path_types - > clear ( ) ;
}
if ( r_path_rids ) {
r_path_rids - > clear ( ) ;
}
if ( r_path_owners ) {
r_path_owners - > clear ( ) ;
}
2022-07-28 22:34:28 +02:00
// Find the start poly and the end poly on this map.
const gd : : Polygon * begin_poly = nullptr ;
const gd : : Polygon * end_poly = nullptr ;
2022-03-15 13:29:32 +01:00
Vector3 begin_point ;
Vector3 end_point ;
2023-06-09 08:53:00 +02:00
real_t begin_d = FLT_MAX ;
real_t end_d = FLT_MAX ;
2022-03-15 13:29:32 +01:00
// Find the initial poly and the end poly on this map.
for ( size_t i ( 0 ) ; i < polygons . size ( ) ; i + + ) {
const gd : : Polygon & p = polygons [ i ] ;
2022-07-28 22:34:28 +02:00
// Only consider the polygon if it in a region with compatible layers.
if ( ( p_navigation_layers & p . owner - > get_navigation_layers ( ) ) = = 0 ) {
continue ;
}
2022-03-15 13:29:32 +01:00
// For each face check the distance between the origin/destination
for ( size_t point_id = 2 ; point_id < p . points . size ( ) ; point_id + + ) {
const Face3 face ( p . points [ 0 ] . pos , p . points [ point_id - 1 ] . pos , p . points [ point_id ] . pos ) ;
2022-07-28 22:34:28 +02:00
Vector3 point = face . get_closest_point_to ( p_origin ) ;
2023-06-09 08:53:00 +02:00
real_t distance_to_point = point . distance_to ( p_origin ) ;
2022-07-28 22:34:28 +02:00
if ( distance_to_point < begin_d ) {
begin_d = distance_to_point ;
2022-03-15 13:29:32 +01:00
begin_poly = & p ;
2022-07-28 22:34:28 +02:00
begin_point = point ;
2022-03-15 13:29:32 +01:00
}
2022-07-28 22:34:28 +02:00
point = face . get_closest_point_to ( p_destination ) ;
distance_to_point = point . distance_to ( p_destination ) ;
if ( distance_to_point < end_d ) {
end_d = distance_to_point ;
2022-03-15 13:29:32 +01:00
end_poly = & p ;
2022-07-28 22:34:28 +02:00
end_point = point ;
2022-03-15 13:29:32 +01:00
}
}
}
2022-07-28 22:34:28 +02:00
// Check for trivial cases
2022-03-15 13:29:32 +01:00
if ( ! begin_poly | | ! end_poly ) {
return Vector < Vector3 > ( ) ;
}
if ( begin_poly = = end_poly ) {
2023-06-09 19:08:38 +02:00
if ( r_path_types ) {
r_path_types - > resize ( 2 ) ;
r_path_types - > write [ 0 ] = begin_poly - > owner - > get_type ( ) ;
r_path_types - > write [ 1 ] = end_poly - > owner - > get_type ( ) ;
}
if ( r_path_rids ) {
r_path_rids - > resize ( 2 ) ;
( * r_path_rids ) [ 0 ] = begin_poly - > owner - > get_self ( ) ;
( * r_path_rids ) [ 1 ] = end_poly - > owner - > get_self ( ) ;
}
if ( r_path_owners ) {
r_path_owners - > resize ( 2 ) ;
r_path_owners - > write [ 0 ] = begin_poly - > owner - > get_owner_id ( ) ;
r_path_owners - > write [ 1 ] = end_poly - > owner - > get_owner_id ( ) ;
}
2022-03-15 13:29:32 +01:00
Vector < Vector3 > path ;
path . resize ( 2 ) ;
path . write [ 0 ] = begin_point ;
path . write [ 1 ] = end_point ;
return path ;
}
2022-07-28 22:34:28 +02:00
// List of all reachable navigation polys.
2022-07-30 00:44:53 +02:00
LocalVector < gd : : NavigationPoly > navigation_polys ;
2022-03-15 13:29:32 +01:00
navigation_polys . reserve ( polygons . size ( ) * 0.75 ) ;
2022-07-28 22:34:28 +02:00
// Add the start polygon to the reachable navigation polygons.
gd : : NavigationPoly begin_navigation_poly = gd : : NavigationPoly ( begin_poly ) ;
begin_navigation_poly . self_id = 0 ;
begin_navigation_poly . entry = begin_point ;
begin_navigation_poly . back_navigation_edge_pathway_start = begin_point ;
begin_navigation_poly . back_navigation_edge_pathway_end = begin_point ;
navigation_polys . push_back ( begin_navigation_poly ) ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
// List of polygon IDs to visit.
List < uint32_t > to_visit ;
to_visit . push_back ( 0 ) ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
// This is an implementation of the A* algorithm.
int least_cost_id = 0 ;
2022-11-22 05:46:43 +01:00
int prev_least_cost_id = - 1 ;
2022-07-28 22:34:28 +02:00
bool found_route = false ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
const gd : : Polygon * reachable_end = nullptr ;
2023-06-09 08:53:00 +02:00
real_t reachable_d = FLT_MAX ;
2022-03-15 13:29:32 +01:00
bool is_reachable = true ;
2022-07-28 22:34:28 +02:00
while ( true ) {
// Takes the current least_cost_poly neighbors (iterating over its edges) and compute the traveled_distance.
for ( size_t i = 0 ; i < navigation_polys [ least_cost_id ] . poly - > edges . size ( ) ; i + + ) {
2022-11-22 05:46:43 +01:00
const gd : : Edge & edge = navigation_polys [ least_cost_id ] . poly - > edges [ i ] ;
2022-04-22 18:19:11 +02:00
2022-07-28 22:34:28 +02:00
// Iterate over connections in this edge, then compute the new optimized travel distance assigned to this polygon.
for ( int connection_index = 0 ; connection_index < edge . connections . size ( ) ; connection_index + + ) {
const gd : : Edge : : Connection & connection = edge . connections [ connection_index ] ;
// Only consider the connection to another polygon if this polygon is in a region with compatible layers.
if ( ( p_navigation_layers & connection . polygon - > owner - > get_navigation_layers ( ) ) = = 0 ) {
2022-03-15 13:29:32 +01:00
continue ;
2022-07-28 22:34:28 +02:00
}
2022-03-15 13:29:32 +01:00
2022-11-22 05:46:43 +01:00
const gd : : NavigationPoly & least_cost_poly = navigation_polys [ least_cost_id ] ;
2023-06-09 08:53:00 +02:00
real_t poly_enter_cost = 0.0 ;
real_t poly_travel_cost = least_cost_poly . poly - > owner - > get_travel_cost ( ) ;
2022-03-15 13:29:32 +01:00
2022-11-22 05:46:43 +01:00
if ( prev_least_cost_id ! = - 1 & & ! ( navigation_polys [ prev_least_cost_id ] . poly - > owner - > get_self ( ) = = least_cost_poly . poly - > owner - > get_self ( ) ) ) {
2023-06-08 17:16:11 +02:00
poly_enter_cost = least_cost_poly . poly - > owner - > get_enter_cost ( ) ;
2022-07-28 22:34:28 +02:00
}
2022-11-22 05:46:43 +01:00
prev_least_cost_id = least_cost_id ;
2022-07-28 22:34:28 +02:00
Vector3 pathway [ 2 ] = { connection . pathway_start , connection . pathway_end } ;
2022-11-22 05:46:43 +01:00
const Vector3 new_entry = Geometry : : get_closest_point_to_segment ( least_cost_poly . entry , pathway ) ;
2023-06-09 08:53:00 +02:00
const real_t new_distance = ( least_cost_poly . entry . distance_to ( new_entry ) * poly_travel_cost ) + poly_enter_cost + least_cost_poly . traveled_distance ;
2022-03-15 13:29:32 +01:00
2022-07-30 00:44:53 +02:00
int64_t already_visited_polygon_index = navigation_polys . find ( gd : : NavigationPoly ( connection . polygon ) ) ;
2022-03-15 13:29:32 +01:00
2022-07-30 00:44:53 +02:00
if ( already_visited_polygon_index ! = - 1 ) {
2022-07-28 22:34:28 +02:00
// Polygon already visited, check if we can reduce the travel cost.
2022-07-30 00:44:53 +02:00
gd : : NavigationPoly & avp = navigation_polys [ already_visited_polygon_index ] ;
if ( new_distance < avp . traveled_distance ) {
avp . back_navigation_poly_id = least_cost_id ;
avp . back_navigation_edge = connection . edge ;
avp . back_navigation_edge_pathway_start = connection . pathway_start ;
avp . back_navigation_edge_pathway_end = connection . pathway_end ;
avp . traveled_distance = new_distance ;
avp . entry = new_entry ;
2022-03-15 13:29:32 +01:00
}
} else {
2022-07-28 22:34:28 +02:00
// Add the neighbour polygon to the reachable ones.
gd : : NavigationPoly new_navigation_poly = gd : : NavigationPoly ( connection . polygon ) ;
new_navigation_poly . self_id = navigation_polys . size ( ) ;
new_navigation_poly . back_navigation_poly_id = least_cost_id ;
new_navigation_poly . back_navigation_edge = connection . edge ;
new_navigation_poly . back_navigation_edge_pathway_start = connection . pathway_start ;
new_navigation_poly . back_navigation_edge_pathway_end = connection . pathway_end ;
new_navigation_poly . traveled_distance = new_distance ;
new_navigation_poly . entry = new_entry ;
navigation_polys . push_back ( new_navigation_poly ) ;
// Add the neighbour polygon to the polygons to visit.
to_visit . push_back ( navigation_polys . size ( ) - 1 ) ;
2022-03-15 13:29:32 +01:00
}
}
}
2022-07-28 22:34:28 +02:00
// Removes the least cost polygon from the list of polygons to visit so we can advance.
to_visit . erase ( least_cost_id ) ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
// When the list of polygons to visit is empty at this point it means the End Polygon is not reachable
if ( to_visit . size ( ) = = 0 ) {
// Thus use the further reachable polygon
2022-03-15 13:29:32 +01:00
ERR_BREAK_MSG ( is_reachable = = false , " It's not expect to not find the most reachable polygons " ) ;
is_reachable = false ;
2022-07-28 22:34:28 +02:00
if ( reachable_end = = nullptr ) {
2022-03-15 13:29:32 +01:00
// The path is not found and there is not a way out.
break ;
}
// Set as end point the furthest reachable point.
end_poly = reachable_end ;
2023-06-09 08:53:00 +02:00
end_d = FLT_MAX ;
2022-03-15 13:29:32 +01:00
for ( size_t point_id = 2 ; point_id < end_poly - > points . size ( ) ; point_id + + ) {
Face3 f ( end_poly - > points [ 0 ] . pos , end_poly - > points [ point_id - 1 ] . pos , end_poly - > points [ point_id ] . pos ) ;
Vector3 spoint = f . get_closest_point_to ( p_destination ) ;
2023-06-09 08:53:00 +02:00
real_t dpoint = spoint . distance_to ( p_destination ) ;
2022-03-15 13:29:32 +01:00
if ( dpoint < end_d ) {
end_point = spoint ;
end_d = dpoint ;
}
}
2023-09-02 13:33:32 +02:00
// Search all faces of start polygon as well.
bool closest_point_on_start_poly = false ;
for ( size_t point_id = 2 ; point_id < begin_poly - > points . size ( ) ; point_id + + ) {
Face3 f ( begin_poly - > points [ 0 ] . pos , begin_poly - > points [ point_id - 1 ] . pos , begin_poly - > points [ point_id ] . pos ) ;
Vector3 spoint = f . get_closest_point_to ( p_destination ) ;
real_t dpoint = spoint . distance_to ( p_destination ) ;
if ( dpoint < end_d ) {
end_point = spoint ;
end_d = dpoint ;
closest_point_on_start_poly = true ;
}
}
if ( closest_point_on_start_poly ) {
// No point to run PostProcessing when start and end convex polygon is the same.
if ( r_path_types ) {
r_path_types - > resize ( 2 ) ;
r_path_types - > write [ 0 ] = begin_poly - > owner - > get_type ( ) ;
r_path_types - > write [ 1 ] = begin_poly - > owner - > get_type ( ) ;
}
if ( r_path_rids ) {
r_path_rids - > resize ( 2 ) ;
( * r_path_rids ) [ 0 ] = begin_poly - > owner - > get_self ( ) ;
( * r_path_rids ) [ 1 ] = begin_poly - > owner - > get_self ( ) ;
}
if ( r_path_owners ) {
r_path_owners - > resize ( 2 ) ;
r_path_owners - > write [ 0 ] = begin_poly - > owner - > get_owner_id ( ) ;
r_path_owners - > write [ 1 ] = begin_poly - > owner - > get_owner_id ( ) ;
}
Vector < Vector3 > path ;
path . resize ( 2 ) ;
path . write [ 0 ] = begin_point ;
path . write [ 1 ] = end_point ;
return path ;
}
2022-03-15 13:29:32 +01:00
// Reset open and navigation_polys
gd : : NavigationPoly np = navigation_polys [ 0 ] ;
navigation_polys . clear ( ) ;
navigation_polys . push_back ( np ) ;
2022-07-28 22:34:28 +02:00
to_visit . clear ( ) ;
to_visit . push_back ( 0 ) ;
2022-04-14 17:15:39 +02:00
least_cost_id = 0 ;
2022-11-22 05:46:43 +01:00
prev_least_cost_id = - 1 ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
reachable_end = nullptr ;
2022-03-15 13:29:32 +01:00
continue ;
}
2022-07-28 22:34:28 +02:00
// Find the polygon with the minimum cost from the list of polygons to visit.
2022-03-15 13:29:32 +01:00
least_cost_id = - 1 ;
2023-06-09 08:53:00 +02:00
real_t least_cost = FLT_MAX ;
2022-07-28 22:34:28 +02:00
for ( List < uint32_t > : : Element * element = to_visit . front ( ) ; element ! = nullptr ; element = element - > next ( ) ) {
2022-03-15 13:29:32 +01:00
gd : : NavigationPoly * np = & navigation_polys [ element - > get ( ) ] ;
2023-06-09 08:53:00 +02:00
real_t cost = np - > traveled_distance ;
2022-07-28 22:34:28 +02:00
cost + = ( np - > entry . distance_to ( end_point ) * np - > poly - > owner - > get_travel_cost ( ) ) ;
2022-03-15 13:29:32 +01:00
if ( cost < least_cost ) {
least_cost_id = np - > self_id ;
least_cost = cost ;
}
}
2022-04-14 17:15:39 +02:00
ERR_BREAK ( least_cost_id = = - 1 ) ;
2022-03-15 13:29:32 +01:00
// Stores the further reachable end polygon, in case our goal is not reachable.
if ( is_reachable ) {
2023-06-09 08:53:00 +02:00
real_t d = navigation_polys [ least_cost_id ] . entry . distance_to ( p_destination ) * navigation_polys [ least_cost_id ] . poly - > owner - > get_travel_cost ( ) ;
2022-03-15 13:29:32 +01:00
if ( reachable_d > d ) {
reachable_d = d ;
reachable_end = navigation_polys [ least_cost_id ] . poly ;
}
}
// Check if we reached the end
if ( navigation_polys [ least_cost_id ] . poly = = end_poly ) {
found_route = true ;
break ;
}
}
2023-09-02 13:33:32 +02:00
// We did not find a route but we have both a start polygon and an end polygon at this point.
// Usually this happens because there was not a single external or internal connected edge, e.g. our start polygon is an isolated, single convex polygon.
2022-07-28 22:34:28 +02:00
if ( ! found_route ) {
2023-09-02 13:33:32 +02:00
end_d = FLT_MAX ;
// Search all faces of the start polygon for the closest point to our target position.
for ( size_t point_id = 2 ; point_id < begin_poly - > points . size ( ) ; point_id + + ) {
Face3 f ( begin_poly - > points [ 0 ] . pos , begin_poly - > points [ point_id - 1 ] . pos , begin_poly - > points [ point_id ] . pos ) ;
Vector3 spoint = f . get_closest_point_to ( p_destination ) ;
real_t dpoint = spoint . distance_to ( p_destination ) ;
if ( dpoint < end_d ) {
end_point = spoint ;
end_d = dpoint ;
}
}
if ( r_path_types ) {
r_path_types - > resize ( 2 ) ;
r_path_types - > write [ 0 ] = begin_poly - > owner - > get_type ( ) ;
r_path_types - > write [ 1 ] = begin_poly - > owner - > get_type ( ) ;
}
if ( r_path_rids ) {
r_path_rids - > resize ( 2 ) ;
( * r_path_rids ) [ 0 ] = begin_poly - > owner - > get_self ( ) ;
( * r_path_rids ) [ 1 ] = begin_poly - > owner - > get_self ( ) ;
}
if ( r_path_owners ) {
r_path_owners - > resize ( 2 ) ;
r_path_owners - > write [ 0 ] = begin_poly - > owner - > get_owner_id ( ) ;
r_path_owners - > write [ 1 ] = begin_poly - > owner - > get_owner_id ( ) ;
}
Vector < Vector3 > path ;
path . resize ( 2 ) ;
path . write [ 0 ] = begin_point ;
path . write [ 1 ] = end_point ;
return path ;
2022-07-28 22:34:28 +02:00
}
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
Vector < Vector3 > path ;
// Optimize the path.
if ( p_optimize ) {
// Set the apex poly/point to the end point
gd : : NavigationPoly * apex_poly = & navigation_polys [ least_cost_id ] ;
2023-09-04 17:40:28 +02:00
Vector3 back_pathway [ 2 ] = { apex_poly - > back_navigation_edge_pathway_start , apex_poly - > back_navigation_edge_pathway_end } ;
const Vector3 back_edge_closest_point = Geometry : : get_closest_point_to_segment ( end_point , back_pathway ) ;
if ( end_point . is_equal_approx ( back_edge_closest_point ) ) {
// The end point is basically on top of the last crossed edge, funneling around the corners would at best do nothing.
// At worst it would add an unwanted path point before the last point due to precision issues so skip to the next polygon.
if ( apex_poly - > back_navigation_poly_id ! = - 1 ) {
apex_poly = & navigation_polys [ apex_poly - > back_navigation_poly_id ] ;
}
}
2022-07-28 22:34:28 +02:00
Vector3 apex_point = end_point ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
gd : : NavigationPoly * left_poly = apex_poly ;
Vector3 left_portal = apex_point ;
gd : : NavigationPoly * right_poly = apex_poly ;
Vector3 right_portal = apex_point ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
gd : : NavigationPoly * p = apex_poly ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
path . push_back ( end_point ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( end_poly ) ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
while ( p ) {
// Set left and right points of the pathway between polygons.
Vector3 left = p - > back_navigation_edge_pathway_start ;
Vector3 right = p - > back_navigation_edge_pathway_end ;
if ( THREE_POINTS_CROSS_PRODUCT ( apex_point , left , right ) . dot ( up ) < 0 ) {
SWAP ( left , right ) ;
}
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
bool skip = false ;
if ( THREE_POINTS_CROSS_PRODUCT ( apex_point , left_portal , left ) . dot ( up ) > = 0 ) {
//process
if ( left_portal = = apex_point | | THREE_POINTS_CROSS_PRODUCT ( apex_point , left , right_portal ) . dot ( up ) > 0 ) {
left_poly = p ;
left_portal = left ;
} else {
2023-06-09 19:08:38 +02:00
clip_path ( navigation_polys , path , apex_poly , right_portal , right_poly , r_path_types , r_path_rids , r_path_owners ) ;
2022-07-28 22:34:28 +02:00
apex_point = right_portal ;
p = right_poly ;
left_poly = p ;
apex_poly = p ;
left_portal = apex_point ;
right_portal = apex_point ;
2023-06-09 20:54:05 +02:00
2022-07-28 22:34:28 +02:00
path . push_back ( apex_point ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( apex_poly - > poly ) ;
2022-07-28 22:34:28 +02:00
skip = true ;
2022-03-15 13:29:32 +01:00
}
2022-07-28 22:34:28 +02:00
}
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
if ( ! skip & & THREE_POINTS_CROSS_PRODUCT ( apex_point , right_portal , right ) . dot ( up ) < = 0 ) {
//process
if ( right_portal = = apex_point | | THREE_POINTS_CROSS_PRODUCT ( apex_point , right , left_portal ) . dot ( up ) < 0 ) {
right_poly = p ;
right_portal = right ;
} else {
2023-06-09 19:08:38 +02:00
clip_path ( navigation_polys , path , apex_poly , left_portal , left_poly , r_path_types , r_path_rids , r_path_owners ) ;
2022-07-28 22:34:28 +02:00
apex_point = left_portal ;
p = left_poly ;
right_poly = p ;
apex_poly = p ;
right_portal = apex_point ;
left_portal = apex_point ;
2023-06-09 19:08:38 +02:00
2022-07-28 22:34:28 +02:00
path . push_back ( apex_point ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( apex_poly - > poly ) ;
2022-03-15 13:29:32 +01:00
}
}
2022-07-28 22:34:28 +02:00
// Go to the previous polygon.
if ( p - > back_navigation_poly_id ! = - 1 ) {
p = & navigation_polys [ p - > back_navigation_poly_id ] ;
} else {
// The end
p = nullptr ;
2022-03-15 13:29:32 +01:00
}
2022-07-28 22:34:28 +02:00
}
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
// If the last point is not the begin point, add it to the list.
if ( path [ path . size ( ) - 1 ] ! = begin_point ) {
2022-02-16 23:22:13 +01:00
path . push_back ( begin_point ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( begin_poly ) ;
2022-07-28 22:34:28 +02:00
}
path . invert ( ) ;
2023-06-09 19:08:38 +02:00
if ( r_path_types ) {
r_path_types - > invert ( ) ;
}
if ( r_path_rids ) {
r_path_rids - > invert ( ) ;
}
if ( r_path_owners ) {
r_path_owners - > invert ( ) ;
}
2022-07-28 22:34:28 +02:00
} else {
path . push_back ( end_point ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( end_poly ) ;
2022-02-16 23:22:13 +01:00
2022-07-28 22:34:28 +02:00
// Add mid points
int np_id = least_cost_id ;
while ( np_id ! = - 1 & & navigation_polys [ np_id ] . back_navigation_poly_id ! = - 1 ) {
2023-06-08 17:16:11 +02:00
if ( navigation_polys [ np_id ] . back_navigation_edge ! = - 1 ) {
int prev = navigation_polys [ np_id ] . back_navigation_edge ;
int prev_n = ( navigation_polys [ np_id ] . back_navigation_edge + 1 ) % navigation_polys [ np_id ] . poly - > points . size ( ) ;
Vector3 point = ( navigation_polys [ np_id ] . poly - > points [ prev ] . pos + navigation_polys [ np_id ] . poly - > points [ prev_n ] . pos ) * 0.5 ;
2023-06-09 19:08:38 +02:00
2023-06-08 17:16:11 +02:00
path . push_back ( point ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( navigation_polys [ np_id ] . poly ) ;
2023-06-08 17:16:11 +02:00
} else {
path . push_back ( navigation_polys [ np_id ] . entry ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( navigation_polys [ np_id ] . poly ) ;
2023-06-08 17:16:11 +02:00
}
2022-07-28 22:34:28 +02:00
np_id = navigation_polys [ np_id ] . back_navigation_poly_id ;
2022-03-15 13:29:32 +01:00
}
2022-07-28 22:34:28 +02:00
path . push_back ( begin_point ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( begin_poly ) ;
2022-07-28 22:34:28 +02:00
path . invert ( ) ;
2023-06-09 19:08:38 +02:00
if ( r_path_types ) {
r_path_types - > invert ( ) ;
}
if ( r_path_rids ) {
r_path_rids - > invert ( ) ;
}
if ( r_path_owners ) {
r_path_owners - > invert ( ) ;
}
2022-03-15 13:29:32 +01:00
}
2022-07-28 22:34:28 +02:00
2023-06-09 19:08:38 +02:00
// Ensure post conditions (path arrays MUST match in size).
CRASH_COND ( r_path_types & & path . size ( ) ! = r_path_types - > size ( ) ) ;
CRASH_COND ( r_path_rids & & path . size ( ) ! = r_path_rids - > size ( ) ) ;
CRASH_COND ( r_path_owners & & path . size ( ) ! = r_path_owners - > size ( ) ) ;
2022-07-28 22:34:28 +02:00
return path ;
2022-03-15 13:29:32 +01:00
}
Vector3 NavMap : : get_closest_point_to_segment ( const Vector3 & p_from , const Vector3 & p_to , const bool p_use_collision ) const {
2023-09-02 13:05:19 +02:00
ERR_FAIL_COND_V_MSG ( map_update_id = = 0 , Vector3 ( ) , " NavigationServer map query failed because it was made before first map synchronization. " ) ;
2022-03-15 13:29:32 +01:00
bool use_collision = p_use_collision ;
Vector3 closest_point ;
2023-06-09 08:53:00 +02:00
real_t closest_point_d = FLT_MAX ;
2022-03-15 13:29:32 +01:00
for ( size_t i ( 0 ) ; i < polygons . size ( ) ; i + + ) {
const gd : : Polygon & p = polygons [ i ] ;
// For each face check the distance to the segment
for ( size_t point_id = 2 ; point_id < p . points . size ( ) ; point_id + = 1 ) {
const Face3 f ( p . points [ 0 ] . pos , p . points [ point_id - 1 ] . pos , p . points [ point_id ] . pos ) ;
Vector3 inters ;
if ( f . intersects_segment ( p_from , p_to , & inters ) ) {
const real_t d = closest_point_d = p_from . distance_to ( inters ) ;
if ( use_collision = = false ) {
closest_point = inters ;
use_collision = true ;
closest_point_d = d ;
} else if ( closest_point_d > d ) {
closest_point = inters ;
closest_point_d = d ;
}
}
}
if ( use_collision = = false ) {
for ( size_t point_id = 0 ; point_id < p . points . size ( ) ; point_id + = 1 ) {
Vector3 a , b ;
Geometry : : get_closest_points_between_segments (
p_from ,
p_to ,
p . points [ point_id ] . pos ,
p . points [ ( point_id + 1 ) % p . points . size ( ) ] . pos ,
a ,
b ) ;
const real_t d = a . distance_to ( b ) ;
if ( d < closest_point_d ) {
closest_point_d = d ;
closest_point = b ;
}
}
}
}
return closest_point ;
}
Vector3 NavMap : : get_closest_point ( const Vector3 & p_point ) const {
2023-09-02 13:05:19 +02:00
ERR_FAIL_COND_V_MSG ( map_update_id = = 0 , Vector3 ( ) , " NavigationServer map query failed because it was made before first map synchronization. " ) ;
2022-03-15 13:29:32 +01:00
gd : : ClosestPointQueryResult cp = get_closest_point_info ( p_point ) ;
return cp . point ;
}
Vector3 NavMap : : get_closest_point_normal ( const Vector3 & p_point ) const {
2023-09-02 13:05:19 +02:00
ERR_FAIL_COND_V_MSG ( map_update_id = = 0 , Vector3 ( ) , " NavigationServer map query failed because it was made before first map synchronization. " ) ;
2022-03-15 13:29:32 +01:00
gd : : ClosestPointQueryResult cp = get_closest_point_info ( p_point ) ;
return cp . normal ;
}
RID NavMap : : get_closest_point_owner ( const Vector3 & p_point ) const {
2023-09-02 13:05:19 +02:00
ERR_FAIL_COND_V_MSG ( map_update_id = = 0 , RID ( ) , " NavigationServer map query failed because it was made before first map synchronization. " ) ;
2022-03-15 13:29:32 +01:00
gd : : ClosestPointQueryResult cp = get_closest_point_info ( p_point ) ;
return cp . owner ;
}
gd : : ClosestPointQueryResult NavMap : : get_closest_point_info ( const Vector3 & p_point ) const {
gd : : ClosestPointQueryResult result ;
2023-06-09 08:53:00 +02:00
real_t closest_point_ds = FLT_MAX ;
2022-03-15 13:29:32 +01:00
for ( size_t i ( 0 ) ; i < polygons . size ( ) ; i + + ) {
const gd : : Polygon & p = polygons [ i ] ;
// For each face check the distance to the point
for ( size_t point_id = 2 ; point_id < p . points . size ( ) ; point_id + = 1 ) {
const Face3 f ( p . points [ 0 ] . pos , p . points [ point_id - 1 ] . pos , p . points [ point_id ] . pos ) ;
const Vector3 inters = f . get_closest_point_to ( p_point ) ;
const real_t ds = inters . distance_squared_to ( p_point ) ;
if ( ds < closest_point_ds ) {
result . point = inters ;
result . normal = f . get_plane ( ) . normal ;
result . owner = p . owner - > get_self ( ) ;
closest_point_ds = ds ;
}
}
}
return result ;
}
void NavMap : : add_region ( NavRegion * p_region ) {
regions . push_back ( p_region ) ;
regenerate_links = true ;
}
void NavMap : : remove_region ( NavRegion * p_region ) {
2022-07-30 00:44:53 +02:00
int64_t region_index = regions . find ( p_region ) ;
2023-06-10 20:58:49 +02:00
if ( region_index > = 0 ) {
2022-07-30 00:44:53 +02:00
regions . remove_unordered ( region_index ) ;
2022-03-15 13:29:32 +01:00
regenerate_links = true ;
}
}
2023-06-08 17:16:11 +02:00
void NavMap : : add_link ( NavLink * p_link ) {
links . push_back ( p_link ) ;
regenerate_links = true ;
}
void NavMap : : remove_link ( NavLink * p_link ) {
int64_t link_index = links . find ( p_link ) ;
2023-06-10 20:58:49 +02:00
if ( link_index > = 0 ) {
2023-06-08 17:16:11 +02:00
links . remove_unordered ( link_index ) ;
regenerate_links = true ;
}
}
2023-06-10 20:58:49 +02:00
bool NavMap : : has_agent ( NavAgent * agent ) const {
return ( agents . find ( agent ) > = 0 ) ;
2022-03-15 13:29:32 +01:00
}
2023-06-10 20:58:49 +02:00
void NavMap : : add_agent ( NavAgent * agent ) {
2022-03-15 13:29:32 +01:00
if ( ! has_agent ( agent ) ) {
agents . push_back ( agent ) ;
agents_dirty = true ;
}
}
2023-06-10 20:58:49 +02:00
void NavMap : : remove_agent ( NavAgent * agent ) {
2022-03-15 13:29:32 +01:00
remove_agent_as_controlled ( agent ) ;
2022-07-30 00:44:53 +02:00
int64_t agent_index = agents . find ( agent ) ;
2023-06-10 20:58:49 +02:00
if ( agent_index > = 0 ) {
2022-07-30 00:44:53 +02:00
agents . remove_unordered ( agent_index ) ;
2022-03-15 13:29:32 +01:00
agents_dirty = true ;
}
}
2023-06-10 20:58:49 +02:00
bool NavMap : : has_obstacle ( NavObstacle * obstacle ) const {
return ( obstacles . find ( obstacle ) > = 0 ) ;
}
void NavMap : : add_obstacle ( NavObstacle * obstacle ) {
2023-09-04 19:04:29 +02:00
if ( obstacle - > get_paused ( ) ) {
// No point in adding a paused obstacle, it will add itself when unpaused again.
return ;
}
2023-06-10 20:58:49 +02:00
if ( ! has_obstacle ( obstacle ) ) {
obstacles . push_back ( obstacle ) ;
obstacles_dirty = true ;
}
}
void NavMap : : remove_obstacle ( NavObstacle * obstacle ) {
int64_t obstacle_index = obstacles . find ( obstacle ) ;
if ( obstacle_index > = 0 ) {
obstacles . remove_unordered ( obstacle_index ) ;
obstacles_dirty = true ;
}
}
2022-07-30 00:44:53 +02:00
2023-06-10 20:58:49 +02:00
void NavMap : : set_agent_as_controlled ( NavAgent * agent ) {
remove_agent_as_controlled ( agent ) ;
2023-09-04 19:04:29 +02:00
if ( agent - > get_paused ( ) ) {
// No point in adding a paused agent, it will add itself when unpaused again.
return ;
}
2023-06-10 20:58:49 +02:00
if ( agent - > get_use_3d_avoidance ( ) ) {
int64_t agent_3d_index = active_3d_avoidance_agents . find ( agent ) ;
if ( agent_3d_index < 0 ) {
active_3d_avoidance_agents . push_back ( agent ) ;
agents_dirty = true ;
}
} else {
int64_t agent_2d_index = active_2d_avoidance_agents . find ( agent ) ;
if ( agent_2d_index < 0 ) {
active_2d_avoidance_agents . push_back ( agent ) ;
agents_dirty = true ;
}
2022-03-15 13:29:32 +01:00
}
}
2023-06-10 20:58:49 +02:00
void NavMap : : remove_agent_as_controlled ( NavAgent * agent ) {
int64_t agent_3d_index = active_3d_avoidance_agents . find ( agent ) ;
if ( agent_3d_index > = 0 ) {
active_3d_avoidance_agents . remove_unordered ( agent_3d_index ) ;
agents_dirty = true ;
}
int64_t agent_2d_index = active_2d_avoidance_agents . find ( agent ) ;
if ( agent_2d_index > = 0 ) {
active_2d_avoidance_agents . remove_unordered ( agent_2d_index ) ;
2022-07-30 00:44:53 +02:00
agents_dirty = true ;
2022-03-15 13:29:32 +01:00
}
}
void NavMap : : sync ( ) {
2023-06-09 10:17:43 +02:00
// Performance Monitor
int _new_pm_region_count = regions . size ( ) ;
int _new_pm_agent_count = agents . size ( ) ;
int _new_pm_link_count = links . size ( ) ;
int _new_pm_polygon_count = pm_polygon_count ;
int _new_pm_edge_count = pm_edge_count ;
int _new_pm_edge_merge_count = pm_edge_merge_count ;
int _new_pm_edge_connection_count = pm_edge_connection_count ;
int _new_pm_edge_free_count = pm_edge_free_count ;
2022-07-28 22:34:28 +02:00
// Check if we need to update the links.
2022-03-15 13:29:32 +01:00
if ( regenerate_polygons ) {
2022-07-30 00:44:53 +02:00
for ( uint32_t r = 0 ; r < regions . size ( ) ; r + + ) {
2022-03-15 13:29:32 +01:00
regions [ r ] - > scratch_polygons ( ) ;
}
regenerate_links = true ;
}
2022-07-30 00:44:53 +02:00
for ( uint32_t r = 0 ; r < regions . size ( ) ; r + + ) {
2022-03-15 13:29:32 +01:00
if ( regions [ r ] - > sync ( ) ) {
regenerate_links = true ;
}
}
2023-06-08 17:16:11 +02:00
for ( uint32_t l = 0 ; l < links . size ( ) ; l + + ) {
if ( links [ l ] - > check_dirty ( ) ) {
regenerate_links = true ;
}
}
2022-03-15 13:29:32 +01:00
if ( regenerate_links ) {
2023-06-09 10:17:43 +02:00
_new_pm_polygon_count = 0 ;
_new_pm_edge_count = 0 ;
_new_pm_edge_merge_count = 0 ;
_new_pm_edge_connection_count = 0 ;
_new_pm_edge_free_count = 0 ;
2022-07-28 22:34:28 +02:00
// Remove regions connections.
2022-07-30 00:44:53 +02:00
for ( uint32_t r = 0 ; r < regions . size ( ) ; r + + ) {
2022-07-28 22:34:28 +02:00
regions [ r ] - > get_connections ( ) . clear ( ) ;
}
// Resize the polygon count.
2022-03-15 13:29:32 +01:00
int count = 0 ;
2022-07-30 00:44:53 +02:00
for ( uint32_t r = 0 ; r < regions . size ( ) ; r + + ) {
2023-09-04 20:22:28 +02:00
NavRegion * region = regions [ r ] ;
if ( ! region - > get_enabled ( ) ) {
continue ;
}
count + = region - > get_polygons ( ) . size ( ) ;
2022-03-15 13:29:32 +01:00
}
polygons . resize ( count ) ;
2022-07-28 22:34:28 +02:00
// Copy all region polygons in the map.
count = 0 ;
2022-07-30 00:44:53 +02:00
for ( uint32_t r = 0 ; r < regions . size ( ) ; r + + ) {
2023-09-04 20:22:28 +02:00
NavRegion * region = regions [ r ] ;
if ( ! region - > get_enabled ( ) ) {
continue ;
}
const LocalVector < gd : : Polygon > & polygons_source = region - > get_polygons ( ) ;
2022-07-30 00:44:53 +02:00
for ( uint32_t n = 0 ; n < polygons_source . size ( ) ; n + + ) {
polygons [ count + n ] = polygons_source [ n ] ;
}
2022-03-15 13:29:32 +01:00
2023-09-04 20:22:28 +02:00
count + = region - > get_polygons ( ) . size ( ) ;
2022-03-15 13:29:32 +01:00
}
2023-06-09 10:17:43 +02:00
_new_pm_polygon_count = polygons . size ( ) ;
2022-07-28 22:34:28 +02:00
// Group all edges per key.
2023-09-04 22:00:26 +02:00
HashMap < gd : : EdgeKey , Vector < gd : : Edge : : Connection > , gd : : EdgeKey > connections ;
2022-07-30 00:44:53 +02:00
for ( uint32_t poly_id = 0 ; poly_id < polygons . size ( ) ; poly_id + + ) {
2022-03-15 13:29:32 +01:00
gd : : Polygon & poly ( polygons [ poly_id ] ) ;
2022-07-30 00:44:53 +02:00
for ( uint32_t p = 0 ; p < poly . points . size ( ) ; p + + ) {
2022-03-15 13:29:32 +01:00
int next_point = ( p + 1 ) % poly . points . size ( ) ;
gd : : EdgeKey ek ( poly . points [ p ] . key , poly . points [ next_point ] . key ) ;
2023-09-04 22:00:26 +02:00
HashMap < gd : : EdgeKey , Vector < gd : : Edge : : Connection > , gd : : EdgeKey > : : Element * connection = connections . find ( ek ) ;
2022-07-30 00:44:53 +02:00
2022-03-15 13:29:32 +01:00
if ( ! connection ) {
2022-07-28 22:34:28 +02:00
connections [ ek ] = Vector < gd : : Edge : : Connection > ( ) ;
2023-06-09 10:17:43 +02:00
_new_pm_edge_count + = 1 ;
2022-07-28 22:34:28 +02:00
}
2022-07-30 00:44:53 +02:00
2022-07-28 22:34:28 +02:00
if ( connections [ ek ] . size ( ) < = 1 ) {
// Add the polygon/edge tuple to this key.
gd : : Edge : : Connection new_connection ;
new_connection . polygon = & poly ;
new_connection . edge = p ;
new_connection . pathway_start = poly . points [ p ] . pos ;
new_connection . pathway_end = poly . points [ next_point ] . pos ;
connections [ ek ] . push_back ( new_connection ) ;
2023-09-04 22:00:26 +02:00
} else {
2022-03-15 13:29:32 +01:00
// The edge is already connected with another edge, skip.
2023-09-02 13:07:00 +02:00
ERR_PRINT_ONCE ( " Navigation map synchronization error. Attempted to merge a navigation mesh polygon edge with another already-merged edge. This is usually caused by crossing edges, overlapping polygons, or a mismatch of the NavigationMesh / NavigationPolygon baked 'cell_size' and navigation map 'cell_size'. " ) ;
2022-03-15 13:29:32 +01:00
}
}
}
2022-07-28 22:34:28 +02:00
Vector < gd : : Edge : : Connection > free_edges ;
2023-09-04 22:00:26 +02:00
for ( HashMap < gd : : EdgeKey , Vector < gd : : Edge : : Connection > , gd : : EdgeKey > : : Element * E = connections . front ( ) ; E ; E = E - > next ) {
2022-07-28 22:34:28 +02:00
if ( E - > get ( ) . size ( ) = = 2 ) {
// Connect edge that are shared in different polygons.
gd : : Edge : : Connection & c1 = E - > get ( ) . write [ 0 ] ;
gd : : Edge : : Connection & c2 = E - > get ( ) . write [ 1 ] ;
c1 . polygon - > edges [ c1 . edge ] . connections . push_back ( c2 ) ;
c2 . polygon - > edges [ c2 . edge ] . connections . push_back ( c1 ) ;
// Note: The pathway_start/end are full for those connection and do not need to be modified.
2023-06-09 10:17:43 +02:00
_new_pm_edge_merge_count + = 1 ;
2022-07-28 22:34:28 +02:00
} else {
CRASH_COND_MSG ( E - > get ( ) . size ( ) ! = 1 , vformat ( " Number of connection != 1. Found: %d " , E - > get ( ) . size ( ) ) ) ;
2023-06-09 20:54:05 +02:00
if ( use_edge_connections & & E - > get ( ) [ 0 ] . polygon - > owner - > get_use_edge_connections ( ) ) {
free_edges . push_back ( E - > get ( ) [ 0 ] ) ;
}
2022-03-15 13:29:32 +01:00
}
}
// Find the compatible near edges.
//
// Note:
// Considering that the edges must be compatible (for obvious reasons)
// to be connected, create new polygons to remove that small gap is
// not really useful and would result in wasteful computation during
// connection, integration and path finding.
2023-06-09 10:17:43 +02:00
_new_pm_edge_free_count = free_edges . size ( ) ;
2022-07-28 22:34:28 +02:00
for ( int i = 0 ; i < free_edges . size ( ) ; i + + ) {
const gd : : Edge : : Connection & free_edge = free_edges [ i ] ;
Vector3 edge_p1 = free_edge . polygon - > points [ free_edge . edge ] . pos ;
Vector3 edge_p2 = free_edge . polygon - > points [ ( free_edge . edge + 1 ) % free_edge . polygon - > points . size ( ) ] . pos ;
for ( int j = 0 ; j < free_edges . size ( ) ; j + + ) {
const gd : : Edge : : Connection & other_edge = free_edges [ j ] ;
if ( i = = j | | free_edge . polygon - > owner = = other_edge . polygon - > owner ) {
continue ;
}
Vector3 other_edge_p1 = other_edge . polygon - > points [ other_edge . edge ] . pos ;
Vector3 other_edge_p2 = other_edge . polygon - > points [ ( other_edge . edge + 1 ) % other_edge . polygon - > points . size ( ) ] . pos ;
// Compute the projection of the opposite edge on the current one
Vector3 edge_vector = edge_p2 - edge_p1 ;
2023-06-09 08:53:00 +02:00
real_t projected_p1_ratio = edge_vector . dot ( other_edge_p1 - edge_p1 ) / ( edge_vector . length_squared ( ) ) ;
real_t projected_p2_ratio = edge_vector . dot ( other_edge_p2 - edge_p1 ) / ( edge_vector . length_squared ( ) ) ;
2022-07-28 22:34:28 +02:00
if ( ( projected_p1_ratio < 0.0 & & projected_p2_ratio < 0.0 ) | | ( projected_p1_ratio > 1.0 & & projected_p2_ratio > 1.0 ) ) {
continue ;
}
// Check if the two edges are close to each other enough and compute a pathway between the two regions.
Vector3 self1 = edge_vector * CLAMP ( projected_p1_ratio , 0.0 , 1.0 ) + edge_p1 ;
Vector3 other1 ;
if ( projected_p1_ratio > = 0.0 & & projected_p1_ratio < = 1.0 ) {
other1 = other_edge_p1 ;
} else {
other1 = other_edge_p1 . linear_interpolate ( other_edge_p2 , ( 1.0 - projected_p1_ratio ) / ( projected_p2_ratio - projected_p1_ratio ) ) ;
}
if ( other1 . distance_to ( self1 ) > edge_connection_margin ) {
2022-03-15 13:29:32 +01:00
continue ;
}
2022-07-28 22:34:28 +02:00
Vector3 self2 = edge_vector * CLAMP ( projected_p2_ratio , 0.0 , 1.0 ) + edge_p1 ;
Vector3 other2 ;
if ( projected_p2_ratio > = 0.0 & & projected_p2_ratio < = 1.0 ) {
other2 = other_edge_p2 ;
} else {
other2 = other_edge_p1 . linear_interpolate ( other_edge_p2 , ( 0.0 - projected_p1_ratio ) / ( projected_p2_ratio - projected_p1_ratio ) ) ;
}
if ( other2 . distance_to ( self2 ) > edge_connection_margin ) {
continue ;
2022-03-15 13:29:32 +01:00
}
2022-07-28 22:34:28 +02:00
// The edges can now be connected.
gd : : Edge : : Connection new_connection = other_edge ;
new_connection . pathway_start = ( self1 + other1 ) / 2.0 ;
new_connection . pathway_end = ( self2 + other2 ) / 2.0 ;
free_edge . polygon - > edges [ free_edge . edge ] . connections . push_back ( new_connection ) ;
// Add the connection to the region_connection map.
2023-06-08 17:16:11 +02:00
( ( NavRegion * ) free_edge . polygon - > owner ) - > get_connections ( ) . push_back ( new_connection ) ;
2023-06-09 10:17:43 +02:00
_new_pm_edge_connection_count + = 1 ;
2023-06-08 17:16:11 +02:00
}
}
uint32_t link_poly_idx = 0 ;
link_polygons . resize ( links . size ( ) ) ;
// Search for polygons within range of a nav link.
for ( uint32_t l = 0 ; l < links . size ( ) ; l + + ) {
const NavLink * link = links [ l ] ;
const Vector3 start = link - > get_start_position ( ) ;
const Vector3 end = link - > get_end_position ( ) ;
gd : : Polygon * closest_start_polygon = nullptr ;
real_t closest_start_distance = link_connection_radius ;
Vector3 closest_start_point ;
gd : : Polygon * closest_end_polygon = nullptr ;
real_t closest_end_distance = link_connection_radius ;
Vector3 closest_end_point ;
// Create link to any polygons within the search radius of the start point.
for ( uint32_t start_index = 0 ; start_index < polygons . size ( ) ; start_index + + ) {
gd : : Polygon & start_poly = polygons [ start_index ] ;
// For each face check the distance to the start
for ( uint32_t start_point_id = 2 ; start_point_id < start_poly . points . size ( ) ; start_point_id + = 1 ) {
const Face3 start_face ( start_poly . points [ 0 ] . pos , start_poly . points [ start_point_id - 1 ] . pos , start_poly . points [ start_point_id ] . pos ) ;
const Vector3 start_point = start_face . get_closest_point_to ( start ) ;
const real_t start_distance = start_point . distance_to ( start ) ;
// Pick the polygon that is within our radius and is closer than anything we've seen yet.
if ( start_distance < = link_connection_radius & & start_distance < closest_start_distance ) {
closest_start_distance = start_distance ;
closest_start_point = start_point ;
closest_start_polygon = & start_poly ;
}
}
}
// Find any polygons within the search radius of the end point.
for ( uint32_t end_index = 0 ; end_index < polygons . size ( ) ; end_index + + ) {
gd : : Polygon & end_poly = polygons [ end_index ] ;
// For each face check the distance to the end
for ( uint32_t end_point_id = 2 ; end_point_id < end_poly . points . size ( ) ; end_point_id + = 1 ) {
const Face3 end_face ( end_poly . points [ 0 ] . pos , end_poly . points [ end_point_id - 1 ] . pos , end_poly . points [ end_point_id ] . pos ) ;
const Vector3 end_point = end_face . get_closest_point_to ( end ) ;
const real_t end_distance = end_point . distance_to ( end ) ;
// Pick the polygon that is within our radius and is closer than anything we've seen yet.
if ( end_distance < = link_connection_radius & & end_distance < closest_end_distance ) {
closest_end_distance = end_distance ;
closest_end_point = end_point ;
closest_end_polygon = & end_poly ;
}
}
}
// If we have both a start and end point, then create a synthetic polygon to route through.
if ( closest_start_polygon & & closest_end_polygon ) {
gd : : Polygon & new_polygon = link_polygons [ link_poly_idx + + ] ;
new_polygon . owner = const_cast < NavLink * > ( link ) ;
new_polygon . edges . clear ( ) ;
new_polygon . edges . resize ( 4 ) ;
new_polygon . points . clear ( ) ;
new_polygon . points . reserve ( 4 ) ;
// Build a set of vertices that create a thin polygon going from the start to the end point.
new_polygon . points . push_back ( { closest_start_point , get_point_key ( closest_start_point ) } ) ;
new_polygon . points . push_back ( { closest_start_point , get_point_key ( closest_start_point ) } ) ;
new_polygon . points . push_back ( { closest_end_point , get_point_key ( closest_end_point ) } ) ;
new_polygon . points . push_back ( { closest_end_point , get_point_key ( closest_end_point ) } ) ;
Vector3 center ;
for ( int p = 0 ; p < 4 ; + + p ) {
center + = new_polygon . points [ p ] . pos ;
}
new_polygon . center = center / real_t ( new_polygon . points . size ( ) ) ;
new_polygon . clockwise = true ;
// Setup connections to go forward in the link.
{
gd : : Edge : : Connection entry_connection ;
entry_connection . polygon = & new_polygon ;
entry_connection . edge = - 1 ;
entry_connection . pathway_start = new_polygon . points [ 0 ] . pos ;
entry_connection . pathway_end = new_polygon . points [ 1 ] . pos ;
closest_start_polygon - > edges [ 0 ] . connections . push_back ( entry_connection ) ;
gd : : Edge : : Connection exit_connection ;
exit_connection . polygon = closest_end_polygon ;
exit_connection . edge = - 1 ;
exit_connection . pathway_start = new_polygon . points [ 2 ] . pos ;
exit_connection . pathway_end = new_polygon . points [ 3 ] . pos ;
new_polygon . edges [ 2 ] . connections . push_back ( exit_connection ) ;
}
// If the link is bi-directional, create connections from the end to the start.
if ( link - > is_bidirectional ( ) ) {
gd : : Edge : : Connection entry_connection ;
entry_connection . polygon = & new_polygon ;
entry_connection . edge = - 1 ;
entry_connection . pathway_start = new_polygon . points [ 2 ] . pos ;
entry_connection . pathway_end = new_polygon . points [ 3 ] . pos ;
closest_end_polygon - > edges [ 0 ] . connections . push_back ( entry_connection ) ;
gd : : Edge : : Connection exit_connection ;
exit_connection . polygon = closest_start_polygon ;
exit_connection . edge = - 1 ;
exit_connection . pathway_start = new_polygon . points [ 0 ] . pos ;
exit_connection . pathway_end = new_polygon . points [ 1 ] . pos ;
new_polygon . edges [ 0 ] . connections . push_back ( exit_connection ) ;
}
2022-03-15 13:29:32 +01:00
}
}
2022-07-28 22:34:28 +02:00
// Update the update ID.
2023-09-04 17:56:34 +02:00
// Some code treats 0 as a failure case, so we avoid returning 0.
map_update_id = map_update_id % 9999999 + 1 ;
2022-03-15 13:29:32 +01:00
}
2023-06-10 20:58:49 +02:00
// Do we have modified obstacle positions?
for ( uint32_t i = 0 ; i < obstacles . size ( ) ; + + i ) {
NavObstacle * obstacle = obstacles [ i ] ;
if ( obstacle - > check_dirty ( ) ) {
obstacles_dirty = true ;
}
}
// Do we have modified agent arrays?
for ( uint32_t i = 0 ; i < agents . size ( ) ; + + i ) {
NavAgent * agent = agents [ i ] ;
if ( agent - > check_dirty ( ) ) {
agents_dirty = true ;
2022-07-28 22:34:28 +02:00
}
2023-06-10 20:58:49 +02:00
}
2022-07-28 22:34:28 +02:00
2023-06-10 20:58:49 +02:00
// Update avoidance worlds.
if ( obstacles_dirty | | agents_dirty ) {
_update_rvo_simulation ( ) ;
2022-03-15 13:29:32 +01:00
}
regenerate_polygons = false ;
regenerate_links = false ;
2023-06-10 20:58:49 +02:00
obstacles_dirty = false ;
2022-03-15 13:29:32 +01:00
agents_dirty = false ;
2023-06-09 10:17:43 +02:00
// Performance Monitor
pm_region_count = _new_pm_region_count ;
pm_agent_count = _new_pm_agent_count ;
pm_link_count = _new_pm_link_count ;
pm_polygon_count = _new_pm_polygon_count ;
pm_edge_count = _new_pm_edge_count ;
pm_edge_merge_count = _new_pm_edge_merge_count ;
pm_edge_connection_count = _new_pm_edge_connection_count ;
pm_edge_free_count = _new_pm_edge_free_count ;
2022-03-15 13:29:32 +01:00
}
2023-06-10 20:58:49 +02:00
void NavMap : : _update_rvo_obstacles_tree_2d ( ) {
int obstacle_vertex_count = 0 ;
for ( uint32_t i = 0 ; i < obstacles . size ( ) ; + + i ) {
NavObstacle * obstacle = obstacles [ i ] ;
obstacle_vertex_count + = obstacle - > get_vertices ( ) . size ( ) ;
}
// Cannot use LocalVector here as RVO library expects std::vector to build KdTree
std : : vector < RVO2D : : Obstacle2D * > raw_obstacles ;
raw_obstacles . reserve ( obstacle_vertex_count ) ;
// The following block is modified copy from RVO2D::AddObstacle()
// Obstacles are linked and depend on all other obstacles.
for ( uint32_t k = 0 ; k < obstacles . size ( ) ; + + k ) {
NavObstacle * obstacle = obstacles [ k ] ;
const Vector3 & _obstacle_position = obstacle - > get_position ( ) ;
const Vector < Vector3 > & _obstacle_vertices = obstacle - > get_vertices ( ) ;
if ( _obstacle_vertices . size ( ) < 2 ) {
continue ;
}
std : : vector < RVO2D : : Vector2 > rvo_2d_vertices ;
rvo_2d_vertices . reserve ( _obstacle_vertices . size ( ) ) ;
uint32_t _obstacle_avoidance_layers = obstacle - > get_avoidance_layers ( ) ;
for ( int i = 0 ; i < _obstacle_vertices . size ( ) ; + + i ) {
const Vector3 & _obstacle_vertex = _obstacle_vertices [ i ] ;
rvo_2d_vertices . push_back ( RVO2D : : Vector2 ( _obstacle_vertex . x + _obstacle_position . x , _obstacle_vertex . z + _obstacle_position . z ) ) ;
}
const size_t obstacleNo = raw_obstacles . size ( ) ;
for ( size_t i = 0 ; i < rvo_2d_vertices . size ( ) ; i + + ) {
RVO2D : : Obstacle2D * rvo_2d_obstacle = new RVO2D : : Obstacle2D ( ) ;
rvo_2d_obstacle - > point_ = rvo_2d_vertices [ i ] ;
rvo_2d_obstacle - > avoidance_layers_ = _obstacle_avoidance_layers ;
if ( i ! = 0 ) {
rvo_2d_obstacle - > prevObstacle_ = raw_obstacles . back ( ) ;
rvo_2d_obstacle - > prevObstacle_ - > nextObstacle_ = rvo_2d_obstacle ;
}
if ( i = = rvo_2d_vertices . size ( ) - 1 ) {
rvo_2d_obstacle - > nextObstacle_ = raw_obstacles [ obstacleNo ] ;
rvo_2d_obstacle - > nextObstacle_ - > prevObstacle_ = rvo_2d_obstacle ;
}
rvo_2d_obstacle - > unitDir_ = normalize ( rvo_2d_vertices [ ( i = = rvo_2d_vertices . size ( ) - 1 ? 0 : i + 1 ) ] - rvo_2d_vertices [ i ] ) ;
if ( rvo_2d_vertices . size ( ) = = 2 ) {
rvo_2d_obstacle - > isConvex_ = true ;
} else {
rvo_2d_obstacle - > isConvex_ = ( leftOf ( rvo_2d_vertices [ ( i = = 0 ? rvo_2d_vertices . size ( ) - 1 : i - 1 ) ] , rvo_2d_vertices [ i ] , rvo_2d_vertices [ ( i = = rvo_2d_vertices . size ( ) - 1 ? 0 : i + 1 ) ] ) > = 0.0f ) ;
}
rvo_2d_obstacle - > id_ = raw_obstacles . size ( ) ;
raw_obstacles . push_back ( rvo_2d_obstacle ) ;
}
}
rvo_simulation_2d . kdTree_ - > buildObstacleTree ( raw_obstacles ) ;
}
void NavMap : : _update_rvo_agents_tree_2d ( ) {
// Cannot use LocalVector here as RVO library expects std::vector to build KdTree.
std : : vector < RVO2D : : Agent2D * > raw_agents ;
raw_agents . reserve ( active_2d_avoidance_agents . size ( ) ) ;
for ( uint32_t k = 0 ; k < active_2d_avoidance_agents . size ( ) ; + + k ) {
NavAgent * agent = active_2d_avoidance_agents [ k ] ;
raw_agents . push_back ( agent - > get_rvo_agent_2d ( ) ) ;
}
rvo_simulation_2d . kdTree_ - > buildAgentTree ( raw_agents ) ;
}
void NavMap : : _update_rvo_agents_tree_3d ( ) {
// Cannot use LocalVector here as RVO library expects std::vector to build KdTree.
std : : vector < RVO3D : : Agent3D * > raw_agents ;
raw_agents . reserve ( active_3d_avoidance_agents . size ( ) ) ;
for ( uint32_t k = 0 ; k < active_3d_avoidance_agents . size ( ) ; + + k ) {
NavAgent * agent = active_3d_avoidance_agents [ k ] ;
raw_agents . push_back ( agent - > get_rvo_agent_3d ( ) ) ;
}
rvo_simulation_3d . kdTree_ - > buildAgentTree ( raw_agents ) ;
}
void NavMap : : _update_rvo_simulation ( ) {
if ( obstacles_dirty ) {
_update_rvo_obstacles_tree_2d ( ) ;
}
if ( agents_dirty ) {
_update_rvo_agents_tree_2d ( ) ;
_update_rvo_agents_tree_3d ( ) ;
}
}
void NavMap : : compute_single_avoidance_step_2d ( uint32_t index , NavAgent * * agent ) {
( * ( agent + index ) ) - > get_rvo_agent_2d ( ) - > computeNeighbors ( & rvo_simulation_2d ) ;
( * ( agent + index ) ) - > get_rvo_agent_2d ( ) - > computeNewVelocity ( & rvo_simulation_2d ) ;
( * ( agent + index ) ) - > get_rvo_agent_2d ( ) - > update ( & rvo_simulation_2d ) ;
( * ( agent + index ) ) - > update ( ) ;
}
void NavMap : : compute_single_avoidance_step_3d ( uint32_t index , NavAgent * * agent ) {
( * ( agent + index ) ) - > get_rvo_agent_3d ( ) - > computeNeighbors ( & rvo_simulation_3d ) ;
( * ( agent + index ) ) - > get_rvo_agent_3d ( ) - > computeNewVelocity ( & rvo_simulation_3d ) ;
( * ( agent + index ) ) - > get_rvo_agent_3d ( ) - > update ( & rvo_simulation_3d ) ;
( * ( agent + index ) ) - > update ( ) ;
2022-03-15 13:29:32 +01:00
}
void NavMap : : step ( real_t p_deltatime ) {
deltatime = p_deltatime ;
2023-06-10 20:58:49 +02:00
if ( active_2d_avoidance_agents . size ( ) > 0 ) {
2022-10-03 11:51:17 +02:00
# ifndef NO_THREADS
2023-06-10 20:58:49 +02:00
if ( use_threads & & avoidance_use_multiple_threads ) {
if ( step_work_pool . get_thread_count ( ) = = 0 ) {
step_work_pool . init ( ) ;
}
step_work_pool . do_work (
active_2d_avoidance_agents . size ( ) ,
this ,
& NavMap : : compute_single_avoidance_step_2d ,
active_2d_avoidance_agents . ptr ( ) ) ;
} else {
for ( int i ( 0 ) ; i < static_cast < int > ( active_2d_avoidance_agents . size ( ) ) ; i + + ) {
NavAgent * agent = active_2d_avoidance_agents [ i ] ;
agent - > get_rvo_agent_2d ( ) - > computeNeighbors ( & rvo_simulation_2d ) ;
agent - > get_rvo_agent_2d ( ) - > computeNewVelocity ( & rvo_simulation_2d ) ;
agent - > get_rvo_agent_2d ( ) - > update ( & rvo_simulation_2d ) ;
agent - > update ( ) ;
}
2022-07-28 22:34:28 +02:00
}
2022-10-03 11:51:17 +02:00
# else
2023-06-10 20:58:49 +02:00
for ( int i ( 0 ) ; i < static_cast < int > ( active_2d_avoidance_agents . size ( ) ) ; i + + ) {
NavAgent * agent = active_2d_avoidance_agents [ i ] ;
agent - > get_rvo_agent_2d ( ) - > computeNeighbors ( & rvo_simulation_2d ) ;
agent - > get_rvo_agent_2d ( ) - > computeNewVelocity ( & rvo_simulation_2d ) ;
agent - > get_rvo_agent_2d ( ) - > update ( & rvo_simulation_2d ) ;
agent - > update ( ) ;
}
# endif // NO_THREADS
}
if ( active_3d_avoidance_agents . size ( ) > 0 ) {
# ifndef NO_THREADS
if ( use_threads & & avoidance_use_multiple_threads ) {
if ( step_work_pool . get_thread_count ( ) = = 0 ) {
step_work_pool . init ( ) ;
}
step_work_pool . do_work (
active_3d_avoidance_agents . size ( ) ,
this ,
& NavMap : : compute_single_avoidance_step_3d ,
active_3d_avoidance_agents . ptr ( ) ) ;
} else {
for ( int i ( 0 ) ; i < static_cast < int > ( active_3d_avoidance_agents . size ( ) ) ; i + + ) {
NavAgent * agent = active_3d_avoidance_agents [ i ] ;
agent - > get_rvo_agent_3d ( ) - > computeNeighbors ( & rvo_simulation_3d ) ;
agent - > get_rvo_agent_3d ( ) - > computeNewVelocity ( & rvo_simulation_3d ) ;
agent - > get_rvo_agent_3d ( ) - > update ( & rvo_simulation_3d ) ;
agent - > update ( ) ;
}
}
# else
for ( int i ( 0 ) ; i < static_cast < int > ( active_3d_avoidance_agents . size ( ) ) ; i + + ) {
NavAgent * agent = active_3d_avoidance_agents [ i ] ;
agent - > get_rvo_agent_3d ( ) - > computeNeighbors ( & rvo_simulation_3d ) ;
agent - > get_rvo_agent_3d ( ) - > computeNewVelocity ( & rvo_simulation_3d ) ;
agent - > get_rvo_agent_3d ( ) - > update ( & rvo_simulation_3d ) ;
agent - > update ( ) ;
2022-10-03 11:51:17 +02:00
}
# endif // NO_THREADS
2022-03-15 13:29:32 +01:00
}
}
void NavMap : : dispatch_callbacks ( ) {
2023-06-10 20:58:49 +02:00
for ( int i ( 0 ) ; i < static_cast < int > ( active_2d_avoidance_agents . size ( ) ) ; i + + ) {
active_2d_avoidance_agents [ i ] - > dispatch_avoidance_callback ( ) ;
}
for ( int i ( 0 ) ; i < static_cast < int > ( active_3d_avoidance_agents . size ( ) ) ; i + + ) {
active_3d_avoidance_agents [ i ] - > dispatch_avoidance_callback ( ) ;
2022-03-15 13:29:32 +01:00
}
}
2023-06-09 19:08:38 +02:00
void NavMap : : clip_path ( const LocalVector < gd : : NavigationPoly > & p_navigation_polys , Vector < Vector3 > & path , const gd : : NavigationPoly * from_poly , const Vector3 & p_to_point , const gd : : NavigationPoly * p_to_poly , Vector < int32_t > * r_path_types , Array * r_path_rids , Vector < uint64_t > * r_path_owners ) const {
2022-03-15 13:29:32 +01:00
Vector3 from = path [ path . size ( ) - 1 ] ;
2022-07-28 22:34:28 +02:00
if ( from . is_equal_approx ( p_to_point ) ) {
2022-03-15 13:29:32 +01:00
return ;
2022-07-28 22:34:28 +02:00
}
2022-03-15 13:29:32 +01:00
Plane cut_plane ;
cut_plane . normal = ( from - p_to_point ) . cross ( up ) ;
2022-07-28 22:34:28 +02:00
if ( cut_plane . normal = = Vector3 ( ) ) {
2022-03-15 13:29:32 +01:00
return ;
2022-07-28 22:34:28 +02:00
}
2022-03-15 13:29:32 +01:00
cut_plane . normal . normalize ( ) ;
cut_plane . d = cut_plane . normal . dot ( from ) ;
while ( from_poly ! = p_to_poly ) {
2022-07-28 22:34:28 +02:00
Vector3 pathway_start = from_poly - > back_navigation_edge_pathway_start ;
Vector3 pathway_end = from_poly - > back_navigation_edge_pathway_end ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
ERR_FAIL_COND ( from_poly - > back_navigation_poly_id = = - 1 ) ;
from_poly = & p_navigation_polys [ from_poly - > back_navigation_poly_id ] ;
2022-03-15 13:29:32 +01:00
2022-07-28 22:34:28 +02:00
if ( ! pathway_start . is_equal_approx ( pathway_end ) ) {
2022-03-15 13:29:32 +01:00
Vector3 inters ;
2022-07-28 22:34:28 +02:00
if ( cut_plane . intersects_segment ( pathway_start , pathway_end , & inters ) ) {
if ( ! inters . is_equal_approx ( p_to_point ) & & ! inters . is_equal_approx ( path [ path . size ( ) - 1 ] ) ) {
2022-03-15 13:29:32 +01:00
path . push_back ( inters ) ;
2023-06-09 19:08:38 +02:00
APPEND_METADATA ( from_poly - > poly ) ;
2022-03-15 13:29:32 +01:00
}
}
}
}
}
2022-07-28 22:34:28 +02:00
NavMap : : NavMap ( ) {
up = Vector3 ( 0 , 1 , 0 ) ;
2023-09-04 18:34:15 +02:00
/// each cell has the following cell_size and cell_height.
cell_size = 0.25 ; // Must match ProjectSettings default 3D cell_size and NavigationMesh cell_size.
cell_height = 0.25 ; // Must match ProjectSettings default 3D cell_height and NavigationMesh cell_height.
2022-07-28 22:34:28 +02:00
edge_connection_margin = 0.25 ;
regenerate_polygons = true ;
regenerate_links = true ;
agents_dirty = false ;
2023-06-10 20:58:49 +02:00
agents_dirty = true ;
obstacles_dirty = true ;
use_threads = true ;
avoidance_use_multiple_threads = true ;
avoidance_use_high_priority_threads = true ;
2022-07-28 22:34:28 +02:00
deltatime = 0.0 ;
map_update_id = 0 ;
2023-06-08 17:16:11 +02:00
link_connection_radius = 1.0 ;
2023-06-09 20:54:05 +02:00
use_edge_connections = true ;
2023-06-09 10:17:43 +02:00
// Performance Monitor
pm_region_count = 0 ;
pm_agent_count = 0 ;
pm_link_count = 0 ;
pm_polygon_count = 0 ;
pm_edge_count = 0 ;
pm_edge_merge_count = 0 ;
pm_edge_connection_count = 0 ;
pm_edge_free_count = 0 ;
2023-06-10 20:58:49 +02:00
avoidance_use_multiple_threads = GLOBAL_GET ( " navigation/avoidance/thread_model/avoidance_use_multiple_threads " ) ;
avoidance_use_high_priority_threads = GLOBAL_GET ( " navigation/avoidance/thread_model/avoidance_use_high_priority_threads " ) ;
2022-07-28 22:34:28 +02:00
}
NavMap : : ~ NavMap ( ) {
2022-10-03 11:51:17 +02:00
# ifndef NO_THREADS
2022-07-28 22:34:28 +02:00
step_work_pool . finish ( ) ;
2022-10-03 11:51:17 +02:00
# endif // !NO_THREADS
2022-07-28 22:34:28 +02:00
}