2024-05-28 20:20:10 -07:00
# COPYRIGHT Colormatic Studios
# MIT licence
# Quality Godot First Person Controller v2
2023-12-01 22:26:46 -08:00
extends CharacterBody3D
2024-05-28 20:23:39 -07:00
2024-07-17 17:21:27 -07:00
## The settings for the character's movement and feel.
2023-12-01 22:26:46 -08:00
@ export_category ( " Character " )
2024-07-17 17:21:27 -07:00
## The speed that the character moves at without crouching or sprinting.
2023-12-01 22:26:46 -08:00
@ export var base_speed : float = 3.0
2024-07-17 17:21:27 -07:00
## The speed that the character moves at when sprinting.
2023-12-01 22:26:46 -08:00
@ export var sprint_speed : float = 6.0
2024-07-17 17:21:27 -07:00
## The speed that the character moves at when crouching.
2023-12-01 22:26:46 -08:00
@ export var crouch_speed : float = 1.0
2024-07-17 17:21:27 -07:00
## How fast the character speeds up and slows down when Motion Smoothing is on.
2023-12-01 22:26:46 -08:00
@ export var acceleration : float = 10.0
2024-07-17 17:21:27 -07:00
## How high the player jumps.
2023-12-01 22:26:46 -08:00
@ export var jump_velocity : float = 4.5
2024-07-17 17:21:27 -07:00
## How far the player turns when the mouse is moved.
2023-12-01 22:26:46 -08:00
@ export var mouse_sensitivity : float = 0.1
2024-07-24 11:29:38 -07:00
## Invert the Y input for mouse and joystick
2024-07-24 11:35:05 -07:00
@ export var invert_mouse_y : bool = false # Possibly add an invert mouse X in the future
2024-07-17 17:21:27 -07:00
## Wether the player can use movement inputs. Does not stop outside forces or jumping. See Jumping Enabled.
2024-02-29 19:45:26 -08:00
@ export var immobile : bool = false
2024-07-17 17:21:27 -07:00
## The reticle file to import at runtime. By default are in res://addons/fpc/reticles/. Set to an empty string to remove.
2024-03-01 16:02:23 -08:00
@ export_file var default_reticle
2023-12-01 22:26:46 -08:00
@ export_group ( " Nodes " )
2024-07-17 17:21:27 -07:00
## The node that holds the camera. This is rotated instead of the camera for mouse input.
2023-12-01 22:26:46 -08:00
@ export var HEAD : Node3D
@ export var CAMERA : Camera3D
2024-01-19 20:31:04 -08:00
@ export var HEADBOB_ANIMATION : AnimationPlayer
@ export var JUMP_ANIMATION : AnimationPlayer
2024-01-20 14:42:32 -08:00
@ export var CROUCH_ANIMATION : AnimationPlayer
2023-12-19 18:34:49 -08:00
@ export var COLLISION_MESH : CollisionShape3D
2023-12-01 22:26:46 -08:00
@ export_group ( " Controls " )
# We are using UI controls because they are built into Godot Engine so they can be used right away
@ export var JUMP : String = " ui_accept "
@ export var LEFT : String = " ui_left "
@ export var RIGHT : String = " ui_right "
@ export var FORWARD : String = " ui_up "
@ export var BACKWARD : String = " ui_down "
2024-07-23 19:05:26 -07:00
## By default this does not pause the game, but that can be changed in _process.
2023-12-01 22:26:46 -08:00
@ export var PAUSE : String = " ui_cancel "
2024-07-02 08:56:27 -07:00
@ export var CROUCH : String = " crouch "
@ export var SPRINT : String = " sprint "
2023-12-01 22:26:46 -08:00
2024-07-23 19:05:26 -07:00
# Uncomment if you want controller support
2024-07-17 17:25:17 -07:00
#@export var controller_sensitivity : float = 0.035
2024-07-15 15:10:44 -07:00
#@export var LOOK_LEFT : String = "look_left"
#@export var LOOK_RIGHT : String = "look_right"
#@export var LOOK_UP : String = "look_up"
#@export var LOOK_DOWN : String = "look_down"
2023-12-19 18:34:49 -08:00
2023-12-01 22:26:46 -08:00
@ export_group ( " Feature Settings " )
2024-07-17 17:21:27 -07:00
## Enable or disable jumping. Useful for restrictive storytelling environments.
2023-12-01 22:26:46 -08:00
@ export var jumping_enabled : bool = true
2024-07-17 17:21:27 -07:00
## Wether the player can move in the air or not.
2023-12-01 22:26:46 -08:00
@ export var in_air_momentum : bool = true
2024-07-17 17:21:27 -07:00
## Smooths the feel of walking.
2023-12-01 22:26:46 -08:00
@ export var motion_smoothing : bool = true
@ export var sprint_enabled : bool = true
@ export var crouch_enabled : bool = true
@ export_enum ( " Hold to Crouch " , " Toggle Crouch " ) var crouch_mode : int = 0
@ export_enum ( " Hold to Sprint " , " Toggle Sprint " ) var sprint_mode : int = 0
2024-07-17 17:21:27 -07:00
## Wether sprinting should effect FOV.
2023-12-01 22:26:46 -08:00
@ export var dynamic_fov : bool = true
2024-07-17 17:21:27 -07:00
## If the player holds down the jump button, should the player keep hopping.
2023-12-01 22:26:46 -08:00
@ export var continuous_jumping : bool = true
2024-07-17 17:21:27 -07:00
## Enables the view bobbing animation.
2023-12-01 22:26:46 -08:00
@ export var view_bobbing : bool = true
2024-07-17 17:21:27 -07:00
## Enables an immersive animation when the player jumps and hits the ground.
2024-01-19 20:31:04 -08:00
@ export var jump_animation : bool = true
2024-07-17 17:21:27 -07:00
## This determines wether the player can use the pause button, not wether the game will actually pause.
2024-03-21 16:02:14 -07:00
@ export var pausing_enabled : bool = true
2024-07-17 17:21:27 -07:00
## Use with caution.
2024-05-28 20:21:53 -07:00
@ export var gravity_enabled : bool = true
2023-12-01 22:26:46 -08:00
# Member variables
var speed : float = base_speed
2024-01-20 14:45:12 -08:00
var current_speed : float = 0.0
2023-12-19 18:34:49 -08:00
# States: normal, crouching, sprinting
var state : String = " normal "
var low_ceiling : bool = false # This is for when the cieling is too low and the player needs to crouch.
2024-05-28 20:23:39 -07:00
var was_on_floor : bool = true # Was the player on the floor last frame (for landing animation)
2023-12-01 22:26:46 -08:00
2024-05-28 20:23:39 -07:00
# The reticle should always have a Control node as the root
2024-03-01 16:02:23 -08:00
var RETICLE : Control
2023-12-01 22:26:46 -08:00
# Get the gravity from the project settings to be synced with RigidBody nodes
var gravity : float = ProjectSettings . get_setting ( " physics/3d/default_gravity " ) # Don't set this as a const, see the gravity section in _physics_process
2024-07-15 17:24:58 +02:00
# Stores mouse input for rotating the camera in the phyhsics process
var mouseInput : Vector2 = Vector2 ( 0 , 0 )
2023-12-01 22:26:46 -08:00
func _ready ( ) :
2024-05-28 20:23:39 -07:00
#It is safe to comment this line if your game doesn't start with the mouse captured
2023-12-01 22:26:46 -08:00
Input . mouse_mode = Input . MOUSE_MODE_CAPTURED
2024-01-09 12:51:48 -08:00
2024-05-28 20:23:39 -07:00
# If the controller is rotated in a certain direction for game design purposes, redirect this rotation into the head.
2024-05-28 20:24:53 -07:00
HEAD . rotation . y = rotation . y
rotation . y = 0
2024-01-11 15:01:53 -08:00
2024-03-01 16:02:23 -08:00
if default_reticle :
change_reticle ( default_reticle )
2024-01-09 12:51:48 -08:00
# Reset the camera position
2024-05-28 20:23:39 -07:00
# If you want to change the default head height, change these animations.
2024-01-19 20:31:04 -08:00
HEADBOB_ANIMATION . play ( " RESET " )
JUMP_ANIMATION . play ( " RESET " )
2024-02-29 19:53:23 -08:00
CROUCH_ANIMATION . play ( " RESET " )
2024-03-21 16:02:14 -07:00
check_controls ( )
2024-09-21 17:31:43 -04:00
enter_normal_state ( )
2024-03-21 16:02:14 -07:00
func check_controls ( ) : # If you add a control, you might want to add a check for it here.
2024-07-01 19:52:49 -07:00
# The actions are being disabled so the engine doesn't halt the entire project in debug mode
2024-03-21 16:02:14 -07:00
if ! InputMap . has_action ( JUMP ) :
push_error ( " No control mapped for jumping. Please add an input map control. Disabling jump. " )
jumping_enabled = false
if ! InputMap . has_action ( LEFT ) :
push_error ( " No control mapped for move left. Please add an input map control. Disabling movement. " )
immobile = true
if ! InputMap . has_action ( RIGHT ) :
push_error ( " No control mapped for move right. Please add an input map control. Disabling movement. " )
immobile = true
if ! InputMap . has_action ( FORWARD ) :
push_error ( " No control mapped for move forward. Please add an input map control. Disabling movement. " )
immobile = true
if ! InputMap . has_action ( BACKWARD ) :
push_error ( " No control mapped for move backward. Please add an input map control. Disabling movement. " )
immobile = true
if ! InputMap . has_action ( PAUSE ) :
2024-07-01 19:55:15 -06:00
push_error ( " No control mapped for pause. Please add an input map control. Disabling pausing. " )
2024-03-21 16:02:14 -07:00
pausing_enabled = false
if ! InputMap . has_action ( CROUCH ) :
push_error ( " No control mapped for crouch. Please add an input map control. Disabling crouching. " )
crouch_enabled = false
if ! InputMap . has_action ( SPRINT ) :
push_error ( " No control mapped for sprint. Please add an input map control. Disabling sprinting. " )
sprint_enabled = false
2023-12-01 22:26:46 -08:00
2023-12-19 18:34:49 -08:00
2024-05-28 20:23:39 -07:00
func change_reticle ( reticle ) : # Yup, this function is kinda strange
2024-03-01 16:02:23 -08:00
if RETICLE :
RETICLE . queue_free ( )
RETICLE = load ( reticle ) . instantiate ( )
RETICLE . character = self
$ UserInterface . add_child ( RETICLE )
2023-12-01 22:26:46 -08:00
func _physics_process ( delta ) :
2024-05-28 20:23:39 -07:00
# Big thanks to github.com/LorenzoAncora for the concept of the improved debug values
2024-01-20 14:45:12 -08:00
current_speed = Vector3 . ZERO . distance_to ( get_real_velocity ( ) )
$ UserInterface / DebugPanel . add_property ( " Speed " , snappedf ( current_speed , 0.001 ) , 1 )
$ UserInterface / DebugPanel . add_property ( " Target speed " , speed , 2 )
2024-01-09 09:55:33 -08:00
var cv : Vector3 = get_real_velocity ( )
var vd : Array [ float ] = [
snappedf ( cv . x , 0.001 ) ,
snappedf ( cv . y , 0.001 ) ,
snappedf ( cv . z , 0.001 )
]
var readable_velocity : String = " X: " + str ( vd [ 0 ] ) + " Y: " + str ( vd [ 1 ] ) + " Z: " + str ( vd [ 2 ] )
2024-01-20 14:45:12 -08:00
$ UserInterface / DebugPanel . add_property ( " Velocity " , readable_velocity , 3 )
2023-12-01 22:26:46 -08:00
# Gravity
#gravity = ProjectSettings.get_setting("physics/3d/default_gravity") # If the gravity changes during your game, uncomment this code
2024-05-28 20:21:53 -07:00
if not is_on_floor ( ) and gravity and gravity_enabled :
2023-12-01 22:26:46 -08:00
velocity . y -= gravity * delta
handle_jumping ( )
var input_dir = Vector2 . ZERO
2024-05-28 20:23:39 -07:00
if ! immobile : # Immobility works by interrupting user input, so other forces can still be applied to the player
2023-12-01 22:26:46 -08:00
input_dir = Input . get_vector ( LEFT , RIGHT , FORWARD , BACKWARD )
handle_movement ( delta , input_dir )
2024-07-15 17:24:58 +02:00
2024-07-15 15:10:20 -07:00
handle_head_rotation ( )
2023-12-01 22:26:46 -08:00
2024-05-28 20:23:39 -07:00
# The player is not able to stand up if the ceiling is too low
2023-12-19 18:34:49 -08:00
low_ceiling = $ CrouchCeilingDetection . is_colliding ( )
2023-12-01 22:26:46 -08:00
2023-12-19 18:34:49 -08:00
handle_state ( input_dir )
2024-05-28 20:23:39 -07:00
if dynamic_fov : # This may be changed to an AnimationPlayer
2023-12-21 21:48:37 -08:00
update_camera_fov ( )
2023-12-01 22:26:46 -08:00
if view_bobbing :
headbob_animation ( input_dir )
2024-01-19 20:31:04 -08:00
if jump_animation :
2024-05-28 20:23:39 -07:00
if ! was_on_floor and is_on_floor ( ) : # The player just landed
match randi ( ) % 2 : #TODO: Change this to detecting velocity direction
2024-02-29 19:36:50 -08:00
0 :
2024-05-07 15:48:37 -07:00
JUMP_ANIMATION . play ( " land_left " , 0.25 )
2024-02-29 19:36:50 -08:00
1 :
2024-05-07 15:48:37 -07:00
JUMP_ANIMATION . play ( " land_right " , 0.25 )
2024-05-28 20:23:39 -07:00
was_on_floor = is_on_floor ( ) # This must always be at the end of physics_process
2023-12-01 22:26:46 -08:00
func handle_jumping ( ) :
if jumping_enabled :
2024-05-28 20:23:39 -07:00
if continuous_jumping : # Hold down the jump button
2024-01-20 19:43:41 -08:00
if Input . is_action_pressed ( JUMP ) and is_on_floor ( ) and ! low_ceiling :
2024-01-19 20:31:04 -08:00
if jump_animation :
2024-05-07 15:48:37 -07:00
JUMP_ANIMATION . play ( " jump " , 0.25 )
2024-05-28 20:23:39 -07:00
velocity . y += jump_velocity # Adding instead of setting so jumping on slopes works properly
2023-12-01 22:26:46 -08:00
else :
2024-01-20 19:43:41 -08:00
if Input . is_action_just_pressed ( JUMP ) and is_on_floor ( ) and ! low_ceiling :
2024-01-19 20:31:04 -08:00
if jump_animation :
2024-05-07 15:48:37 -07:00
JUMP_ANIMATION . play ( " jump " , 0.25 )
2023-12-01 22:26:46 -08:00
velocity . y += jump_velocity
2023-12-19 18:34:49 -08:00
2023-12-01 22:26:46 -08:00
func handle_movement ( delta , input_dir ) :
var direction = input_dir . rotated ( - HEAD . rotation . y )
direction = Vector3 ( direction . x , 0 , direction . y )
move_and_slide ( )
if in_air_momentum :
2023-12-19 18:34:49 -08:00
if is_on_floor ( ) :
2023-12-01 22:26:46 -08:00
if motion_smoothing :
velocity . x = lerp ( velocity . x , direction . x * speed , acceleration * delta )
velocity . z = lerp ( velocity . z , direction . z * speed , acceleration * delta )
else :
velocity . x = direction . x * speed
velocity . z = direction . z * speed
else :
if motion_smoothing :
velocity . x = lerp ( velocity . x , direction . x * speed , acceleration * delta )
velocity . z = lerp ( velocity . z , direction . z * speed , acceleration * delta )
else :
velocity . x = direction . x * speed
velocity . z = direction . z * speed
2024-07-15 15:10:20 -07:00
func handle_head_rotation ( ) :
2024-07-15 17:24:58 +02:00
HEAD . rotation_degrees . y -= mouseInput . x * mouse_sensitivity
2024-07-24 11:29:38 -07:00
if invert_mouse_y :
HEAD . rotation_degrees . x -= mouseInput . y * mouse_sensitivity * - 1.0
else :
HEAD . rotation_degrees . x -= mouseInput . y * mouse_sensitivity
2024-07-15 17:24:58 +02:00
2024-07-15 15:10:44 -07:00
# Uncomment for controller support
2024-07-17 17:25:17 -07:00
#var controller_view_rotation = Input.get_vector(LOOK_DOWN, LOOK_UP, LOOK_RIGHT, LOOK_LEFT) * controller_sensitivity # These are inverted because of the nature of 3D rotation.
2024-07-15 15:10:44 -07:00
#HEAD.rotation.x += controller_view_rotation.x
2024-07-24 11:29:38 -07:00
#if invert_mouse_y:
#HEAD.rotation.y += controller_view_rotation.y * -1.0
#else:
#HEAD.rotation.y += controller_view_rotation.y
2024-07-15 15:10:44 -07:00
mouseInput = Vector2 ( 0 , 0 )
2024-07-15 17:24:58 +02:00
HEAD . rotation . x = clamp ( HEAD . rotation . x , deg_to_rad ( - 90 ) , deg_to_rad ( 90 ) )
2023-12-01 22:26:46 -08:00
2023-12-19 18:34:49 -08:00
func handle_state ( moving ) :
if sprint_enabled :
if sprint_mode == 0 :
2024-01-20 14:42:32 -08:00
if Input . is_action_pressed ( SPRINT ) and state != " crouching " :
2023-12-19 18:34:49 -08:00
if moving :
if state != " sprinting " :
enter_sprint_state ( )
else :
if state == " sprinting " :
enter_normal_state ( )
elif state == " sprinting " :
enter_normal_state ( )
elif sprint_mode == 1 :
if moving :
2024-01-20 19:19:13 -08:00
# If the player is holding sprint before moving, handle that cenerio
if Input . is_action_pressed ( SPRINT ) and state == " normal " :
enter_sprint_state ( )
2024-01-08 11:58:14 -08:00
if Input . is_action_just_pressed ( SPRINT ) :
2023-12-19 18:34:49 -08:00
match state :
" normal " :
enter_sprint_state ( )
" sprinting " :
enter_normal_state ( )
elif state == " sprinting " :
enter_normal_state ( )
2023-12-01 22:26:46 -08:00
if crouch_enabled :
if crouch_mode == 0 :
2024-01-20 19:19:13 -08:00
if Input . is_action_pressed ( CROUCH ) and state != " sprinting " :
2023-12-19 18:34:49 -08:00
if state != " crouching " :
enter_crouch_state ( )
elif state == " crouching " and ! $ CrouchCeilingDetection . is_colliding ( ) :
enter_normal_state ( )
2023-12-01 22:26:46 -08:00
elif crouch_mode == 1 :
2024-01-08 11:58:14 -08:00
if Input . is_action_just_pressed ( CROUCH ) :
2023-12-19 18:34:49 -08:00
match state :
" normal " :
enter_crouch_state ( )
" crouching " :
if ! $ CrouchCeilingDetection . is_colliding ( ) :
enter_normal_state ( )
2024-01-09 13:57:39 -08:00
# Any enter state function should only be called once when you want to enter that state, not every frame.
2023-12-19 18:34:49 -08:00
func enter_normal_state ( ) :
#print("entering normal state")
var prev_state = state
2024-01-20 14:42:32 -08:00
if prev_state == " crouching " :
CROUCH_ANIMATION . play_backwards ( " crouch " )
2023-12-19 18:34:49 -08:00
state = " normal "
speed = base_speed
func enter_crouch_state ( ) :
#print("entering crouch state")
var prev_state = state
state = " crouching "
speed = crouch_speed
2024-01-20 14:42:32 -08:00
CROUCH_ANIMATION . play ( " crouch " )
2023-12-19 18:34:49 -08:00
func enter_sprint_state ( ) :
#print("entering sprint state")
var prev_state = state
2024-01-20 14:42:32 -08:00
if prev_state == " crouching " :
CROUCH_ANIMATION . play_backwards ( " crouch " )
2023-12-19 18:34:49 -08:00
state = " sprinting "
speed = sprint_speed
func update_camera_fov ( ) :
if state == " sprinting " :
CAMERA . fov = lerp ( CAMERA . fov , 85.0 , 0.3 )
else :
CAMERA . fov = lerp ( CAMERA . fov , 75.0 , 0.3 )
2023-12-01 22:26:46 -08:00
func headbob_animation ( moving ) :
if moving and is_on_floor ( ) :
2024-02-29 19:36:50 -08:00
var use_headbob_animation : String
match state :
" normal " , " crouching " :
use_headbob_animation = " walk "
" sprinting " :
use_headbob_animation = " sprint "
2024-01-20 18:05:15 -08:00
var was_playing : bool = false
2024-02-29 19:36:50 -08:00
if HEADBOB_ANIMATION . current_animation == use_headbob_animation :
2024-01-20 18:05:15 -08:00
was_playing = true
2024-02-29 19:36:50 -08:00
HEADBOB_ANIMATION . play ( use_headbob_animation , 0.25 )
2024-01-20 14:45:12 -08:00
HEADBOB_ANIMATION . speed_scale = ( current_speed / base_speed ) * 1.75
2024-01-20 18:05:15 -08:00
if ! was_playing :
HEADBOB_ANIMATION . seek ( float ( randi ( ) % 2 ) ) # Randomize the initial headbob direction
2024-01-24 16:43:18 -08:00
# Let me explain that piece of code because it looks like it does the opposite of what it actually does.
# The headbob animation has two starting positions. One is at 0 and the other is at 1.
# randi() % 2 returns either 0 or 1, and so the animation randomly starts at one of the starting positions.
2024-02-29 19:36:50 -08:00
# This code is extremely performant but it makes no sense.
2024-01-20 18:05:15 -08:00
2023-12-01 22:26:46 -08:00
else :
2024-06-21 00:11:36 -07:00
if HEADBOB_ANIMATION . current_animation == " sprint " or HEADBOB_ANIMATION . current_animation == " walk " :
2024-05-28 20:26:27 -07:00
HEADBOB_ANIMATION . speed_scale = 1
2024-06-21 00:11:36 -07:00
HEADBOB_ANIMATION . play ( " RESET " , 1 )
2023-12-19 18:34:49 -08:00
func _process ( delta ) :
2024-01-09 09:55:33 -08:00
$ UserInterface / DebugPanel . add_property ( " FPS " , Performance . get_monitor ( Performance . TIME_FPS ) , 0 )
var status : String = state
if ! is_on_floor ( ) :
status += " in the air "
2024-01-20 14:45:12 -08:00
$ UserInterface / DebugPanel . add_property ( " State " , status , 4 )
2023-12-19 18:34:49 -08:00
2024-03-21 16:02:14 -07:00
if pausing_enabled :
if Input . is_action_just_pressed ( PAUSE ) :
2024-07-17 17:21:27 -07:00
# You may want another node to handle pausing, because this player may get paused too.
2024-05-28 20:27:12 -07:00
match Input . mouse_mode :
Input . MOUSE_MODE_CAPTURED :
Input . mouse_mode = Input . MOUSE_MODE_VISIBLE
2024-07-17 17:21:27 -07:00
#get_tree().paused = false
2024-05-28 20:27:12 -07:00
Input . MOUSE_MODE_VISIBLE :
Input . mouse_mode = Input . MOUSE_MODE_CAPTURED
2024-07-17 17:21:27 -07:00
#get_tree().paused = false
2023-12-19 18:34:49 -08:00
2024-07-23 19:05:26 -07:00
func _unhandled_input ( event : InputEvent ) :
2023-12-19 18:34:49 -08:00
if event is InputEventMouseMotion and Input . mouse_mode == Input . MOUSE_MODE_CAPTURED :
2024-07-15 17:24:58 +02:00
mouseInput . x += event . relative . x
mouseInput . y += event . relative . y
2024-07-23 19:05:26 -07:00
# Toggle debug menu
elif event is InputEventKey :
if event . is_released ( ) :
# Where we're going, we don't need InputMap
if event . keycode == 4194338 : # F7
$ UserInterface / DebugPanel . visible = ! $ UserInterface / DebugPanel . visible