Единство заземленного состояния мерцает для характера..Используется, но мигает. - PullRequest
0 голосов
/ 25 ноября 2018

Заземленное состояние моего контроллера персонажей постоянно мигает и включается, как кажется, в каждом кадре.Из того, что я знаю, он должен проверить, заземлен ли игрок через player.isGounded, но что-то еще перемещает его обратно.

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class PlayerCharacterController: MonoBehaviour {

    static Animator anim;

    public bool walking;

    public GameObject playerModel, Hero;
    //Transforms
    public Transform playerCam, character, centerPoint;

    private Vector3 moveDirection;

    //character controller declaration
    CharacterController player;

    //Mouse Rotation
    private float rotX, rotY;
    //Mouse Y Position
    public float mouseYPosition = 1f; 
    //Mouse Sensitivity
    public float Sensitivity = 10f;
    //Mouse Zoom
    private float zoom;
    public float zoomSpeed = 2;
    //Clamping Zoom
    public float zoomMin = -2f;
    public float zoomMax = -10f;
    public float rotationSpeed = 5f;
    //Move Front Back left & Right
    private float moveFB, moveLR;   
    //Movement Speed
    public float Speed = 2f;
    //Velocity of Gravity
    public float verticalVelocity;
    //Jump Distance
    public float jumpDist = 5f;
    //Multiple Jumps
    int jumpTimes;
    //To use with Dialogue Manager
    public DialogueManager DiagM;

    public AudioClip jumpSound;
    public AudioClip HurtSound;
    public AudioClip PunchSound;

    AudioSource audioSource;

    //knockback

    public float knockBackForce;
    public float knockBackTime;
    private float knockBackCounter;


    // Use this for initialization
    void Start ()   
    {
        //character controller
        player = GameObject.Find("Player").GetComponent<CharacterController> ();

        StartCoroutine(MyCoroutine(character));

        anim = GetComponent<Animator>();

        //mouse zoom
        zoom = -3;  
        centerPoint.transform.position = playerCam.transform.position;
        centerPoint.transform.parent = null;
        audioSource = GetComponent<AudioSource>();
    }

    IEnumerator MyCoroutine (Transform character)
    {
        if (player.isGrounded == true)

        {
        anim.SetBool("isFalling",false);
        //anim.SetBool("isIdling", true);
        yield return new WaitForSeconds(0);
        }

    }
    // Update is called once per frame

    void Update ()
    {        
        //Mouse Zoom Input
        zoom += Input.GetAxis ("Mouse ScrollWheel") * zoomSpeed;

        if (zoom > zoomMin)
            zoom = zoomMin;
        if (zoom < zoomMax)
            zoom = zoomMax;

        //Mouse Camera Input
        playerCam.transform.localPosition = new Vector3 (0, 0, zoom);

        //Mouse Rotation

        rotX += Input.GetAxis ("Mouse X") * Sensitivity;
        rotY -= Input.GetAxis ("Mouse Y") * Sensitivity;      

        //Clamp Camera
        rotY = Mathf.Clamp (rotY, -60f, 60f);
        playerCam.LookAt (centerPoint);
        centerPoint.localRotation = Quaternion.Euler (rotY, rotX, 0);

        //Movement Speed
        if (knockBackCounter <= 0)
        {   
            moveDirection = (transform.forward * Input.GetAxis("Vertical")) + (transform.right * Input.GetAxis("Horizontal"));
            moveDirection = moveDirection * Speed;
            moveDirection.y = verticalVelocity;

            player.Move(moveDirection * Time.deltaTime);



            //Movement Rotation

            centerPoint.position = new Vector3 (character.position.x, character.position.y + mouseYPosition, character.position.z);

            //knockback disable
            //Movement Input

            if (Input.GetAxis("Vertical") != 0 || Input.GetAxis("Horizontal") != 0)

            {
                transform.rotation = Quaternion.Euler(0f, centerPoint.rotation.eulerAngles.y, 0f);
                Quaternion turnAngle = Quaternion.LookRotation(new Vector3(moveDirection.x, 0f, moveDirection.z));
                playerModel.transform.rotation = Quaternion.Slerp(playerModel.transform.rotation, turnAngle, Time.deltaTime * rotationSpeed);

                if (player.isGrounded == true)
                {
                    anim.SetBool("isWalking", true);
                    anim.Play("Running");
                }
            }
            else 
            {
                StartCoroutine(MyCoroutine(character));
            }

            if (Input.GetButtonDown("LHand"))
            {
                audioSource.PlayOneShot(PunchSound, 1F);
                anim.Play("RPunch");
            }

            if (player.isGrounded == true)
            {
                jumpTimes = 0;
                //verticalVelocity = -Physics.gravity.y * Time.deltaTime;
                verticalVelocity = 0;
            }
            else
            {
                verticalVelocity += Physics.gravity.y * Time.deltaTime;
                anim.SetBool("isWalking", false);
                anim.SetBool("isFalling", true);
            }


            if (jumpTimes < 1)
            {
                if (Input.GetButtonDown("Jump"))
                {
                    verticalVelocity += jumpDist;
                    anim.Play("Jump");
                    audioSource.PlayOneShot(jumpSound, 1F);
                    jumpTimes += 1;
                }
            }
        }
        else
        {
            knockBackCounter -= Time.deltaTime;
        }
    }

    public void Knockback(Vector3 direction)
    {
        knockBackCounter = knockBackTime;
        anim.Play("Jump");
        audioSource.PlayOneShot(HurtSound, 50F);
        moveDirection = direction * knockBackForce;
        moveDirection.y = knockBackForce;
    }
}

Похоже, что это связано с линиями вертикальной скорости, но до сих пор я пытался установить только вертикальную скорость = 0, и это работает, пока я фактически не переместил символ.Что я могу изменить, чтобы остановить мерцание?

Ответы [ 2 ]

0 голосов
/ 23 июля 2019

Возможно, это уже решено, но причина в том, что если вы используете контроллер персонажей, вы должны применять гравитацию ВСЕ время к персонажу.

Когда персонаж сталкивается с объектом, он фактически входитнемного внутри этого объекта, затем Unity отталкивает персонажа от объекта, пока он больше не касается его.В этот момент ваша сила тяжести начинает действовать снова и заново инициализирует цикл.

Вам нужно применять силу тяжести 100% времени, чтобы создать достаточно силы, чтобы «сбалансировать» этот бой с полом.Может быть меньше "гравитация", как 1. Не нужно быть вашей гравитационной переменной.

Кроме того, я хотел бы добавить "время Койота", и сделать мой метод IsGounded (),следующим образом:

    public bool IsGrounded()
{
    return CoyoteTime < CoyoteTimeMax;
}

public void CoyoteControl()
{
    if (CharController.isGrounded)
    {
        CoyoteTime = 0;
    }
    else
    {
        CoyoteTime += Time.deltaTime;
    }
}

И затем я вызываю CoyoteControl () для Update (), и я могу вызывать IsGounded () всякий раз, когда мне нужно.На инспекторе я обычно устанавливаю CoyoteTimeMax равным 0,1, и это делает падение более плавным.

0 голосов
/ 26 ноября 2018

Согласно вашему комментарию.Вы не должны определять, заземлен ли ваш игрок, проверяя параметр анимации.Лучший способ - использовать RayCast ().Итак, что вам нужно сделать:

  1. Создайте слой с именем Ground и добавьте все платформы вашей сцены в этот слой.
  2. Создайте переменную bool

т.е.

bool isGrounded;
Создайте функцию, чтобы проверить, заземлен ли символ

Что-то вроде:

bool checkGrounded(){

    return Physics.Raycast(transform.position, Vector3.down, 2f, 1 << LayerMask.NameToLayer("Ground")));

} 

В этом ответе вы можете прочитать о задействованных параметрахв Raycast

Наконец внутри обновления проверьте, заземлен плеер или нет

Что-то вроде:

void Update(){
    isGrounded = checkGrounded();
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...