2025-08-09 17:43:37 +04:00
|
|
|
using System.Collections;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using UnityEngine;
|
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
namespace Polyart
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
public class FirstPersonController : MonoBehaviour
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
public static FirstPersonController instance;
|
|
|
|
public bool CanMove { get; private set; } = true;
|
|
|
|
private bool isSprinting => canSprint && Input.GetKey(sprintKey);
|
|
|
|
private bool ShouldJump => Input.GetKeyDown(jumpKey) && characterController.isGrounded;
|
|
|
|
private bool ShouldCrouch => Input.GetKeyDown(crouchKey) && !duringCrouchAnim && characterController.isGrounded;
|
|
|
|
|
|
|
|
[Header("Character Options")]
|
|
|
|
[SerializeField] private bool canSprint = true;
|
|
|
|
[SerializeField] private bool canJump = true;
|
|
|
|
[SerializeField] private bool canCrouch = true;
|
|
|
|
[SerializeField] private bool canHeadBob = true;
|
|
|
|
[SerializeField] private bool canInteract = true;
|
|
|
|
[SerializeField] private bool useFootsteps = true;
|
|
|
|
|
|
|
|
[Header("Controls")]
|
|
|
|
[SerializeField] private KeyCode sprintKey = KeyCode.LeftShift;
|
|
|
|
[SerializeField] private KeyCode jumpKey = KeyCode.Space;
|
|
|
|
[SerializeField] private KeyCode crouchKey = KeyCode.LeftControl;
|
|
|
|
[SerializeField] private KeyCode interactKey = KeyCode.E;
|
|
|
|
|
|
|
|
[Header("Interaction")]
|
|
|
|
[SerializeField] private Vector3 interactionRayPoint = default;
|
|
|
|
[SerializeField] private float interactionDistance = default;
|
|
|
|
[SerializeField] private LayerMask interactionLayer = default;
|
|
|
|
private Interactable_Dreamscape currentInteractable;
|
|
|
|
|
|
|
|
[Header("Movement Parameters")]
|
|
|
|
[SerializeField] private float walkSpeed = 3.0f;
|
|
|
|
[SerializeField] private float sprintSpeed = 6.0f;
|
|
|
|
[SerializeField] private float crouchSpeed = 1.5f;
|
|
|
|
|
|
|
|
[Header("Camera Parameters")]
|
|
|
|
[SerializeField, Range(1, 10)] private float lookSpeedX = 2.0f;
|
|
|
|
[SerializeField, Range(1, 10)] private float lookSpeedY = 2.0f;
|
|
|
|
[SerializeField, Range(1, 180)] private float lowerLookLimit = 80.0f;
|
|
|
|
[SerializeField, Range(1, 180)] private float upperLookLimit = 80.0f;
|
|
|
|
|
|
|
|
[Header("Jumping Parameters")]
|
|
|
|
[SerializeField] private float jumpForce = 8.0f;
|
|
|
|
[SerializeField] private float gravity = 30.0f;
|
|
|
|
|
|
|
|
[Header("Crouching Parameters")]
|
|
|
|
|
|
|
|
[SerializeField] private float crouchHeight = 0.5f;
|
|
|
|
[SerializeField] private float standinghHeight = 2f;
|
|
|
|
[SerializeField] private float timeToCrouch = 0.25f;
|
|
|
|
[SerializeField] private Vector3 crouchCenter = new Vector3(0, 0.5f, 0);
|
|
|
|
[SerializeField] private Vector3 standingCenter = new Vector3(0, 0, 0);
|
|
|
|
private bool isCrouching;
|
|
|
|
private bool duringCrouchAnim;
|
|
|
|
|
|
|
|
[Header("Headbob Parameters")]
|
|
|
|
[SerializeField] private float walkBobSpeed = 14.0f;
|
|
|
|
[SerializeField] private float walkBobAmount = 0.05f;
|
|
|
|
[SerializeField] private float sprintBobSpeed = 18.0f;
|
|
|
|
[SerializeField] private float sprintBobAmount = 0.11f;
|
|
|
|
|
|
|
|
[Header("Footstep Parameters")]
|
|
|
|
[SerializeField] private float baseStepSpeed = 0.5f;
|
|
|
|
[SerializeField] private float crouchStepMultiplier = 1.5f;
|
|
|
|
[SerializeField] private float sprintStepMultiplier = 0.6f;
|
|
|
|
[SerializeField] private AudioSource footstepAudioSource = default;
|
|
|
|
[SerializeField] private AudioClip[] woodClips = default;
|
|
|
|
[SerializeField] private AudioClip[] stoneClips = default;
|
|
|
|
[SerializeField] private AudioClip[] waterClips = default;
|
|
|
|
[SerializeField] private AudioClip[] grassClips = default;
|
|
|
|
public Camera PlayerCamera => playerCamera;
|
|
|
|
|
|
|
|
private float footstepTimer = 0;
|
|
|
|
private float GetCurrentOffset => isCrouching ? baseStepSpeed * crouchStepMultiplier : isSprinting ? baseStepSpeed * sprintStepMultiplier : baseStepSpeed;
|
|
|
|
|
|
|
|
private float defaultYPos = 0;
|
|
|
|
private float timer;
|
|
|
|
|
|
|
|
private Camera playerCamera;
|
|
|
|
private CharacterController characterController;
|
|
|
|
|
|
|
|
private Vector3 moveDirection;
|
|
|
|
private Vector2 currentInput;
|
|
|
|
|
|
|
|
private float rotationX = 0;
|
|
|
|
|
|
|
|
public Animator[] HandsAnimator;
|
|
|
|
|
|
|
|
void Awake()
|
|
|
|
{
|
|
|
|
playerCamera = GetComponentInChildren<Camera>();
|
|
|
|
characterController = GetComponent<CharacterController>();
|
|
|
|
Cursor.lockState = CursorLockMode.Locked;
|
|
|
|
Cursor.visible = false;
|
|
|
|
defaultYPos = playerCamera.transform.localPosition.y;
|
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
|
|
|
|
private void Start()
|
|
|
|
{
|
|
|
|
if (instance != null)
|
|
|
|
{
|
|
|
|
Destroy(gameObject);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
instance = this;
|
|
|
|
}
|
|
|
|
DontDestroyOnLoad(gameObject);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update is called once per frame
|
|
|
|
void Update()
|
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
if (CanMove)
|
|
|
|
{
|
|
|
|
HandleMovementInput();
|
|
|
|
HandleMouseLook();
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
if (canJump)
|
|
|
|
HandleJump();
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
if (canCrouch)
|
|
|
|
HandleCrouch();
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
if (canHeadBob)
|
|
|
|
{
|
|
|
|
HandleHeadBob();
|
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
if (canInteract)
|
|
|
|
{
|
|
|
|
HandleInteractionCheck();
|
|
|
|
HandleInteractionInput();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (useFootsteps)
|
|
|
|
{
|
|
|
|
HandleFootsteps();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ApplyFinalMovements();
|
2025-08-09 17:43:37 +04:00
|
|
|
|
|
|
|
}
|
2025-09-29 15:38:48 +05:00
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
private void HandleMovementInput()
|
|
|
|
{
|
|
|
|
currentInput = new Vector2((isCrouching ? crouchSpeed : isSprinting ? sprintSpeed : walkSpeed) * Input.GetAxis("Vertical"), (isCrouching ? crouchSpeed : isSprinting ? sprintSpeed : walkSpeed) * Input.GetAxis("Horizontal"));
|
2025-08-09 17:43:37 +04:00
|
|
|
|
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
float moveDirectionY = moveDirection.y;
|
|
|
|
moveDirection = (transform.TransformDirection(Vector3.forward) * currentInput.x) + (transform.TransformDirection(Vector3.right) * currentInput.y);
|
|
|
|
moveDirection.y = moveDirectionY;
|
|
|
|
HandleArmsMoveInput((int)currentInput.x);
|
2025-08-09 17:43:37 +04:00
|
|
|
}
|
2025-09-29 15:38:48 +05:00
|
|
|
void HandleArmsMoveInput(float input)
|
|
|
|
{
|
|
|
|
int stateValue;
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
if (input != 0)
|
|
|
|
stateValue = isSprinting ? 2 : 1;
|
|
|
|
else
|
|
|
|
stateValue = 0;
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
foreach (Animator handAnim in HandsAnimator)
|
|
|
|
handAnim.SetInteger("State", stateValue);
|
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
private void HandleHeadBob()
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
if (!characterController.isGrounded) return;
|
|
|
|
if (Mathf.Abs(moveDirection.x) > 0.1f || Mathf.Abs(moveDirection.z) > 0.1f)
|
|
|
|
{
|
|
|
|
timer += Time.deltaTime * (isSprinting ? sprintBobSpeed : walkBobSpeed);
|
|
|
|
playerCamera.transform.localPosition = new Vector3(
|
|
|
|
playerCamera.transform.localPosition.x,
|
|
|
|
defaultYPos + Mathf.Sin(timer) * (isSprinting ? sprintBobAmount : walkBobAmount),
|
|
|
|
playerCamera.transform.localPosition.z);
|
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
private void HandleMouseLook()
|
|
|
|
{
|
|
|
|
rotationX -= Input.GetAxis("Mouse Y") * lookSpeedY;
|
|
|
|
rotationX = Mathf.Clamp(rotationX, -upperLookLimit, lowerLookLimit);
|
|
|
|
playerCamera.transform.localRotation = Quaternion.Euler(rotationX, 0, 0);
|
|
|
|
transform.rotation *= Quaternion.Euler(0, Input.GetAxis("Mouse X") * lookSpeedX, 0);
|
|
|
|
}
|
|
|
|
private void HandleJump()
|
|
|
|
{
|
|
|
|
if (ShouldJump)
|
|
|
|
moveDirection.y = jumpForce;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void HandleCrouch()
|
|
|
|
{
|
|
|
|
if (ShouldCrouch)
|
|
|
|
StartCoroutine(CrouchStand());
|
|
|
|
|
|
|
|
}
|
|
|
|
private void ApplyFinalMovements()
|
|
|
|
{
|
|
|
|
if (!characterController.isGrounded)
|
|
|
|
moveDirection.y -= gravity * Time.deltaTime;
|
|
|
|
|
|
|
|
characterController.Move(moveDirection * Time.deltaTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
private IEnumerator CrouchStand()
|
|
|
|
{
|
|
|
|
if (isCrouching && Physics.Raycast(playerCamera.transform.position, Vector3.up, 1f))
|
|
|
|
yield break;
|
|
|
|
|
|
|
|
duringCrouchAnim = true;
|
|
|
|
|
|
|
|
float timeElapsed = 0;
|
|
|
|
float targetHeight = isCrouching ? standinghHeight : crouchHeight;
|
|
|
|
float currentHeight = characterController.height;
|
|
|
|
Vector3 targetCenter = isCrouching ? standingCenter : crouchCenter;
|
|
|
|
Vector3 currentCenter = characterController.center;
|
|
|
|
|
|
|
|
while (timeElapsed < timeToCrouch)
|
|
|
|
{
|
|
|
|
characterController.height = Mathf.Lerp(currentHeight, targetHeight, timeElapsed / timeToCrouch);
|
|
|
|
characterController.center = Vector3.Lerp(currentCenter, targetCenter, timeElapsed / timeToCrouch);
|
|
|
|
timeElapsed += Time.deltaTime;
|
|
|
|
yield return null;
|
|
|
|
}
|
|
|
|
characterController.height = targetHeight;
|
|
|
|
characterController.center = targetCenter;
|
|
|
|
|
|
|
|
isCrouching = !isCrouching;
|
|
|
|
|
|
|
|
duringCrouchAnim = false;
|
|
|
|
}
|
2025-09-29 15:38:48 +05:00
|
|
|
private void HandleInteractionCheck()
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
if (Physics.Raycast(playerCamera.ViewportPointToRay(interactionRayPoint), out RaycastHit hit, interactionDistance))
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
if (hit.collider.gameObject.layer == 7 && (currentInteractable == null || hit.collider.gameObject.GetInstanceID() != currentInteractable.gameObject.GetInstanceID()))
|
|
|
|
{
|
|
|
|
hit.collider.TryGetComponent(out currentInteractable);
|
|
|
|
if (currentInteractable)
|
|
|
|
currentInteractable.OnFocus();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (currentInteractable)
|
|
|
|
{
|
|
|
|
currentInteractable.OnLoseFocus();
|
|
|
|
currentInteractable = null;
|
2025-08-09 17:43:37 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
private void HandleFootsteps()
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
if (!characterController.isGrounded) return;
|
|
|
|
if (currentInput == Vector2.zero) return;
|
|
|
|
|
|
|
|
footstepTimer -= Time.deltaTime;
|
|
|
|
if (footstepTimer <= 0)
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
if (Physics.Raycast(playerCamera.transform.position, Vector3.down, out RaycastHit hit, 2))
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
switch (hit.collider.tag)
|
|
|
|
{
|
|
|
|
case "Footsteps/Grass":
|
|
|
|
footstepAudioSource.PlayOneShot(grassClips[Random.Range(0, grassClips.Length - 1)]);
|
|
|
|
break;
|
|
|
|
case "Footsteps/Stone":
|
|
|
|
footstepAudioSource.PlayOneShot(stoneClips[Random.Range(0, stoneClips.Length - 1)]);
|
|
|
|
break;
|
|
|
|
case "Footsteps/Water":
|
|
|
|
footstepAudioSource.PlayOneShot(waterClips[Random.Range(0, waterClips.Length - 1)]);
|
|
|
|
break;
|
|
|
|
case "Footsteps/Wood":
|
|
|
|
footstepAudioSource.PlayOneShot(woodClips[Random.Range(0, woodClips.Length - 1)]);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
footstepAudioSource.PlayOneShot(grassClips[Random.Range(0, grassClips.Length - 1)]);
|
|
|
|
break;
|
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
}
|
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
footstepTimer = GetCurrentOffset;
|
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
}
|
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
private void HandleInteractionInput()
|
2025-08-09 17:43:37 +04:00
|
|
|
{
|
2025-09-29 15:38:48 +05:00
|
|
|
if (Input.GetKeyDown(interactKey) && currentInteractable != null && Physics.Raycast(playerCamera.ViewportPointToRay(interactionRayPoint), out RaycastHit hit, interactionDistance, interactionLayer))
|
|
|
|
{
|
|
|
|
currentInteractable.OnInteract();
|
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
}
|
|
|
|
|
2025-09-29 15:38:48 +05:00
|
|
|
}
|
2025-08-09 17:43:37 +04:00
|
|
|
|
|
|
|
}
|