2015-10-08 18:00:40 +00:00
/*************************************************************************/
2020-03-28 12:19:05 +00:00
/* space_3d_sw.cpp */
2015-10-08 18:00:40 +00:00
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
2017-08-27 12:16:55 +00:00
/* https://godotengine.org */
2015-10-08 18:00:40 +00:00
/*************************************************************************/
2020-01-01 10:16:22 +00:00
/* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md). */
2015-10-08 18:00:40 +00:00
/* */
/* 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. */
/*************************************************************************/
2018-01-04 23:50:27 +00:00
2020-03-27 18:21:27 +00:00
# include "space_3d_sw.h"
2017-08-27 19:07:15 +00:00
2020-03-27 18:21:27 +00:00
# include "collision_solver_3d_sw.h"
2018-09-11 16:13:45 +00:00
# include "core/project_settings.h"
2020-03-27 18:21:27 +00:00
# include "physics_server_3d_sw.h"
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
_FORCE_INLINE_ static bool _can_collide_with ( CollisionObject3DSW * p_object , uint32_t p_collision_mask , bool p_collide_with_bodies , bool p_collide_with_areas ) {
2018-08-21 18:30:41 +00:00
if ( ! ( p_object - > get_collision_layer ( ) & p_collision_mask ) ) {
return false ;
}
2020-03-27 18:21:27 +00:00
if ( p_object - > get_type ( ) = = CollisionObject3DSW : : TYPE_AREA & & ! p_collide_with_areas )
2018-08-21 18:30:41 +00:00
return false ;
2020-03-27 18:21:27 +00:00
if ( p_object - > get_type ( ) = = CollisionObject3DSW : : TYPE_BODY & & ! p_collide_with_bodies )
2018-08-21 18:30:41 +00:00
return false ;
return true ;
2015-10-08 18:00:40 +00:00
}
2020-03-27 18:21:27 +00:00
int PhysicsDirectSpaceState3DSW : : intersect_point ( const Vector3 & p_point , ShapeResult * r_results , int p_result_max , const Set < RID > & p_exclude , uint32_t p_collision_mask , bool p_collide_with_bodies , bool p_collide_with_areas ) {
2017-07-15 04:23:10 +00:00
ERR_FAIL_COND_V ( space - > locked , false ) ;
2020-03-27 18:21:27 +00:00
int amount = space - > broadphase - > cull_point ( p_point , space - > intersection_query_results , Space3DSW : : INTERSECTION_QUERY_MAX , space - > intersection_query_subindex_results ) ;
2017-07-15 04:23:10 +00:00
int cc = 0 ;
//Transform ai = p_xform.affine_inverse();
for ( int i = 0 ; i < amount ; i + + ) {
if ( cc > = p_result_max )
break ;
2018-08-21 18:30:41 +00:00
if ( ! _can_collide_with ( space - > intersection_query_results [ i ] , p_collision_mask , p_collide_with_bodies , p_collide_with_areas ) )
2017-07-15 04:23:10 +00:00
continue ;
//area can't be picked by ray (default)
if ( p_exclude . has ( space - > intersection_query_results [ i ] - > get_self ( ) ) )
continue ;
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = space - > intersection_query_results [ i ] ;
2017-07-15 04:23:10 +00:00
int shape_idx = space - > intersection_query_subindex_results [ i ] ;
Transform inv_xform = col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) ;
inv_xform . affine_invert ( ) ;
if ( ! col_obj - > get_shape ( shape_idx ) - > intersect_point ( inv_xform . xform ( p_point ) ) )
continue ;
r_results [ cc ] . collider_id = col_obj - > get_instance_id ( ) ;
2020-02-12 17:24:06 +00:00
if ( r_results [ cc ] . collider_id . is_valid ( ) )
2017-07-15 04:23:10 +00:00
r_results [ cc ] . collider = ObjectDB : : get_instance ( r_results [ cc ] . collider_id ) ;
else
2020-04-01 23:20:12 +00:00
r_results [ cc ] . collider = nullptr ;
2017-07-15 04:23:10 +00:00
r_results [ cc ] . rid = col_obj - > get_self ( ) ;
r_results [ cc ] . shape = shape_idx ;
cc + + ;
}
return cc ;
}
2020-03-27 18:21:27 +00:00
bool PhysicsDirectSpaceState3DSW : : intersect_ray ( const Vector3 & p_from , const Vector3 & p_to , RayResult & r_result , const Set < RID > & p_exclude , uint32_t p_collision_mask , bool p_collide_with_bodies , bool p_collide_with_areas , bool p_pick_ray ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_COND_V ( space - > locked , false ) ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
Vector3 begin , end ;
2015-10-08 18:00:40 +00:00
Vector3 normal ;
2017-03-05 15:44:50 +00:00
begin = p_from ;
end = p_to ;
normal = ( end - begin ) . normalized ( ) ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
int amount = space - > broadphase - > cull_segment ( begin , end , space - > intersection_query_results , Space3DSW : : INTERSECTION_QUERY_MAX , space - > intersection_query_subindex_results ) ;
2015-10-08 18:00:40 +00:00
2018-09-13 01:38:39 +00:00
//todo, create another array that references results, compute AABBs and check closest point to ray origin, sort, and stop evaluating results when beyond first collision
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
bool collided = false ;
Vector3 res_point , res_normal ;
2015-10-08 18:00:40 +00:00
int res_shape ;
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * res_obj ;
2017-03-05 15:44:50 +00:00
real_t min_d = 1e10 ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
for ( int i = 0 ; i < amount ; i + + ) {
2018-08-21 18:30:41 +00:00
if ( ! _can_collide_with ( space - > intersection_query_results [ i ] , p_collision_mask , p_collide_with_bodies , p_collide_with_areas ) )
2015-10-08 18:00:40 +00:00
continue ;
2019-06-26 13:08:25 +00:00
if ( p_pick_ray & & ! ( space - > intersection_query_results [ i ] - > is_ray_pickable ( ) ) )
2015-10-08 18:00:40 +00:00
continue ;
2017-03-05 15:44:50 +00:00
if ( p_exclude . has ( space - > intersection_query_results [ i ] - > get_self ( ) ) )
2015-10-08 18:00:40 +00:00
continue ;
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = space - > intersection_query_results [ i ] ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
int shape_idx = space - > intersection_query_subindex_results [ i ] ;
2015-10-08 18:00:40 +00:00
Transform inv_xform = col_obj - > get_shape_inv_transform ( shape_idx ) * col_obj - > get_inv_transform ( ) ;
Vector3 local_from = inv_xform . xform ( begin ) ;
Vector3 local_to = inv_xform . xform ( end ) ;
2020-03-27 18:21:27 +00:00
const Shape3DSW * shape = col_obj - > get_shape ( shape_idx ) ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
Vector3 shape_point , shape_normal ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
if ( shape - > intersect_segment ( local_from , local_to , shape_point , shape_normal ) ) {
2015-10-08 18:00:40 +00:00
Transform xform = col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) ;
2017-03-05 15:44:50 +00:00
shape_point = xform . xform ( shape_point ) ;
2015-10-08 18:00:40 +00:00
real_t ld = normal . dot ( shape_point ) ;
2017-03-05 15:44:50 +00:00
if ( ld < min_d ) {
min_d = ld ;
res_point = shape_point ;
res_normal = inv_xform . basis . xform_inv ( shape_normal ) . normalized ( ) ;
res_shape = shape_idx ;
res_obj = col_obj ;
collided = true ;
2015-10-08 18:00:40 +00:00
}
}
}
if ( ! collided )
return false ;
2017-03-05 15:44:50 +00:00
r_result . collider_id = res_obj - > get_instance_id ( ) ;
2020-02-12 17:24:06 +00:00
if ( r_result . collider_id . is_valid ( ) )
2017-03-05 15:44:50 +00:00
r_result . collider = ObjectDB : : get_instance ( r_result . collider_id ) ;
2015-10-08 18:00:40 +00:00
else
2020-04-01 23:20:12 +00:00
r_result . collider = nullptr ;
2017-03-05 15:44:50 +00:00
r_result . normal = res_normal ;
r_result . position = res_point ;
r_result . rid = res_obj - > get_self ( ) ;
r_result . shape = res_shape ;
2015-10-08 18:00:40 +00:00
return true ;
}
2020-03-27 18:21:27 +00:00
int PhysicsDirectSpaceState3DSW : : intersect_shape ( const RID & p_shape , const Transform & p_xform , real_t p_margin , ShapeResult * r_results , int p_result_max , const Set < RID > & p_exclude , uint32_t p_collision_mask , bool p_collide_with_bodies , bool p_collide_with_areas ) {
2017-03-05 15:44:50 +00:00
if ( p_result_max < = 0 )
2015-10-08 18:00:40 +00:00
return 0 ;
2020-03-27 18:21:27 +00:00
Shape3DSW * shape = static_cast < PhysicsServer3DSW * > ( PhysicsServer3D : : get_singleton ( ) ) - > shape_owner . getornull ( p_shape ) ;
2017-03-05 15:44:50 +00:00
ERR_FAIL_COND_V ( ! shape , 0 ) ;
2015-10-08 18:00:40 +00:00
2017-11-17 02:09:00 +00:00
AABB aabb = p_xform . xform ( shape - > get_aabb ( ) ) ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
int amount = space - > broadphase - > cull_aabb ( aabb , space - > intersection_query_results , Space3DSW : : INTERSECTION_QUERY_MAX , space - > intersection_query_subindex_results ) ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
int cc = 0 ;
2015-10-08 18:00:40 +00:00
//Transform ai = p_xform.affine_inverse();
2017-03-05 15:44:50 +00:00
for ( int i = 0 ; i < amount ; i + + ) {
if ( cc > = p_result_max )
2015-10-08 18:00:40 +00:00
break ;
2018-08-21 18:30:41 +00:00
if ( ! _can_collide_with ( space - > intersection_query_results [ i ] , p_collision_mask , p_collide_with_bodies , p_collide_with_areas ) )
2015-10-08 18:00:40 +00:00
continue ;
2017-03-24 20:45:31 +00:00
//area can't be picked by ray (default)
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
if ( p_exclude . has ( space - > intersection_query_results [ i ] - > get_self ( ) ) )
2015-10-08 18:00:40 +00:00
continue ;
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = space - > intersection_query_results [ i ] ;
2017-03-05 15:44:50 +00:00
int shape_idx = space - > intersection_query_subindex_results [ i ] ;
2015-10-08 18:00:40 +00:00
2020-04-01 23:20:12 +00:00
if ( ! CollisionSolver3DSW : : solve_static ( shape , p_xform , col_obj - > get_shape ( shape_idx ) , col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) , nullptr , nullptr , nullptr , p_margin , 0 ) )
2015-10-08 18:00:40 +00:00
continue ;
2015-12-04 08:31:01 +00:00
if ( r_results ) {
2017-03-05 15:44:50 +00:00
r_results [ cc ] . collider_id = col_obj - > get_instance_id ( ) ;
2020-02-12 17:24:06 +00:00
if ( r_results [ cc ] . collider_id . is_valid ( ) )
2017-03-05 15:44:50 +00:00
r_results [ cc ] . collider = ObjectDB : : get_instance ( r_results [ cc ] . collider_id ) ;
2015-12-04 08:31:01 +00:00
else
2020-04-01 23:20:12 +00:00
r_results [ cc ] . collider = nullptr ;
2017-03-05 15:44:50 +00:00
r_results [ cc ] . rid = col_obj - > get_self ( ) ;
r_results [ cc ] . shape = shape_idx ;
2015-12-04 08:31:01 +00:00
}
2015-11-22 13:14:07 +00:00
2015-12-04 08:31:01 +00:00
cc + + ;
2015-10-08 18:00:40 +00:00
}
return cc ;
}
2020-03-27 18:21:27 +00:00
bool PhysicsDirectSpaceState3DSW : : cast_motion ( const RID & p_shape , const Transform & p_xform , const Vector3 & p_motion , real_t p_margin , real_t & p_closest_safe , real_t & p_closest_unsafe , const Set < RID > & p_exclude , uint32_t p_collision_mask , bool p_collide_with_bodies , bool p_collide_with_areas , ShapeRestInfo * r_info ) {
Shape3DSW * shape = static_cast < PhysicsServer3DSW * > ( PhysicsServer3D : : get_singleton ( ) ) - > shape_owner . getornull ( p_shape ) ;
2017-03-05 15:44:50 +00:00
ERR_FAIL_COND_V ( ! shape , false ) ;
2015-10-08 18:00:40 +00:00
2017-11-17 02:09:00 +00:00
AABB aabb = p_xform . xform ( shape - > get_aabb ( ) ) ;
aabb = aabb . merge ( AABB ( aabb . position + p_motion , aabb . size ) ) ; //motion
2017-03-05 15:44:50 +00:00
aabb = aabb . grow ( p_margin ) ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
int amount = space - > broadphase - > cull_aabb ( aabb , space - > intersection_query_results , Space3DSW : : INTERSECTION_QUERY_MAX , space - > intersection_query_subindex_results ) ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
real_t best_safe = 1 ;
real_t best_unsafe = 1 ;
2015-10-08 18:00:40 +00:00
Transform xform_inv = p_xform . affine_inverse ( ) ;
2020-03-27 18:21:27 +00:00
MotionShape3DSW mshape ;
2017-03-05 15:44:50 +00:00
mshape . shape = shape ;
mshape . motion = xform_inv . basis . xform ( p_motion ) ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
bool best_first = true ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
Vector3 closest_A , closest_B ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
for ( int i = 0 ; i < amount ; i + + ) {
2018-08-21 18:30:41 +00:00
if ( ! _can_collide_with ( space - > intersection_query_results [ i ] , p_collision_mask , p_collide_with_bodies , p_collide_with_areas ) )
2015-10-08 18:00:40 +00:00
continue ;
2017-03-05 15:44:50 +00:00
if ( p_exclude . has ( space - > intersection_query_results [ i ] - > get_self ( ) ) )
2015-10-08 18:00:40 +00:00
continue ; //ignore excluded
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = space - > intersection_query_results [ i ] ;
2017-03-05 15:44:50 +00:00
int shape_idx = space - > intersection_query_subindex_results [ i ] ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
Vector3 point_A , point_B ;
Vector3 sep_axis = p_motion . normalized ( ) ;
2015-10-08 18:00:40 +00:00
Transform col_obj_xform = col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) ;
//test initial overlap, does it collide if going all the way?
2020-03-27 18:21:27 +00:00
if ( CollisionSolver3DSW : : solve_distance ( & mshape , p_xform , col_obj - > get_shape ( shape_idx ) , col_obj_xform , point_A , point_B , aabb , & sep_axis ) ) {
2015-10-08 18:00:40 +00:00
continue ;
}
2017-08-27 19:07:15 +00:00
//test initial overlap
2017-03-05 15:44:50 +00:00
sep_axis = p_motion . normalized ( ) ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
if ( ! CollisionSolver3DSW : : solve_distance ( shape , p_xform , col_obj - > get_shape ( shape_idx ) , col_obj_xform , point_A , point_B , aabb , & sep_axis ) ) {
2015-10-08 18:00:40 +00:00
return false ;
}
//just do kinematic solving
2017-03-05 15:44:50 +00:00
real_t low = 0 ;
real_t hi = 1 ;
Vector3 mnormal = p_motion . normalized ( ) ;
2015-10-08 18:00:40 +00:00
2019-02-12 20:10:08 +00:00
for ( int j = 0 ; j < 8 ; j + + ) { //steps should be customizable..
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
real_t ofs = ( low + hi ) * 0.5 ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
Vector3 sep = mnormal ; //important optimization for this to work fast enough
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
mshape . motion = xform_inv . basis . xform ( p_motion * ofs ) ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
Vector3 lA , lB ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
bool collided = ! CollisionSolver3DSW : : solve_distance ( & mshape , p_xform , col_obj - > get_shape ( shape_idx ) , col_obj_xform , lA , lB , aabb , & sep ) ;
2015-10-08 18:00:40 +00:00
if ( collided ) {
2017-03-05 15:44:50 +00:00
hi = ofs ;
2015-10-08 18:00:40 +00:00
} else {
2017-03-05 15:44:50 +00:00
point_A = lA ;
point_B = lB ;
low = ofs ;
2015-10-08 18:00:40 +00:00
}
}
2017-03-05 15:44:50 +00:00
if ( low < best_safe ) {
best_first = true ; //force reset
best_safe = low ;
best_unsafe = hi ;
2015-10-08 18:00:40 +00:00
}
2017-03-05 15:44:50 +00:00
if ( r_info & & ( best_first | | ( point_A . distance_squared_to ( point_B ) < closest_A . distance_squared_to ( closest_B ) & & low < = best_safe ) ) ) {
closest_A = point_A ;
closest_B = point_B ;
r_info - > collider_id = col_obj - > get_instance_id ( ) ;
r_info - > rid = col_obj - > get_self ( ) ;
r_info - > shape = shape_idx ;
r_info - > point = closest_B ;
r_info - > normal = ( closest_A - closest_B ) . normalized ( ) ;
best_first = false ;
2020-03-27 18:21:27 +00:00
if ( col_obj - > get_type ( ) = = CollisionObject3DSW : : TYPE_BODY ) {
const Body3DSW * body = static_cast < const Body3DSW * > ( col_obj ) ;
2017-03-05 15:44:50 +00:00
r_info - > linear_velocity = body - > get_linear_velocity ( ) + ( body - > get_angular_velocity ( ) ) . cross ( body - > get_transform ( ) . origin - closest_B ) ;
2015-10-08 18:00:40 +00:00
}
}
}
2017-03-05 15:44:50 +00:00
p_closest_safe = best_safe ;
p_closest_unsafe = best_unsafe ;
2015-10-08 18:00:40 +00:00
return true ;
}
2020-03-27 18:21:27 +00:00
bool PhysicsDirectSpaceState3DSW : : collide_shape ( RID p_shape , const Transform & p_shape_xform , real_t p_margin , Vector3 * r_results , int p_result_max , int & r_result_count , const Set < RID > & p_exclude , uint32_t p_collision_mask , bool p_collide_with_bodies , bool p_collide_with_areas ) {
2017-03-05 15:44:50 +00:00
if ( p_result_max < = 0 )
2020-05-14 09:00:19 +00:00
return false ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
Shape3DSW * shape = static_cast < PhysicsServer3DSW * > ( PhysicsServer3D : : get_singleton ( ) ) - > shape_owner . getornull ( p_shape ) ;
2017-03-05 15:44:50 +00:00
ERR_FAIL_COND_V ( ! shape , 0 ) ;
2015-10-08 18:00:40 +00:00
2017-11-17 02:09:00 +00:00
AABB aabb = p_shape_xform . xform ( shape - > get_aabb ( ) ) ;
2017-03-05 15:44:50 +00:00
aabb = aabb . grow ( p_margin ) ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
int amount = space - > broadphase - > cull_aabb ( aabb , space - > intersection_query_results , Space3DSW : : INTERSECTION_QUERY_MAX , space - > intersection_query_subindex_results ) ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
bool collided = false ;
r_result_count = 0 ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
PhysicsServer3DSW : : CollCbkData cbk ;
2017-03-05 15:44:50 +00:00
cbk . max = p_result_max ;
cbk . amount = 0 ;
cbk . ptr = r_results ;
2020-03-27 18:21:27 +00:00
CollisionSolver3DSW : : CallbackResult cbkres = PhysicsServer3DSW : : _shape_col_cbk ;
2017-03-05 15:44:50 +00:00
2020-03-27 18:21:27 +00:00
PhysicsServer3DSW : : CollCbkData * cbkptr = & cbk ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
for ( int i = 0 ; i < amount ; i + + ) {
2018-08-21 18:30:41 +00:00
if ( ! _can_collide_with ( space - > intersection_query_results [ i ] , p_collision_mask , p_collide_with_bodies , p_collide_with_areas ) )
2015-10-08 18:00:40 +00:00
continue ;
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = space - > intersection_query_results [ i ] ;
2017-03-05 15:44:50 +00:00
int shape_idx = space - > intersection_query_subindex_results [ i ] ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
if ( p_exclude . has ( col_obj - > get_self ( ) ) ) {
2015-10-08 18:00:40 +00:00
continue ;
}
2020-04-01 23:20:12 +00:00
if ( CollisionSolver3DSW : : solve_static ( shape , p_shape_xform , col_obj - > get_shape ( shape_idx ) , col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) , cbkres , cbkptr , nullptr , p_margin ) ) {
2017-03-05 15:44:50 +00:00
collided = true ;
2015-10-08 18:00:40 +00:00
}
}
2017-03-05 15:44:50 +00:00
r_result_count = cbk . amount ;
2015-10-08 18:00:40 +00:00
return collided ;
}
struct _RestCallbackData {
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * object ;
const CollisionObject3DSW * best_object ;
2015-10-08 18:00:40 +00:00
int shape ;
int best_shape ;
Vector3 best_contact ;
Vector3 best_normal ;
2017-02-13 23:25:05 +00:00
real_t best_len ;
2019-02-16 16:45:01 +00:00
real_t min_allowed_depth ;
2015-10-08 18:00:40 +00:00
} ;
2017-03-05 15:44:50 +00:00
static void _rest_cbk_result ( const Vector3 & p_point_A , const Vector3 & p_point_B , void * p_userdata ) {
_RestCallbackData * rd = ( _RestCallbackData * ) p_userdata ;
2015-10-08 18:00:40 +00:00
Vector3 contact_rel = p_point_B - p_point_A ;
2017-02-13 23:25:05 +00:00
real_t len = contact_rel . length ( ) ;
2019-02-16 16:45:01 +00:00
if ( len < rd - > min_allowed_depth )
return ;
2015-10-08 18:00:40 +00:00
if ( len < = rd - > best_len )
return ;
2017-03-05 15:44:50 +00:00
rd - > best_len = len ;
rd - > best_contact = p_point_B ;
rd - > best_normal = contact_rel / len ;
rd - > best_object = rd - > object ;
rd - > best_shape = rd - > shape ;
2015-10-08 18:00:40 +00:00
}
2020-03-27 18:21:27 +00:00
bool PhysicsDirectSpaceState3DSW : : rest_info ( RID p_shape , const Transform & p_shape_xform , real_t p_margin , ShapeRestInfo * r_info , const Set < RID > & p_exclude , uint32_t p_collision_mask , bool p_collide_with_bodies , bool p_collide_with_areas ) {
Shape3DSW * shape = static_cast < PhysicsServer3DSW * > ( PhysicsServer3D : : get_singleton ( ) ) - > shape_owner . getornull ( p_shape ) ;
2017-03-05 15:44:50 +00:00
ERR_FAIL_COND_V ( ! shape , 0 ) ;
2015-10-08 18:00:40 +00:00
2017-11-17 02:09:00 +00:00
AABB aabb = p_shape_xform . xform ( shape - > get_aabb ( ) ) ;
2017-03-05 15:44:50 +00:00
aabb = aabb . grow ( p_margin ) ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
int amount = space - > broadphase - > cull_aabb ( aabb , space - > intersection_query_results , Space3DSW : : INTERSECTION_QUERY_MAX , space - > intersection_query_subindex_results ) ;
2015-10-08 18:00:40 +00:00
_RestCallbackData rcd ;
2017-03-05 15:44:50 +00:00
rcd . best_len = 0 ;
2020-04-01 23:20:12 +00:00
rcd . best_object = nullptr ;
2017-03-05 15:44:50 +00:00
rcd . best_shape = 0 ;
2019-02-16 16:45:01 +00:00
rcd . min_allowed_depth = space - > test_motion_min_contact_depth ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
for ( int i = 0 ; i < amount ; i + + ) {
2018-08-21 18:30:41 +00:00
if ( ! _can_collide_with ( space - > intersection_query_results [ i ] , p_collision_mask , p_collide_with_bodies , p_collide_with_areas ) )
2015-10-08 18:00:40 +00:00
continue ;
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = space - > intersection_query_results [ i ] ;
2017-03-05 15:44:50 +00:00
int shape_idx = space - > intersection_query_subindex_results [ i ] ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
if ( p_exclude . has ( col_obj - > get_self ( ) ) )
2015-10-08 18:00:40 +00:00
continue ;
2017-03-05 15:44:50 +00:00
rcd . object = col_obj ;
rcd . shape = shape_idx ;
2020-04-01 23:20:12 +00:00
bool sc = CollisionSolver3DSW : : solve_static ( shape , p_shape_xform , col_obj - > get_shape ( shape_idx ) , col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) , _rest_cbk_result , & rcd , nullptr , p_margin ) ;
2015-10-08 18:00:40 +00:00
if ( ! sc )
continue ;
}
2019-06-26 13:08:25 +00:00
if ( rcd . best_len = = 0 | | ! rcd . best_object )
2015-10-08 18:00:40 +00:00
return false ;
2017-03-05 15:44:50 +00:00
r_info - > collider_id = rcd . best_object - > get_instance_id ( ) ;
r_info - > shape = rcd . best_shape ;
r_info - > normal = rcd . best_normal ;
r_info - > point = rcd . best_contact ;
r_info - > rid = rcd . best_object - > get_self ( ) ;
2020-03-27 18:21:27 +00:00
if ( rcd . best_object - > get_type ( ) = = CollisionObject3DSW : : TYPE_BODY ) {
const Body3DSW * body = static_cast < const Body3DSW * > ( rcd . best_object ) ;
2015-10-08 18:00:40 +00:00
r_info - > linear_velocity = body - > get_linear_velocity ( ) +
2017-03-05 15:44:50 +00:00
( body - > get_angular_velocity ( ) ) . cross ( body - > get_transform ( ) . origin - rcd . best_contact ) ; // * mPos);
2015-10-08 18:00:40 +00:00
} else {
2017-03-05 15:44:50 +00:00
r_info - > linear_velocity = Vector3 ( ) ;
2015-10-08 18:00:40 +00:00
}
return true ;
}
2020-03-27 18:21:27 +00:00
Vector3 PhysicsDirectSpaceState3DSW : : get_closest_point_to_object_volume ( RID p_object , const Vector3 p_point ) const {
CollisionObject3DSW * obj = PhysicsServer3DSW : : singleton - > area_owner . getornull ( p_object ) ;
2017-07-15 04:23:10 +00:00
if ( ! obj ) {
2020-03-27 18:21:27 +00:00
obj = PhysicsServer3DSW : : singleton - > body_owner . getornull ( p_object ) ;
2017-07-15 04:23:10 +00:00
}
ERR_FAIL_COND_V ( ! obj , Vector3 ( ) ) ;
ERR_FAIL_COND_V ( obj - > get_space ( ) ! = space , Vector3 ( ) ) ;
float min_distance = 1e20 ;
Vector3 min_point ;
bool shapes_found = false ;
for ( int i = 0 ; i < obj - > get_shape_count ( ) ; i + + ) {
if ( obj - > is_shape_set_as_disabled ( i ) )
continue ;
Transform shape_xform = obj - > get_transform ( ) * obj - > get_shape_transform ( i ) ;
2020-03-27 18:21:27 +00:00
Shape3DSW * shape = obj - > get_shape ( i ) ;
2017-07-15 04:23:10 +00:00
Vector3 point = shape - > get_closest_point_to ( shape_xform . affine_inverse ( ) . xform ( p_point ) ) ;
point = shape_xform . xform ( point ) ;
float dist = point . distance_to ( p_point ) ;
if ( dist < min_distance ) {
min_distance = dist ;
min_point = point ;
}
shapes_found = true ;
}
if ( ! shapes_found ) {
return obj - > get_transform ( ) . origin ; //no shapes found, use distance to origin.
} else {
return min_point ;
}
}
2020-03-27 18:21:27 +00:00
PhysicsDirectSpaceState3DSW : : PhysicsDirectSpaceState3DSW ( ) {
2020-04-01 23:20:12 +00:00
space = nullptr ;
2015-10-08 18:00:40 +00:00
}
////////////////////////////////////////////////////////////////////////////////////////////////////////////
2020-03-27 18:21:27 +00:00
int Space3DSW : : _cull_aabb_for_body ( Body3DSW * p_body , const AABB & p_aabb ) {
2017-07-15 04:23:10 +00:00
int amount = broadphase - > cull_aabb ( p_aabb , intersection_query_results , INTERSECTION_QUERY_MAX , intersection_query_subindex_results ) ;
for ( int i = 0 ; i < amount ; i + + ) {
bool keep = true ;
if ( intersection_query_results [ i ] = = p_body )
keep = false ;
2020-03-27 18:21:27 +00:00
else if ( intersection_query_results [ i ] - > get_type ( ) = = CollisionObject3DSW : : TYPE_AREA )
2017-07-15 04:23:10 +00:00
keep = false ;
2020-03-27 18:21:27 +00:00
else if ( ( static_cast < Body3DSW * > ( intersection_query_results [ i ] ) - > test_collision_mask ( p_body ) ) = = 0 )
2017-07-15 04:23:10 +00:00
keep = false ;
2020-03-27 18:21:27 +00:00
else if ( static_cast < Body3DSW * > ( intersection_query_results [ i ] ) - > has_exception ( p_body - > get_self ( ) ) | | p_body - > has_exception ( intersection_query_results [ i ] - > get_self ( ) ) )
2017-07-15 04:23:10 +00:00
keep = false ;
2020-03-27 18:21:27 +00:00
else if ( static_cast < Body3DSW * > ( intersection_query_results [ i ] ) - > is_shape_set_as_disabled ( intersection_query_subindex_results [ i ] ) )
2017-07-15 04:23:10 +00:00
keep = false ;
if ( ! keep ) {
if ( i < amount - 1 ) {
SWAP ( intersection_query_results [ i ] , intersection_query_results [ amount - 1 ] ) ;
SWAP ( intersection_query_subindex_results [ i ] , intersection_query_subindex_results [ amount - 1 ] ) ;
}
amount - - ;
i - - ;
}
}
return amount ;
}
2020-03-27 18:21:27 +00:00
int Space3DSW : : test_body_ray_separation ( Body3DSW * p_body , const Transform & p_transform , bool p_infinite_inertia , Vector3 & r_recover_motion , PhysicsServer3D : : SeparationResult * r_results , int p_result_max , real_t p_margin ) {
2018-08-20 20:31:55 +00:00
AABB body_aabb ;
2018-11-02 18:44:29 +00:00
bool shapes_found = false ;
2018-08-20 20:31:55 +00:00
for ( int i = 0 ; i < p_body - > get_shape_count ( ) ; i + + ) {
2018-11-02 18:44:29 +00:00
if ( p_body - > is_shape_set_as_disabled ( i ) )
continue ;
if ( ! shapes_found ) {
2018-08-20 20:31:55 +00:00
body_aabb = p_body - > get_shape_aabb ( i ) ;
2018-11-02 18:44:29 +00:00
shapes_found = true ;
} else {
2018-08-20 20:31:55 +00:00
body_aabb = body_aabb . merge ( p_body - > get_shape_aabb ( i ) ) ;
2018-11-02 18:44:29 +00:00
}
2018-08-20 20:31:55 +00:00
}
2018-11-02 18:44:29 +00:00
if ( ! shapes_found ) {
return 0 ;
}
2018-08-20 20:31:55 +00:00
// Undo the currently transform the physics server is aware of and apply the provided one
body_aabb = p_transform . xform ( p_body - > get_inv_transform ( ) . xform ( body_aabb ) ) ;
body_aabb = body_aabb . grow ( p_margin ) ;
Transform body_transform = p_transform ;
for ( int i = 0 ; i < p_result_max ; i + + ) {
//reset results
r_results [ i ] . collision_depth = 0 ;
}
int rays_found = 0 ;
{
// raycast AND separate
const int max_results = 32 ;
int recover_attempts = 4 ;
Vector3 sr [ max_results * 2 ] ;
2020-03-27 18:21:27 +00:00
PhysicsServer3DSW : : CollCbkData cbk ;
2018-08-20 20:31:55 +00:00
cbk . max = max_results ;
2020-03-27 18:21:27 +00:00
PhysicsServer3DSW : : CollCbkData * cbkptr = & cbk ;
CollisionSolver3DSW : : CallbackResult cbkres = PhysicsServer3DSW : : _shape_col_cbk ;
2018-08-20 20:31:55 +00:00
do {
Vector3 recover_motion ;
bool collided = false ;
int amount = _cull_aabb_for_body ( p_body , body_aabb ) ;
for ( int j = 0 ; j < p_body - > get_shape_count ( ) ; j + + ) {
if ( p_body - > is_shape_set_as_disabled ( j ) )
continue ;
2020-03-27 18:21:27 +00:00
Shape3DSW * body_shape = p_body - > get_shape ( j ) ;
2018-08-20 20:31:55 +00:00
2020-03-27 18:21:27 +00:00
if ( body_shape - > get_type ( ) ! = PhysicsServer3D : : SHAPE_RAY )
2018-08-20 20:31:55 +00:00
continue ;
Transform body_shape_xform = body_transform * p_body - > get_shape_transform ( j ) ;
for ( int i = 0 ; i < amount ; i + + ) {
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = intersection_query_results [ i ] ;
2018-08-20 20:31:55 +00:00
int shape_idx = intersection_query_subindex_results [ i ] ;
cbk . amount = 0 ;
cbk . ptr = sr ;
2020-03-27 18:21:27 +00:00
if ( CollisionObject3DSW : : TYPE_BODY = = col_obj - > get_type ( ) ) {
const Body3DSW * b = static_cast < const Body3DSW * > ( col_obj ) ;
if ( p_infinite_inertia & & PhysicsServer3D : : BODY_MODE_STATIC ! = b - > get_mode ( ) & & PhysicsServer3D : : BODY_MODE_KINEMATIC ! = b - > get_mode ( ) ) {
2018-08-20 20:31:55 +00:00
continue ;
}
}
2020-03-27 18:21:27 +00:00
Shape3DSW * against_shape = col_obj - > get_shape ( shape_idx ) ;
2020-04-01 23:20:12 +00:00
if ( CollisionSolver3DSW : : solve_static ( body_shape , body_shape_xform , against_shape , col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) , cbkres , cbkptr , nullptr , p_margin ) ) {
2018-08-20 20:31:55 +00:00
if ( cbk . amount > 0 ) {
collided = true ;
}
2019-02-16 19:14:57 +00:00
int ray_index = - 1 ; //reuse shape
for ( int k = 0 ; k < rays_found ; k + + ) {
2019-03-25 21:46:26 +00:00
if ( r_results [ k ] . collision_local_shape = = j ) {
2019-02-16 19:14:57 +00:00
ray_index = k ;
}
}
if ( ray_index = = - 1 & & rays_found < p_result_max ) {
ray_index = rays_found ;
rays_found + + ;
}
if ( ray_index ! = - 1 ) {
2020-03-27 18:21:27 +00:00
PhysicsServer3D : : SeparationResult & result = r_results [ ray_index ] ;
2018-08-20 20:31:55 +00:00
for ( int k = 0 ; k < cbk . amount ; k + + ) {
Vector3 a = sr [ k * 2 + 0 ] ;
Vector3 b = sr [ k * 2 + 1 ] ;
recover_motion + = ( b - a ) * 0.4 ;
float depth = a . distance_to ( b ) ;
if ( depth > result . collision_depth ) {
result . collision_depth = depth ;
result . collision_point = b ;
result . collision_normal = ( b - a ) . normalized ( ) ;
2019-02-16 19:14:57 +00:00
result . collision_local_shape = j ;
2018-08-20 20:31:55 +00:00
result . collider = col_obj - > get_self ( ) ;
result . collider_id = col_obj - > get_instance_id ( ) ;
2019-02-16 19:14:57 +00:00
result . collider_shape = shape_idx ;
2018-08-20 20:31:55 +00:00
//result.collider_metadata = col_obj->get_shape_metadata(shape_idx);
2020-03-27 18:21:27 +00:00
if ( col_obj - > get_type ( ) = = CollisionObject3DSW : : TYPE_BODY ) {
Body3DSW * body = ( Body3DSW * ) col_obj ;
2018-08-20 20:31:55 +00:00
Vector3 rel_vec = b - body - > get_transform ( ) . get_origin ( ) ;
//result.collider_velocity = Vector3(-body->get_angular_velocity() * rel_vec.y, body->get_angular_velocity() * rel_vec.x) + body->get_linear_velocity();
result . collider_velocity = body - > get_linear_velocity ( ) + ( body - > get_angular_velocity ( ) ) . cross ( body - > get_transform ( ) . origin - rel_vec ) ; // * mPos);
}
}
}
}
}
}
}
if ( ! collided | | recover_motion = = Vector3 ( ) ) {
break ;
}
body_transform . origin + = recover_motion ;
body_aabb . position + = recover_motion ;
recover_attempts - - ;
} while ( recover_attempts ) ;
}
//optimize results (remove non colliding)
for ( int i = 0 ; i < rays_found ; i + + ) {
if ( r_results [ i ] . collision_depth = = 0 ) {
rays_found - - ;
SWAP ( r_results [ i ] , r_results [ rays_found ] ) ;
}
}
r_recover_motion = body_transform . origin - p_transform . origin ;
return rays_found ;
}
2020-03-27 18:21:27 +00:00
bool Space3DSW : : test_body_motion ( Body3DSW * p_body , const Transform & p_from , const Vector3 & p_motion , bool p_infinite_inertia , real_t p_margin , PhysicsServer3D : : MotionResult * r_result , bool p_exclude_raycast_shapes ) {
2017-07-15 04:23:10 +00:00
//give me back regular physics engine logic
//this is madness
//and most people using this function will think
//what it does is simpler than using physics
//this took about a week to get right..
//but is it right? who knows at this point..
if ( r_result ) {
2020-02-12 17:24:06 +00:00
r_result - > collider_id = ObjectID ( ) ;
2017-07-15 04:23:10 +00:00
r_result - > collider_shape = 0 ;
}
2017-11-17 02:09:00 +00:00
AABB body_aabb ;
2018-11-02 18:44:29 +00:00
bool shapes_found = false ;
2017-07-15 04:23:10 +00:00
for ( int i = 0 ; i < p_body - > get_shape_count ( ) ; i + + ) {
2018-11-02 18:44:29 +00:00
if ( p_body - > is_shape_set_as_disabled ( i ) )
continue ;
if ( ! shapes_found ) {
2017-07-15 04:23:10 +00:00
body_aabb = p_body - > get_shape_aabb ( i ) ;
2018-11-02 18:44:29 +00:00
shapes_found = true ;
} else {
2017-07-15 04:23:10 +00:00
body_aabb = body_aabb . merge ( p_body - > get_shape_aabb ( i ) ) ;
2018-11-02 18:44:29 +00:00
}
}
if ( ! shapes_found ) {
2019-02-16 14:06:17 +00:00
if ( r_result ) {
2020-03-27 18:21:27 +00:00
* r_result = PhysicsServer3D : : MotionResult ( ) ;
2019-02-16 14:06:17 +00:00
r_result - > motion = p_motion ;
}
2018-11-02 18:44:29 +00:00
return false ;
2017-07-15 04:23:10 +00:00
}
// Undo the currently transform the physics server is aware of and apply the provided one
body_aabb = p_from . xform ( p_body - > get_inv_transform ( ) . xform ( body_aabb ) ) ;
body_aabb = body_aabb . grow ( p_margin ) ;
Transform body_transform = p_from ;
{
//STEP 1, FREE BODY IF STUCK
const int max_results = 32 ;
int recover_attempts = 4 ;
Vector3 sr [ max_results * 2 ] ;
do {
2020-03-27 18:21:27 +00:00
PhysicsServer3DSW : : CollCbkData cbk ;
2017-07-15 04:23:10 +00:00
cbk . max = max_results ;
cbk . amount = 0 ;
cbk . ptr = sr ;
2020-03-27 18:21:27 +00:00
PhysicsServer3DSW : : CollCbkData * cbkptr = & cbk ;
CollisionSolver3DSW : : CallbackResult cbkres = PhysicsServer3DSW : : _shape_col_cbk ;
2017-07-15 04:23:10 +00:00
bool collided = false ;
int amount = _cull_aabb_for_body ( p_body , body_aabb ) ;
for ( int j = 0 ; j < p_body - > get_shape_count ( ) ; j + + ) {
if ( p_body - > is_shape_set_as_disabled ( j ) )
continue ;
Transform body_shape_xform = body_transform * p_body - > get_shape_transform ( j ) ;
2020-03-27 18:21:27 +00:00
Shape3DSW * body_shape = p_body - > get_shape ( j ) ;
if ( p_exclude_raycast_shapes & & body_shape - > get_type ( ) = = PhysicsServer3D : : SHAPE_RAY ) {
2018-08-20 20:31:55 +00:00
continue ;
}
2017-07-15 04:23:10 +00:00
for ( int i = 0 ; i < amount ; i + + ) {
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = intersection_query_results [ i ] ;
2017-07-15 04:23:10 +00:00
int shape_idx = intersection_query_subindex_results [ i ] ;
2020-04-01 23:20:12 +00:00
if ( CollisionSolver3DSW : : solve_static ( body_shape , body_shape_xform , col_obj - > get_shape ( shape_idx ) , col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) , cbkres , cbkptr , nullptr , p_margin ) ) {
2017-07-15 04:23:10 +00:00
collided = cbk . amount > 0 ;
}
}
}
if ( ! collided ) {
break ;
}
Vector3 recover_motion ;
for ( int i = 0 ; i < cbk . amount ; i + + ) {
Vector3 a = sr [ i * 2 + 0 ] ;
Vector3 b = sr [ i * 2 + 1 ] ;
recover_motion + = ( b - a ) * 0.4 ;
}
if ( recover_motion = = Vector3 ( ) ) {
collided = false ;
break ;
}
body_transform . origin + = recover_motion ;
body_aabb . position + = recover_motion ;
recover_attempts - - ;
} while ( recover_attempts ) ;
}
real_t safe = 1.0 ;
real_t unsafe = 1.0 ;
int best_shape = - 1 ;
{
// STEP 2 ATTEMPT MOTION
2017-11-17 02:09:00 +00:00
AABB motion_aabb = body_aabb ;
2017-07-15 04:23:10 +00:00
motion_aabb . position + = p_motion ;
motion_aabb = motion_aabb . merge ( body_aabb ) ;
int amount = _cull_aabb_for_body ( p_body , motion_aabb ) ;
for ( int j = 0 ; j < p_body - > get_shape_count ( ) ; j + + ) {
if ( p_body - > is_shape_set_as_disabled ( j ) )
continue ;
Transform body_shape_xform = body_transform * p_body - > get_shape_transform ( j ) ;
2020-03-27 18:21:27 +00:00
Shape3DSW * body_shape = p_body - > get_shape ( j ) ;
2017-07-15 04:23:10 +00:00
2020-03-27 18:21:27 +00:00
if ( p_exclude_raycast_shapes & & body_shape - > get_type ( ) = = PhysicsServer3D : : SHAPE_RAY ) {
2018-08-20 20:31:55 +00:00
continue ;
}
2017-07-15 04:23:10 +00:00
Transform body_shape_xform_inv = body_shape_xform . affine_inverse ( ) ;
2020-03-27 18:21:27 +00:00
MotionShape3DSW mshape ;
2017-07-15 04:23:10 +00:00
mshape . shape = body_shape ;
mshape . motion = body_shape_xform_inv . basis . xform ( p_motion ) ;
bool stuck = false ;
real_t best_safe = 1 ;
real_t best_unsafe = 1 ;
for ( int i = 0 ; i < amount ; i + + ) {
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = intersection_query_results [ i ] ;
2017-07-15 04:23:10 +00:00
int shape_idx = intersection_query_subindex_results [ i ] ;
//test initial overlap, does it collide if going all the way?
Vector3 point_A , point_B ;
Vector3 sep_axis = p_motion . normalized ( ) ;
Transform col_obj_xform = col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) ;
//test initial overlap, does it collide if going all the way?
2020-03-27 18:21:27 +00:00
if ( CollisionSolver3DSW : : solve_distance ( & mshape , body_shape_xform , col_obj - > get_shape ( shape_idx ) , col_obj_xform , point_A , point_B , motion_aabb , & sep_axis ) ) {
2017-07-15 04:23:10 +00:00
continue ;
}
sep_axis = p_motion . normalized ( ) ;
2020-03-27 18:21:27 +00:00
if ( ! CollisionSolver3DSW : : solve_distance ( body_shape , body_shape_xform , col_obj - > get_shape ( shape_idx ) , col_obj_xform , point_A , point_B , motion_aabb , & sep_axis ) ) {
2017-07-15 04:23:10 +00:00
stuck = true ;
break ;
}
//just do kinematic solving
real_t low = 0 ;
real_t hi = 1 ;
Vector3 mnormal = p_motion . normalized ( ) ;
2019-02-12 20:10:08 +00:00
for ( int k = 0 ; k < 8 ; k + + ) { //steps should be customizable..
2017-07-15 04:23:10 +00:00
real_t ofs = ( low + hi ) * 0.5 ;
Vector3 sep = mnormal ; //important optimization for this to work fast enough
mshape . motion = body_shape_xform_inv . basis . xform ( p_motion * ofs ) ;
Vector3 lA , lB ;
2020-03-27 18:21:27 +00:00
bool collided = ! CollisionSolver3DSW : : solve_distance ( & mshape , body_shape_xform , col_obj - > get_shape ( shape_idx ) , col_obj_xform , lA , lB , motion_aabb , & sep ) ;
2017-07-15 04:23:10 +00:00
if ( collided ) {
hi = ofs ;
} else {
point_A = lA ;
point_B = lB ;
low = ofs ;
}
}
if ( low < best_safe ) {
best_safe = low ;
best_unsafe = hi ;
}
}
if ( stuck ) {
safe = 0 ;
unsafe = 0 ;
best_shape = j ; //sadly it's the best
break ;
}
if ( best_safe = = 1.0 ) {
continue ;
}
if ( best_safe < safe ) {
safe = best_safe ;
unsafe = best_unsafe ;
best_shape = j ;
}
}
}
bool collided = false ;
if ( safe > = 1 ) {
//not collided
collided = false ;
if ( r_result ) {
r_result - > motion = p_motion ;
r_result - > remainder = Vector3 ( ) ;
r_result - > motion + = ( body_transform . get_origin ( ) - p_from . get_origin ( ) ) ;
}
} else {
//it collided, let's get the rest info in unsafe advance
Transform ugt = body_transform ;
ugt . origin + = p_motion * unsafe ;
_RestCallbackData rcd ;
rcd . best_len = 0 ;
2020-04-01 23:20:12 +00:00
rcd . best_object = nullptr ;
2017-07-15 04:23:10 +00:00
rcd . best_shape = 0 ;
2019-02-16 16:45:01 +00:00
rcd . min_allowed_depth = test_motion_min_contact_depth ;
2017-07-15 04:23:10 +00:00
Transform body_shape_xform = ugt * p_body - > get_shape_transform ( best_shape ) ;
2020-03-27 18:21:27 +00:00
Shape3DSW * body_shape = p_body - > get_shape ( best_shape ) ;
2017-07-15 04:23:10 +00:00
body_aabb . position + = p_motion * unsafe ;
int amount = _cull_aabb_for_body ( p_body , body_aabb ) ;
for ( int i = 0 ; i < amount ; i + + ) {
2020-03-27 18:21:27 +00:00
const CollisionObject3DSW * col_obj = intersection_query_results [ i ] ;
2017-07-15 04:23:10 +00:00
int shape_idx = intersection_query_subindex_results [ i ] ;
rcd . object = col_obj ;
rcd . shape = shape_idx ;
2020-04-01 23:20:12 +00:00
bool sc = CollisionSolver3DSW : : solve_static ( body_shape , body_shape_xform , col_obj - > get_shape ( shape_idx ) , col_obj - > get_transform ( ) * col_obj - > get_shape_transform ( shape_idx ) , _rest_cbk_result , & rcd , nullptr , p_margin ) ;
2017-07-15 04:23:10 +00:00
if ( ! sc )
continue ;
}
if ( rcd . best_len ! = 0 ) {
if ( r_result ) {
r_result - > collider = rcd . best_object - > get_self ( ) ;
r_result - > collider_id = rcd . best_object - > get_instance_id ( ) ;
r_result - > collider_shape = rcd . best_shape ;
r_result - > collision_local_shape = best_shape ;
r_result - > collision_normal = rcd . best_normal ;
r_result - > collision_point = rcd . best_contact ;
//r_result->collider_metadata = rcd.best_object->get_shape_metadata(rcd.best_shape);
2020-03-27 18:21:27 +00:00
const Body3DSW * body = static_cast < const Body3DSW * > ( rcd . best_object ) ;
2017-07-15 04:23:10 +00:00
//Vector3 rel_vec = r_result->collision_point - body->get_transform().get_origin();
// r_result->collider_velocity = Vector3(-body->get_angular_velocity() * rel_vec.y, body->get_angular_velocity() * rel_vec.x) + body->get_linear_velocity();
r_result - > collider_velocity = body - > get_linear_velocity ( ) + ( body - > get_angular_velocity ( ) ) . cross ( body - > get_transform ( ) . origin - rcd . best_contact ) ; // * mPos);
r_result - > motion = safe * p_motion ;
r_result - > remainder = p_motion - safe * p_motion ;
r_result - > motion + = ( body_transform . get_origin ( ) - p_from . get_origin ( ) ) ;
}
collided = true ;
} else {
if ( r_result ) {
r_result - > motion = p_motion ;
r_result - > remainder = Vector3 ( ) ;
r_result - > motion + = ( body_transform . get_origin ( ) - p_from . get_origin ( ) ) ;
}
collided = false ;
}
}
return collided ;
}
2020-03-27 18:21:27 +00:00
void * Space3DSW : : _broadphase_pair ( CollisionObject3DSW * A , int p_subindex_A , CollisionObject3DSW * B , int p_subindex_B , void * p_self ) {
CollisionObject3DSW : : Type type_A = A - > get_type ( ) ;
CollisionObject3DSW : : Type type_B = B - > get_type ( ) ;
2017-03-05 15:44:50 +00:00
if ( type_A > type_B ) {
SWAP ( A , B ) ;
SWAP ( p_subindex_A , p_subindex_B ) ;
SWAP ( type_A , type_B ) ;
2015-10-08 18:00:40 +00:00
}
2020-03-27 18:21:27 +00:00
Space3DSW * self = ( Space3DSW * ) p_self ;
2015-10-08 18:00:40 +00:00
self - > collision_pairs + + ;
2020-03-27 18:21:27 +00:00
if ( type_A = = CollisionObject3DSW : : TYPE_AREA ) {
Area3DSW * area = static_cast < Area3DSW * > ( A ) ;
if ( type_B = = CollisionObject3DSW : : TYPE_AREA ) {
Area3DSW * area_b = static_cast < Area3DSW * > ( B ) ;
Area2Pair3DSW * area2_pair = memnew ( Area2Pair3DSW ( area_b , p_subindex_B , area , p_subindex_A ) ) ;
2015-10-08 18:00:40 +00:00
return area2_pair ;
} else {
2020-03-27 18:21:27 +00:00
Body3DSW * body = static_cast < Body3DSW * > ( B ) ;
AreaPair3DSW * area_pair = memnew ( AreaPair3DSW ( body , p_subindex_B , area , p_subindex_A ) ) ;
2015-10-08 18:00:40 +00:00
return area_pair ;
}
} else {
2020-03-27 18:21:27 +00:00
BodyPair3DSW * b = memnew ( BodyPair3DSW ( ( Body3DSW * ) A , p_subindex_A , ( Body3DSW * ) B , p_subindex_B ) ) ;
2015-10-08 18:00:40 +00:00
return b ;
}
2020-04-01 23:20:12 +00:00
return nullptr ;
2015-10-08 18:00:40 +00:00
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : _broadphase_unpair ( CollisionObject3DSW * A , int p_subindex_A , CollisionObject3DSW * B , int p_subindex_B , void * p_data , void * p_self ) {
Space3DSW * self = ( Space3DSW * ) p_self ;
2015-10-08 18:00:40 +00:00
self - > collision_pairs - - ;
2020-03-27 18:21:27 +00:00
Constraint3DSW * c = ( Constraint3DSW * ) p_data ;
2015-10-08 18:00:40 +00:00
memdelete ( c ) ;
}
2020-03-27 18:21:27 +00:00
const SelfList < Body3DSW > : : List & Space3DSW : : get_active_body_list ( ) const {
2015-10-08 18:00:40 +00:00
return active_list ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : body_add_to_active_list ( SelfList < Body3DSW > * p_body ) {
2015-10-08 18:00:40 +00:00
active_list . add ( p_body ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : body_remove_from_active_list ( SelfList < Body3DSW > * p_body ) {
2015-10-08 18:00:40 +00:00
active_list . remove ( p_body ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : body_add_to_inertia_update_list ( SelfList < Body3DSW > * p_body ) {
2015-10-08 18:00:40 +00:00
inertia_update_list . add ( p_body ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : body_remove_from_inertia_update_list ( SelfList < Body3DSW > * p_body ) {
2015-10-08 18:00:40 +00:00
inertia_update_list . remove ( p_body ) ;
}
2020-03-27 18:21:27 +00:00
BroadPhase3DSW * Space3DSW : : get_broadphase ( ) {
2015-10-08 18:00:40 +00:00
return broadphase ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : add_object ( CollisionObject3DSW * p_object ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_COND ( objects . has ( p_object ) ) ;
2015-10-08 18:00:40 +00:00
objects . insert ( p_object ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : remove_object ( CollisionObject3DSW * p_object ) {
2017-03-05 15:44:50 +00:00
ERR_FAIL_COND ( ! objects . has ( p_object ) ) ;
2015-10-08 18:00:40 +00:00
objects . erase ( p_object ) ;
}
2020-03-27 18:21:27 +00:00
const Set < CollisionObject3DSW * > & Space3DSW : : get_objects ( ) const {
2015-10-08 18:00:40 +00:00
return objects ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : body_add_to_state_query_list ( SelfList < Body3DSW > * p_body ) {
2015-10-08 18:00:40 +00:00
state_query_list . add ( p_body ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : body_remove_from_state_query_list ( SelfList < Body3DSW > * p_body ) {
2015-10-08 18:00:40 +00:00
state_query_list . remove ( p_body ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : area_add_to_monitor_query_list ( SelfList < Area3DSW > * p_area ) {
2015-10-08 18:00:40 +00:00
monitor_query_list . add ( p_area ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : area_remove_from_monitor_query_list ( SelfList < Area3DSW > * p_area ) {
2015-10-08 18:00:40 +00:00
monitor_query_list . remove ( p_area ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : area_add_to_moved_list ( SelfList < Area3DSW > * p_area ) {
2015-10-08 18:00:40 +00:00
area_moved_list . add ( p_area ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : area_remove_from_moved_list ( SelfList < Area3DSW > * p_area ) {
2015-10-08 18:00:40 +00:00
area_moved_list . remove ( p_area ) ;
}
2020-03-27 18:21:27 +00:00
const SelfList < Area3DSW > : : List & Space3DSW : : get_moved_area_list ( ) const {
2015-10-08 18:00:40 +00:00
return area_moved_list ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : call_queries ( ) {
2017-03-05 15:44:50 +00:00
while ( state_query_list . first ( ) ) {
2020-03-27 18:21:27 +00:00
Body3DSW * b = state_query_list . first ( ) - > self ( ) ;
2015-10-08 18:00:40 +00:00
state_query_list . remove ( state_query_list . first ( ) ) ;
2017-11-10 12:21:33 +00:00
b - > call_queries ( ) ;
2015-10-08 18:00:40 +00:00
}
2017-03-05 15:44:50 +00:00
while ( monitor_query_list . first ( ) ) {
2020-03-27 18:21:27 +00:00
Area3DSW * a = monitor_query_list . first ( ) - > self ( ) ;
2015-10-08 18:00:40 +00:00
monitor_query_list . remove ( monitor_query_list . first ( ) ) ;
2017-11-10 12:21:33 +00:00
a - > call_queries ( ) ;
2015-10-08 18:00:40 +00:00
}
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : setup ( ) {
2017-03-05 15:44:50 +00:00
contact_debug_count = 0 ;
while ( inertia_update_list . first ( ) ) {
2015-10-08 18:00:40 +00:00
inertia_update_list . first ( ) - > self ( ) - > update_inertias ( ) ;
inertia_update_list . remove ( inertia_update_list . first ( ) ) ;
}
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : update ( ) {
2015-10-08 18:00:40 +00:00
broadphase - > update ( ) ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : set_param ( PhysicsServer3D : : SpaceParameter p_param , real_t p_value ) {
2017-03-05 15:44:50 +00:00
switch ( p_param ) {
2020-05-10 11:00:47 +00:00
case PhysicsServer3D : : SPACE_PARAM_CONTACT_RECYCLE_RADIUS :
contact_recycle_radius = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_CONTACT_MAX_SEPARATION :
contact_max_separation = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_BODY_MAX_ALLOWED_PENETRATION :
contact_max_allowed_penetration = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_BODY_LINEAR_VELOCITY_SLEEP_THRESHOLD :
body_linear_velocity_sleep_threshold = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_BODY_ANGULAR_VELOCITY_SLEEP_THRESHOLD :
body_angular_velocity_sleep_threshold = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_BODY_TIME_TO_SLEEP :
body_time_to_sleep = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_BODY_ANGULAR_VELOCITY_DAMP_RATIO :
body_angular_velocity_damp_ratio = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_CONSTRAINT_DEFAULT_BIAS :
constraint_bias = p_value ;
break ;
case PhysicsServer3D : : SPACE_PARAM_TEST_MOTION_MIN_CONTACT_DEPTH :
test_motion_min_contact_depth = p_value ;
break ;
2015-10-08 18:00:40 +00:00
}
}
2020-03-27 18:21:27 +00:00
real_t Space3DSW : : get_param ( PhysicsServer3D : : SpaceParameter p_param ) const {
2017-03-05 15:44:50 +00:00
switch ( p_param ) {
2020-05-10 11:00:47 +00:00
case PhysicsServer3D : : SPACE_PARAM_CONTACT_RECYCLE_RADIUS :
return contact_recycle_radius ;
case PhysicsServer3D : : SPACE_PARAM_CONTACT_MAX_SEPARATION :
return contact_max_separation ;
case PhysicsServer3D : : SPACE_PARAM_BODY_MAX_ALLOWED_PENETRATION :
return contact_max_allowed_penetration ;
case PhysicsServer3D : : SPACE_PARAM_BODY_LINEAR_VELOCITY_SLEEP_THRESHOLD :
return body_linear_velocity_sleep_threshold ;
case PhysicsServer3D : : SPACE_PARAM_BODY_ANGULAR_VELOCITY_SLEEP_THRESHOLD :
return body_angular_velocity_sleep_threshold ;
case PhysicsServer3D : : SPACE_PARAM_BODY_TIME_TO_SLEEP :
return body_time_to_sleep ;
case PhysicsServer3D : : SPACE_PARAM_BODY_ANGULAR_VELOCITY_DAMP_RATIO :
return body_angular_velocity_damp_ratio ;
case PhysicsServer3D : : SPACE_PARAM_CONSTRAINT_DEFAULT_BIAS :
return constraint_bias ;
case PhysicsServer3D : : SPACE_PARAM_TEST_MOTION_MIN_CONTACT_DEPTH :
return test_motion_min_contact_depth ;
2015-10-08 18:00:40 +00:00
}
return 0 ;
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : lock ( ) {
2017-03-05 15:44:50 +00:00
locked = true ;
2015-10-08 18:00:40 +00:00
}
2020-03-27 18:21:27 +00:00
void Space3DSW : : unlock ( ) {
2017-03-05 15:44:50 +00:00
locked = false ;
2015-10-08 18:00:40 +00:00
}
2020-03-27 18:21:27 +00:00
bool Space3DSW : : is_locked ( ) const {
2015-10-08 18:00:40 +00:00
return locked ;
}
2020-03-27 18:21:27 +00:00
PhysicsDirectSpaceState3DSW * Space3DSW : : get_direct_state ( ) {
2015-10-08 18:00:40 +00:00
return direct_access ;
}
2020-03-27 18:21:27 +00:00
Space3DSW : : Space3DSW ( ) {
2017-03-05 15:44:50 +00:00
collision_pairs = 0 ;
active_objects = 0 ;
island_count = 0 ;
contact_debug_count = 0 ;
2015-10-08 18:00:40 +00:00
2017-03-05 15:44:50 +00:00
locked = false ;
contact_recycle_radius = 0.01 ;
contact_max_separation = 0.05 ;
contact_max_allowed_penetration = 0.01 ;
2019-02-16 16:45:01 +00:00
test_motion_min_contact_depth = 0.00001 ;
2015-10-08 18:00:40 +00:00
constraint_bias = 0.01 ;
2017-03-05 15:44:50 +00:00
body_linear_velocity_sleep_threshold = GLOBAL_DEF ( " physics/3d/sleep_threshold_linear " , 0.1 ) ;
body_angular_velocity_sleep_threshold = GLOBAL_DEF ( " physics/3d/sleep_threshold_angular " , ( 8.0 / 180.0 * Math_PI ) ) ;
body_time_to_sleep = GLOBAL_DEF ( " physics/3d/time_before_sleep " , 0.5 ) ;
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
ProjectSettings : : get_singleton ( ) - > set_custom_property_info ( " physics/3d/time_before_sleep " , PropertyInfo ( Variant : : FLOAT , " physics/3d/time_before_sleep " , PROPERTY_HINT_RANGE , " 0,5,0.01,or_greater " ) ) ;
2017-03-05 15:44:50 +00:00
body_angular_velocity_damp_ratio = 10 ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
broadphase = BroadPhase3DSW : : create_func ( ) ;
2017-03-05 15:44:50 +00:00
broadphase - > set_pair_callback ( _broadphase_pair , this ) ;
broadphase - > set_unpair_callback ( _broadphase_unpair , this ) ;
2020-04-01 23:20:12 +00:00
area = nullptr ;
2015-10-08 18:00:40 +00:00
2020-03-27 18:21:27 +00:00
direct_access = memnew ( PhysicsDirectSpaceState3DSW ) ;
2017-03-05 15:44:50 +00:00
direct_access - > space = this ;
2016-05-22 00:18:16 +00:00
2017-03-05 15:44:50 +00:00
for ( int i = 0 ; i < ELAPSED_TIME_MAX ; i + + )
elapsed_time [ i ] = 0 ;
2015-10-08 18:00:40 +00:00
}
2020-03-27 18:21:27 +00:00
Space3DSW : : ~ Space3DSW ( ) {
2015-10-08 18:00:40 +00:00
memdelete ( broadphase ) ;
2017-03-05 15:44:50 +00:00
memdelete ( direct_access ) ;
2015-10-08 18:00:40 +00:00
}