100 lines
2.8 KiB
GDScript

class_name FootstepsComponent
extends Node3D
@export var skeleton: Skeleton3D
@export var movement: MovementComponent
@export var footsteps: AudioStreamRandomizer
@export var step_threshold := 0.1
@export var ik_step_threshold := 0.2
@export var ik_speed := 5.0
const feet_names := ["LeftFoot", "RightFoot"]
var feet: Array[Foot] = []
var ik_time: float
func _ready():
if owner != Global.player:
set_enabled(false)
for foot_name in feet_names:
var foot = Foot.new()
foot.bone_name = foot_name
foot.bone_id = skeleton.find_bone(foot_name)
foot.audio = get_node(foot_name)
foot.position = skeleton.to_global(skeleton.get_bone_global_pose(foot.bone_id).origin)
foot.floor_distance = INF
foot.normal = Vector3.UP
foot.attachment = BoneAttachment3D.new()
foot.attachment.bone_name = foot_name
foot.attachment.bone_idx = foot.bone_id
foot.audio.reparent(foot.attachment)
foot.audio.position = Vector3.ZERO
skeleton.add_child(foot.attachment)
feet.append(foot)
func _process(delta: float):
for foot in feet:
var old_position := foot.position
var global_pose := skeleton.get_bone_global_pose(foot.bone_id)
var world_pose := skeleton.global_transform * global_pose
foot.position = world_pose.origin
if foot.floor_distance < ik_step_threshold:
ik_time = minf(ik_time + ik_speed * delta, 1.0)
align_surface_normal(foot, world_pose)
else:
ik_time = 0
if foot.floor_distance > step_threshold:
continue
foot.velocity = (foot.position - old_position) / delta
if foot.velocity.y > 0:
continue
if foot.velocity.length_squared() < 1.0:
continue
if foot.audio.playing && foot.audio.get_playback_position() < 0.3:
continue
play(foot.audio)
func _physics_process(_delta: float):
var space_state = get_world_3d().direct_space_state
for foot in feet:
var from := foot.position
var to := foot.position + Vector3(0, -10, 0)
var query = PhysicsRayQueryParameters3D.create(from, to, 1)
var result = space_state.intersect_ray(query)
if result:
foot.floor_distance = (foot.position - result.position).length()
foot.normal = result.normal
else:
foot.floor_distance = INF
foot.normal = Vector3.UP
func play(audio_player: AudioStreamPlayer3D):
audio_player.stream = footsteps
audio_player.play()
func align_surface_normal(foot: Foot, world_pose: Transform3D):
var pose := skeleton.get_bone_pose(foot.bone_id)
var local_normal := world_pose.basis.transposed() * foot.normal
var new_basis := get_foot_rotation(pose.basis, local_normal)
skeleton.set_bone_pose_rotation(foot.bone_id, pose.basis.slerp(new_basis, ik_time))
func get_foot_rotation(old: Basis, normal: Vector3) -> Basis:
var new := Basis(-old.x, normal, old.z)
new = new.orthonormalized()
new = new.rotated(Vector3.UP, PI)
return new
func set_enabled(enabled: bool):
set_process(enabled)
set_physics_process(enabled)