2024-07-20 09:49:49 +02:00
|
|
|
class_name Player
|
2024-07-09 22:33:38 +02:00
|
|
|
extends CharacterBody3D
|
|
|
|
|
2024-09-10 22:26:26 +02:00
|
|
|
const SPEED = 3.0
|
2024-07-09 22:33:38 +02:00
|
|
|
const JUMP_VELOCITY = 2.5
|
|
|
|
|
2024-09-08 16:57:39 +02:00
|
|
|
@onready var geometry:Node3D = %Geometry
|
2024-07-10 22:52:28 +02:00
|
|
|
@onready var actionable_detector = %ActionableDetector
|
2024-09-08 16:57:39 +02:00
|
|
|
@onready var animation_tree:AnimationTree = $Geometry/Rogue/AnimationTree
|
2024-09-10 22:26:26 +02:00
|
|
|
@onready var build_location:Node3D = %BuildLocation
|
2024-09-27 11:46:31 +02:00
|
|
|
@onready var right_hand_attachement:Node3D = %RightHandAttachement
|
2024-07-10 22:52:28 +02:00
|
|
|
|
2024-07-09 22:33:38 +02:00
|
|
|
# Get the gravity from the project settings to be synced with RigidBody nodes.
|
|
|
|
var gravity = ProjectSettings.get_setting("physics/3d/default_gravity")
|
|
|
|
var inventory:Inventory = Inventory.new()
|
2024-09-14 12:45:42 +02:00
|
|
|
var selected_tool_slot_index:int = 0
|
2024-09-27 18:09:30 +02:00
|
|
|
var current_tool:Item = null
|
2024-07-09 22:33:38 +02:00
|
|
|
|
2024-09-08 16:57:39 +02:00
|
|
|
var look_direction:Vector3 = Vector3.BACK
|
2024-09-08 21:11:02 +02:00
|
|
|
var look_direction_damper:SpringDamper = SpringDamper.new(Vector3.ZERO)
|
2024-07-10 22:52:28 +02:00
|
|
|
|
2024-07-09 22:33:38 +02:00
|
|
|
signal trigger_message(message:String)
|
|
|
|
|
2024-09-08 16:57:39 +02:00
|
|
|
var interaction_state:bool = false
|
|
|
|
|
|
|
|
func _process(_delta):
|
2024-09-30 18:02:20 +02:00
|
|
|
animation_tree.set("parameters/conditions/interact", interaction_state)
|
2024-09-08 16:57:39 +02:00
|
|
|
|
|
|
|
if interaction_state:
|
|
|
|
interaction_state = false
|
|
|
|
|
2024-07-09 22:33:38 +02:00
|
|
|
func _physics_process(delta):
|
|
|
|
# Add the gravity.
|
|
|
|
if not is_on_floor():
|
|
|
|
velocity.y -= gravity * delta
|
|
|
|
|
|
|
|
# Get the input direction and handle the movement/deceleration.
|
|
|
|
# As good practice, you should replace UI actions with custom gameplay actions.
|
|
|
|
var input_dir = Input.get_vector("walk_left", "walk_right", "walk_forward", "walk_back")
|
|
|
|
var direction = (transform.basis * Vector3(input_dir.x, 0, input_dir.y)).normalized()
|
|
|
|
if direction:
|
|
|
|
velocity.x = direction.x * SPEED
|
|
|
|
velocity.z = direction.z * SPEED
|
|
|
|
else:
|
|
|
|
velocity.x = move_toward(velocity.x, 0, SPEED)
|
|
|
|
velocity.z = move_toward(velocity.z, 0, SPEED)
|
|
|
|
|
|
|
|
move_and_slide()
|
2024-07-10 22:52:28 +02:00
|
|
|
|
2024-07-12 11:59:33 +02:00
|
|
|
var ground_velocity:Vector2 = Vector2(velocity.x, velocity.z)
|
2024-09-08 16:57:39 +02:00
|
|
|
|
|
|
|
var is_moving:bool = ground_velocity.length_squared() > 0.1 * 0.1
|
2024-07-12 11:59:33 +02:00
|
|
|
|
2024-09-08 16:57:39 +02:00
|
|
|
if is_moving:
|
|
|
|
look_direction = Vector3(ground_velocity.x, 0, ground_velocity.y).normalized()
|
|
|
|
elif direction.length_squared() > 0.1 * 0.1:
|
|
|
|
look_direction = direction
|
2024-07-12 11:27:10 +02:00
|
|
|
|
2024-09-08 21:11:02 +02:00
|
|
|
var damped_look_direction = look_direction_damper.calc(geometry.global_basis.z, look_direction, delta)
|
|
|
|
|
2024-09-08 16:57:39 +02:00
|
|
|
animation_tree.set("parameters/conditions/running", is_moving)
|
|
|
|
animation_tree.set("parameters/conditions/idle", not is_moving)
|
2024-09-08 21:11:02 +02:00
|
|
|
geometry.look_at(position - damped_look_direction, Vector3.UP)
|
2024-07-12 14:32:33 +02:00
|
|
|
|
2024-07-09 22:33:38 +02:00
|
|
|
|
|
|
|
func on_item_picked_up(item:Item):
|
|
|
|
emit_signal("trigger_message", "Picked up a " + item.name)
|
2024-07-12 14:32:33 +02:00
|
|
|
inventory.add_item(item)
|
2024-07-10 22:52:28 +02:00
|
|
|
|
|
|
|
|
2024-09-08 22:06:34 +02:00
|
|
|
func get_actionable_global_transform() -> Vector3:
|
2024-09-10 22:26:26 +02:00
|
|
|
return build_location.global_position
|
2024-09-08 22:06:34 +02:00
|
|
|
|
2024-09-14 12:45:42 +02:00
|
|
|
|
2024-09-27 11:46:31 +02:00
|
|
|
func set_right_hand_item(item:Item) -> void:
|
|
|
|
for child in right_hand_attachement.get_children():
|
|
|
|
child.queue_free()
|
2024-09-27 18:09:30 +02:00
|
|
|
|
|
|
|
current_tool = null
|
2024-09-27 11:46:31 +02:00
|
|
|
|
|
|
|
if item != null and item.is_tool:
|
|
|
|
right_hand_attachement.add_child(item.scene.instantiate())
|
2024-09-27 18:09:30 +02:00
|
|
|
current_tool = item
|
2024-09-27 11:46:31 +02:00
|
|
|
|
2024-09-30 18:02:20 +02:00
|
|
|
animation_tree.set("parameters/Interact/conditions/tool_action_slice", false)
|
|
|
|
animation_tree.set("parameters/Interact/conditions/tool_action_chop", false)
|
|
|
|
|
|
|
|
match item.tool_action:
|
|
|
|
1:
|
|
|
|
animation_tree.set("parameters/Interact/conditions/tool_action_chop", true)
|
|
|
|
2:
|
|
|
|
animation_tree.set("parameters/Interact/conditions/tool_action_slice", true)
|
|
|
|
|
2024-09-27 18:09:30 +02:00
|
|
|
func has_build_tool_active() -> bool:
|
|
|
|
if current_tool and current_tool.name == "Pickaxe":
|
|
|
|
return true
|
|
|
|
|
|
|
|
return false
|
2024-09-27 11:46:31 +02:00
|
|
|
|
2024-09-27 18:09:30 +02:00
|
|
|
func handle_tool_slot_input_events(_event:InputEvent) -> bool:
|
2024-09-14 12:45:42 +02:00
|
|
|
var key_event:InputEventKey = _event as InputEventKey
|
|
|
|
|
|
|
|
if key_event and key_event.is_released():
|
|
|
|
if key_event.keycode >= KEY_1 and key_event.keycode <= KEY_9:
|
|
|
|
selected_tool_slot_index = key_event.keycode - KEY_1
|
2024-09-27 11:46:31 +02:00
|
|
|
set_right_hand_item(inventory.get_tool_item_stacks()[selected_tool_slot_index].item)
|
2024-09-14 12:45:42 +02:00
|
|
|
return true
|
|
|
|
|
|
|
|
if _event.is_action_released("item_next"):
|
|
|
|
selected_tool_slot_index = (selected_tool_slot_index + 1) % (inventory.get_tool_item_stacks().size())
|
2024-09-27 11:46:31 +02:00
|
|
|
set_right_hand_item(inventory.get_tool_item_stacks()[selected_tool_slot_index].item)
|
2024-09-14 12:45:42 +02:00
|
|
|
return true
|
|
|
|
|
|
|
|
if _event.is_action_released("item_prev"):
|
|
|
|
var tool_slot_size = inventory.get_tool_item_stacks().size()
|
|
|
|
selected_tool_slot_index = (selected_tool_slot_index + tool_slot_size - 1) % (tool_slot_size)
|
2024-09-27 11:46:31 +02:00
|
|
|
set_right_hand_item(inventory.get_tool_item_stacks()[selected_tool_slot_index].item)
|
2024-09-14 12:45:42 +02:00
|
|
|
return true
|
|
|
|
|
|
|
|
return false
|
|
|
|
|
2024-07-10 22:52:28 +02:00
|
|
|
func _unhandled_input(_event: InputEvent) -> void:
|
|
|
|
if Input.is_action_just_pressed("ui_accept"):
|
|
|
|
var actionables = actionable_detector.get_overlapping_areas()
|
|
|
|
if actionables.size() > 0:
|
2024-09-08 16:57:39 +02:00
|
|
|
look_direction = (actionables[0].global_position - position).normalized()
|
2024-08-16 00:24:27 +02:00
|
|
|
actionables[0].action()
|
2024-09-08 16:57:39 +02:00
|
|
|
|
|
|
|
get_viewport().set_input_as_handled()
|
|
|
|
return
|
|
|
|
|
|
|
|
if Input.is_action_just_pressed("interaction"):
|
|
|
|
interaction_state = true
|
|
|
|
|
|
|
|
get_viewport().set_input_as_handled()
|
|
|
|
return
|