xrjamfeb2025/addons/godot-xr-tools/objects/force_body/force_body.gd

182 lines
4.9 KiB
GDScript

@tool
class_name XRToolsForceBody
extends AnimatableBody3D
## XRTools Force Body script
##
## This script enhances AnimatableBody3D with move_and_slide and the ability
## to push bodies by emparting forces on them.
## Force Body Collision
class ForceBodyCollision:
## Collider object
var collider : Node3D
## Collision point
var position : Vector3
## Collision normal
var normal : Vector3
## Enables or disables pushing bodies
@export var push_bodies : bool = true
## Control the stiffness of the body
@export var stiffness : float = 10.0
## Control the maximum push force
@export var maximum_force : float = 1.0
## Maximum slides
@export var max_slides : int = 4
## Add support for is_xr_class on XRTools classes
func is_xr_class(name : String) -> bool:
return name == "XRToolsForceBody"
## This function moves and slides along the [param move] vector. It returns
## information about the last collision, or null if no collision
func move_and_slide(move : Vector3) -> ForceBodyCollision:
# Make sure this is off or weird shit happens...
sync_to_physics = false
# Loop performing the movement steps
var step_move := move
var ret : ForceBodyCollision = null
for step in max_slides:
# Take the next step
var collision := move_and_collide(step_move)
# If we didn't collide with anything then we have finished the entire
# move_and_slide operation
if not collision:
break
# Save relevant collision information
var collider := collision.get_collider()
var postion := collision.get_position()
var normal := collision.get_normal()
# Save the collision information
if not ret:
ret = ForceBodyCollision.new()
ret.collider = collider
ret.position = postion
ret.normal = normal
# Calculate the next move
var next_move := collision.get_remainder().slide(normal)
# Handle pushing bodies
if push_bodies:
var body := collider as RigidBody3D
if body:
# Calculate the momentum lost by the collision
var lost_momentum := step_move - next_move
# TODO: We should consider the velocity of the body such that
# we never push it away faster than our own velocity.
# Apply the lost momentum as an impulse to the body we hit
body.apply_impulse(
(lost_momentum * stiffness).limit_length(maximum_force),
position - body.global_position)
# Update the remaining movement
step_move = next_move
# Prevent bouncing back along movement path
if next_move.dot(move) <= 0:
break
# Return the last collision data
return ret
## Attempts to rotate our object until it collides
func rotate_and_collide( \
target_global_basis : Basis, \
step_angle : float = deg_to_rad(5.0) \
) -> ForceBodyCollision:
# Make sure this is off or weird shit happens...
sync_to_physics = false
var ret : ForceBodyCollision = null
var space = PhysicsServer3D.body_get_space(get_rid())
var direct_state = PhysicsServer3D.space_get_direct_state(space)
# We don't seem to have a rotational movement query for collisions,
# so best we can do is to rotate in steps and test
var from_quat : Quaternion = Quaternion(global_basis)
var to_quat : Quaternion = Quaternion(target_global_basis)
var angle : float = from_quat.angle_to(to_quat)
var steps : float = ceil(angle / step_angle)
# Convert collision exceptions to a RID array
var exception_rids : Array[RID]
for collision_exception in get_collision_exceptions():
exception_rids.push_back(collision_exception.get_rid())
# Prevent collisions with ourselves
exception_rids.push_back(get_rid())
# Find our shape ids
var shape_rids : Array[RID]
for node in get_children(true):
if node is CollisionShape3D:
var col_shape : CollisionShape3D = node
if not col_shape.disabled:
shape_rids.push_back(col_shape.shape.get_rid())
# Our physics query
var query : PhysicsShapeQueryParameters3D = PhysicsShapeQueryParameters3D.new()
query.collide_with_areas = false
query.collide_with_bodies = true
query.collision_mask = collision_mask
query.exclude = exception_rids
# Check our collisions
var step : float = 0.0
var new_quat : Quaternion = from_quat
var t = global_transform
while step < steps and not ret:
step += 1.0
var test_quat : Quaternion = from_quat.slerp(to_quat, step / steps)
t.basis = Basis(test_quat)
query.transform = t
for rid in shape_rids:
query.shape_rid = rid
var collision = direct_state.get_rest_info(query)
if not collision.is_empty():
ret = ForceBodyCollision.new()
ret.collider = instance_from_id(collision["collider_id"])
ret.position = collision["point"]
ret.normal = collision["normal"]
# TODO May need to see about applying a rotational force
# if pushbodies is true
break
if not ret:
# No collision, we can rotate this far!
new_quat = test_quat
# Update our rotation to our last successful rotation
global_basis = Basis(new_quat)
# Return the last collision data
return ret
func _ready():
process_physics_priority = -90