killbox/godot/scenes/characters/character_wrapper.gd

101 lines
4.1 KiB
GDScript3
Raw Normal View History

2025-02-09 19:14:24 +00:00
extends CharacterBody3D
2025-02-06 08:56:39 +00:00
class_name CharacterWrapper
@export var die_script: GDScript = null
2025-02-09 19:14:24 +00:00
@export var owner_placeholder: CharacterBody3D = null
2025-02-09 19:28:44 +00:00
@export var interpolation_speed: float = 50.0 # How quickly the model corrects position
@export var velocity_influence: float = 1.0 # How much velocity is used for smoothing
2025-02-09 21:21:42 +00:00
@export var snap_threshold: float = 2.0 # If desync is larger than this, snap faster
2025-02-06 08:56:39 +00:00
2025-02-09 21:25:10 +00:00
@export var interpolation_delay: float = 0.1 # Stay 100ms behind for smooth interpolation
2025-02-09 21:43:37 +00:00
2025-02-09 21:25:10 +00:00
var position_buffer = [] # Stores past positions (tuples of (timestamp, position))
2025-02-09 21:43:37 +00:00
var previous_position: Vector3 = Vector3.ZERO
var pseudo_velocity: Vector3 = Vector3.ZERO # Approximate velocity without CharacterBody3D
var previous_server_pos: Vector3 = Vector3.ZERO
2025-02-09 21:25:10 +00:00
func _process(delta):
# Record the latest server position with timestamp
2025-02-09 21:48:05 +00:00
global_rotation = owner_placeholder.global_rotation
2025-02-09 22:05:59 +00:00
# -- This code is 100% generated by chatGPT and I can't understand it fully yet
2025-02-09 21:43:37 +00:00
2025-02-09 22:05:59 +00:00
var position_difference := global_position - owner_placeholder.global_position
if is_vector_a_lower_than_b(position_difference, Vector3(0.3, 0.3, 0.3)) and \
is_vector_a_lower_than_b(Vector3(-0.3, -0.3, -0.3), position_difference):
var server_pos = owner_placeholder.global_transform.origin
pseudo_velocity = (server_pos - previous_position) / delta
previous_position = server_pos # Store for next frame
position_buffer.append([Time.get_ticks_msec() / 1000.0, server_pos])
while position_buffer.size() > 2 and position_buffer[1][0] < (Time.get_ticks_msec() / 1000.0) - interpolation_delay:
position_buffer.pop_front()
if position_buffer.size() >= 2:
var t_now = (Time.get_ticks_msec() / 1000.0) - interpolation_delay
var prev_point = position_buffer[0]
var next_point = position_buffer[1]
2025-02-09 21:43:37 +00:00
2025-02-09 22:05:59 +00:00
var alpha = (t_now - prev_point[0]) / (next_point[0] - prev_point[0])
alpha = clamp(alpha, 0.0, 1.0)
global_transform.origin = prev_point[1].lerp(next_point[1], alpha)
else:
global_position = owner_placeholder.global_position
2025-02-09 21:43:37 +00:00
2025-02-06 08:56:39 +00:00
func _ready() -> void:
2025-02-09 19:14:24 +00:00
set_multiplayer_authority(multiplayer.get_unique_id())
global_position = owner_placeholder.global_position
2025-02-06 08:56:39 +00:00
2025-02-09 21:25:10 +00:00
#func _physics_process(delta: float) -> void:
##if not is_on_floor():
##velocity += get_gravity() * delta
##global_position = owner_placeholder.global_position
#
#var server_pos = owner_placeholder.global_transform.origin
#var server_velocity = (server_pos - previous_server_pos) / delta
#
#if server_velocity.length() < 0.1:
#velocity = Vector3.ZERO # Stop predicting movement when nearly still
#var distance_to_server = global_transform.origin.distance_to(server_pos)
#if distance_to_server > snap_threshold:
## If the desync is too large, quickly correct it
#global_transform.origin = server_pos
#else:
## Otherwise, smoothly adjust position
#global_transform.origin = global_transform.origin.lerp(server_pos, interpolation_speed * delta)
#
2025-02-09 21:21:42 +00:00
#global_transform.origin += velocity * velocity_influence * delta
2025-02-09 21:25:10 +00:00
## Update stored velocity
2025-02-09 21:21:42 +00:00
#velocity = velocity.lerp(server_velocity, delta * interpolation_speed)
2025-02-09 21:25:10 +00:00
#previous_server_pos = server_pos
##velocity = owner_placeholder.velocity
##global_transform.origin = global_transform.origin.lerp(server_pos, interpolation_speed * delta)
##global_transform.origin += velocity * velocity_influence * delta
##velocity = velocity.lerp(server_velocity, delta * interpolation_speed)
2025-02-09 19:28:44 +00:00
#move_and_slide()
2025-02-09 19:14:24 +00:00
2025-02-06 08:56:39 +00:00
# Set the owner placeholder, so the characters can send the requests to a node
# it depends on
2025-02-09 19:14:24 +00:00
func set_owner_placeholder(owner_placeholder: Node3D):
owner_placeholder = owner_placeholder
2025-02-06 08:56:39 +00:00
2025-02-08 18:30:32 +00:00
func die():
2025-02-06 08:56:39 +00:00
push_warning("TODO: Implement ragdoll kind of dying and respawn character as an object")
queue_free()
func _on_area_body_part_hit(damage: int) -> void:
# The owner node should be aware of how to take damage, so we need to
# pass the value.
if owner_placeholder:
if owner_placeholder.has_method("take_damage"):
owner_placeholder.take_damage(damage)
else:
push_warning("Node doesn't know how to take the damage")
2025-02-09 22:05:59 +00:00
func is_vector_a_lower_than_b(vec_a: Vector3, vec_b: Vector3) -> bool:
return vec_a.x < vec_b.x and vec_a.y < vec_b.y and vec_a.z < vec_b.z