unity 范围 敌人 搜索 unity敌人随机移动_unity

总目录

9.Unity2D 简单AI 之 敌人跳跃条件优化+自动范围内检测敌人发起攻击(索敌)+对象池优化+主角受伤死亡_ζั͡ ั͡雾 ั͡狼 ั͡✾的博客-Unity2D 简单AI 之 敌人跳跃条件优化+自动范围内检测敌人攻击+敌人二连击。在敌人预制体下,创建空物体EnemyCanAttack,改成不受攻击检测标签,加上触发器,加上触发器脚本。 在主角到该范围内,敌人开始攻击,加入脚本EnemyAttackBox。在1中已经增加了主角受伤动画,现在优化攻击判定的代码,使得敌人也可以使用脚本。。。......

上一节课我们专门做了进入敌人前方范围,敌人自动攻击,所以这节课我们专门做AI寻路功能,将敌人动起来,既要随机一点,也要智能一些,接上一节的自动攻击AI。这样敌人就可以随机移动而攻击了,这个功能困难的一点是如何让敌人跳跃上方头顶地面和跳跃下方地面。我们先从 简单的做起。

1.优化CharacterPanel(角色面板)脚本

首先我们对角色PlayerControl脚本(主角控制)中提取主角和敌人共有的变量和函数(像跳跃,移动之类)到CharacterPanel(角色面板)脚本中,这样写敌人行为脚本的时候,好调用且好共同调整修改。

playercontrol脚本(只有主角有)

现阶段主角物体组件

unity 范围 敌人 搜索 unity敌人随机移动_经验分享_02

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

public class playerControl : MonoBehaviour
{
  
    private Rigidbody2D rig;//2D刚体
    private CharacterPanel characterPanel;//获取角色面板
    private Animator ani;//动画控制器
    private AnimatorStateInfo state;//动画状态
                         // Start is called before the first frame update
    void Start()
    {
        rig = GetComponent<Rigidbody2D>();//获取刚体
        ani = GetComponent<Animator>();//获取动画控制器
        characterPanel = GetComponent<CharacterPanel>();//获取角色面板


    }

    // Update is called once per frame
    void Update()
    {
        move();//移动函数
        attack();//攻击函数-四连击

    }
    private void move()
    {    //水平,垂直俩个轴系
        float h = Input.GetAxis("Horizontal");
        float v = Input.GetAxis("Vertical");
        float dir = Input.GetAxisRaw("Horizontal");//方向-1 0 1
        //跳跃
        if (v > 0 &&transform.GetComponent<CharacterPanel>().Iscanjump == true)
        {
            characterPanel.jump();
        }
        //长按高跳
        if (rig.velocity.y > 0 && Input.GetKey(KeyCode.W)|| Input.GetKey(KeyCode.UpArrow) && transform.GetComponent<CharacterPanel>().Iscanjump == false)
        {
            rig.velocity += Vector2.up * 0.2f;//长按高跳额外得到向上速度
        }

        //方向改变
        if (dir != 0)
        {
            transform.localScale = new Vector3(dir* Mathf.Abs(transform.localScale.x), transform.localScale.y, transform.localScale.z);//通过改变scale改变方向
        }
        //按键左右移动
        Vector3 vt = new Vector3(h, 0, 0).normalized;//vt为俩个轴系合成的方向向量,normalized单位化
            //移动动画
        if (dir != 0)
        {
            ani.SetBool("Ismove", true);
        }
        else { ani.SetBool("Ismove", false); }
   
        //空中左右移动,为地面jumpcharacterPanel.speedVertiacal倍
        if (h != 0 && transform.GetComponent<CharacterPanel>().Iscanjump == false)
        {
            gameObject.transform.Translate(vt * characterPanel.speed * characterPanel.jumpspeedVertiacal * Time.deltaTime);//通过这个函数来使用vt使得左右移动
        }
        //地面左右移动
        else { gameObject.transform.Translate(vt * characterPanel.speed * Time.deltaTime); }
    }
    private void attack() {
        

        state = ani.GetCurrentAnimatorStateInfo(0);
        //判断播放完
        if ((state.IsName("attack1") || state.IsName("attack2") || state.IsName("attack3") || state.IsName("attack4")) && state.normalizedTime >= 1.0f)
        {
            
            ani.SetInteger("attack", 0);

        }

        if (Input.GetKey(KeyCode.J))
        {
    
            if (state.IsName("idle")||  state.IsName("move") && ani.GetInteger("attack")==0 )
            {
            
                ani.SetInteger("attack", 1);
            } else if (state.IsName("attack1")&& ani.GetInteger("attack") == 1 )
            {
                ani.SetInteger("attack", 2);
            }
            else if (state.IsName("attack2") && ani.GetInteger("attack") == 2)
            {
                ani.SetInteger("attack", 3);
            }
            else if (state.IsName("attack3") && ani.GetInteger("attack") == 3)
            {
                ani.SetInteger("attack", 4);
            }
        }
        if (state.normalizedTime >=1.0f) 
        {
            ani.SetFloat("normalizedTime", state.normalizedTime);
        }


        }
}

characterpanel脚本(主角敌人共有)

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class CharacterPanel : MonoBehaviour
{
    public bool Iscanjump = false;//是否能跳跃,默认不能
    public float Hpmax = 100;//最大生命
    public float Hp = 100;//生命
    public float Atk = 10;//攻击力
    public float AtkRan = 1;//攻击浮动
    public float Def = 10;//防御力
    public float lookdir;//获取初始Scale.x,用于转向 
    public float dropConst = 15;//下坠常数
    public float speed = 10;//地面移动速度
    public float jumpspeedUp = 20;//上升速度
    public float jumpspeedVertiacal = 0.5f;//空中左右移动速度
    private Rigidbody2D rig;//2D刚体
    private Animator ani;
    private Transform Canvas;//获取角色个人UI面板
    // Start is called before the first frame update
    void Start()
    {
        Canvas = transform.Find("Canvas");
        ani = transform.GetComponent<Animator>();
      
        rig = GetComponent<Rigidbody2D>();//获取刚体
    }

    // Update is called once per frame
    void FixedUpdate()
    {
        lookdir = transform.localScale.x;
        check();
        //跳跃优化手感
        Quickjump();

    }
    //标准化检查
    private void check()
    {
        if (Hp > Hpmax) Hp = Hpmax;//血量不超过上限
        if (Hp <= 0) { Hp = 0;death(); };//血量不超过下限,且死亡
    }
    //受伤,其他脚本调用
    public void hurt(float atk)
    {
       float hurtnum= (20 * atk / (20 + Def)) + Random.Range(-AtkRan, AtkRan);
        //受伤动画 为什么要将触发器中的受伤动画移动在角色面板的hurt函数下?因为受伤不仅有角色攻击受伤,还有陷阱,掉落等伤害,所以直接血量减少时播放受伤动画更加好
        transform.GetComponent<Animator>().SetBool("Ishurt", true);
        StartCoroutine(endHurt());//开启协程结束受伤动画
        //伤害数值显示
        Canvas.GetComponent<CharaCanvas>().ShowHurtText(hurtnum);
        Hp -= hurtnum;
    
    }
    IEnumerator endHurt()
    {
        yield return 0;//此处暂停,下一帧执行
        transform.GetComponent<Animator>().SetBool("Ishurt", false);
    }
    //死亡
    private void death()
    {
        ani.SetBool("Isdeath",true);
    }
    //跳跃
    public void jump()
    {
        if (Iscanjump == true)
        {
            rig.velocity = new Vector2(0, jumpspeedUp);//设置刚体速度,给予向量
        }
    }
    //优化跳跃手感,迅速下落,放入帧频率更新函数里面
    public void Quickjump()
    {
        float a = dropConst * 5 - Mathf.Abs(rig.velocity.y);//通过下坠常数,空中速度快为0时,下坠常数a越大,即越快速 度过这个状态
        rig.velocity -= Vector2.up * a * Time.deltaTime;
    }
    //以下是敌人的调运行为函数,主角有自己的控制脚本
    //转向
    public void Turndir()
    {
          transform.localScale = new Vector3(-lookdir, transform.localScale.y, transform.localScale.z);//通过改变scale改变方向
    }
    //移动
    public void move()
    {
        Vector3 vt = new Vector3(lookdir/Mathf.Abs(lookdir), 0, 0);
        //空中左右移动,为地面jumpcharacterPanel.speedVertiacal倍
        if (Iscanjump == false)
        {
            gameObject.transform.Translate(vt * speed * jumpspeedVertiacal * Time.deltaTime);//通过这个函数来使用vt使得左右移动
        }
        //地面左右移动
        else { gameObject.transform.Translate(vt * speed * Time.deltaTime); }
        ani.SetBool("Ismove", true);

    }
    //等待
    public void idle()
    {
        ani.SetBool("Ismove", false);
    }
}

2.敌人行为脚本EnemyBehavior,放在在敌人物体上

现阶段敌人物体组件

unity 范围 敌人 搜索 unity敌人随机移动_游戏程序_03

(1)随机移动

unity 范围 敌人 搜索 unity敌人随机移动_unity 范围 敌人 搜索_04

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

public class EnemyBehavior : MonoBehaviour
{
    private CharacterPanel characterPanel;
    public float moveTime=3;//移动时间
    private float _moveTime;//定值
    private float waitTime=3;//等待时间
    private float _waitTime;//定值
    // Start is called before the first frame update
    void Start()
    {
        characterPanel = transform.GetComponent<CharacterPanel>();
        _moveTime = moveTime;
        _waitTime = waitTime;
    }

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

    }
    //随机横向移动
    private void RandomMove()
    {
        if (moveTime > 0)//如果处于移动时间
        {
            characterPanel.move();
            moveTime -= Time.deltaTime;//移动时间减少一秒
            if (moveTime < 0)
            {
                waitTime = _waitTime;//初始化等待时间
            }
        }
        else
        {
            characterPanel.idle();
            if (waitTime > 0)//如果处于等待时间
            {
                waitTime -= Time.deltaTime;//等待时间减少一秒
            }
            else { moveTime = _moveTime;//初始化移动时间
                //等待结束,随机转向
                bool Isturn = (Random.value > 0.5f);
                if(Isturn)
                {
                    characterPanel.Turndir();//转向
                }
                

            }
        }
    }

    
}

我们能够发现问题,随机移动会跳下悬崖,也会前面有遮挡,确一直往前走,不能自动跳跃。解决这个问题我们需要进行射线检测

教学链接

Unity 常用射线检测方法_昵称好难写的博客_unity射线检测1.普通射线检测(一般用于检测某一个物体)Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition); Debug.DrawRay(ray.origin ,ray.direction , Color.red); RaycastHit hit; if(Physics .Raycast (ray,

(2)自动跳跃障碍物+悬崖勒马+射线检测

unity 范围 敌人 搜索 unity敌人随机移动_游戏_05

白线三条是检测跳跃视线,蓝线是检测转向线。

白线中下线检测到物体直接跳跃,上线检测,进行一次判定,是跳上去还是钻过去

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

public class EnemyBehavior : MonoBehaviour
{
    private CharacterPanel characterPanel;
    public float moveTime=3;//移动时间
    private float _moveTime;//定值
    private float waitTime=3;//等待时间
    private float _waitTime;//定值

    private bool OneIsMustJump=true;//一次头顶检测,跳跃判断
    private int OneTurn = 1;//一次转向,防止抽搐
    // Start is called before the first frame update
    void Start()
    {
        characterPanel = transform.GetComponent<CharacterPanel>();
        _moveTime = moveTime;
        _waitTime = waitTime;
    }

    // Update is called once per frame
    void Update()
    {
        RandomMove();
        JumpObstacle();
        CliffTurn();
    }
    //随机横向移动
    private void RandomMove()
    {
        if (moveTime > 0)//如果处于移动时间
        {
            characterPanel.move();
            moveTime -= Time.deltaTime;//移动时间减少一秒
            if (moveTime < 0)
            {
                waitTime = _waitTime;//初始化等待时间
            }
        }
        else
        {
            characterPanel.idle();
            if (waitTime > 0)//如果处于等待时间
            {
                waitTime -= Time.deltaTime;//等待时间减少一秒
            }
            else { moveTime = _moveTime;//初始化移动时间
                //等待结束,随机转向
                bool Isturn = (Random.value > 0.5f);
                if(Isturn)
                {
                    characterPanel.Turndir();//转向
                }
                

            }
        }
    }
    //跳跃障碍物
    private void JumpObstacle()
    {

        for (int i = -1;i<=1;i++)//i取-1,0,1,使得y轴变正负,三条射线检测,形成一个小扇角
        {
            RaycastHit2D hit= Physics2D.Raycast(transform.position, characterPanel.lookdir * new Vector3(0.25f, -0.15f*i, 0), 
                               2, LayerMask.GetMask("tilemap","items"));//检测这俩种图层
            if (hit)//检测到了有碰赚点
            {
                Debug.DrawLine(transform.position, hit.point);
                if (hit.point.y <=transform.position.y)//脚底前方或前方检测到
                {
                    characterPanel.jump();
                }
                else
                { //头顶前方检测到 下面代码含义为,头顶检测到首先判断要不要跳,还是钻过去,
                  //如果钻过去下面,就要使得3秒内OneIsMustJump为flase,走过去,三秒后如果头顶前方有东西,再进行一次判定再
                    bool mustJump=false;
                    if (OneIsMustJump)
                    {
                        mustJump = (Random.value > 0.5f);//进行一次判断是否跳跃
                        StartCoroutine(reMustJump()); //开启协程恢复OneIsMustJump为true
                         OneIsMustJump = false;
                        Debug.Log(mustJump);
                    }
                    if (mustJump)
                    {
                        if (moveTime < _moveTime)
                        {
                            moveTime += Time.deltaTime;//移动时间加一秒确定能让他跳过去;
                        }
                        characterPanel.jump();
                    }

               
                }
            }
            else {
             
                Debug.DrawRay(transform.position, characterPanel.lookdir * new Vector3(0.25f, -0.15f * i, 0));
            }

        }

       
       



    }
    IEnumerator reMustJump()
    {
        yield return new WaitForSeconds(3);
        waitTime += Time.deltaTime;
        OneIsMustJump = true;
    }
    //悬崖勒马,不让角色跳崖
    private void CliffTurn()
    { 
        RaycastHit2D hit = Physics2D.Raycast(transform.position, new Vector2(0,-1),
                            Mathf.Infinity, LayerMask.GetMask("tilemap"));
        if (!hit)//检测没有有碰撞点
        {
            if (OneTurn >= 1)
            {
                StartCoroutine(turnDir());//协程转向,转向一次后延迟0.5秒才能下一次转向,防止过短时间内频繁转向,抽搐
                OneTurn--;
            }
            characterPanel.move();
        }
        Debug.DrawRay(transform.position , new Vector2(0, -1),Color.blue);
    }
    IEnumerator turnDir()
    {
        characterPanel.Turndir();//转向
        yield return new WaitForSeconds(0.5f);
        OneTurn = 1;
    }
}

 如果不想敌人把敌人推下去,我们可以先给敌人预制体赋予enemy层(不是标签),然后在项目设置里关闭敌人层与敌人层的碰撞

unity 范围 敌人 搜索 unity敌人随机移动_游戏程序_06

结束了!!

其实学完射线检测我们可以做到很多东西,像简单的攻击范围判定,跳跃条件判定都可以通过射线检测来完成(我的程序是通过碰撞盒和触发器完成的)。