2023-01-10 15:26:54 +01:00
/**************************************************************************/
/* collision_shape_2d.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
2014-02-10 02:10:30 +01:00
# include "collision_shape_2d.h"
2017-08-19 01:02:56 +02:00
2014-02-10 02:10:30 +01:00
# include "collision_object_2d.h"
2018-09-11 18:13:45 +02:00
# include "core/engine.h"
2022-05-08 00:31:40 +02:00
# include "scene/2d/area_2d.h"
2014-02-10 02:10:30 +01:00
# include "scene/resources/capsule_shape_2d.h"
2017-03-05 16:44:50 +01:00
# include "scene/resources/circle_shape_2d.h"
2014-02-10 02:10:30 +01:00
# include "scene/resources/concave_polygon_shape_2d.h"
2017-03-05 16:44:50 +01:00
# include "scene/resources/convex_polygon_shape_2d.h"
2019-02-12 17:18:13 +01:00
# include "scene/resources/line_shape_2d.h"
2017-03-05 16:44:50 +01:00
# include "scene/resources/rectangle_shape_2d.h"
# include "scene/resources/segment_shape_2d.h"
2014-02-10 02:10:30 +01:00
void CollisionShape2D : : _shape_changed ( ) {
update ( ) ;
}
2018-01-11 21:05:42 +01:00
void CollisionShape2D : : _update_in_shape_owner ( bool p_xform_only ) {
parent - > shape_owner_set_transform ( owner_id , get_transform ( ) ) ;
2021-05-05 12:44:11 +02:00
if ( p_xform_only ) {
2018-01-11 21:05:42 +01:00
return ;
2021-05-05 12:44:11 +02:00
}
2018-01-11 21:05:42 +01:00
parent - > shape_owner_set_disabled ( owner_id , disabled ) ;
parent - > shape_owner_set_one_way_collision ( owner_id , one_way_collision ) ;
2019-01-18 18:15:05 +01:00
parent - > shape_owner_set_one_way_collision_margin ( owner_id , one_way_collision_margin ) ;
2018-01-11 21:05:42 +01:00
}
2014-02-10 02:10:30 +01:00
void CollisionShape2D : : _notification ( int p_what ) {
2017-03-05 16:44:50 +01:00
switch ( p_what ) {
2017-06-24 04:30:43 +02:00
case NOTIFICATION_PARENTED : {
2017-08-24 22:58:51 +02:00
parent = Object : : cast_to < CollisionObject2D > ( get_parent ( ) ) ;
2017-06-24 04:30:43 +02:00
if ( parent ) {
owner_id = parent - > create_shape_owner ( this ) ;
if ( shape . is_valid ( ) ) {
parent - > shape_owner_add_shape ( owner_id , shape ) ;
}
2018-01-11 21:05:42 +01:00
_update_in_shape_owner ( ) ;
2017-06-24 04:30:43 +02:00
}
2017-08-19 01:02:56 +02:00
/*if (Engine::get_singleton()->is_editor_hint()) {
2015-12-12 12:11:36 +01:00
//display above all else
set_z_as_relative ( false ) ;
2017-12-29 23:06:03 +01:00
set_z_index ( VS : : CANVAS_ITEM_Z_MAX - 1 ) ;
2017-06-24 04:30:43 +02:00
} */
2015-09-16 03:07:03 +02:00
2018-01-11 21:05:42 +01:00
} break ;
case NOTIFICATION_ENTER_TREE : {
if ( parent ) {
_update_in_shape_owner ( ) ;
}
2014-09-22 05:50:48 +02:00
} break ;
2014-02-10 02:10:30 +01:00
case NOTIFICATION_LOCAL_TRANSFORM_CHANGED : {
2017-06-24 04:30:43 +02:00
if ( parent ) {
2018-01-11 21:05:42 +01:00
_update_in_shape_owner ( true ) ;
2015-09-16 03:07:03 +02:00
}
} break ;
2017-06-24 04:30:43 +02:00
case NOTIFICATION_UNPARENTED : {
if ( parent ) {
parent - > remove_shape_owner ( owner_id ) ;
}
owner_id = 0 ;
2021-05-04 16:00:45 +02:00
parent = nullptr ;
2014-02-10 02:10:30 +01:00
2019-07-12 23:46:22 +02:00
} break ;
2014-02-10 02:10:30 +01:00
case NOTIFICATION_DRAW : {
2021-10-22 13:44:38 +02:00
ERR_FAIL_COND ( ! is_inside_tree ( ) ) ;
2017-08-19 01:02:56 +02:00
if ( ! Engine : : get_singleton ( ) - > is_editor_hint ( ) & & ! get_tree ( ) - > is_debugging_collisions_hint ( ) ) {
2015-09-19 04:10:58 +02:00
break ;
2015-09-20 18:03:46 +02:00
}
2015-09-19 04:10:58 +02:00
2015-09-23 02:30:08 +02:00
if ( ! shape . is_valid ( ) ) {
break ;
}
2017-03-05 16:44:50 +01:00
rect = Rect2 ( ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
Color draw_col = get_tree ( ) - > get_debug_collisions_color ( ) ;
2017-06-24 04:30:43 +02:00
if ( disabled ) {
2018-08-21 17:47:31 +02:00
float g = draw_col . get_v ( ) ;
2017-06-24 04:30:43 +02:00
draw_col . r = g ;
draw_col . g = g ;
draw_col . b = g ;
2020-08-31 06:20:13 +02:00
draw_col . a * = 0.5 ;
2017-06-24 04:30:43 +02:00
}
2017-03-05 16:44:50 +01:00
shape - > draw ( get_canvas_item ( ) , draw_col ) ;
2014-02-10 02:10:30 +01:00
2017-03-05 16:44:50 +01:00
rect = shape - > get_rect ( ) ;
rect = rect . grow ( 3 ) ;
2014-02-10 02:10:30 +01:00
2017-06-24 04:30:43 +02:00
if ( one_way_collision ) {
2019-07-12 23:46:22 +02:00
// Draw an arrow indicating the one-way collision direction
draw_col = get_tree ( ) - > get_debug_collisions_color ( ) . inverted ( ) ;
if ( disabled ) {
draw_col = draw_col . darkened ( 0.25 ) ;
}
2017-06-24 04:30:43 +02:00
Vector2 line_to ( 0 , 20 ) ;
2019-07-12 23:46:22 +02:00
draw_line ( Vector2 ( ) , line_to , draw_col , 2 , true ) ;
2017-06-24 04:30:43 +02:00
Vector < Vector2 > pts ;
float tsize = 8 ;
pts . push_back ( line_to + ( Vector2 ( 0 , tsize ) ) ) ;
2019-12-10 05:13:02 +01:00
pts . push_back ( line_to + ( Vector2 ( Math_SQRT12 * tsize , 0 ) ) ) ;
pts . push_back ( line_to + ( Vector2 ( - Math_SQRT12 * tsize , 0 ) ) ) ;
2017-06-24 04:30:43 +02:00
Vector < Color > cols ;
2021-05-05 12:44:11 +02:00
for ( int i = 0 ; i < 3 ; i + + ) {
2019-07-12 23:46:22 +02:00
cols . push_back ( draw_col ) ;
2021-05-05 12:44:11 +02:00
}
2017-06-24 04:30:43 +02:00
2019-07-12 23:46:22 +02:00
draw_primitive ( pts , cols , Vector < Vector2 > ( ) ) ;
2017-06-24 04:30:43 +02:00
}
2014-03-15 06:58:53 +01:00
} break ;
2014-02-10 02:10:30 +01:00
}
}
2017-03-05 16:44:50 +01:00
void CollisionShape2D : : set_shape ( const Ref < Shape2D > & p_shape ) {
2021-01-11 17:04:08 +01:00
if ( p_shape = = shape ) {
return ;
}
2021-05-05 12:44:11 +02:00
if ( shape . is_valid ( ) ) {
2017-03-05 16:44:50 +01:00
shape - > disconnect ( " changed " , this , " _shape_changed " ) ;
2021-05-05 12:44:11 +02:00
}
2017-03-05 16:44:50 +01:00
shape = p_shape ;
2014-02-10 02:10:30 +01:00
update ( ) ;
2017-06-24 04:30:43 +02:00
if ( parent ) {
parent - > shape_owner_clear_shapes ( owner_id ) ;
if ( shape . is_valid ( ) ) {
parent - > shape_owner_add_shape ( owner_id , shape ) ;
2015-09-16 03:07:03 +02:00
}
2021-01-11 17:04:08 +01:00
_update_in_shape_owner ( ) ;
2015-09-16 03:07:03 +02:00
}
2017-06-24 04:30:43 +02:00
2021-05-05 12:44:11 +02:00
if ( shape . is_valid ( ) ) {
2017-03-05 16:44:50 +01:00
shape - > connect ( " changed " , this , " _shape_changed " ) ;
2021-05-05 12:44:11 +02:00
}
2014-02-10 02:10:30 +01:00
2017-01-22 14:09:41 +01:00
update_configuration_warning ( ) ;
2014-02-10 02:10:30 +01:00
}
Ref < Shape2D > CollisionShape2D : : get_shape ( ) const {
return shape ;
}
2017-12-27 09:28:02 +01:00
bool CollisionShape2D : : _edit_is_selected_on_click ( const Point2 & p_point , double p_tolerance ) const {
2021-05-05 12:44:11 +02:00
if ( ! shape . is_valid ( ) ) {
2017-12-27 09:28:02 +01:00
return false ;
2021-05-05 12:44:11 +02:00
}
2017-12-27 09:28:02 +01:00
return shape - > _edit_is_selected_on_click ( p_point , p_tolerance ) ;
}
2017-06-24 04:30:43 +02:00
String CollisionShape2D : : get_configuration_warning ( ) const {
2020-03-22 09:31:09 +01:00
String warning = Node2D : : get_configuration_warning ( ) ;
2020-10-02 09:51:45 +02:00
2017-08-24 22:58:51 +02:00
if ( ! Object : : cast_to < CollisionObject2D > ( get_parent ( ) ) ) {
2020-03-22 09:31:09 +01:00
if ( warning ! = String ( ) ) {
warning + = " \n \n " ;
}
warning + = TTR ( " CollisionShape2D only serves to provide a collision shape to a CollisionObject2D derived node. Please only use it as a child of Area2D, StaticBody2D, RigidBody2D, KinematicBody2D, etc. to give them a shape. " ) ;
2015-09-16 03:07:03 +02:00
}
2020-10-02 09:51:45 +02:00
2017-06-24 04:30:43 +02:00
if ( ! shape . is_valid ( ) ) {
2020-03-22 09:31:09 +01:00
if ( warning ! = String ( ) ) {
warning + = " \n \n " ;
}
warning + = TTR ( " A shape must be provided for CollisionShape2D to function. Please create a shape resource for it! " ) ;
2020-10-02 09:51:45 +02:00
} else {
Ref < ConvexPolygonShape2D > convex = shape ;
Ref < ConcavePolygonShape2D > concave = shape ;
if ( convex . is_valid ( ) | | concave . is_valid ( ) ) {
if ( warning ! = String ( ) ) {
warning + = " \n \n " ;
}
warning + = TTR ( " Polygon-based shapes are not meant be used nor edited directly through the CollisionShape2D node. Please use the CollisionPolygon2D node instead. " ) ;
2020-03-22 09:31:09 +01:00
}
2020-07-03 14:45:17 +02:00
}
2022-05-08 00:31:40 +02:00
if ( one_way_collision & & Object : : cast_to < Area2D > ( get_parent ( ) ) ) {
warning + = TTR ( " The One Way Collision property will be ignored when the parent is an Area2D. " ) ;
}
2020-10-02 09:51:45 +02:00
2020-03-22 09:31:09 +01:00
return warning ;
2014-02-10 02:10:30 +01:00
}
2017-06-24 04:30:43 +02:00
void CollisionShape2D : : set_disabled ( bool p_disabled ) {
disabled = p_disabled ;
update ( ) ;
if ( parent ) {
parent - > shape_owner_set_disabled ( owner_id , p_disabled ) ;
}
2015-09-16 03:07:03 +02:00
}
2017-06-24 04:30:43 +02:00
bool CollisionShape2D : : is_disabled ( ) const {
return disabled ;
2015-09-16 03:07:03 +02:00
}
2017-06-24 04:30:43 +02:00
void CollisionShape2D : : set_one_way_collision ( bool p_enable ) {
one_way_collision = p_enable ;
update ( ) ;
if ( parent ) {
parent - > shape_owner_set_one_way_collision ( owner_id , p_enable ) ;
2016-05-17 23:27:15 +02:00
}
2022-05-08 00:31:40 +02:00
update_configuration_warning ( ) ;
2017-06-24 04:30:43 +02:00
}
2016-05-17 23:27:15 +02:00
2017-06-24 04:30:43 +02:00
bool CollisionShape2D : : is_one_way_collision_enabled ( ) const {
return one_way_collision ;
2016-05-17 23:27:15 +02:00
}
2019-01-18 18:15:05 +01:00
void CollisionShape2D : : set_one_way_collision_margin ( float p_margin ) {
one_way_collision_margin = p_margin ;
if ( parent ) {
parent - > shape_owner_set_one_way_collision_margin ( owner_id , one_way_collision_margin ) ;
}
}
float CollisionShape2D : : get_one_way_collision_margin ( ) const {
return one_way_collision_margin ;
}
2014-02-10 02:10:30 +01:00
void CollisionShape2D : : _bind_methods ( ) {
2017-08-09 13:19:41 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_shape " , " shape " ) , & CollisionShape2D : : set_shape ) ;
ClassDB : : bind_method ( D_METHOD ( " get_shape " ) , & CollisionShape2D : : get_shape ) ;
2017-06-24 04:30:43 +02:00
ClassDB : : bind_method ( D_METHOD ( " set_disabled " , " disabled " ) , & CollisionShape2D : : set_disabled ) ;
ClassDB : : bind_method ( D_METHOD ( " is_disabled " ) , & CollisionShape2D : : is_disabled ) ;
ClassDB : : bind_method ( D_METHOD ( " set_one_way_collision " , " enabled " ) , & CollisionShape2D : : set_one_way_collision ) ;
ClassDB : : bind_method ( D_METHOD ( " is_one_way_collision_enabled " ) , & CollisionShape2D : : is_one_way_collision_enabled ) ;
2019-01-18 18:15:05 +01:00
ClassDB : : bind_method ( D_METHOD ( " set_one_way_collision_margin " , " margin " ) , & CollisionShape2D : : set_one_way_collision_margin ) ;
ClassDB : : bind_method ( D_METHOD ( " get_one_way_collision_margin " ) , & CollisionShape2D : : get_one_way_collision_margin ) ;
2017-03-05 16:44:50 +01:00
ClassDB : : bind_method ( D_METHOD ( " _shape_changed " ) , & CollisionShape2D : : _shape_changed ) ;
2015-09-16 03:07:03 +02:00
2018-11-08 15:30:02 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : OBJECT , " shape " , PROPERTY_HINT_RESOURCE_TYPE , " Shape2D " ) , " set_shape " , " get_shape " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " disabled " ) , " set_disabled " , " is_disabled " ) ;
ADD_PROPERTY ( PropertyInfo ( Variant : : BOOL , " one_way_collision " ) , " set_one_way_collision " , " is_one_way_collision_enabled " ) ;
2019-01-18 18:15:05 +01:00
ADD_PROPERTY ( PropertyInfo ( Variant : : REAL , " one_way_collision_margin " , PROPERTY_HINT_RANGE , " 0,128,0.1 " ) , " set_one_way_collision_margin " , " get_one_way_collision_margin " ) ;
2014-02-10 02:10:30 +01:00
}
CollisionShape2D : : CollisionShape2D ( ) {
2017-03-05 16:44:50 +01:00
rect = Rect2 ( - Point2 ( 10 , 10 ) , Point2 ( 20 , 20 ) ) ;
2015-09-16 03:07:03 +02:00
set_notify_local_transform ( true ) ;
2017-06-24 04:30:43 +02:00
owner_id = 0 ;
2021-05-04 16:00:45 +02:00
parent = nullptr ;
2017-06-24 04:30:43 +02:00
disabled = false ;
one_way_collision = false ;
2019-01-18 18:15:05 +01:00
one_way_collision_margin = 1.0 ;
2014-02-10 02:10:30 +01:00
}