2023-01-05 13:25:55 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								/**************************************************************************/  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								/*  nav_map.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  "nav_map.h" 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-02-01 22:11:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  "nav_agent.h" 
  
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  "nav_link.h" 
  
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  "nav_obstacle.h" 
  
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								# include  "nav_region.h" 
  
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-13 16:56:21 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  "core/config/project_settings.h" 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# include  "core/object/worker_thread_pool.h" 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# include  <Obstacle2d.h> 
  
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# define THREE_POINTS_CROSS_PRODUCT(m_a, m_b, m_c) (((m_c) - (m_a)).cross((m_b) - (m_a))) 
  
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06: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 ( ) ) ;  \
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-02-22 12:50:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# ifdef DEBUG_ENABLED 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# define NAVMAP_ITERATION_ZERO_ERROR_MSG() \ 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									ERR_PRINT_ONCE ( " NavigationServer navigation map query failed because it was made before first map synchronization. \n \
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									NavigationServer  ' map_changed '  signal  can  be  used  to  receive  update  notifications . \ n \
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									NavigationServer  ' map_get_iteration_id ( ) '  can  be  used  to  check  if  a  map  has  finished  its  newest  iteration . " ); 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# else 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# define NAVMAP_ITERATION_ZERO_ERROR_MSG() 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# endif  // DEBUG_ENABLED
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								void  NavMap : : set_up ( Vector3  p_up )  {  
						 
					
						
							
								
									
										
										
										
											2023-04-05 02:15:25 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( up  = =  p_up )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									up  =  p_up ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regenerate_polygons  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : set_cell_size ( real_t  p_cell_size )  {  
						 
					
						
							
								
									
										
										
										
											2023-04-05 02:15:25 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( cell_size  = =  p_cell_size )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									cell_size  =  p_cell_size ; 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-04 21:31:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									_update_merge_rasterizer_cell_dimensions ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									regenerate_polygons  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-13 13:36:05 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : set_cell_height ( real_t  p_cell_height )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( cell_height  = =  p_cell_height )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									cell_height  =  p_cell_height ; 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-04 21:31:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									_update_merge_rasterizer_cell_dimensions ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regenerate_polygons  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  NavMap : : set_merge_rasterizer_cell_scale ( float  p_value )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( merge_rasterizer_cell_scale  = =  p_value )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									merge_rasterizer_cell_scale  =  p_value ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									_update_merge_rasterizer_cell_dimensions ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-06-13 13:36:05 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									regenerate_polygons  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-04-01 01:49:43 +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-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : set_edge_connection_margin ( real_t  p_edge_connection_margin )  {  
						 
					
						
							
								
									
										
										
										
											2023-04-05 02:15:25 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( edge_connection_margin  = =  p_edge_connection_margin )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									edge_connection_margin  =  p_edge_connection_margin ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : set_link_connection_radius ( real_t  p_link_connection_radius )  {  
						 
					
						
							
								
									
										
										
										
											2023-04-05 02:15:25 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( link_connection_radius  = =  p_link_connection_radius )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									link_connection_radius  =  p_link_connection_radius ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								gd : : PointKey  NavMap : : get_point_key ( const  Vector3  & p_pos )  const  {  
						 
					
						
							
								
									
										
										
										
											2024-02-04 21:31:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									const  int  x  =  static_cast < int > ( Math : : floor ( p_pos . x  /  merge_rasterizer_cell_size ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									const  int  y  =  static_cast < int > ( Math : : floor ( p_pos . y  /  merge_rasterizer_cell_height ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									const  int  z  =  static_cast < int > ( Math : : floor ( p_pos . z  /  merge_rasterizer_cell_size ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									gd : : PointKey  p ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									p . key  =  0 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									p . x  =  x ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									p . y  =  y ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									p . z  =  z ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  p ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06: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 ,  TypedArray < RID >  * r_path_rids ,  Vector < int64_t >  * r_path_owners )  const  {  
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockRead  read_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 03:02:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( iteration_id  = =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-22 12:50:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										NAVMAP_ITERATION_ZERO_ERROR_MSG ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  Vector < Vector3 > ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06: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 ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// Find the start poly and the end poly on this map.
 
							 
						 
					
						
							
								
									
										
										
										
											2020-04-02 01:20:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									const  gd : : Polygon  * begin_poly  =  nullptr ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									const  gd : : Polygon  * end_poly  =  nullptr ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									Vector3  begin_point ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									Vector3  end_point ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									real_t  begin_d  =  FLT_MAX ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									real_t  end_d  =  FLT_MAX ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									// Find the initial poly and the end poly on this map.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									for  ( const  gd : : Polygon  & p  :  polygons )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-08 20:56:33 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Only consider the polygon if it in a region with compatible layers.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-14 23:34:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  ( ( p_navigation_layers  &  p . owner - > get_navigation_layers ( ) )  = =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-08 20:56:33 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +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 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											Vector3  point  =  face . get_closest_point_to ( p_origin ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											real_t  distance_to_point  =  point . distance_to ( p_origin ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											if  ( distance_to_point  <  begin_d )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												begin_d  =  distance_to_point ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												begin_poly  =  & p ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												begin_point  =  point ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												end_poly  =  & p ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												end_point  =  point ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-05-20 12:07:26 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// Check for trivial cases
 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  ( ! begin_poly  | |  ! end_poly )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  Vector < Vector3 > ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( begin_poly  = =  end_poly )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06: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 ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										Vector < Vector3 >  path ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										path . resize ( 2 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										path . write [ 0 ]  =  begin_point ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										path . write [ 1 ]  =  end_point ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  path ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// List of all reachable navigation polys.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									LocalVector < gd : : NavigationPoly >  navigation_polys ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									navigation_polys . reserve ( polygons . size ( )  *  0.75 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// List of polygon IDs to visit.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									List < uint32_t >  to_visit ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									to_visit . push_back ( 0 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// This is an implementation of the A* algorithm.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									int  least_cost_id  =  0 ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-09-27 18:51:41 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									int  prev_least_cost_id  =  - 1 ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									bool  found_route  =  false ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-04-02 01:20:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									const  gd : : Polygon  * reachable_end  =  nullptr ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									real_t  reachable_d  =  FLT_MAX ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									bool  is_reachable  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									while  ( true )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// Takes the current least_cost_poly neighbors (iterating over its edges) and compute the traveled_distance.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( const  gd : : Edge  & edge  :  navigation_polys [ least_cost_id ] . poly - > edges )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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 ] ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-08 20:56:33 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												// Only consider the connection to another polygon if this polygon is in a region with compatible layers.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-14 23:34:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( ( p_navigation_layers  &  connection . polygon - > owner - > get_navigation_layers ( ) )  = =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
													continue ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-05-14 16:41:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-09-27 18:51:41 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												const  gd : : NavigationPoly  & least_cost_poly  =  navigation_polys [ least_cost_id ] ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												real_t  poly_enter_cost  =  0.0 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												real_t  poly_travel_cost  =  least_cost_poly . poly - > owner - > get_travel_cost ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-06 05:24:11 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-09-27 18:51:41 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( prev_least_cost_id  ! =  - 1  & &  ( navigation_polys [ prev_least_cost_id ] . poly - > owner - > get_self ( )  ! =  least_cost_poly . poly - > owner - > get_self ( ) ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													poly_enter_cost  =  least_cost_poly . poly - > owner - > get_enter_cost ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-06 05:24:11 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2022-09-27 18:51:41 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												prev_least_cost_id  =  least_cost_id ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-06 05:24:11 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												Vector3  pathway [ 2 ]  =  {  connection . pathway_start ,  connection . pathway_end  } ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-09-27 18:51:41 +08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												const  Vector3  new_entry  =  Geometry3D : : get_closest_point_to_segment ( least_cost_poly . entry ,  pathway ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01: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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												int64_t  already_visited_polygon_index  =  navigation_polys . find ( gd : : NavigationPoly ( connection . polygon ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( already_visited_polygon_index  ! =  - 1 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													// Polygon already visited, check if we can reduce the travel cost.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
													} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												}  else  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-21 12:25:29 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													// Add the neighbor polygon to the reachable ones.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													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 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-21 12:25:29 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													// Add the neighbor polygon to the polygons to visit.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													to_visit . push_back ( navigation_polys . size ( )  -  1 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Removes the least cost polygon from the list of polygons to visit so we can advance.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										to_visit . erase ( least_cost_id ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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
 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											ERR_BREAK_MSG ( is_reachable  = =  false ,  " It's not expect to not find the most reachable polygons " ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											is_reachable  =  false ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-04-02 01:20:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											if  ( reachable_end  = =  nullptr )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +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-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											end_d  =  FLT_MAX ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											for  ( size_t  point_id  =  2 ;  point_id  <  end_poly - > points . size ( ) ;  point_id + + )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												Face3  f ( end_poly - > points [ 0 ] . pos ,  end_poly - > points [ point_id  -  1 ] . pos ,  end_poly - > points [ point_id ] . pos ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												Vector3  spoint  =  f . get_closest_point_to ( p_destination ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												real_t  dpoint  =  spoint . distance_to ( p_destination ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												if  ( dpoint  <  end_d )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													end_point  =  spoint ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													end_d  =  dpoint ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-07-04 01:25:08 +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 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											// Reset open and navigation_polys
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											gd : : NavigationPoly  np  =  navigation_polys [ 0 ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											navigation_polys . clear ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											navigation_polys . push_back ( np ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											to_visit . clear ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											to_visit . push_back ( 0 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-04-14 17:13:53 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											least_cost_id  =  0 ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-02 13:18:33 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											prev_least_cost_id  =  - 1 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-04-02 01:20:12 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											reachable_end  =  nullptr ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Find the polygon with the minimum cost from the list of polygons to visit.
 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										least_cost_id  =  - 1 ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										real_t  least_cost  =  FLT_MAX ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( List < uint32_t > : : Element  * element  =  to_visit . front ( ) ;  element  ! =  nullptr ;  element  =  element - > next ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											gd : : NavigationPoly  * np  =  & navigation_polys [ element - > get ( ) ] ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											real_t  cost  =  np - > traveled_distance ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-06-06 05:24:11 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											cost  + =  ( np - > entry . distance_to ( end_point )  *  np - > poly - > owner - > get_travel_cost ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											if  ( cost  <  least_cost )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												least_cost_id  =  np - > self_id ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												least_cost  =  cost ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-14 17:13:53 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ERR_BREAK ( least_cost_id  = =  - 1 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										// Stores the further reachable end polygon, in case our goal is not reachable.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( is_reachable )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-22 19:36:18 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											real_t  d  =  navigation_polys [ least_cost_id ] . entry . distance_to ( p_destination ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +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-07-04 01:25:08 +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.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( ! found_route )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-04 01:25:08 +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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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-07-09 02:41:51 +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  =  Geometry3D : : 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 ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										Vector3  apex_point  =  end_point ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										gd : : NavigationPoly  * left_poly  =  apex_poly ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										Vector3  left_portal  =  apex_point ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										gd : : NavigationPoly  * right_poly  =  apex_poly ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										Vector3  right_portal  =  apex_point ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										gd : : NavigationPoly  * p  =  apex_poly ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										path . push_back ( end_point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										APPEND_METADATA ( end_poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													clip_path ( navigation_polys ,  path ,  apex_poly ,  right_portal ,  right_poly ,  r_path_types ,  r_path_rids ,  r_path_owners ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													apex_point  =  right_portal ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													p  =  right_poly ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													left_poly  =  p ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													apex_poly  =  p ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													left_portal  =  apex_point ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													right_portal  =  apex_point ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													path . push_back ( apex_point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													APPEND_METADATA ( apex_poly - > poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													skip  =  true ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-05-14 16:41:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												}  else  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													clip_path ( navigation_polys ,  path ,  apex_poly ,  left_portal ,  left_poly ,  r_path_types ,  r_path_rids ,  r_path_owners ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													apex_point  =  left_portal ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													p  =  left_poly ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													right_poly  =  p ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													apex_poly  =  p ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													right_portal  =  apex_point ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													left_portal  =  apex_point ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													path . push_back ( apex_point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													APPEND_METADATA ( apex_poly - > poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-05-14 16:41:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-05-14 16:41:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// If the last point is not the begin point, add it to the list.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( path [ path . size ( )  -  1 ]  ! =  begin_point )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											path . push_back ( begin_point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											APPEND_METADATA ( begin_poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-14 07:21:32 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										path . reverse ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  ( r_path_types )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											r_path_types - > reverse ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( r_path_rids )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											r_path_rids - > reverse ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( r_path_owners )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											r_path_owners - > reverse ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									}  else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										path . push_back ( end_point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										APPEND_METADATA ( end_poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Add mid points
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										int  np_id  =  least_cost_id ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-04-05 23:24:03 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										while  ( np_id  ! =  - 1  & &  navigation_polys [ np_id ] . back_navigation_poly_id  ! =  - 1 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08: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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												path . push_back ( point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												APPEND_METADATA ( navigation_polys [ np_id ] . poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											}  else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												path . push_back ( navigation_polys [ np_id ] . entry ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												APPEND_METADATA ( navigation_polys [ np_id ] . poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											np_id  =  navigation_polys [ np_id ] . back_navigation_poly_id ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-05 23:24:03 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										path . push_back ( begin_point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										APPEND_METADATA ( begin_poly ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-14 07:21:32 +00:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										path . reverse ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  ( r_path_types )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											r_path_types - > reverse ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( r_path_rids )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											r_path_rids - > reverse ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( r_path_owners )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											r_path_owners - > reverse ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06: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 ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									return  path ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Vector3  NavMap : : get_closest_point_to_segment ( const  Vector3  & p_from ,  const  Vector3  & p_to ,  const  bool  p_use_collision )  const  {  
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockRead  read_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 03:02:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( iteration_id  = =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-22 12:50:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										NAVMAP_ITERATION_ZERO_ERROR_MSG ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  Vector3 ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									bool  use_collision  =  p_use_collision ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									Vector3  closest_point ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									real_t  closest_point_d  =  FLT_MAX ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									for  ( const  gd : : Polygon  & p  :  polygons )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// For each face check the distance to the segment
 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( size_t  point_id  =  2 ;  point_id  <  p . points . size ( ) ;  point_id  + =  1 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											const  Face3  f ( p . points [ 0 ] . pos ,  p . points [ point_id  -  1 ] . pos ,  p . points [ point_id ] . pos ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											Vector3  inters ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  ( f . intersects_segment ( p_from ,  p_to ,  & inters ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-06-06 23:20:05 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												const  real_t  d  =  p_from . distance_to ( inters ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												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 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-05-25 20:20:45 +03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												Geometry3D : : get_closest_points_between_segments ( 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
														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-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockRead  read_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 03:02:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( iteration_id  = =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-22 12:50:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										NAVMAP_ITERATION_ZERO_ERROR_MSG ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  Vector3 ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									gd : : ClosestPointQueryResult  cp  =  get_closest_point_info ( p_point ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  cp . point ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Vector3  NavMap : : get_closest_point_normal ( const  Vector3  & p_point )  const  {  
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockRead  read_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 03:02:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( iteration_id  = =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-22 12:50:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										NAVMAP_ITERATION_ZERO_ERROR_MSG ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  Vector3 ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									gd : : ClosestPointQueryResult  cp  =  get_closest_point_info ( p_point ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  cp . normal ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								RID  NavMap : : get_closest_point_owner ( const  Vector3  & p_point )  const  {  
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockRead  read_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 03:02:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( iteration_id  = =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-22 12:50:20 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										NAVMAP_ITERATION_ZERO_ERROR_MSG ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  RID ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									gd : : ClosestPointQueryResult  cp  =  get_closest_point_info ( p_point ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									return  cp . owner ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								gd : : ClosestPointQueryResult  NavMap : : get_closest_point_info ( const  Vector3  & p_point )  const  {  
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockRead  read_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									gd : : ClosestPointQueryResult  result ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-03-07 15:16:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									real_t  closest_point_ds  =  FLT_MAX ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									for  ( const  gd : : Polygon  & p  :  polygons )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// For each face check the distance to the point
 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( size_t  point_id  =  2 ;  point_id  <  p . points . size ( ) ;  point_id  + =  1 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											const  Face3  f ( p . points [ 0 ] . pos ,  p . points [ point_id  -  1 ] . pos ,  p . points [ point_id ] . pos ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											const  Vector3  inters  =  f . get_closest_point_to ( p_point ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-13 16:07:01 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									return  result ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-18 17:08:34 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								void  NavMap : : add_region ( NavRegion  * p_region )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regions . push_back ( p_region ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  NavMap : : remove_region ( NavRegion  * p_region )  {  
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									int64_t  region_index  =  regions . find ( p_region ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( region_index  > =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										regions . remove_at_unordered ( region_index ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-02-24 18:09:14 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08: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-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( link_index  > =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										links . remove_at_unordered ( link_index ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-02-01 22:11:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								bool  NavMap : : has_agent ( NavAgent  * agent )  const  {  
						 
					
						
							
								
									
										
										
										
											2024-05-06 16:43:04 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									return  agents . has ( agent ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-02-01 22:11:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : add_agent ( NavAgent  * agent )  {  
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  ( ! has_agent ( agent ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										agents . push_back ( agent ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										agents_dirty  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-02-01 22:11:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : remove_agent ( NavAgent  * agent )  {  
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									remove_agent_as_controlled ( agent ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									int64_t  agent_index  =  agents . find ( agent ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( agent_index  > =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										agents . remove_at_unordered ( agent_index ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										agents_dirty  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								bool  NavMap : : has_obstacle ( NavObstacle  * obstacle )  const  {  
						 
					
						
							
								
									
										
										
										
											2024-05-06 16:43:04 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									return  obstacles . has ( obstacle ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  NavMap : : add_obstacle ( NavObstacle  * obstacle )  {  
						 
					
						
							
								
									
										
										
										
											2023-06-15 15:35:53 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( obstacle - > get_paused ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// No point in adding a paused obstacle, it will add itself when unpaused again.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01: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_at_unordered ( obstacle_index ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										obstacles_dirty  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-02-01 22:11:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : set_agent_as_controlled ( NavAgent  * agent )  {  
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									remove_agent_as_controlled ( agent ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-06-15 15:35:53 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( agent - > get_paused ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// No point in adding a paused agent, it will add itself when unpaused again.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01: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 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-02-01 22:11:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : remove_agent_as_controlled ( NavAgent  * agent )  {  
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									int64_t  agent_3d_index  =  active_3d_avoidance_agents . find ( agent ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( agent_3d_index  > =  0 )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										active_3d_avoidance_agents . remove_at_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_at_unordered ( agent_2d_index ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										agents_dirty  =  true ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-10-23 18:16:05 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Vector3  NavMap : : get_random_point ( uint32_t  p_navigation_layers ,  bool  p_uniformly )  const  {  
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockRead  read_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-10-23 18:16:05 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									const  LocalVector < NavRegion  * >  map_regions  =  get_regions ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( map_regions . is_empty ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  Vector3 ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									LocalVector < const  NavRegion  * >  accessible_regions ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  ( const  NavRegion  * region  :  map_regions )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( ! region - > get_enabled ( )  | |  ( p_navigation_layers  &  region - > get_navigation_layers ( ) )  = =  0 )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										accessible_regions . push_back ( region ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( accessible_regions . is_empty ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// All existing region polygons are disabled.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  Vector3 ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( p_uniformly )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										real_t  accumulated_region_surface_area  =  0 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										RBMap < real_t ,  uint32_t >  accessible_regions_area_map ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										for  ( uint32_t  accessible_region_index  =  0 ;  accessible_region_index  <  accessible_regions . size ( ) ;  accessible_region_index + + )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											const  NavRegion  * region  =  accessible_regions [ accessible_region_index ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											real_t  region_surface_area  =  region - > get_surface_area ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  ( region_surface_area  = =  0.0f )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											accessible_regions_area_map [ accumulated_region_surface_area ]  =  accessible_region_index ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											accumulated_region_surface_area  + =  region_surface_area ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( accessible_regions_area_map . is_empty ( )  | |  accumulated_region_surface_area  = =  0 )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											// All faces have no real surface / no area.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											return  Vector3 ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										real_t  random_accessible_regions_area_map  =  Math : : random ( real_t ( 0 ) ,  accumulated_region_surface_area ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										RBMap < real_t ,  uint32_t > : : Iterator  E  =  accessible_regions_area_map . find_closest ( random_accessible_regions_area_map ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ERR_FAIL_COND_V ( ! E ,  Vector3 ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										uint32_t  random_region_index  =  E - > value ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-12-12 11:00:06 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ERR_FAIL_UNSIGNED_INDEX_V ( random_region_index ,  accessible_regions . size ( ) ,  Vector3 ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-23 18:16:05 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										const  NavRegion  * random_region  =  accessible_regions [ random_region_index ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ERR_FAIL_NULL_V ( random_region ,  Vector3 ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  random_region - > get_random_point ( p_navigation_layers ,  p_uniformly ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									}  else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										uint32_t  random_region_index  =  Math : : random ( int ( 0 ) ,  accessible_regions . size ( )  -  1 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										const  NavRegion  * random_region  =  accessible_regions [ random_region_index ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										ERR_FAIL_NULL_V ( random_region ,  Vector3 ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return  random_region - > get_random_point ( p_navigation_layers ,  p_uniformly ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								void  NavMap : : sync ( )  {  
						 
					
						
							
								
									
										
										
										
											2023-07-17 12:20:09 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									RWLockWrite  write_lock ( map_rwlock ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01: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 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// Check if we need to update the links.
 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  ( regenerate_polygons )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( NavRegion  * region  :  regions )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											region - > scratch_polygons ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									for  ( NavRegion  * region  :  regions )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( region - > sync ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									for  ( NavLink  * link  :  links )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( link - > check_dirty ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											regenerate_links  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									if  ( regenerate_links )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01: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 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Remove regions connections.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( NavRegion  * region  :  regions )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											region - > get_connections ( ) . clear ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// Resize the polygon count.
 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										int  count  =  0 ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( const  NavRegion  * region  :  regions )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-06 23:01:19 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											if  ( ! region - > get_enabled ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											count  + =  region - > get_polygons ( ) . size ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										polygons . resize ( count ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Copy all region polygons in the map.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										count  =  0 ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( const  NavRegion  * region  :  regions )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-07-06 23:01:19 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											if  ( ! region - > get_enabled ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											const  LocalVector < gd : : Polygon >  & polygons_source  =  region - > get_polygons ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											for  ( uint32_t  n  =  0 ;  n  <  polygons_source . size ( ) ;  n + + )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												polygons [ count  +  n ]  =  polygons_source [ n ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											count  + =  region - > get_polygons ( ) . size ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										_new_pm_polygon_count  =  polygons . size ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Group all edges per key.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-05-13 15:04:37 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										HashMap < gd : : EdgeKey ,  Vector < gd : : Edge : : Connection > ,  gd : : EdgeKey >  connections ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( gd : : Polygon  & poly  :  polygons )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-07-28 19:24:14 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											for  ( uint32_t  p  =  0 ;  p  <  poly . points . size ( ) ;  p + + )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												int  next_point  =  ( p  +  1 )  %  poly . points . size ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												gd : : EdgeKey  ek ( poly . points [ p ] . key ,  poly . points [ next_point ] . key ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-05-13 15:04:37 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												HashMap < gd : : EdgeKey ,  Vector < gd : : Edge : : Connection > ,  gd : : EdgeKey > : : Iterator  connection  =  connections . find ( ek ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												if  ( ! connection )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													connections [ ek ]  =  Vector < gd : : Edge : : Connection > ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													_new_pm_edge_count  + =  1 ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												}  else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													// The edge is already connected with another edge, skip.
 
							 
						 
					
						
							
								
									
										
										
										
											2024-02-04 21:31:07 +01: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'. If you're certain none of above is the case, change 'navigation/3d/merge_rasterizer_cell_scale' to 0.001. " ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										Vector < gd : : Edge : : Connection >  free_edges ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-08-09 14:13:42 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( KeyValue < gd : : EdgeKey ,  Vector < gd : : Edge : : Connection > >  & E  :  connections )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  ( E . value . size ( )  = =  2 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												// Connect edge that are shared in different polygons.
 
							 
						 
					
						
							
								
									
										
										
										
											2021-08-09 14:13:42 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												gd : : Edge : : Connection  & c1  =  E . value . write [ 0 ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												gd : : Edge : : Connection  & c2  =  E . value . write [ 1 ] ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												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.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												_new_pm_edge_merge_count  + =  1 ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											}  else  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-08-09 14:13:42 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												CRASH_COND_MSG ( E . value . size ( )  ! =  1 ,  vformat ( " Number of connection != 1. Found: %d " ,  E . value . size ( ) ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-04-01 01:49:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( use_edge_connections  & &  E . value [ 0 ] . polygon - > owner - > get_use_edge_connections ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
													free_edges . push_back ( E . value [ 0 ] ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +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.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										_new_pm_edge_free_count  =  free_edges . size ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01: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-03-07 15:16:07 +01: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 ( ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( ( projected_p1_ratio  <  0.0  & &  projected_p2_ratio  <  0.0 )  | |  ( projected_p1_ratio  >  1.0  & &  projected_p2_ratio  >  1.0 ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
													continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												// 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 . lerp ( other_edge_p2 ,  ( 1.0  -  projected_p1_ratio )  /  ( projected_p2_ratio  -  projected_p1_ratio ) ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-29 09:36:34 +05:45 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( other1 . distance_to ( self1 )  >  edge_connection_margin )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												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 . lerp ( other_edge_p2 ,  ( 0.0  -  projected_p1_ratio )  /  ( projected_p2_ratio  -  projected_p1_ratio ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-29 09:36:34 +05:45 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( other2 . distance_to ( self2 )  >  edge_connection_margin )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												// 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.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												( ( NavRegion  * ) free_edge . polygon - > owner ) - > get_connections ( ) . push_back ( new_connection ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												_new_pm_edge_connection_count  + =  1 ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										uint32_t  link_poly_idx  =  0 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										link_polygons . resize ( links . size ( ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										// Search for polygons within range of a nav link.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										for  ( const  NavLink  * link  :  links )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-10-21 03:11:37 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											if  ( ! link - > get_enabled ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												continue ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-06 23:32:11 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											const  Vector3  start  =  link - > get_start_position ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											const  Vector3  end  =  link - > get_end_position ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											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.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-29 01:24:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											for  ( gd : : Polygon  & end_poly  :  polygons )  { 
							 
						 
					
						
							
								
									
										
										
										
											2022-01-30 15:39:52 -08:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												// 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  =  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 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-11-01 03:02:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										// Some code treats 0 as a failure case, so we avoid returning 0 and modulo wrap UINT32_MAX manually.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										iteration_id  =  iteration_id  %  UINT32_MAX  +  1 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// Do we have modified obstacle positions?
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  ( NavObstacle  * obstacle  :  obstacles )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( obstacle - > check_dirty ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											obstacles_dirty  =  true ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// Do we have modified agent arrays?
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  ( NavAgent  * agent  :  agents )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( agent - > check_dirty ( ) )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											agents_dirty  =  true ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-05-14 16:41:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									// Update avoidance worlds.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( obstacles_dirty  | |  agents_dirty )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										_update_rvo_simulation ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regenerate_polygons  =  false ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									regenerate_links  =  false ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									obstacles_dirty  =  false ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									agents_dirty  =  false ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// Performance Monitor.
 
							 
						 
					
						
							
								
									
										
										
										
											2022-12-30 05:19:15 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									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 ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : _update_rvo_obstacles_tree_2d ( )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									int  obstacle_vertex_count  =  0 ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  ( NavObstacle  * obstacle  :  obstacles )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										obstacle_vertex_count  + =  obstacle - > get_vertices ( ) . size ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-11-12 23:05:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// Cleaning old obstacles.
 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  ( size_t  i  =  0 ;  i  <  rvo_simulation_2d . obstacles_ . size ( ) ;  + + i )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										delete  rvo_simulation_2d . obstacles_ [ i ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									rvo_simulation_2d . obstacles_ . clear ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									// Cannot use LocalVector here as RVO library expects std::vector to build KdTree
 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-12 23:05:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									std : : vector < RVO2D : : Obstacle2D  * >  & raw_obstacles  =  rvo_simulation_2d . obstacles_ ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									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  ( NavObstacle  * obstacle  :  obstacles )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										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 ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-13 21:32:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										real_t  _obstacle_height  =  obstacle - > get_height ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										for  ( const  Vector3  & _obstacle_vertex  :  _obstacle_vertices )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-12 23:05:45 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								# ifdef TOOLS_ENABLED 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  ( _obstacle_vertex . y  ! =  0 )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												WARN_PRINT_ONCE ( " Y coordinates of static obstacle vertices are ignored. Please use obstacle position Y to change elevation of obstacle. " ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# endif 
  
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											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 ] ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-13 21:32:22 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											rvo_2d_obstacle - > height_  =  _obstacle_height ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-11-13 06:55:31 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											rvo_2d_obstacle - > elevation_  =  _obstacle_position . y ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											rvo_2d_obstacle - > avoidance_layers_  =  _obstacle_avoidance_layers ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  ( i  ! =  0 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-06-29 12:50:49 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												rvo_2d_obstacle - > prevObstacle_  =  raw_obstacles . back ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												rvo_2d_obstacle - > prevObstacle_ - > nextObstacle_  =  rvo_2d_obstacle ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											if  ( i  = =  rvo_2d_vertices . size ( )  -  1 )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-06-29 12:50:49 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												rvo_2d_obstacle - > nextObstacle_  =  raw_obstacles [ obstacleNo ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												rvo_2d_obstacle - > nextObstacle_ - > prevObstacle_  =  rvo_2d_obstacle ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-29 12:50:49 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											rvo_2d_obstacle - > unitDir_  =  normalize ( rvo_2d_vertices [ ( i  = =  rvo_2d_vertices . size ( )  -  1  ?  0  :  i  +  1 ) ]  -  rvo_2d_vertices [ i ] ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											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  ( NavAgent  * agent  :  active_2d_avoidance_agents )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										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  ( NavAgent  * agent  :  active_3d_avoidance_agents )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										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 )  {  
						 
					
						
							
								
									
										
										
										
											2023-06-29 12:50:49 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									( * ( 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 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									( * ( 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 ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  NavMap : : step ( real_t  p_deltatime )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									deltatime  =  p_deltatime ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									rvo_simulation_2d . setTimeStep ( float ( deltatime ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									rvo_simulation_3d . setTimeStep ( float ( deltatime ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( active_2d_avoidance_agents . size ( )  >  0 )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( use_threads  & &  avoidance_use_multiple_threads )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											WorkerThreadPool : : GroupID  group_task  =  WorkerThreadPool : : get_singleton ( ) - > add_template_group_task ( this ,  & NavMap : : compute_single_avoidance_step_2d ,  active_2d_avoidance_agents . ptr ( ) ,  active_2d_avoidance_agents . size ( ) ,  - 1 ,  true ,  SNAME ( " RVOAvoidanceAgents2D " ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											WorkerThreadPool : : get_singleton ( ) - > wait_for_group_task_completion ( group_task ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										}  else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											for  ( NavAgent  * agent  :  active_2d_avoidance_agents )  { 
							 
						 
					
						
							
								
									
										
										
										
											2023-06-29 12:50:49 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												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 ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												agent - > update ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									if  ( active_3d_avoidance_agents . size ( )  >  0 )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										if  ( use_threads  & &  avoidance_use_multiple_threads )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											WorkerThreadPool : : GroupID  group_task  =  WorkerThreadPool : : get_singleton ( ) - > add_template_group_task ( this ,  & NavMap : : compute_single_avoidance_step_3d ,  active_3d_avoidance_agents . ptr ( ) ,  active_3d_avoidance_agents . size ( ) ,  - 1 ,  true ,  SNAME ( " RVOAvoidanceAgents3D " ) ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											WorkerThreadPool : : get_singleton ( ) - > wait_for_group_task_completion ( group_task ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										}  else  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											for  ( NavAgent  * agent  :  active_3d_avoidance_agents )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
												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 ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								void  NavMap : : dispatch_callbacks ( )  {  
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									for  ( NavAgent  * agent  :  active_2d_avoidance_agents )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										agent - > dispatch_avoidance_callback ( ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									for  ( NavAgent  * agent  :  active_3d_avoidance_agents )  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										agent - > dispatch_avoidance_callback ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06: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 ,  TypedArray < RID >  * r_path_rids ,  Vector < int64_t >  * r_path_owners )  const  {  
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									Vector3  from  =  path [ path . size ( )  -  1 ] ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-07-16 13:19:55 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( from . is_equal_approx ( p_to_point ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											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 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									Plane  cut_plane ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									cut_plane . normal  =  ( from  -  p_to_point ) . cross ( up ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-05-14 16:41:43 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									if  ( cut_plane . normal  = =  Vector3 ( ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											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 
										
									 
								 
							 
							
								
							 
							
								 
							
							
									cut_plane . normal . normalize ( ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-05-10 16:47:11 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
									cut_plane . d  =  cut_plane . normal . dot ( from ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									while  ( from_poly  ! =  p_to_poly )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										Vector3  pathway_start  =  from_poly - > back_navigation_edge_pathway_start ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										Vector3  pathway_end  =  from_poly - > back_navigation_edge_pathway_end ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										ERR_FAIL_COND ( from_poly - > back_navigation_poly_id  = =  - 1 ) ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										from_poly  =  & p_navigation_polys [ from_poly - > back_navigation_poly_id ] ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-07-16 13:19:55 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
										if  ( ! pathway_start . is_equal_approx ( pathway_end ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
											Vector3  inters ; 
							 
						 
					
						
							
								
									
										
										
										
											2021-03-15 12:45:28 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
											if  ( cut_plane . intersects_segment ( pathway_start ,  pathway_end ,  & inters ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2021-07-16 13:19:55 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
												if  ( ! inters . is_equal_approx ( p_to_point )  & &  ! inters . is_equal_approx ( path [ path . size ( )  -  1 ] ) )  { 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
													path . push_back ( inters ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2022-10-05 17:24:45 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
													APPEND_METADATA ( from_poly - > poly ) ; 
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 12:22:34 +01:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
												} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
											} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									} 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
									
										
										
										
											2022-04-18 15:57:23 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-02-04 21:31:07 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								void  NavMap : : _update_merge_rasterizer_cell_dimensions ( )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									merge_rasterizer_cell_size  =  cell_size  *  merge_rasterizer_cell_scale ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									merge_rasterizer_cell_height  =  cell_height  *  merge_rasterizer_cell_scale ; 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-18 15:57:23 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								NavMap : : NavMap ( )  {  
						 
					
						
							
								
									
										
										
										
											2023-01-10 07:14:16 +01: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-04-18 15:57:23 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								NavMap : : ~ NavMap ( )  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}