Character controller getting "stuck" when going through a door that should easily be "fit-into-able"

Essentially my character controller is trying to go through a doorway that is 2 blocks wide and 3 blocks tall (voxel world, each block is 1 unity unit^3 exactly). It walks to the door, gets noticeably slowed down for a few frames, then continues through the door.

It gets stuck if the height is 2 or over, or if the skin width is too high, variable with the height.

The character controller settings are shown here:

I have modified the standard assets script a bit, only the FirstPersonController script, I copied it into a new script to avoid losing changes, here is the script:

Code

Lines that I remember changing are shown in blue for your convenience :smile: edit: didnt show up, you can see them by looking for comments, telling what each change was for

using System;
using UnityEngine;
using UnityStandardAssets.CrossPlatformInput;
using UnityStandardAssets.Utility;
using Random = UnityEngine.Random;

namespace UnityStandardAssets.Characters.FirstPerson
{
    [RequireComponent(typeof (CharacterController))]
    [RequireComponent(typeof (AudioSource))]
    public class PlayerControl : MonoBehaviour
    {
        [SerializeField] private bool m_IsWalking;
        [SerializeField] private bool m_RawInput;
        [SerializeField] private float m_WalkSpeed;
        [SerializeField] private float m_RunSpeed;
        [SerializeField] [Range(0f, 1f)] private float m_RunstepLenghten;
        [SerializeField] private float m_JumpSpeed;

        [SerializeField] private float m_GravityMultiplier;
        [SerializeField] private MouseLook m_MouseLook;
        [SerializeField] private bool m_UseFovKick;
        [SerializeField] private FOVKick m_FovKick = new FOVKick();
        [SerializeField] private bool m_UseHeadBob;
        [SerializeField] private CurveControlledBob m_HeadBob = new CurveControlledBob();
        [SerializeField] private LerpControlledBob m_JumpBob = new LerpControlledBob();
        [SerializeField] private float m_StepInterval;
        [SerializeField] private AudioClip[] m_FootstepSounds;    // an array of footstep sounds that will be randomly selected from.
        [SerializeField] private AudioClip m_JumpSound;           // the sound played when character leaves the ground.
        [SerializeField] private AudioClip m_LandSound;           // the sound played when character touches back on ground.
      
        private Camera m_Camera;
        private bool m_Jump;
        private float m_YRotation;
        private Vector2 m_Input;
        private Vector3 m_MoveDir = Vector3.zero;
        private CharacterController m_CharacterController;
        private CollisionFlags m_CollisionFlags;
        private bool m_PreviouslyGrounded;
        private Vector3 m_OriginalCameraPosition;
        private float m_StepCycle;
        private float m_NextStep;
        private bool m_Jumping;
        private AudioSource m_AudioSource;
        private float horizontal;
        private float vertical;
      
        // Use this for initialization
        private void Start()
        {
            m_CharacterController = GetComponent<CharacterController>();
            m_Camera = Camera.main;
            m_OriginalCameraPosition = m_Camera.transform.localPosition;
            m_FovKick.Setup(m_Camera);
            m_HeadBob.Setup(m_Camera, m_StepInterval);
            m_StepCycle = 0f;
            m_NextStep = m_StepCycle/2f;
            m_Jumping = false;
            m_AudioSource = GetComponent<AudioSource>();
            m_MouseLook.Init(transform , m_Camera.transform);
          
         Screen.lockCursor = true;
       //Temporary, to reduce annoyance
        }

        private void GetInput(out float speed)
        {
            // Read input
            horizontal = (m_RawInput ? Input.GetAxisRaw("Horizontal") : Input.GetAxis("Horizontal"));
            vertical = (m_RawInput ? Input.GetAxisRaw("Vertical") : Input.GetAxis("Vertical"));
          
//Changed to give the option to not continue walking after key was released ^^^
          
            m_IsWalking = !Input.GetKey(KeyCode.LeftShift);
          
            // set the desired speed to be walking or running
            speed = m_IsWalking ? m_WalkSpeed : m_RunSpeed;
            m_Input = new Vector2(horizontal, vertical);
          
            // normalize input if it exceeds 1 in combined length:
            if (m_Input.sqrMagnitude > 1)
            {
                m_Input.Normalize();
            }
          
        }

      
        // Update is called once per frame
        private void Update()
        {

            //   RotateView();
            // the jump state needs to read here to make sure it is not missed
            if (!m_Jump && !m_Jumping)   //Changed because I did not want character to jump after hitting ground
                                                             //if jump was pressed mid air
            {
                m_Jump = Input.GetButton("Jump");
            }
          
            if ( m_CharacterController.isGrounded)
            {
                StartCoroutine(m_JumpBob.DoBobCycle());
        //        PlayLandingSound();
                m_MoveDir.y = 0f;
                m_Jumping = false;
            }
            if (!m_CharacterController.isGrounded && !m_Jumping && m_PreviouslyGrounded)
            {
                m_MoveDir.y = 0f;
            }
          
            m_PreviouslyGrounded = m_CharacterController.isGrounded;
          
          
          
        }
      
      
        private void PlayLandingSound()
        {
            m_AudioSource.clip = m_LandSound;
            m_AudioSource.Play();
            m_NextStep = m_StepCycle + .5f;
        }
      
      
        private void FixedUpdate()
        {
            float speed;
            GetInput(out speed);
          
            // always move along the camera forward as it is the direction that it being aimed at
            Vector3 desiredMove = transform.forward*m_Input.y + transform.right*m_Input.x;
          
            // get a normal for the surface that is being touched to move along it
            RaycastHit hitInfo;
            Physics.SphereCast(transform.position, m_CharacterController.radius, Vector3.down, out hitInfo,
                               m_CharacterController.height/2f);
          //desiredMove = Vector3.ProjectOnPlane(desiredMove, hitInfo.normal).normalized;
           //Commented this line out to avoid having the player sticking to edges that he/she walks off of, along
           //Side note, I don't fully understand what this does, but it seemed to fix my problem, is this fix a bad
           //solution? Also I'll remove the Physics.SphereCast later if this is a good fix, because it will be unneeded
           //correct?



            m_MoveDir.x = desiredMove.x*speed;
            m_MoveDir.z = desiredMove.z*speed;
          
          
            if (m_CharacterController.isGrounded)
            {

                if (m_Jump)
                {
                    m_MoveDir.y = m_JumpSpeed;
        //            PlayJumpSound();
                    m_Jump = false;
                    m_Jumping = true;
                }
            }
            else
            {
                m_MoveDir += Physics.gravity*m_GravityMultiplier*Time.fixedDeltaTime;
            }
            m_CollisionFlags = m_CharacterController.Move(m_MoveDir*Time.fixedDeltaTime);

            if ( (m_CollisionFlags  & CollisionFlags.Above) != 0 ){
                if(m_MoveDir.y > 0){
                    m_MoveDir.y = 0;
                }
            //    m_MoveDir += Physics.gravity*m_GravityMultiplier*Time.fixedDeltaTime;
            }[/COLOR]

            //This is my best guess for why it could be going wrong, is it shoving me into the floor somehow?
            //This edit was to make it so I don't stick to the ceiling momentarily when I hit my head on it.


            ProgressStepCycle(speed);
            UpdateCameraPosition(speed);
            RotateView();
        }
      
      
      
      
        private void ProgressStepCycle(float speed)
        {
            if (m_CharacterController.velocity.sqrMagnitude > 0 && (m_Input.x != 0 || m_Input.y != 0))
            {
                m_StepCycle += (m_CharacterController.velocity.magnitude + (speed*(m_IsWalking ? 1f : m_RunstepLenghten)))*
                    Time.fixedDeltaTime;
            }
          
            if (!(m_StepCycle > m_NextStep))
            {
                return;
            }
          
            m_NextStep = m_StepCycle + m_StepInterval;
          
        //    PlayFootStepAudio();
        }
      
      
        private void PlayJumpSound()
        {
            m_AudioSource.clip = m_JumpSound;
            m_AudioSource.Play();
        }
      
      
      
        private void PlayFootStepAudio()
        {
            if (!m_CharacterController.isGrounded)
            {
                return;
            }
            // pick & play a random footstep sound from the array,
            // excluding sound at index 0
            int n = Random.Range(1, m_FootstepSounds.Length);
            m_AudioSource.clip = m_FootstepSounds[n];
            m_AudioSource.PlayOneShot(m_AudioSource.clip);
            // move picked sound to index 0 so it's not picked next time
            m_FootstepSounds[n] = m_FootstepSounds[0];
            m_FootstepSounds[0] = m_AudioSource.clip;
        }
      
      
        private void UpdateCameraPosition(float speed)
        {
            Vector3 newCameraPosition;
            if (!m_UseHeadBob)
            {
                return;
            }
            if (m_CharacterController.velocity.magnitude > 0 && m_CharacterController.isGrounded)
            {
                m_Camera.transform.localPosition =
                    m_HeadBob.DoHeadBob(m_CharacterController.velocity.magnitude +
                                        (speed*(m_IsWalking ? 1f : m_RunstepLenghten)));
                newCameraPosition = m_Camera.transform.localPosition;
                newCameraPosition.y = m_Camera.transform.localPosition.y - m_JumpBob.Offset();
            }
            else
            {
                newCameraPosition = m_Camera.transform.localPosition;
                newCameraPosition.y = m_OriginalCameraPosition.y - m_JumpBob.Offset();
            }
            m_Camera.transform.localPosition = newCameraPosition;
        }
      
      

      
      
        private void RotateView()
        {
            m_MouseLook.LookRotation (transform, m_Camera.transform);
        }
      
      
        private void OnControllerColliderHit(ControllerColliderHit hit)
        {
            Rigidbody body = hit.collider.attachedRigidbody;
            //dont move the rigidbody if the character is on top of it
            if (m_CollisionFlags == CollisionFlags.Below)
            {
                return;
            }
          
            if (body == null || body.isKinematic)
            {
                return;
            }
            body.AddForceAtPosition(m_CharacterController.velocity*0.1f, hit.point, ForceMode.Impulse);
        }
      
  
      
      
      
    }
  
}

Bonus question: character teleports upwards the step offset value sometimes when walking into a wall sideways. I assume this may be because the slope is set to 90, but I set that to 90 so that the character wouldn’t get stuck when angling against a wall trying to step up to the side.
Shown here: Screenshot - 4154c9a3b8fe67bdc847be90afed2675 - Gyazo
If the slope is set to anything less than 90 the character will not walk up the block to the left, and be stuck until you look farther left.

Bump

Your CharacterController values seem OK so I think you’re going to have to setup some debugging tests to really figure out the problem.

First thing I would try is write a script that simply moves your character forward (characterController.Move(transform.forward * 10 * Time.deltaTime);), disable your FPS controller script, line them up in front of a door and see if they walk through without an issue. If so, you at least know that it’s somewhere in the FPS controller script.

Do the same test for the “bonus question” issue as well.

That will at least give you a starting point to begin debugging. Report your findings.

Video of getting caught on door: https://dl.dropboxusercontent.com/u/85261353/Snagging on door.mp4
Video of teleporting upwards: http://gyazo.com/ce9a9088cbe7175e151b944189e0d2c6

Both of these videos were done using the debug thing you suggested.

Vector3 desiredMove = transform.forward* /*10 * Time.deltaTime */ m_Input.y + transform.right* 20 * Time.deltaTime; //   m_Input.x;

edit:
After further experimentation, I have noticed that when I put my step offset to 2, my character can only go through the door if I jump.

It seems I can fit through the door un-slowed-down if my step offset and my height added together are less than the door height

Nice music :hushed:

It’s not the clearest in a gif, but I’m able to duplicate your issue:

If I delete the ceiling, it appears to work no problem. I’ll start playing with values and see if I can come up with anything.

Setting the Step Offset to 0.75 from 1.2 seems to resolve the issue while still allowing you to walk on blocks:

My guess is it calculates the step offset from the middle of the capsule, not the bottom, so it’s hitting the ceiling of the door frame for a moment (1.1 height + 1.2 step offset > 2 block height?). Just a guess though.

Thats very interesting, then why is it that you cannot set the step offset to anything above your players height?

This is still an unadressed problem in 2017 with Unity 5.6.

FPSController (which I used for the first time out of lazyness, I usually test with my real characters anyway)
has a height of 1.7 and a radius of 0.2, the doorway is 2 high and 1 wide, it couldn’t go through!

Decreasing the Step Offset from Standard 0.3 to 1.5 fixed it. This is a student-level error from Unity.
I have given up hoping Unity getting a decent team, they simply don’t care about us free developers, they just push their agendas and wait for the next paycheck from licensing.

My solution based on the idea in this thread:

  1. Create a box collider that is a child of the character and make its scale 0.1.
  2. Put the child a little above and in front of the head of the character
  3. Attach this script to the child:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class StepChanger : MonoBehaviour {

    GameObject pawn; // parent - character
    CharacterController controller;

    // Use this for initialization
    void Start () {
    pawn = this.gameObject.transform.parent.gameObject;
    controller = pawn.GetComponent<CharacterController>();
    controller.height = 1.8f;
    controller.radius = 0.4f;
    }
 
    // Update is called once per frame
    void Update () {
      
    }
 
    void OnTriggerEnter(Collider other)
    {
       print("colliding with: " + other.gameObject.name);
        if (other.gameObject.name.Contains("Block")) //the name of the object that is the ceilling
        {
        print("we hit ceiling dropping step to 0.1");
        controller.stepOffset = 0.1f;
            
        }
    }
 
     void OnTriggerExit(Collider other)
    {
       print("collision exit");
        if (other.gameObject.name.Contains("Block")) //the name of the object that is the ceilling
        {
        print("collision exit, return step offset to 1.1f");
        controller.stepOffset = 1.1f;
            
        }
    }
}