Unity如何实现在球表面移动并朝向一目标点 (二)

时间:2024-03-15 14:42:54
上一次我们讲述了如何在球上运动并且始终朝向一点,其实之前的代码是有BUG的,因为我们点击的方向不同,他的转向也会不同,所以我们可能会+angle,也可能-angle。我会在这一章的例子中把之前的问题顺便阐述清楚。
我们之前做的是始终朝向一点运动,那么,我们是否能朝向我们所运动的方向呢?似乎这样更符合我们的认知。还有,之前在球上的操作是鼠标长按,鼠标在哪小人就在哪,那么我们是否可以实现点击球上一点,让小人自己走过去呢?今天就来解决这些问题。
首先,我们解决简单的鼠标点击问题,以前我们通过射线碰撞获取碰撞点,直接把碰撞点的位置赋给小人,这样显然有些操作上的不方便,那么我们现在点击一点,小人在球面上运动到目标点。如何实现呢?我们有这样一种思路:
Unity如何实现在球表面移动并朝向一目标点 (二)
如图,我们正常是从直线的一端走向另一端,但是我们现在可以先计算出从起始点到目标点的直线移动坐标变化,每帧记录下这些坐标,然后每帧发射射线指向这些坐标,与球面碰撞的位置就是我们要在曲面上运动的轨迹,有了这个思路,我们的移动问题就解决了。
接下来就是文章一开头提到的转向问题:
Unity如何实现在球表面移动并朝向一目标点 (二)
如图,我们之前只考虑到一种转向问题,就是如图曲线的方向,这只是在面朝自己的这一面向下走时应该转的方向。
首先我们必须明确的一点是小人是有父对象的,并且父对象的正方向始终不变,为的就是和正确方向计算得出正确转向,那么我们的问题就来了,既然父物体对象正方向一直不变,怎么可能始终让子物体朝着一个方向转动呢,我们的角度一直都是非负的!不可能自动调整。所以我们就必须分情况讨论:
首先我们确定一个前提,就是父物体的x轴始终朝向我们,这样方便我们描述方向
1.面朝我们的球的一面:
面朝我们,就拿刚才第二个图来说,此时我们看到的球就是面朝着我们的一面。在这一面上,如果我们点击让小人往下走,就必须让小人顺时针旋转;如果往上走,就必须让小人逆时针旋转。
2.背朝我们这面:
与面朝我们这一面刚好相反,往下走逆时针,往上走顺时针。
这样,我们上一章的问题也理清了,知道了解决的方案,代码也就一起给上:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class MoveOnSphere : MonoBehaviour {
    private Vector3 desPos;
    private bool isMove;//是否移动
    private Vector3 currentPos;
    private RaycastHit hit;
    public GameObject sphere;
   
        void Update () {
        if (Input.GetMouseButtonDown(0))
        {
            Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);           
            if (Physics.Raycast(ray, out hit, 1000, 1 << 8))
            {
                desPos = hit.point;
                isMove = true;
                Debug.Log("开始移动");
            }
        }
        if (isMove)
        {
            //获取每帧移动时当前的点
            currentPos = Vector3.MoveTowards(transform.position,desPos,0.1f);
            Debug.Log(currentPos);
            //每帧发射的射线
            Ray rayEveryFrame = new Ray(Camera.main.transform.localPosition, (currentPos - Camera.main.transform.localPosition).normalized);
            //发射射线
            if (Physics.Raycast(rayEveryFrame, out hit, 1000, 1 << 8))
            {         
                //求当前点的法线
                  Vector3 normal =  (transform.position - sphere.transform.position).normalized;
                //次切线
                  Vector3 binormal = Vector3.Cross(normal,desPos-sphere.transform.position).normalized;
                //切线
                  Vector3 tangent = Vector3.Cross(binormal,normal).normalized;                 
                  transform.parent.position = hit.point;
                  transform.parent.up = normal;
                //计算父级物体正方向和切线的夹角
                  float angle = Vector3.Angle(transform.parent.forward, tangent);
                //将子物体的方向矫正
                //要分在物体上方点击还是下方点击来判断子物体应该往那边偏移
                  if (transform.position.x > sphere.transform.position.x)
                  {
                      if (desPos.y > transform.position.y)
                      {
                          transform.localEulerAngles = new Vector3(0, transform.parent.localEulerAngles.y - angle, 0);
                      }
                      else
                      {
                          transform.localEulerAngles = new Vector3(0, transform.parent.localEulerAngles.y + angle, 0);
                      }
                  }
                  else
                  {
                      if (desPos.y > transform.position.y)
                      {
                          transform.localEulerAngles = new Vector3(0, transform.parent.localEulerAngles.y + angle, 0);
                      }
                      else
                      {
                          transform.localEulerAngles = new Vector3(0, transform.parent.localEulerAngles.y - angle, 0);
                      }
                  }
                 
            }
            if (Vector3.Distance(transform.position, desPos) < 0.1f)
            {
                isMove = false;
            }
        }
        }
}
如果还有BUG,或者有什么疑问,可以留言;我会在第一时间与大家讨论!感谢大家的关注:)