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

上一次我们讲述了如何在球上运动并且始终朝向一点,其实之前的代码是有BUG的,因为我们点击的方向不同,他的转向也会不同,所以我们可能会+angle,也可能-angle。我会在这一章的例子中把之前的问题顺便阐述清楚。
我们之前做的是始终朝向一点运动,那么,我们是否能朝向我们所运动的方向呢?似乎这样更符合我们的认知。还有,之前在球上的操作是鼠标长按,鼠标在哪小人就在哪,那么我们是否可以实现点击球上一点,让小人自己走过去呢?今天就来解决这些问题。
首先,我们解决简单的鼠标点击问题,以前我们 通过射线碰撞获取碰撞点,直接把碰撞点的位置赋给小人,这样显然有些操作上的不方便,那么我们现在点击一点,小人在球面上运动到目标点。如何实现呢?我们有这样一种思路:
如图,我们正常是从直线的一端走向另一端,但是我们现在可以先计算出从起始点到目标点的直线移动坐标变化,每帧记录下这些坐标,然后每帧发射射线指向这些坐标,与球面碰撞的位置就是我们要在曲面上运动的轨迹,有了这个思路,我们的移动问题就解决了。
接下来就是文章一开头提到的转向问题:
如图,我们之前只考虑到一种转向问题,就是如图曲线的方向,这只是在面朝自己的这一面向下走时应该转的方向。
首先我们必须明确的一点是小人是有父对象的,并且父对象的正方向始终不变,为的就是和正确方向计算得出正确转向,那么我们的问题就来了,既然父物体对象正方向一直不变,怎么可能始终让子物体朝着一个方向转动呢,我们的角度一直都是非负的!不可能自动调整。所以我们就必须分情况讨论:
首先我们确定一个前提,就是父物体的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,或者有什么疑问,可以留言;我会在第一时间与大家讨论!感谢大家的关注:)
  • 4
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值