像F-Zero GX

时间:2018-11-08 04:48:01

标签: c# unity3d game-physics rigid-bodies lerp

我已经在这个项目上工作了一段时间-我一直在尝试获得一个简单的立方体,玩家可以使用箭头向左或向右移动,可以绕管道旋转,类似于赛车手可以移动的方式。在F-Zero GX中的管道周围:

https://www.youtube.com/watch?v=RlS1i7aCnvg

现在,我知道我不是第一个尝试实施这些物理技术的人,因为我已经阅读了尝试实施类似技术的人员的两三个主题。问题是我尝试使用这些不同的方法或它们的变体。其中之一已经非常接近我想要的行为了-但是,在立方体从地面移到整个管道壁的一半以上之后,播放器立方体仍会意外地在其X轴上旋转。这是我正在谈论的关卡构建的视觉效果:

enter image description here

整个想法是,立方体可以在管道的整个壁上“移动”或“行走”,就像您在F-Zero视频的管道部分中看到的那样,当然仍然可以向前移动。我有一个圆柱体,它实际上位于您看到的圆柱体内,实际上只是一个凸形触发器-用来确保当立方体在管道模型内时,玩家立方体的重力刻度关闭。

我已经接近完成这项工作了,但这是看玩家能够在一个完整的圆圈内一直移动的一个问题。以圆周运动完全回到底部或玩家进入管道时开始的位置。但是,当完成墙体移动的一半时,立方体喜欢“翻转”。我在另一篇文章中读到有人实际上改变了物体的刚体,以使“船”保持直立,这与它有关吗?

https://forum.unity.com/threads/f-zero-esque-control-question.157909/

“基本上,我给我的车辆悬停了裙子。”

我应该考虑相应地更改刚体的形状吗?最好的资源是什么?还是应该改用字符控制器?我仍然依靠刚体设置,但是在阅读了这些内容之后,我已经对这种可能性敞开心mind。

此Raycast代码非常接近我想要的内容,没有骰子:

float distDown = Mathf.Infinity;
            RaycastHit hitDown;
            if (Physics.SphereCast(transform.position, 0.25f, -transform.up, out hitDown, 5))
            {
                distDown = hitDown.distance;
            }
transform.rotation = Quaternion.Lerp(transform.rotation, Quaternion.LookRotation(Vector3.Cross(transform.right, hitDown.normal), hitDown.normal), Time.deltaTime * 1.0f);

我尝试探索的另一种可能性是创建自己的重力拉力-我什至有一个尝试做到这一点的类FauxGravity,它附着在玩家与之碰撞的物体上(在本例中为圆柱体): / p>

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

public class FauxGravity : MonoBehaviour {

    public Transform attractor;
    public Transform player;
    public Transform collider;
    Rigidbody attractRB;
    Rigidbody playerRB;
    Vector3 myNormal;
    public Vector3 vectorFromPipeCenter;
    public Vector3 forwardPipeVector;
    Vector3 project2Center;
    public Vector3 pipeGravityPull;
    public int gravity = 1;

    // Use this for initialization
    void Start () {

        //Physics.gravity.magnitude = 0;
    }

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

    }

    //private void OnCollisionEnter(Collision collision)
    //{
    //    //collision.gameObject.GetComponent<Rigidbody>().transform.up = Vector3.zero;
    //    //player.gameObject.GetComponent<Rigidbody>().useGravity = false;
    //    myNormal = playerRB.transform.up;
    //    //playerRB = collision.gameObject.GetComponent<Rigidbody>();

    //    gravity = 0;
    //    //playerRB.isKinematic = true;
    //    //player.gameObject.GetComponent<Rigidbody>().AddRelativeForce()
    //}

    public void FixedUpdate()
    {
        if (gravity == 0)
        {
            //playerRB.isKinematic = true;
            //Debug.Log("Gravity is 0.");
            attractRB = attractor.GetComponent<Rigidbody>();
            playerRB = player.GetComponent<Rigidbody>();
            //playerRB.AddForce(-10 * playerRB.mass * myNormal);
            Debug.Log("PlayerRB position: " + playerRB.position);
            Debug.Log("AttractRB position: " + attractRB.position);
            vectorFromPipeCenter = playerRB.position - attractRB.position;
            vectorFromPipeCenter.z = 0;
            //vectorFromPipeCenter.Normalize();
            Debug.Log("Player distance from pipe center: " + vectorFromPipeCenter.magnitude);
            Debug.Log("Player vector from pipe center" + vectorFromPipeCenter);
            //vectorFromPipeCenter = attractRB.position - playerRB.position;
            Debug.Log("playerRB forward is " + playerRB.rotation.z);
            Debug.Log("playerRB magnitude is " + player.forward.magnitude);
            forwardPipeVector = player.forward.magnitude * Vector3.forward;
            Debug.Log("Player forward vector? " + forwardPipeVector);
            // or
            //Vector forwardPipeVector = pipeTransform.forward;
            // And finally
            project2Center = Vector3.Project(vectorFromPipeCenter, forwardPipeVector);
            Debug.Log("What is project2Center? " + project2Center);
            float radiusFromCrossectionCenter = vectorFromPipeCenter.magnitude;
            double playerY = System.Convert.ToDouble(playerRB.position.y);
            double playerX = System.Convert.ToDouble(playerRB.position.x);
            //float inverseTan = System.Convert.ToSingle(System.Math.Atan(playerY / playerX));
            //Debug.Log("Normal is: " + Quaternion.AngleAxis(inverseTan, forwardPipeVector));
            // pipe pull force = distance from pipe center to power 2
            //pipeGravityPull = Quaternion.AngleAxis(inverseTan, playerRB.transform.forward) * project2Center * Mathf.Pow ( (radiusFromCrossectionCenter * 1 ), 2 );
            pipeGravityPull = new Vector3(playerRB.position.x, radiusFromCrossectionCenter - playerRB.position.y, 0)/Mathf.Sqrt(Mathf.Pow(playerRB.position.x,2) + Mathf.Pow((radiusFromCrossectionCenter-playerRB.position.y),2));
            Debug.Log("Pipe gravity vector? " + pipeGravityPull);
            //playerRB.useGravity = true;
            Debug.DrawLine(pipeGravityPull, pipeGravityPull);
            Debug.Log("Adding force from FG");
            //playerRB.AddForce(pipeGravityPull);
        }
        if (gravity == 1)
        {
            player.GetComponent<Rigidbody>().useGravity = true;
            //playerRB.isKinematic = false;
        }
    }

    private void OnCollisionExit(Collision collision)
    {
        //Debug.Log("Gravity is 1 again.");
        //player.gameObject.GetComponent<Rigidbody>().useGravity = true;
        //gravity = 1;
        //playerRB.useGravity = true;
        playerRB.isKinematic = false;
        //playerRB.AddForce(10, 20, 0);
    }

    void gravityAttract(Collider colliderObject)
    {
        var rb = colliderObject.GetComponent<Rigidbody>();
        rb.AddForce(Vector3.down * 30, ForceMode.Force);
        rb.AddForce(Vector3.up * 30, ForceMode.Force);
    }
}

我是否需要将我的FixedUpdate逻辑移到Update方法?根据我父亲的说法(我是天体物理学家),我用这种方法探索的最后一个算法本质上是通过获取玩家的矢量和他/她所行进的管道的横截面来确保拉力始终等于1。

这是我的玩家的移动类,该类进行了多次注释和未注释的尝试来旋转多维数据集,以使多维数据集的下侧在爬升时始终面对管道壁:

using System;
using UnityEngine;

public class PlayerMovement : MonoBehaviour {

    public Rigidbody rb;

    public float forwardForce = 2000f;
    public float sidewaysForce = 500f;
    public Boolean fauxGravity = false;
    public Vector3 distanceFromPipeCenter = new Vector3(0, 0, 0);
    public Vector3 pipePull = new Vector3(0,0,0);

    // Use this for initialization
    void Start () {

    }

    // Update is called once per frame use fixed update for Unity Fizzix
    void FixedUpdate () {
        //distanceFromPipeCenter.Normalize();
        //add forward force
        rb.AddForce(0, 0, forwardForce * Time.deltaTime);

        if (Input.GetKey(KeyCode.RightArrow)/* && !fauxGravity*/)
        {
            rb.AddForce(sidewaysForce * Time.deltaTime, 0, 0, ForceMode.VelocityChange);
        }

        if (Input.GetKey(KeyCode.LeftArrow)/* && !fauxGravity*/)
        {
            rb.AddForce(-sidewaysForce * Time.deltaTime, 0, 0, ForceMode.VelocityChange);
        }
        if (pipePull.x == 0)
        {
            pipePull.x = 1;
        }
        if (pipePull.y == 0)
        {
            pipePull.y = 1;
        }
        //transform.rotation = Quaternion.identity;
        //pipePull.z = 0;
        if (Input.GetKey(KeyCode.RightArrow) && fauxGravity)
        {
            Debug.Log("Right pressed");
            //Debug.Log("Rotation before: " + rb.rotation);
            //rb.rotation = rb.rotation * Quaternion.FromToRotation(rb.transform.up, pipePull);
            //rb.rotation = Quaternion.Lerp(rb.rotation, Quaternion.LookRotation(Vector3.Cross(rb.transform.right, pipePull), pipePull), Time.deltaTime * 5.0f);
            //Debug.Log("Rotation after: " + rb.rotation);
            //if (distanceFromPipeCenter.y < 0)
            //{
            //    Debug.Log("Right A, pull positive: " +pipePull);
                //rb.AddForce(sidewaysForce * pipePull.x * Time.deltaTime, sidewaysForce * pipePull.y * Time.deltaTime, 0, ForceMode.VelocityChange);
            //}
            //else
            //{
            //    Debug.Log("Right B, pull negative: " + distanceFromPipeCenter);
            //    rb.AddForce(sidewaysForce * pipePull.x * Time.deltaTime, -sidewaysForce * pipePull.y * Time.deltaTime, 0, ForceMode.VelocityChange);
            //}
            //Debug.Log(rb.angularVelocity);
            //float headingDeltaAngle = Input.GetAxis("Horizontal") * Time.deltaTime * sidewaysForce;
            //Quaternion headingDelta = Quaternion.AngleAxis(headingDeltaAngle, transform.up);
            //align with surface normal
            //transform.rotation = Quaternion.FromToRotation(transform.up, distanceFromPipeCenter) * transform.rotation;
            //apply heading rotation
            //transform.rotation = headingDelta * transform.rotation;
        }
        if (Input.GetKey(KeyCode.LeftArrow) && fauxGravity)
        {
            Debug.Log("Left pressed");
            //Debug.Log("Rotation before: " + rb.rotation);
            //rb.rotation = rb.rotation * Quaternion.FromToRotation(rb.transform.up, pipePull);
            //rb.rotation = Quaternion.Lerp(rb.rotation, Quaternion.LookRotation(Vector3.Cross(rb.transform.right, pipePull), pipePull), Time.deltaTime * 5.0f);
            //Debug.Log("Rotation after: " + rb.rotation);
            //if (distanceFromPipeCenter.y < 0)
            //{
            //    Debug.Log("Left A, pull positive: " +pipePull);
                //rb.AddForce(-sidewaysForce * pipePull.x * Time.deltaTime, sidewaysForce * pipePull.y * Time.deltaTime, 0, ForceMode.VelocityChange);
            //}
            //else
            //{
            //    Debug.Log("Left B, pull negative: " + distanceFromPipeCenter);
            //    rb.AddForce(-sidewaysForce * pipePull.x * Time.deltaTime, -sidewaysForce * pipePull.y * Time.deltaTime, 0, ForceMode.VelocityChange);
            //}
            //Debug.Log(rb.angularVelocity);
            //float headingDeltaAngle = Input.GetAxis("Horizontal") * Time.deltaTime * sidewaysForce;
            //Quaternion headingDelta = Quaternion.AngleAxis(headingDeltaAngle, transform.up);
            //align with surface normal
            //transform.rotation = Quaternion.FromToRotation(transform.up, distanceFromPipeCenter) * transform.rotation;
            //apply heading rotation
            //transform.rotation = headingDelta * transform.rotation;
        }
        if (fauxGravity)
        {
            rb.useGravity = false;

            ///*We get the user input and modifiy the direction the ship will face towards*/
            //float yaw = Time.deltaTime * Input.GetAxis("Horizontal");
            ///*We want to save our current transform.up vector so we can smoothly change it later*/
            //Vector3 prev_up = rb.transform.up;
            ///*Now we set all angles to zero except for the Y which corresponds to the Yaw*/
            //transform.rotation = Quaternion.Euler(0, yaw, 0);

            //RaycastHit hit;
            //if (Physics.Raycast(transform.position, -prev_up, out hit))
            //{
            //    Debug.DrawLine(transform.position, hit.point);

            //    /*Here are the meat and potatoes: first we calculate the new up vector for the ship using lerp so that it is smoothed*/
            //    Vector3 desired_up = Vector3.Lerp(prev_up, hit.normal, Time.deltaTime /** pitch_smooth*/);
            //    /*Then we get the angle that we have to rotate in quaternion format*/
            //    Quaternion tilt = Quaternion.FromToRotation(transform.up, desired_up);
            //    /*Now we apply it to the ship with the quaternion product property*/
            //    transform.rotation = tilt * transform.rotation;

            //    /*Smoothly adjust our height*/
            //    //smooth_y = Mathf.Lerp(smooth_y, hover_height - hit.distance, Time.deltaTime * height_smooth);
            //    //transform.localPosition += prev_up * smooth_y;
            //}
            //float distForward = Mathf.Infinity;
            //RaycastHit hitForward;

            //if (Physics.SphereCast(transform.position, 0.25f, -transform.up + transform.forward, out hitForward, 5))
            //{
            //    distForward = hitForward.distance;
            //}
            float distDown = Mathf.Infinity;
            RaycastHit hitDown;
            if (Physics.SphereCast(transform.position, 0.25f, -transform.up, out hitDown, 5))
            {
                distDown = hitDown.distance;
            }
            //float distBack = Mathf.Infinity;
            //RaycastHit hitBack;
            //if (Physics.SphereCast(transform.position, 0.25f, -transform.up + -transform.forward, out hitBack, 5))
            //{
            //    distBack = hitBack.distance;
            //}

            //if (distForward < distDown && distForward < distBack)
            //{
            //    transform.rotation = Quaternion.Lerp(transform.rotation, Quaternion.LookRotation(Vector3.Cross(transform.right, hitForward.normal), hitForward.normal), Time.deltaTime * 5.0f);
            //}
            //else if (distDown < distForward && distDown < distBack)
            //{
            transform.rotation = Quaternion.Lerp(transform.rotation, Quaternion.LookRotation(Vector3.Cross(transform.right, hitDown.normal), hitDown.normal), Time.deltaTime * 1.0f);
            //}
            //else if (distBack < distForward && distBack < distDown)
            //{
            //    transform.rotation = Quaternion.Lerp(transform.rotation, Quaternion.LookRotation(Vector3.Cross(transform.right, hitBack.normal), hitBack.normal), Time.deltaTime * 5.0f);
            //}

            //GetComponent<Rigidbody>().AddForce(-transform.up * Time.deltaTime * 10);
        }
        if (rb.position.y <-1f)
        {
            FindObjectOfType<GameManagement>().EndGame();
        }
    }

    //void OnCollisionEnter(Collision collision)
    //{
    //    //collision.gameObject.GetComponent<Rigidbody>().transform.up = Vector3.zero;
    //    //player.gameObject.GetComponent<Rigidbody>().useGravity = false;
    //    System.Console.WriteLine("Player has collided with: " + collision.collider.name);
    //    if(collision.gameObject.name == "PipeBasic 1")
    //    {
    //        System.Console.WriteLine("Player Collided with Pipe");
    //        fauxGravity = true;
    //    }
    //    //playerRB.isKinematic = true;
    //    //player.gameObject.GetComponent<Rigidbody>().AddRelativeForce()
    //}
}

任何人都可以提供其他提示,技巧和窍门吗?我认为使用Raycast甚至SphereCast可能是我想要的最接近的方法,但是我只是坚持如何准确地实现它而不会冒着在旋转播放器立方体的风险。错误的一面,搞砸了物理相互作用。我可能对此是否考虑过多?

非常感谢大家。

更新:

我一直在改变播放器Y速度的逻辑,因为在圆周中移动必须涉及特定的物理原理,例如下页给出的方程式:

https://www.physicsclassroom.com/class/circles/Lesson-1/Speed-and-Velocity

但是,我仍然在努力寻找合适的时间值来对等式的顶部进行除法。这是到目前为止我向右移动时的移动代码:

if (distanceFromPipeCenter.y < 0)
            {
                //Debug.Log("Right A, pull positive: " + pipePull);
                //Debug.Log("X Pull: " + pipePull.x);
                rb.AddForce(sidewaysForce /** pipePull.x*/ * Time.deltaTime, (sidewaysForce * 2 * radiusInPipe * (float)Math.PI) / Time.deltaTime, 0, ForceMode.VelocityChange);
            }
            else if(distanceFromPipeCenter.x>=3)
            {
                //Debug.Log("Right B, pull negative: " + distanceFromPipeCenter);

                rb.AddForce(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, (sidewaysForce * 2 * radiusInPipe * (float)Math.PI) / Time.deltaTime, 0, ForceMode.VelocityChange);
            }
            else if(distanceFromPipeCenter.x>0)
            {

                rb.AddForce(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, (sidewaysForce * 2 * radiusInPipe * (float)Math.PI) / Time.deltaTime, 0, ForceMode.VelocityChange);
            }
            else if(distanceFromPipeCenter.y>0)
            {
                Debug.Log("X distance: " + distanceFromPipeCenter.x);
                Debug.Log("Radius: " + radiusInPipe);
                Debug.Log("Frame count? " + Time.fixedDeltaTime);
                Vector3 pull = new Vector3(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((-sidewaysForce * 2 * radiusInPipe * (float)Math.PI) / Time.fixedDeltaTime) * Time.deltaTime, 0);
                Debug.Log("About to apply: " + pull );
                rb.AddForce(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, (-sidewaysForce * 2 * radiusInPipe * (float)Math.PI) / Time.deltaTime, 0, ForceMode.VelocityChange);
            }
            else
            {
                rb.AddForce(sidewaysForce /** pipePull.x*/ * Time.deltaTime, (-sidewaysForce * 2 * radiusInPipe * (float)Math.PI) / Time.deltaTime, 0, ForceMode.VelocityChange);
            }

似乎被Time.deltaTime所分割只是让玩家跳了一段很大的距离-不是我想要的。我还尝试使用自游戏开始以来的时间作为分频器,然后将整个结果乘以Time.deltaTime-但您可能可以推断出,帧只会增加,然后最终以这种方式降低速度

还有其他想法或建议吗?

2 个答案:

答案 0 :(得分:1)

1。禁用重力

当玩家进入管道时如何触发触发器,您可以使用重力禁用其刚体。我尝试了一下,但似乎可行,尽管我的立方体在靠近顶部时会翻转,因为它没有粘在地板上。

2。仅在循环管道时在播放器下对撞机

对此的一种可能的解决方法是在玩家(作为玩家的孩子)下方产生一个碰撞器,该碰撞器也位于管道之外-因此,玩家将永远不会从管道上漂浮下来。 注意:我自己没有进行测试,因为我的测试缸只有很多四边形。

example image

  • 绿色圆圈是阻止玩家从管道两侧漂浮的对撞机。

  • 绿色方块被触发以激活绿色圆圈对撞机,ontriggerenter激活它,ontriggerexit禁用它。

答案 1 :(得分:0)

所以,事实证明,我对整个游戏的想法太高了,或者说,我忽略了这个游戏引擎的关键和令人惊叹的功能。

我试图计算所有这些运动,因为我认为我们必须计算出游戏世界起源的所有力和运动。在整个过程中,我一直在自言自语,心想:“伙计,如果您只需在对象的局部坐标中添加力/扭矩,那就容易得多了。”

输入AddRelativeForce()而不是AddForce()。

有了它,并利用了我穿过的隧道的半径,并且在没有重力的情况下,我能够更轻松地获得所需的效果。

这时此代码可能有点肿,但是关键要点是我使用的nextLeft和nextRight Vector3对象。我希望这可以帮助某人。让我知道是否可以澄清任何内容或以其他任何方式使这篇文章更好。

void FixedUpdate () {
        //distanceFromPipeCenter.Normalize();
        //add forward force
        rb.AddForce(0, 0, forwardForce * Time.deltaTime);

        if (Input.GetKey(KeyCode.RightArrow) && !fauxGravity)
        {
            rb.AddForce(sidewaysForce * Time.deltaTime, 0, 0, ForceMode.VelocityChange);
        }

        if (Input.GetKey(KeyCode.LeftArrow) && !fauxGravity)
        {
            rb.AddForce(-sidewaysForce * Time.deltaTime, 0, 0, ForceMode.VelocityChange);
        }
        if(radiusInPipe == 0)
        {
            radiusInPipe = 1;
        }
        //transform.rotation = Quaternion.identity;
        //pipePull.z = 0;
        Vector3 nextRight = new Vector3(sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((2 * radiusInPipe * (float)Math.PI) / (sidewaysForce)) * Time.fixedDeltaTime, 0);
        Vector3 nextLeft = new Vector3(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((2 * radiusInPipe * (float)Math.PI) / (sidewaysForce)) * Time.fixedDeltaTime, 0);
        if (Input.GetKey(KeyCode.RightArrow) && fauxGravity)
        {
            //Debug.Log("Right pressed");
            //Debug.Log("Rotation before: " + rb.rotation);
            //rb.rotation = rb.rotation * Quaternion.FromToRotation(rb.transform.up, pipePull);
            //rb.rotation = Quaternion.Lerp(rb.rotation, Quaternion.LookRotation(Vector3.Cross(rb.transform.right, pipePull), pipePull), Time.deltaTime * 5.0f);
            //Debug.Log("Rotation after: " + rb.rotation);
            //Vector3 next = new Vector3(sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((2 * radiusInPipe * (float)Math.PI) / (sidewaysForce)) * Time.fixedDeltaTime, 0);
            if (distanceFromPipeCenter.y < 0)
            {
                //Debug.Log("Right A, pull positive: " + pipePull);
                //Debug.Log("X Pull: " + pipePull.x);
                Debug.Log("Current deltaTime: " + Time.deltaTime);
                //Vector3 next = new Vector3(sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((2 * radiusInPipe * (float)Math.PI)/(sidewaysForce))  * Time.fixedDeltaTime, 0);
                Debug.Log("About to apply: " + nextRight);
                rb.AddRelativeForce(nextRight, ForceMode.VelocityChange);
            }
            else if(distanceFromPipeCenter.x>=3)
            {
                //Debug.Log("Right B, pull negative: " + distanceFromPipeCenter);

                //rb.AddForce(sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((2 * radiusInPipe * (float)Math.PI) * (sidewaysForce)) * Time.deltaTime, 0, ForceMode.VelocityChange);
                rb.AddRelativeForce(nextRight, ForceMode.VelocityChange);
            }
            else if(distanceFromPipeCenter.x>0)
            {
                rb.AddRelativeForce(nextRight, ForceMode.VelocityChange);
                //rb.AddForce(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((2 * radiusInPipe * (float)Math.PI) * (sidewaysForce)) * Time.deltaTime, 0, ForceMode.VelocityChange);
            }
            else if(distanceFromPipeCenter.y>0)
            {
                Debug.Log("X distance: " + distanceFromPipeCenter.x);
                Debug.Log("Radius: " + radiusInPipe);
                Debug.Log("Frame count? " + Time.fixedDeltaTime);
                Vector3 pull = new Vector3(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, -((2 * radiusInPipe * (float)Math.PI) / (sidewaysForce)) * Time.fixedDeltaTime, 0);
                Debug.Log("About to apply: " + pull );

                //rb.AddForce(-sidewaysForce /** pipePull.x*/ * Time.deltaTime, -((2 * radiusInPipe * (float)Math.PI) * sidewaysForce) * Time.deltaTime, 0, ForceMode.VelocityChange);
                rb.AddRelativeForce(nextRight, ForceMode.VelocityChange);
            }
            else
            {
                rb.AddRelativeForce(nextRight, ForceMode.VelocityChange);
                //rb.AddForce(sidewaysForce /** pipePull.x*/ * Time.deltaTime, ((2 * radiusInPipe * (float)Math.PI) / (sidewaysForce)) * Time.fixedDeltaTime, 0, ForceMode.VelocityChange);
            }
            //Debug.Log(rb.angularVelocity);
            float headingDeltaAngle = Input.GetAxis("Horizontal") * Time.deltaTime * sidewaysForce;
            Quaternion headingDelta = Quaternion.AngleAxis(headingDeltaAngle, -transform.up);
            headingDelta.y = 0;
            headingDelta.x = 0;
            //align with surface normal
            transform.rotation = Quaternion.FromToRotation(-transform.up, distanceFromPipeCenter) * transform.rotation;
            //apply heading rotation
            transform.rotation = headingDelta * transform.rotation;
        }
        if (Input.GetKey(KeyCode.LeftArrow) && fauxGravity)
        {
            //Debug.Log("Left pressed");
            //Debug.Log("Rotation before: " + rb.rotation);
            //rb.rotation = rb.rotation * Quaternion.FromToRotation(rb.transform.up, pipePull);
            //rb.rotation = Quaternion.Lerp(rb.rotation, Quaternion.LookRotation(Vector3.Cross(rb.transform.right, pipePull), pipePull), Time.deltaTime * 5.0f);
            //Debug.Log("Rotation after: " + rb.rotation);
            //if (distanceFromPipeCenter.y < 0)
            //{
            //    //Debug.Log("Left A, pull positive: " + pipePull);
            //    rb.AddForce(-sidewaysForce/* * pipePull.x*/ * Time.deltaTime, sidewaysForce /pipePull.y * Time.deltaTime, Math.Abs(pipePull.z) * Time.deltaTime, ForceMode.VelocityChange);
            //}
            //else if(distanceFromPipeCenter.x >=0)
            //{
            //    //Debug.Log("Left B, pull negative: " + distanceFromPipeCenter);
            //    rb.AddForce(sidewaysForce /** pipePull.x*/ * Time.deltaTime, sidewaysForce * -pipePull.y * Time.deltaTime, Math.Abs(pipePull.z) * Time.deltaTime, ForceMode.VelocityChange);
            //}
            //else
            //{
            //    //rb.AddForce(sidewaysForce /** pipePull.x*/ * Time.deltaTime, sidewaysForce * -pipePull.y * Time.deltaTime, 0, ForceMode.VelocityChange);
            //    Debug.Log("Deadzone. PAUSE. Pull: " + pipePull);
            //}
            rb.AddRelativeForce(nextLeft, ForceMode.VelocityChange);
            //Debug.Log(rb.angularVelocity);
            float headingDeltaAngle = Input.GetAxis("Horizontal") * Time.deltaTime * sidewaysForce;
            Quaternion headingDelta = Quaternion.AngleAxis(headingDeltaAngle, -transform.up);
            headingDelta.y = 0;
            headingDelta.x = 0;
            //align with surface normal
            transform.rotation = Quaternion.FromToRotation(-transform.up, distanceFromPipeCenter) * transform.rotation;
            //apply heading rotation
            transform.rotation = headingDelta * transform.rotation;

        }
        if (fauxGravity)
        {
            rb.useGravity = false;
            if(rb.position.y-ground.position.y > 2)
            {
                roller.isTrigger = false;
            }
            else
            {
                roller.isTrigger = true;
            }
            //rb.AddForce(pipePull);
            //roller.isTrigger = false;
            ///*We get the user input and modifiy the direction the ship will face towards*/
            //float yaw = Time.deltaTime * Input.GetAxis("Horizontal");
            ///*We want to save our current transform.up vector so we can smoothly change it later*/
            //Vector3 prev_up = rb.transform.up;
            ///*Now we set all angles to zero except for the Y which corresponds to the Yaw*/
            //transform.rotation = Quaternion.Euler(0, yaw, 0);

            //RaycastHit hit;
            //if (Physics.Raycast(transform.position, -prev_up, out hit))
            //{
            //    Debug.DrawLine(transform.position, hit.point);

            //    /*Here are the meat and potatoes: first we calculate the new up vector for the ship using lerp so that it is smoothed*/
            //    Vector3 desired_up = Vector3.Lerp(prev_up, hit.normal, Time.deltaTime /** pitch_smooth*/);
            //    /*Then we get the angle that we have to rotate in quaternion format*/
            //    Quaternion tilt = Quaternion.FromToRotation(transform.up, desired_up);
            //    /*Now we apply it to the ship with the quaternion product property*/
            //    transform.rotation = tilt * transform.rotation;

            //    /*Smoothly adjust our height*/
            //    //smooth_y = Mathf.Lerp(smooth_y, hover_height - hit.distance, Time.deltaTime * height_smooth);
            //    //transform.localPosition += prev_up * smooth_y;
            //}
            //float distForward = Mathf.Infinity;
            //RaycastHit hitForward;

            //if (Physics.SphereCast(transform.position, 0.25f, -transform.up + transform.forward, out hitForward, 5))
            //{
            //    distForward = hitForward.distance;
            //}
            float distDown = Mathf.Infinity;
            RaycastHit hitDown;
            if (Physics.SphereCast(transform.position, 0.25f, -transform.up, out hitDown, 5))
            {
                distDown = hitDown.distance;
            }
            //float distBack = Mathf.Infinity;
            //RaycastHit hitBack;
            //if (Physics.SphereCast(transform.position, 0.25f, -transform.up + -transform.forward, out hitBack, 5))
            //{
            //    distBack = hitBack.distance;
            //}

            //if (distForward < distDown && distForward < distBack)
            //{
            //    transform.rotation = Quaternion.Lerp(transform.rotation, Quaternion.LookRotation(Vector3.Cross(transform.right, hitForward.normal), hitForward.normal), Time.deltaTime * 5.0f);
            //}
            //else if (distDown < distForward && distDown < distBack)
            //{
            //transform.rotation = Quaternion.Lerp(transform.rotation, Quaternion.LookRotation(Vector3.Cross(transform.right, hitDown.normal), hitDown.normal), Time.deltaTime * 1.0f);
            //}
            //else if (distBack < distForward && distBack < distDown)
            //{
            //    transform.rotation = Quaternion.Lerp(transform.rotation, Quaternion.LookRotation(Vector3.Cross(transform.right, hitBack.normal), hitBack.normal), Time.deltaTime * 5.0f);
            //}

            //GetComponent<Rigidbody>().AddForce(-transform.up * Time.deltaTime * 10);
            transformDifference = rb.position;
            previousFrame = Time.frameCount;
        }
        if (rb.position.y <-1f)
        {
            FindObjectOfType<GameManagement>().EndGame();
        }
    }