2023-01-05 13:25:55 +01:00
/**************************************************************************/
/* collision_polygon_3d.cpp */
/**************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/**************************************************************************/
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
/* */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the */
/* "Software"), to deal in the Software without restriction, including */
/* without limitation the rights to use, copy, modify, merge, publish, */
/* distribute, sublicense, and/or sell copies of the Software, and to */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions: */
/* */
/* The above copyright notice and this permission notice shall be */
/* included in all copies or substantial portions of the Software. */
/* */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
/**************************************************************************/
2018-01-05 00:50:27 +01:00
2020-03-26 22:49:16 +01:00
# include "collision_polygon_3d.h"
2014-09-17 02:19:54 +02:00
2020-03-26 22:49:16 +01:00
# include "collision_object_3d.h"
2020-05-25 19:20:45 +02:00
# include "core/math/geometry_2d.h"
2020-03-26 22:49:16 +01:00
# include "scene/resources/convex_polygon_shape_3d.h"
2014-09-17 02:19:54 +02:00
2020-03-26 22:49:16 +01:00
void CollisionPolygon3D : : _build_polygon ( ) {
2023-06-22 21:06:05 +02:00
if ( ! collision_object ) {
2015-09-16 03:07:03 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2014-09-17 02:19:54 +02:00
2023-06-22 21:06:05 +02:00
collision_object - > shape_owner_clear_shapes ( owner_id ) ;
2014-09-17 02:19:54 +02:00
2020-05-14 16:41:43 +02:00
if ( polygon . size ( ) = = 0 ) {
2014-09-17 02:19:54 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2014-09-17 02:19:54 +02:00
2020-05-25 19:20:45 +02:00
Vector < Vector < Vector2 > > decomp = Geometry2D : : decompose_polygon_in_convex ( polygon ) ;
2020-05-14 16:41:43 +02:00
if ( decomp . size ( ) = = 0 ) {
2014-09-17 02:19:54 +02:00
return ;
2020-05-14 16:41:43 +02:00
}
2014-09-17 02:19:54 +02:00
2017-07-15 06:23:10 +02:00
//here comes the sun, lalalala
//decompose concave into multiple convex polygons and add them
for ( int i = 0 ; i < decomp . size ( ) ; i + + ) {
2020-03-26 22:49:16 +01:00
Ref < ConvexPolygonShape3D > convex = memnew ( ConvexPolygonShape3D ) ;
2020-02-17 22:06:54 +01:00
Vector < Vector3 > cp ;
2017-07-15 06:23:10 +02:00
int cs = decomp [ i ] . size ( ) ;
cp . resize ( cs * 2 ) ;
{
2020-02-17 22:06:54 +01:00
Vector3 * w = cp . ptrw ( ) ;
2017-07-15 06:23:10 +02:00
int idx = 0 ;
for ( int j = 0 ; j < cs ; j + + ) {
Vector2 d = decomp [ i ] [ j ] ;
w [ idx + + ] = Vector3 ( d . x , d . y , depth * 0.5 ) ;
w [ idx + + ] = Vector3 ( d . x , d . y , - depth * 0.5 ) ;
2014-09-17 02:19:54 +02:00
}
}
2017-07-15 06:23:10 +02:00
convex - > set_points ( cp ) ;
2021-02-09 20:37:12 +01:00
convex - > set_margin ( margin ) ;
2023-06-22 21:06:05 +02:00
collision_object - > shape_owner_add_shape ( owner_id , convex ) ;
collision_object - > shape_owner_set_disabled ( owner_id , disabled ) ;
2014-09-17 02:19:54 +02:00
}
2015-09-16 03:07:03 +02:00
}
2020-03-26 22:49:16 +01:00
void CollisionPolygon3D : : _update_in_shape_owner ( bool p_xform_only ) {
2023-06-22 21:06:05 +02:00
collision_object - > shape_owner_set_transform ( owner_id , get_transform ( ) ) ;
2020-05-14 16:41:43 +02:00
if ( p_xform_only ) {
2018-01-11 21:05:42 +01:00
return ;
2020-05-14 16:41:43 +02:00
}
2023-06-22 21:06:05 +02:00
collision_object - > shape_owner_set_disabled ( owner_id , disabled ) ;
2018-01-11 21:05:42 +01:00
}
2020-03-26 22:49:16 +01:00
void CollisionPolygon3D : : _notification ( int p_what ) {
2017-03-05 16:44:50 +01:00
switch ( p_what ) {
2017-07-15 06:23:10 +02:00
case NOTIFICATION_PARENTED : {
2023-06-22 21:06:05 +02:00
collision_object = Object : : cast_to < CollisionObject3D > ( get_parent ( ) ) ;
if ( collision_object ) {
owner_id = collision_object - > create_shape_owner ( this ) ;
2017-07-15 06:23:10 +02:00
_build_polygon ( ) ;
2018-01-11 21:05:42 +01:00
_update_in_shape_owner ( ) ;
2015-09-16 03:07:03 +02:00
}
} break ;
2022-02-15 18:06:48 +01:00
2018-01-11 21:05:42 +01:00
case NOTIFICATION_ENTER_TREE : {
2023-06-22 21:06:05 +02:00
if ( collision_object ) {
2018-01-11 21:05:42 +01:00
_update_in_shape_owner ( ) ;
}
} break ;
2022-02-15 18:06:48 +01:00
2015-09-16 03:07:03 +02:00
case NOTIFICATION_LOCAL_TRANSFORM_CHANGED : {
2023-06-22 21:06:05 +02:00
if ( collision_object ) {
2018-01-11 21:05:42 +01:00
_update_in_shape_owner ( true ) ;
2014-09-17 02:19:54 +02:00
}
2022-10-29 22:15:27 +02:00
update_configuration_warnings ( ) ;
2017-07-15 06:23:10 +02:00
} break ;
2022-02-15 18:06:48 +01:00
2017-07-15 06:23:10 +02:00
case NOTIFICATION_UNPARENTED : {
2023-06-22 21:06:05 +02:00
if ( collision_object ) {
collision_object - > remove_shape_owner ( owner_id ) ;
2014-09-17 02:19:54 +02:00
}
2017-07-15 06:23:10 +02:00
owner_id = 0 ;
2023-06-22 21:06:05 +02:00
collision_object = nullptr ;
2014-09-17 02:19:54 +02:00
} break ;
}
}
2020-03-26 22:49:16 +01:00
void CollisionPolygon3D : : set_polygon ( const Vector < Point2 > & p_polygon ) {
2017-03-05 16:44:50 +01:00
polygon = p_polygon ;
2023-06-22 21:06:05 +02:00
if ( collision_object ) {
2017-07-15 06:23:10 +02:00
_build_polygon ( ) ;
2014-09-17 02:19:54 +02:00
}
2020-10-29 11:01:28 +01:00
update_configuration_warnings ( ) ;
2021-06-23 16:49:50 +02:00
update_gizmos ( ) ;
2014-09-17 02:19:54 +02:00
}
2020-03-26 22:49:16 +01:00
Vector < Point2 > CollisionPolygon3D : : get_polygon ( ) const {
2014-09-17 02:19:54 +02:00
return polygon ;
}
2020-03-26 22:49:16 +01:00
AABB CollisionPolygon3D : : get_item_rect ( ) const {
2014-09-17 02:19:54 +02:00
return aabb ;
}
2021-01-30 00:22:12 +01:00
void CollisionPolygon3D : : set_depth ( real_t p_depth ) {
2017-03-05 16:44:50 +01:00
depth = p_depth ;
2017-07-15 06:23:10 +02:00
_build_polygon ( ) ;
2021-06-23 16:49:50 +02:00
update_gizmos ( ) ;
2014-09-17 02:19:54 +02:00
}
2021-01-30 00:22:12 +01:00
real_t CollisionPolygon3D : : get_depth ( ) const {
2014-09-17 02:19:54 +02:00
return depth ;
}
2020-03-26 22:49:16 +01:00
void CollisionPolygon3D : : set_disabled ( bool p_disabled ) {
2017-07-15 06:23:10 +02:00
disabled = p_disabled ;
2021-06-23 16:49:50 +02:00
update_gizmos ( ) ;
2019-08-21 15:27:02 +02:00
2023-06-22 21:06:05 +02:00
if ( collision_object ) {
collision_object - > shape_owner_set_disabled ( owner_id , p_disabled ) ;
2017-07-15 06:23:10 +02:00
}
}
2020-03-26 22:49:16 +01:00
bool CollisionPolygon3D : : is_disabled ( ) const {
2017-07-15 06:23:10 +02:00
return disabled ;
}
2021-02-09 20:37:12 +01:00
real_t CollisionPolygon3D : : get_margin ( ) const {
return margin ;
}
void CollisionPolygon3D : : set_margin ( real_t p_margin ) {
margin = p_margin ;
2023-06-22 21:06:05 +02:00
if ( collision_object ) {
2021-02-09 20:37:12 +01:00
_build_polygon ( ) ;
}
}
2022-09-19 17:43:15 +02:00
PackedStringArray CollisionPolygon3D : : get_configuration_warnings ( ) const {
PackedStringArray warnings = Node : : get_configuration_warnings ( ) ;
2020-05-14 22:59:27 +02:00
2020-03-26 22:49:16 +01:00
if ( ! Object : : cast_to < CollisionObject3D > ( get_parent ( ) ) ) {
2022-10-24 19:07:26 +02:00
warnings . push_back ( RTR ( " CollisionPolygon3D only serves to provide a collision shape to a CollisionObject3D derived node. \n Please only use it as a child of Area3D, StaticBody3D, RigidBody3D, CharacterBody3D, etc. to give them a shape. " ) ) ;
2016-05-17 23:27:15 +02:00
}
2020-12-15 13:04:21 +01:00
if ( polygon . is_empty ( ) ) {
2022-03-28 15:24:14 +02:00
warnings . push_back ( RTR ( " An empty CollisionPolygon3D has no effect on collision. " ) ) ;
2016-05-17 23:27:15 +02:00
}
2022-10-24 19:07:26 +02:00
Vector3 scale = get_transform ( ) . get_basis ( ) . get_scale ( ) ;
if ( ! ( Math : : is_zero_approx ( scale . x - scale . y ) & & Math : : is_zero_approx ( scale . y - scale . z ) ) ) {
warnings . push_back ( RTR ( " A non-uniformly scaled CollisionPolygon3D node will probably not function as expected. \n Please make its scale uniform (i.e. the same on all axes), and change its polygon's vertices instead. " ) ) ;
}
2020-10-29 11:01:28 +01:00
return warnings ;
2016-05-17 23:27:15 +02:00
}
2014-09-17 02:19:54 +02:00
2020-03-26 22:49:16 +01:00
bool CollisionPolygon3D : : _is_editable_3d_polygon ( ) const {
2018-04-28 02:52:15 +02:00
return true ;
}
2020-05-14 14:29:06 +02:00
2020-03-26 22:49:16 +01:00
void CollisionPolygon3D : : _bind_methods ( ) {
ClassDB : : bind_method ( D_METHOD ( " set_depth " , " depth " ) , & CollisionPolygon3D : : set_depth ) ;
ClassDB : : bind_method ( D_METHOD ( " get_depth " ) , & CollisionPolygon3D : : get_depth ) ;
2014-09-17 02:19:54 +02:00
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_polygon " , " polygon " ) , & CollisionPolygon3D : : set_polygon ) ;
ClassDB : : bind_method ( D_METHOD ( " get_polygon " ) , & CollisionPolygon3D : : get_polygon ) ;
2015-09-16 03:07:03 +02:00
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_disabled " , " disabled " ) , & CollisionPolygon3D : : set_disabled ) ;
ClassDB : : bind_method ( D_METHOD ( " is_disabled " ) , & CollisionPolygon3D : : is_disabled ) ;
2014-09-17 02:19:54 +02:00
2021-02-09 20:37:12 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_margin " , " margin " ) , & CollisionPolygon3D : : set_margin ) ;
ClassDB : : bind_method ( D_METHOD ( " get_margin " ) , & CollisionPolygon3D : : get_margin ) ;
2020-03-26 22:49:16 +01:00
ClassDB : : bind_method ( D_METHOD ( " _is_editable_3d_polygon " ) , & CollisionPolygon3D : : _is_editable_3d_polygon ) ;
2018-04-28 02:52:15 +02:00
2021-12-03 01:09:19 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " depth " , PROPERTY_HINT_NONE , " suffix:m " ) , " set_depth " , " get_depth " ) ;
2017-07-15 06:23:10 +02:00
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " disabled " ) , " set_disabled " , " is_disabled " ) ;
2020-02-17 22:06:54 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : PACKED_VECTOR2_ARRAY , " polygon " ) , " set_polygon " , " get_polygon " ) ;
2021-12-03 01:09:19 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : FLOAT , " margin " , PROPERTY_HINT_RANGE , " 0.001,10,0.001,suffix:m " ) , " set_margin " , " get_margin " ) ;
2014-09-17 02:19:54 +02:00
}
2020-03-26 22:49:16 +01:00
CollisionPolygon3D : : CollisionPolygon3D ( ) {
2017-07-15 06:23:10 +02:00
set_notify_local_transform ( true ) ;
2014-09-17 02:19:54 +02:00
}