b2a3e8cb34
This also changes the code used to hide the custom center of mass property for consistency with other places in the editor (only this approach worked for Contact Monitor as well). The Center of Mass property hint in RigidBody2D was also modified for better usability.
823 lines
32 KiB
C++
823 lines
32 KiB
C++
/**************************************************************************/
|
|
/* rigid_body_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. */
|
|
/**************************************************************************/
|
|
|
|
#include "rigid_body_2d.h"
|
|
|
|
#include "scene/scene_string_names.h"
|
|
|
|
void RigidBody2D::_body_enter_tree(ObjectID p_id) {
|
|
Object *obj = ObjectDB::get_instance(p_id);
|
|
Node *node = Object::cast_to<Node>(obj);
|
|
ERR_FAIL_NULL(node);
|
|
ERR_FAIL_NULL(contact_monitor);
|
|
HashMap<ObjectID, BodyState>::Iterator E = contact_monitor->body_map.find(p_id);
|
|
ERR_FAIL_COND(!E);
|
|
ERR_FAIL_COND(E->value.in_scene);
|
|
|
|
contact_monitor->locked = true;
|
|
|
|
E->value.in_scene = true;
|
|
emit_signal(SceneStringNames::get_singleton()->body_entered, node);
|
|
|
|
for (int i = 0; i < E->value.shapes.size(); i++) {
|
|
emit_signal(SceneStringNames::get_singleton()->body_shape_entered, E->value.rid, node, E->value.shapes[i].body_shape, E->value.shapes[i].local_shape);
|
|
}
|
|
|
|
contact_monitor->locked = false;
|
|
}
|
|
|
|
void RigidBody2D::_body_exit_tree(ObjectID p_id) {
|
|
Object *obj = ObjectDB::get_instance(p_id);
|
|
Node *node = Object::cast_to<Node>(obj);
|
|
ERR_FAIL_NULL(node);
|
|
ERR_FAIL_NULL(contact_monitor);
|
|
HashMap<ObjectID, BodyState>::Iterator E = contact_monitor->body_map.find(p_id);
|
|
ERR_FAIL_COND(!E);
|
|
ERR_FAIL_COND(!E->value.in_scene);
|
|
E->value.in_scene = false;
|
|
|
|
contact_monitor->locked = true;
|
|
|
|
emit_signal(SceneStringNames::get_singleton()->body_exited, node);
|
|
|
|
for (int i = 0; i < E->value.shapes.size(); i++) {
|
|
emit_signal(SceneStringNames::get_singleton()->body_shape_exited, E->value.rid, node, E->value.shapes[i].body_shape, E->value.shapes[i].local_shape);
|
|
}
|
|
|
|
contact_monitor->locked = false;
|
|
}
|
|
|
|
void RigidBody2D::_body_inout(int p_status, const RID &p_body, ObjectID p_instance, int p_body_shape, int p_local_shape) {
|
|
bool body_in = p_status == 1;
|
|
ObjectID objid = p_instance;
|
|
|
|
Object *obj = ObjectDB::get_instance(objid);
|
|
Node *node = Object::cast_to<Node>(obj);
|
|
|
|
ERR_FAIL_NULL(contact_monitor);
|
|
HashMap<ObjectID, BodyState>::Iterator E = contact_monitor->body_map.find(objid);
|
|
|
|
ERR_FAIL_COND(!body_in && !E);
|
|
|
|
if (body_in) {
|
|
if (!E) {
|
|
E = contact_monitor->body_map.insert(objid, BodyState());
|
|
E->value.rid = p_body;
|
|
//E->value.rc=0;
|
|
E->value.in_scene = node && node->is_inside_tree();
|
|
if (node) {
|
|
node->connect(SceneStringNames::get_singleton()->tree_entered, callable_mp(this, &RigidBody2D::_body_enter_tree).bind(objid));
|
|
node->connect(SceneStringNames::get_singleton()->tree_exiting, callable_mp(this, &RigidBody2D::_body_exit_tree).bind(objid));
|
|
if (E->value.in_scene) {
|
|
emit_signal(SceneStringNames::get_singleton()->body_entered, node);
|
|
}
|
|
}
|
|
|
|
//E->value.rc++;
|
|
}
|
|
|
|
if (node) {
|
|
E->value.shapes.insert(ShapePair(p_body_shape, p_local_shape));
|
|
}
|
|
|
|
if (E->value.in_scene) {
|
|
emit_signal(SceneStringNames::get_singleton()->body_shape_entered, p_body, node, p_body_shape, p_local_shape);
|
|
}
|
|
|
|
} else {
|
|
//E->value.rc--;
|
|
|
|
if (node) {
|
|
E->value.shapes.erase(ShapePair(p_body_shape, p_local_shape));
|
|
}
|
|
|
|
bool in_scene = E->value.in_scene;
|
|
|
|
if (E->value.shapes.is_empty()) {
|
|
if (node) {
|
|
node->disconnect(SceneStringNames::get_singleton()->tree_entered, callable_mp(this, &RigidBody2D::_body_enter_tree));
|
|
node->disconnect(SceneStringNames::get_singleton()->tree_exiting, callable_mp(this, &RigidBody2D::_body_exit_tree));
|
|
if (in_scene) {
|
|
emit_signal(SceneStringNames::get_singleton()->body_exited, node);
|
|
}
|
|
}
|
|
|
|
contact_monitor->body_map.remove(E);
|
|
}
|
|
if (node && in_scene) {
|
|
emit_signal(SceneStringNames::get_singleton()->body_shape_exited, p_body, node, p_body_shape, p_local_shape);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct _RigidBody2DInOut {
|
|
RID rid;
|
|
ObjectID id;
|
|
int shape = 0;
|
|
int local_shape = 0;
|
|
};
|
|
|
|
void RigidBody2D::_sync_body_state(PhysicsDirectBodyState2D *p_state) {
|
|
if (!freeze || freeze_mode != FREEZE_MODE_KINEMATIC) {
|
|
set_block_transform_notify(true);
|
|
set_global_transform(p_state->get_transform());
|
|
set_block_transform_notify(false);
|
|
}
|
|
|
|
linear_velocity = p_state->get_linear_velocity();
|
|
angular_velocity = p_state->get_angular_velocity();
|
|
|
|
contact_count = p_state->get_contact_count();
|
|
|
|
if (sleeping != p_state->is_sleeping()) {
|
|
sleeping = p_state->is_sleeping();
|
|
emit_signal(SceneStringNames::get_singleton()->sleeping_state_changed);
|
|
}
|
|
}
|
|
|
|
void RigidBody2D::_body_state_changed(PhysicsDirectBodyState2D *p_state) {
|
|
lock_callback();
|
|
|
|
if (GDVIRTUAL_IS_OVERRIDDEN(_integrate_forces)) {
|
|
_sync_body_state(p_state);
|
|
|
|
Transform2D old_transform = get_global_transform();
|
|
GDVIRTUAL_CALL(_integrate_forces, p_state);
|
|
Transform2D new_transform = get_global_transform();
|
|
|
|
if (new_transform != old_transform) {
|
|
// Update the physics server with the new transform, to prevent it from being overwritten at the sync below.
|
|
PhysicsServer2D::get_singleton()->body_set_state(get_rid(), PhysicsServer2D::BODY_STATE_TRANSFORM, new_transform);
|
|
}
|
|
}
|
|
|
|
_sync_body_state(p_state);
|
|
|
|
if (contact_monitor) {
|
|
contact_monitor->locked = true;
|
|
|
|
//untag all
|
|
int rc = 0;
|
|
for (KeyValue<ObjectID, BodyState> &E : contact_monitor->body_map) {
|
|
for (int i = 0; i < E.value.shapes.size(); i++) {
|
|
E.value.shapes[i].tagged = false;
|
|
rc++;
|
|
}
|
|
}
|
|
|
|
_RigidBody2DInOut *toadd = (_RigidBody2DInOut *)alloca(p_state->get_contact_count() * sizeof(_RigidBody2DInOut));
|
|
int toadd_count = 0; //state->get_contact_count();
|
|
RigidBody2D_RemoveAction *toremove = (RigidBody2D_RemoveAction *)alloca(rc * sizeof(RigidBody2D_RemoveAction));
|
|
int toremove_count = 0;
|
|
|
|
//put the ones to add
|
|
|
|
for (int i = 0; i < p_state->get_contact_count(); i++) {
|
|
RID col_rid = p_state->get_contact_collider(i);
|
|
ObjectID col_obj = p_state->get_contact_collider_id(i);
|
|
int local_shape = p_state->get_contact_local_shape(i);
|
|
int col_shape = p_state->get_contact_collider_shape(i);
|
|
|
|
HashMap<ObjectID, BodyState>::Iterator E = contact_monitor->body_map.find(col_obj);
|
|
if (!E) {
|
|
toadd[toadd_count].rid = col_rid;
|
|
toadd[toadd_count].local_shape = local_shape;
|
|
toadd[toadd_count].id = col_obj;
|
|
toadd[toadd_count].shape = col_shape;
|
|
toadd_count++;
|
|
continue;
|
|
}
|
|
|
|
ShapePair sp(col_shape, local_shape);
|
|
int idx = E->value.shapes.find(sp);
|
|
if (idx == -1) {
|
|
toadd[toadd_count].rid = col_rid;
|
|
toadd[toadd_count].local_shape = local_shape;
|
|
toadd[toadd_count].id = col_obj;
|
|
toadd[toadd_count].shape = col_shape;
|
|
toadd_count++;
|
|
continue;
|
|
}
|
|
|
|
E->value.shapes[idx].tagged = true;
|
|
}
|
|
|
|
//put the ones to remove
|
|
|
|
for (const KeyValue<ObjectID, BodyState> &E : contact_monitor->body_map) {
|
|
for (int i = 0; i < E.value.shapes.size(); i++) {
|
|
if (!E.value.shapes[i].tagged) {
|
|
toremove[toremove_count].rid = E.value.rid;
|
|
toremove[toremove_count].body_id = E.key;
|
|
toremove[toremove_count].pair = E.value.shapes[i];
|
|
toremove_count++;
|
|
}
|
|
}
|
|
}
|
|
|
|
//process removals
|
|
|
|
for (int i = 0; i < toremove_count; i++) {
|
|
_body_inout(0, toremove[i].rid, toremove[i].body_id, toremove[i].pair.body_shape, toremove[i].pair.local_shape);
|
|
}
|
|
|
|
//process additions
|
|
|
|
for (int i = 0; i < toadd_count; i++) {
|
|
_body_inout(1, toadd[i].rid, toadd[i].id, toadd[i].shape, toadd[i].local_shape);
|
|
}
|
|
|
|
contact_monitor->locked = false;
|
|
}
|
|
|
|
unlock_callback();
|
|
}
|
|
|
|
void RigidBody2D::_apply_body_mode() {
|
|
if (freeze) {
|
|
switch (freeze_mode) {
|
|
case FREEZE_MODE_STATIC: {
|
|
set_body_mode(PhysicsServer2D::BODY_MODE_STATIC);
|
|
} break;
|
|
case FREEZE_MODE_KINEMATIC: {
|
|
set_body_mode(PhysicsServer2D::BODY_MODE_KINEMATIC);
|
|
} break;
|
|
}
|
|
} else if (lock_rotation) {
|
|
set_body_mode(PhysicsServer2D::BODY_MODE_RIGID_LINEAR);
|
|
} else {
|
|
set_body_mode(PhysicsServer2D::BODY_MODE_RIGID);
|
|
}
|
|
}
|
|
|
|
void RigidBody2D::set_lock_rotation_enabled(bool p_lock_rotation) {
|
|
if (p_lock_rotation == lock_rotation) {
|
|
return;
|
|
}
|
|
|
|
lock_rotation = p_lock_rotation;
|
|
_apply_body_mode();
|
|
}
|
|
|
|
bool RigidBody2D::is_lock_rotation_enabled() const {
|
|
return lock_rotation;
|
|
}
|
|
|
|
void RigidBody2D::set_freeze_enabled(bool p_freeze) {
|
|
if (p_freeze == freeze) {
|
|
return;
|
|
}
|
|
|
|
freeze = p_freeze;
|
|
_apply_body_mode();
|
|
}
|
|
|
|
bool RigidBody2D::is_freeze_enabled() const {
|
|
return freeze;
|
|
}
|
|
|
|
void RigidBody2D::set_freeze_mode(FreezeMode p_freeze_mode) {
|
|
if (p_freeze_mode == freeze_mode) {
|
|
return;
|
|
}
|
|
|
|
freeze_mode = p_freeze_mode;
|
|
_apply_body_mode();
|
|
}
|
|
|
|
RigidBody2D::FreezeMode RigidBody2D::get_freeze_mode() const {
|
|
return freeze_mode;
|
|
}
|
|
|
|
void RigidBody2D::set_mass(real_t p_mass) {
|
|
ERR_FAIL_COND(p_mass <= 0);
|
|
mass = p_mass;
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_MASS, mass);
|
|
}
|
|
|
|
real_t RigidBody2D::get_mass() const {
|
|
return mass;
|
|
}
|
|
|
|
void RigidBody2D::set_inertia(real_t p_inertia) {
|
|
ERR_FAIL_COND(p_inertia < 0);
|
|
inertia = p_inertia;
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_INERTIA, inertia);
|
|
}
|
|
|
|
real_t RigidBody2D::get_inertia() const {
|
|
return inertia;
|
|
}
|
|
|
|
void RigidBody2D::set_center_of_mass_mode(CenterOfMassMode p_mode) {
|
|
if (center_of_mass_mode == p_mode) {
|
|
return;
|
|
}
|
|
|
|
center_of_mass_mode = p_mode;
|
|
|
|
switch (center_of_mass_mode) {
|
|
case CENTER_OF_MASS_MODE_AUTO: {
|
|
center_of_mass = Vector2();
|
|
PhysicsServer2D::get_singleton()->body_reset_mass_properties(get_rid());
|
|
if (inertia != 0.0) {
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_INERTIA, inertia);
|
|
}
|
|
} break;
|
|
|
|
case CENTER_OF_MASS_MODE_CUSTOM: {
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_CENTER_OF_MASS, center_of_mass);
|
|
} break;
|
|
}
|
|
|
|
notify_property_list_changed();
|
|
}
|
|
|
|
RigidBody2D::CenterOfMassMode RigidBody2D::get_center_of_mass_mode() const {
|
|
return center_of_mass_mode;
|
|
}
|
|
|
|
void RigidBody2D::set_center_of_mass(const Vector2 &p_center_of_mass) {
|
|
if (center_of_mass == p_center_of_mass) {
|
|
return;
|
|
}
|
|
|
|
ERR_FAIL_COND(center_of_mass_mode != CENTER_OF_MASS_MODE_CUSTOM);
|
|
center_of_mass = p_center_of_mass;
|
|
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_CENTER_OF_MASS, center_of_mass);
|
|
}
|
|
|
|
const Vector2 &RigidBody2D::get_center_of_mass() const {
|
|
return center_of_mass;
|
|
}
|
|
|
|
void RigidBody2D::set_physics_material_override(const Ref<PhysicsMaterial> &p_physics_material_override) {
|
|
if (physics_material_override.is_valid()) {
|
|
physics_material_override->disconnect_changed(callable_mp(this, &RigidBody2D::_reload_physics_characteristics));
|
|
}
|
|
|
|
physics_material_override = p_physics_material_override;
|
|
|
|
if (physics_material_override.is_valid()) {
|
|
physics_material_override->connect_changed(callable_mp(this, &RigidBody2D::_reload_physics_characteristics));
|
|
}
|
|
_reload_physics_characteristics();
|
|
}
|
|
|
|
Ref<PhysicsMaterial> RigidBody2D::get_physics_material_override() const {
|
|
return physics_material_override;
|
|
}
|
|
|
|
void RigidBody2D::set_gravity_scale(real_t p_gravity_scale) {
|
|
gravity_scale = p_gravity_scale;
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_GRAVITY_SCALE, gravity_scale);
|
|
}
|
|
|
|
real_t RigidBody2D::get_gravity_scale() const {
|
|
return gravity_scale;
|
|
}
|
|
|
|
void RigidBody2D::set_linear_damp_mode(DampMode p_mode) {
|
|
linear_damp_mode = p_mode;
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_LINEAR_DAMP_MODE, linear_damp_mode);
|
|
}
|
|
|
|
RigidBody2D::DampMode RigidBody2D::get_linear_damp_mode() const {
|
|
return linear_damp_mode;
|
|
}
|
|
|
|
void RigidBody2D::set_angular_damp_mode(DampMode p_mode) {
|
|
angular_damp_mode = p_mode;
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_ANGULAR_DAMP_MODE, angular_damp_mode);
|
|
}
|
|
|
|
RigidBody2D::DampMode RigidBody2D::get_angular_damp_mode() const {
|
|
return angular_damp_mode;
|
|
}
|
|
|
|
void RigidBody2D::set_linear_damp(real_t p_linear_damp) {
|
|
ERR_FAIL_COND(p_linear_damp < -1);
|
|
linear_damp = p_linear_damp;
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_LINEAR_DAMP, linear_damp);
|
|
}
|
|
|
|
real_t RigidBody2D::get_linear_damp() const {
|
|
return linear_damp;
|
|
}
|
|
|
|
void RigidBody2D::set_angular_damp(real_t p_angular_damp) {
|
|
ERR_FAIL_COND(p_angular_damp < -1);
|
|
angular_damp = p_angular_damp;
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_ANGULAR_DAMP, angular_damp);
|
|
}
|
|
|
|
real_t RigidBody2D::get_angular_damp() const {
|
|
return angular_damp;
|
|
}
|
|
|
|
void RigidBody2D::set_axis_velocity(const Vector2 &p_axis) {
|
|
Vector2 axis = p_axis.normalized();
|
|
linear_velocity -= axis * axis.dot(linear_velocity);
|
|
linear_velocity += p_axis;
|
|
PhysicsServer2D::get_singleton()->body_set_state(get_rid(), PhysicsServer2D::BODY_STATE_LINEAR_VELOCITY, linear_velocity);
|
|
}
|
|
|
|
void RigidBody2D::set_linear_velocity(const Vector2 &p_velocity) {
|
|
linear_velocity = p_velocity;
|
|
PhysicsServer2D::get_singleton()->body_set_state(get_rid(), PhysicsServer2D::BODY_STATE_LINEAR_VELOCITY, linear_velocity);
|
|
}
|
|
|
|
Vector2 RigidBody2D::get_linear_velocity() const {
|
|
return linear_velocity;
|
|
}
|
|
|
|
void RigidBody2D::set_angular_velocity(real_t p_velocity) {
|
|
angular_velocity = p_velocity;
|
|
PhysicsServer2D::get_singleton()->body_set_state(get_rid(), PhysicsServer2D::BODY_STATE_ANGULAR_VELOCITY, angular_velocity);
|
|
}
|
|
|
|
real_t RigidBody2D::get_angular_velocity() const {
|
|
return angular_velocity;
|
|
}
|
|
|
|
void RigidBody2D::set_use_custom_integrator(bool p_enable) {
|
|
if (custom_integrator == p_enable) {
|
|
return;
|
|
}
|
|
|
|
custom_integrator = p_enable;
|
|
PhysicsServer2D::get_singleton()->body_set_omit_force_integration(get_rid(), p_enable);
|
|
}
|
|
|
|
bool RigidBody2D::is_using_custom_integrator() {
|
|
return custom_integrator;
|
|
}
|
|
|
|
void RigidBody2D::set_sleeping(bool p_sleeping) {
|
|
sleeping = p_sleeping;
|
|
PhysicsServer2D::get_singleton()->body_set_state(get_rid(), PhysicsServer2D::BODY_STATE_SLEEPING, sleeping);
|
|
}
|
|
|
|
void RigidBody2D::set_can_sleep(bool p_active) {
|
|
can_sleep = p_active;
|
|
PhysicsServer2D::get_singleton()->body_set_state(get_rid(), PhysicsServer2D::BODY_STATE_CAN_SLEEP, p_active);
|
|
}
|
|
|
|
bool RigidBody2D::is_able_to_sleep() const {
|
|
return can_sleep;
|
|
}
|
|
|
|
bool RigidBody2D::is_sleeping() const {
|
|
return sleeping;
|
|
}
|
|
|
|
void RigidBody2D::set_max_contacts_reported(int p_amount) {
|
|
max_contacts_reported = p_amount;
|
|
PhysicsServer2D::get_singleton()->body_set_max_contacts_reported(get_rid(), p_amount);
|
|
}
|
|
|
|
int RigidBody2D::get_max_contacts_reported() const {
|
|
return max_contacts_reported;
|
|
}
|
|
|
|
int RigidBody2D::get_contact_count() const {
|
|
return contact_count;
|
|
}
|
|
|
|
void RigidBody2D::apply_central_impulse(const Vector2 &p_impulse) {
|
|
PhysicsServer2D::get_singleton()->body_apply_central_impulse(get_rid(), p_impulse);
|
|
}
|
|
|
|
void RigidBody2D::apply_impulse(const Vector2 &p_impulse, const Vector2 &p_position) {
|
|
PhysicsServer2D::get_singleton()->body_apply_impulse(get_rid(), p_impulse, p_position);
|
|
}
|
|
|
|
void RigidBody2D::apply_torque_impulse(real_t p_torque) {
|
|
PhysicsServer2D::get_singleton()->body_apply_torque_impulse(get_rid(), p_torque);
|
|
}
|
|
|
|
void RigidBody2D::apply_central_force(const Vector2 &p_force) {
|
|
PhysicsServer2D::get_singleton()->body_apply_central_force(get_rid(), p_force);
|
|
}
|
|
|
|
void RigidBody2D::apply_force(const Vector2 &p_force, const Vector2 &p_position) {
|
|
PhysicsServer2D::get_singleton()->body_apply_force(get_rid(), p_force, p_position);
|
|
}
|
|
|
|
void RigidBody2D::apply_torque(real_t p_torque) {
|
|
PhysicsServer2D::get_singleton()->body_apply_torque(get_rid(), p_torque);
|
|
}
|
|
|
|
void RigidBody2D::add_constant_central_force(const Vector2 &p_force) {
|
|
PhysicsServer2D::get_singleton()->body_add_constant_central_force(get_rid(), p_force);
|
|
}
|
|
|
|
void RigidBody2D::add_constant_force(const Vector2 &p_force, const Vector2 &p_position) {
|
|
PhysicsServer2D::get_singleton()->body_add_constant_force(get_rid(), p_force, p_position);
|
|
}
|
|
|
|
void RigidBody2D::add_constant_torque(const real_t p_torque) {
|
|
PhysicsServer2D::get_singleton()->body_add_constant_torque(get_rid(), p_torque);
|
|
}
|
|
|
|
void RigidBody2D::set_constant_force(const Vector2 &p_force) {
|
|
PhysicsServer2D::get_singleton()->body_set_constant_force(get_rid(), p_force);
|
|
}
|
|
|
|
Vector2 RigidBody2D::get_constant_force() const {
|
|
return PhysicsServer2D::get_singleton()->body_get_constant_force(get_rid());
|
|
}
|
|
|
|
void RigidBody2D::set_constant_torque(real_t p_torque) {
|
|
PhysicsServer2D::get_singleton()->body_set_constant_torque(get_rid(), p_torque);
|
|
}
|
|
|
|
real_t RigidBody2D::get_constant_torque() const {
|
|
return PhysicsServer2D::get_singleton()->body_get_constant_torque(get_rid());
|
|
}
|
|
|
|
void RigidBody2D::set_continuous_collision_detection_mode(CCDMode p_mode) {
|
|
ccd_mode = p_mode;
|
|
PhysicsServer2D::get_singleton()->body_set_continuous_collision_detection_mode(get_rid(), PhysicsServer2D::CCDMode(p_mode));
|
|
}
|
|
|
|
RigidBody2D::CCDMode RigidBody2D::get_continuous_collision_detection_mode() const {
|
|
return ccd_mode;
|
|
}
|
|
|
|
TypedArray<Node2D> RigidBody2D::get_colliding_bodies() const {
|
|
ERR_FAIL_NULL_V(contact_monitor, TypedArray<Node2D>());
|
|
|
|
TypedArray<Node2D> ret;
|
|
ret.resize(contact_monitor->body_map.size());
|
|
int idx = 0;
|
|
for (const KeyValue<ObjectID, BodyState> &E : contact_monitor->body_map) {
|
|
Object *obj = ObjectDB::get_instance(E.key);
|
|
if (!obj) {
|
|
ret.resize(ret.size() - 1); //ops
|
|
} else {
|
|
ret[idx++] = obj;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void RigidBody2D::set_contact_monitor(bool p_enabled) {
|
|
if (p_enabled == is_contact_monitor_enabled()) {
|
|
return;
|
|
}
|
|
|
|
if (!p_enabled) {
|
|
ERR_FAIL_COND_MSG(contact_monitor->locked, "Can't disable contact monitoring during in/out callback. Use call_deferred(\"set_contact_monitor\", false) instead.");
|
|
|
|
for (const KeyValue<ObjectID, BodyState> &E : contact_monitor->body_map) {
|
|
//clean up mess
|
|
Object *obj = ObjectDB::get_instance(E.key);
|
|
Node *node = Object::cast_to<Node>(obj);
|
|
|
|
if (node) {
|
|
node->disconnect(SceneStringNames::get_singleton()->tree_entered, callable_mp(this, &RigidBody2D::_body_enter_tree));
|
|
node->disconnect(SceneStringNames::get_singleton()->tree_exiting, callable_mp(this, &RigidBody2D::_body_exit_tree));
|
|
}
|
|
}
|
|
|
|
memdelete(contact_monitor);
|
|
contact_monitor = nullptr;
|
|
} else {
|
|
contact_monitor = memnew(ContactMonitor);
|
|
contact_monitor->locked = false;
|
|
}
|
|
|
|
notify_property_list_changed();
|
|
}
|
|
|
|
bool RigidBody2D::is_contact_monitor_enabled() const {
|
|
return contact_monitor != nullptr;
|
|
}
|
|
|
|
void RigidBody2D::_notification(int p_what) {
|
|
#ifdef TOOLS_ENABLED
|
|
switch (p_what) {
|
|
case NOTIFICATION_ENTER_TREE: {
|
|
if (Engine::get_singleton()->is_editor_hint()) {
|
|
set_notify_local_transform(true); // Used for warnings and only in editor.
|
|
}
|
|
} break;
|
|
|
|
case NOTIFICATION_LOCAL_TRANSFORM_CHANGED: {
|
|
update_configuration_warnings();
|
|
} break;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
PackedStringArray RigidBody2D::get_configuration_warnings() const {
|
|
Transform2D t = get_transform();
|
|
|
|
PackedStringArray warnings = CollisionObject2D::get_configuration_warnings();
|
|
|
|
if (ABS(t.columns[0].length() - 1.0) > 0.05 || ABS(t.columns[1].length() - 1.0) > 0.05) {
|
|
warnings.push_back(RTR("Size changes to RigidBody2D will be overridden by the physics engine when running.\nChange the size in children collision shapes instead."));
|
|
}
|
|
|
|
return warnings;
|
|
}
|
|
|
|
void RigidBody2D::_bind_methods() {
|
|
ClassDB::bind_method(D_METHOD("set_mass", "mass"), &RigidBody2D::set_mass);
|
|
ClassDB::bind_method(D_METHOD("get_mass"), &RigidBody2D::get_mass);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_inertia"), &RigidBody2D::get_inertia);
|
|
ClassDB::bind_method(D_METHOD("set_inertia", "inertia"), &RigidBody2D::set_inertia);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_center_of_mass_mode", "mode"), &RigidBody2D::set_center_of_mass_mode);
|
|
ClassDB::bind_method(D_METHOD("get_center_of_mass_mode"), &RigidBody2D::get_center_of_mass_mode);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_center_of_mass", "center_of_mass"), &RigidBody2D::set_center_of_mass);
|
|
ClassDB::bind_method(D_METHOD("get_center_of_mass"), &RigidBody2D::get_center_of_mass);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_physics_material_override", "physics_material_override"), &RigidBody2D::set_physics_material_override);
|
|
ClassDB::bind_method(D_METHOD("get_physics_material_override"), &RigidBody2D::get_physics_material_override);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_gravity_scale", "gravity_scale"), &RigidBody2D::set_gravity_scale);
|
|
ClassDB::bind_method(D_METHOD("get_gravity_scale"), &RigidBody2D::get_gravity_scale);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_linear_damp_mode", "linear_damp_mode"), &RigidBody2D::set_linear_damp_mode);
|
|
ClassDB::bind_method(D_METHOD("get_linear_damp_mode"), &RigidBody2D::get_linear_damp_mode);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_angular_damp_mode", "angular_damp_mode"), &RigidBody2D::set_angular_damp_mode);
|
|
ClassDB::bind_method(D_METHOD("get_angular_damp_mode"), &RigidBody2D::get_angular_damp_mode);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_linear_damp", "linear_damp"), &RigidBody2D::set_linear_damp);
|
|
ClassDB::bind_method(D_METHOD("get_linear_damp"), &RigidBody2D::get_linear_damp);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_angular_damp", "angular_damp"), &RigidBody2D::set_angular_damp);
|
|
ClassDB::bind_method(D_METHOD("get_angular_damp"), &RigidBody2D::get_angular_damp);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_linear_velocity", "linear_velocity"), &RigidBody2D::set_linear_velocity);
|
|
ClassDB::bind_method(D_METHOD("get_linear_velocity"), &RigidBody2D::get_linear_velocity);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_angular_velocity", "angular_velocity"), &RigidBody2D::set_angular_velocity);
|
|
ClassDB::bind_method(D_METHOD("get_angular_velocity"), &RigidBody2D::get_angular_velocity);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_max_contacts_reported", "amount"), &RigidBody2D::set_max_contacts_reported);
|
|
ClassDB::bind_method(D_METHOD("get_max_contacts_reported"), &RigidBody2D::get_max_contacts_reported);
|
|
ClassDB::bind_method(D_METHOD("get_contact_count"), &RigidBody2D::get_contact_count);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_use_custom_integrator", "enable"), &RigidBody2D::set_use_custom_integrator);
|
|
ClassDB::bind_method(D_METHOD("is_using_custom_integrator"), &RigidBody2D::is_using_custom_integrator);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_contact_monitor", "enabled"), &RigidBody2D::set_contact_monitor);
|
|
ClassDB::bind_method(D_METHOD("is_contact_monitor_enabled"), &RigidBody2D::is_contact_monitor_enabled);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_continuous_collision_detection_mode", "mode"), &RigidBody2D::set_continuous_collision_detection_mode);
|
|
ClassDB::bind_method(D_METHOD("get_continuous_collision_detection_mode"), &RigidBody2D::get_continuous_collision_detection_mode);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_axis_velocity", "axis_velocity"), &RigidBody2D::set_axis_velocity);
|
|
ClassDB::bind_method(D_METHOD("apply_central_impulse", "impulse"), &RigidBody2D::apply_central_impulse, Vector2());
|
|
ClassDB::bind_method(D_METHOD("apply_impulse", "impulse", "position"), &RigidBody2D::apply_impulse, Vector2());
|
|
ClassDB::bind_method(D_METHOD("apply_torque_impulse", "torque"), &RigidBody2D::apply_torque_impulse);
|
|
|
|
ClassDB::bind_method(D_METHOD("apply_central_force", "force"), &RigidBody2D::apply_central_force);
|
|
ClassDB::bind_method(D_METHOD("apply_force", "force", "position"), &RigidBody2D::apply_force, Vector2());
|
|
ClassDB::bind_method(D_METHOD("apply_torque", "torque"), &RigidBody2D::apply_torque);
|
|
|
|
ClassDB::bind_method(D_METHOD("add_constant_central_force", "force"), &RigidBody2D::add_constant_central_force);
|
|
ClassDB::bind_method(D_METHOD("add_constant_force", "force", "position"), &RigidBody2D::add_constant_force, Vector2());
|
|
ClassDB::bind_method(D_METHOD("add_constant_torque", "torque"), &RigidBody2D::add_constant_torque);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_constant_force", "force"), &RigidBody2D::set_constant_force);
|
|
ClassDB::bind_method(D_METHOD("get_constant_force"), &RigidBody2D::get_constant_force);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_constant_torque", "torque"), &RigidBody2D::set_constant_torque);
|
|
ClassDB::bind_method(D_METHOD("get_constant_torque"), &RigidBody2D::get_constant_torque);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_sleeping", "sleeping"), &RigidBody2D::set_sleeping);
|
|
ClassDB::bind_method(D_METHOD("is_sleeping"), &RigidBody2D::is_sleeping);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_can_sleep", "able_to_sleep"), &RigidBody2D::set_can_sleep);
|
|
ClassDB::bind_method(D_METHOD("is_able_to_sleep"), &RigidBody2D::is_able_to_sleep);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_lock_rotation_enabled", "lock_rotation"), &RigidBody2D::set_lock_rotation_enabled);
|
|
ClassDB::bind_method(D_METHOD("is_lock_rotation_enabled"), &RigidBody2D::is_lock_rotation_enabled);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_freeze_enabled", "freeze_mode"), &RigidBody2D::set_freeze_enabled);
|
|
ClassDB::bind_method(D_METHOD("is_freeze_enabled"), &RigidBody2D::is_freeze_enabled);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_freeze_mode", "freeze_mode"), &RigidBody2D::set_freeze_mode);
|
|
ClassDB::bind_method(D_METHOD("get_freeze_mode"), &RigidBody2D::get_freeze_mode);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_colliding_bodies"), &RigidBody2D::get_colliding_bodies);
|
|
|
|
GDVIRTUAL_BIND(_integrate_forces, "state");
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "mass", PROPERTY_HINT_RANGE, "0.01,1000,0.01,or_greater,exp,suffix:kg"), "set_mass", "get_mass");
|
|
ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "physics_material_override", PROPERTY_HINT_RESOURCE_TYPE, "PhysicsMaterial"), "set_physics_material_override", "get_physics_material_override");
|
|
ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "gravity_scale", PROPERTY_HINT_RANGE, "-8,8,0.001,or_less,or_greater"), "set_gravity_scale", "get_gravity_scale");
|
|
ADD_GROUP("Mass Distribution", "");
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "center_of_mass_mode", PROPERTY_HINT_ENUM, "Auto,Custom"), "set_center_of_mass_mode", "get_center_of_mass_mode");
|
|
ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "center_of_mass", PROPERTY_HINT_RANGE, "-1000,1000,0.01,or_less,or_greater,suffix:px"), "set_center_of_mass", "get_center_of_mass");
|
|
ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "inertia", PROPERTY_HINT_RANGE, U"0,1000,0.01,or_greater,exp,suffix:kg\u22C5px\u00B2"), "set_inertia", "get_inertia");
|
|
ADD_GROUP("Deactivation", "");
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "sleeping"), "set_sleeping", "is_sleeping");
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "can_sleep"), "set_can_sleep", "is_able_to_sleep");
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "lock_rotation"), "set_lock_rotation_enabled", "is_lock_rotation_enabled");
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "freeze"), "set_freeze_enabled", "is_freeze_enabled");
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "freeze_mode", PROPERTY_HINT_ENUM, "Static,Kinematic"), "set_freeze_mode", "get_freeze_mode");
|
|
ADD_GROUP("Solver", "");
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "custom_integrator"), "set_use_custom_integrator", "is_using_custom_integrator");
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "continuous_cd", PROPERTY_HINT_ENUM, "Disabled,Cast Ray,Cast Shape"), "set_continuous_collision_detection_mode", "get_continuous_collision_detection_mode");
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "contact_monitor"), "set_contact_monitor", "is_contact_monitor_enabled");
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "max_contacts_reported", PROPERTY_HINT_RANGE, "0,64,1,or_greater"), "set_max_contacts_reported", "get_max_contacts_reported");
|
|
ADD_GROUP("Linear", "linear_");
|
|
ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "linear_velocity", PROPERTY_HINT_NONE, "suffix:px/s"), "set_linear_velocity", "get_linear_velocity");
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "linear_damp_mode", PROPERTY_HINT_ENUM, "Combine,Replace"), "set_linear_damp_mode", "get_linear_damp_mode");
|
|
ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "linear_damp", PROPERTY_HINT_RANGE, "-1,100,0.001,or_greater"), "set_linear_damp", "get_linear_damp");
|
|
ADD_GROUP("Angular", "angular_");
|
|
ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "angular_velocity", PROPERTY_HINT_NONE, U"radians_as_degrees,suffix:\u00B0/s"), "set_angular_velocity", "get_angular_velocity");
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "angular_damp_mode", PROPERTY_HINT_ENUM, "Combine,Replace"), "set_angular_damp_mode", "get_angular_damp_mode");
|
|
ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "angular_damp", PROPERTY_HINT_RANGE, "-1,100,0.001,or_greater"), "set_angular_damp", "get_angular_damp");
|
|
ADD_GROUP("Constant Forces", "constant_");
|
|
ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "constant_force", PROPERTY_HINT_NONE, U"suffix:kg\u22C5px/s\u00B2"), "set_constant_force", "get_constant_force");
|
|
ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "constant_torque", PROPERTY_HINT_NONE, U"suffix:kg\u22C5px\u00B2/s\u00B2/rad"), "set_constant_torque", "get_constant_torque");
|
|
|
|
ADD_SIGNAL(MethodInfo("body_shape_entered", PropertyInfo(Variant::RID, "body_rid"), PropertyInfo(Variant::OBJECT, "body", PROPERTY_HINT_RESOURCE_TYPE, "Node"), PropertyInfo(Variant::INT, "body_shape_index"), PropertyInfo(Variant::INT, "local_shape_index")));
|
|
ADD_SIGNAL(MethodInfo("body_shape_exited", PropertyInfo(Variant::RID, "body_rid"), PropertyInfo(Variant::OBJECT, "body", PROPERTY_HINT_RESOURCE_TYPE, "Node"), PropertyInfo(Variant::INT, "body_shape_index"), PropertyInfo(Variant::INT, "local_shape_index")));
|
|
ADD_SIGNAL(MethodInfo("body_entered", PropertyInfo(Variant::OBJECT, "body", PROPERTY_HINT_RESOURCE_TYPE, "Node")));
|
|
ADD_SIGNAL(MethodInfo("body_exited", PropertyInfo(Variant::OBJECT, "body", PROPERTY_HINT_RESOURCE_TYPE, "Node")));
|
|
ADD_SIGNAL(MethodInfo("sleeping_state_changed"));
|
|
|
|
BIND_ENUM_CONSTANT(FREEZE_MODE_STATIC);
|
|
BIND_ENUM_CONSTANT(FREEZE_MODE_KINEMATIC);
|
|
|
|
BIND_ENUM_CONSTANT(CENTER_OF_MASS_MODE_AUTO);
|
|
BIND_ENUM_CONSTANT(CENTER_OF_MASS_MODE_CUSTOM);
|
|
|
|
BIND_ENUM_CONSTANT(DAMP_MODE_COMBINE);
|
|
BIND_ENUM_CONSTANT(DAMP_MODE_REPLACE);
|
|
|
|
BIND_ENUM_CONSTANT(CCD_MODE_DISABLED);
|
|
BIND_ENUM_CONSTANT(CCD_MODE_CAST_RAY);
|
|
BIND_ENUM_CONSTANT(CCD_MODE_CAST_SHAPE);
|
|
}
|
|
|
|
void RigidBody2D::_validate_property(PropertyInfo &p_property) const {
|
|
if (center_of_mass_mode != CENTER_OF_MASS_MODE_CUSTOM && p_property.name == "center_of_mass") {
|
|
p_property.usage = PROPERTY_USAGE_NO_EDITOR;
|
|
}
|
|
|
|
if (!contact_monitor && p_property.name == "max_contacts_reported") {
|
|
p_property.usage = PROPERTY_USAGE_NO_EDITOR;
|
|
}
|
|
}
|
|
|
|
RigidBody2D::RigidBody2D() :
|
|
PhysicsBody2D(PhysicsServer2D::BODY_MODE_RIGID) {
|
|
PhysicsServer2D::get_singleton()->body_set_state_sync_callback(get_rid(), callable_mp(this, &RigidBody2D::_body_state_changed));
|
|
}
|
|
|
|
RigidBody2D::~RigidBody2D() {
|
|
if (contact_monitor) {
|
|
memdelete(contact_monitor);
|
|
}
|
|
}
|
|
|
|
void RigidBody2D::_reload_physics_characteristics() {
|
|
if (physics_material_override.is_null()) {
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_BOUNCE, 0);
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_FRICTION, 1);
|
|
} else {
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_BOUNCE, physics_material_override->computed_bounce());
|
|
PhysicsServer2D::get_singleton()->body_set_param(get_rid(), PhysicsServer2D::BODY_PARAM_FRICTION, physics_material_override->computed_friction());
|
|
}
|
|
}
|