3D游戏编程与设计作业七

智能巡逻兵

要求

游戏设计要求:
  1. 创建一个地图和若干巡逻兵(使用动画);
  2. 每个巡逻兵走一个3~5个边的凸多边型,位置数据是相对地址。即每次确定下一个目标位置,用自己当前位置为原点计算;
  3. 巡逻兵碰撞到障碍物,则会自动选下一个点为目标;
  4. 巡逻兵在设定范围内感知到玩家,会自动追击玩家;
  5. 巡逻兵失去玩家目标后,继续巡逻;
  6. 计分:玩家每次甩掉一个巡逻兵计一分,与巡逻兵碰撞游戏结束;
程序设计要求:
  1. 必须使用订阅与发布模式传消息
    subject:OnLostGoal
    Publisher: ?
    Subscriber: ?
  2. 工厂模式生产巡逻兵

实现

代码挺多的:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
挑一些重要的说一下思路。
这是巡逻兵的Animator:
在这里插入图片描述
右上角的状态是杀死玩家,转移到这个状态之后游戏结束,第二行左边的状态是发现玩家并扑向玩家,第二行右边的是行走(巡逻)状态。

在这里插入图片描述
而玩家的状态有:站立(第一行左边),跑(第二行右边)和死亡(第二行左边)。死亡是与巡逻兵接触的时候触发,因此可以从站立和跑的状态转移过去,而按方向键移动玩家则会从站立状态转为跑的状态。

玩家触碰到巡逻兵死亡是通过以下代码实现的(PlayerCollideDetection.cs):

void OnCollisionEnter(Collision other){
        if (other.gameObject.tag == "Player")
        {
            other.gameObject.GetComponent<Animator>().SetBool("death",true);
            this.GetComponent<Animator>().SetTrigger("shoot");
            Singleton<GameEventManager>.Instance.PlayerGameover();
        }
    }

因为碰撞的物体除了玩家还可能有墙,因此要在一开始加一个判断条件。

而巡逻兵跟随玩家的实现是在(PatrolFollowAction.cs)中:

void Follow(){
        //向着玩家的方向移动
        transform.position = Vector3.MoveTowards(this.transform.position, player.transform.position, speed * Time.deltaTime);
        this.transform.LookAt(player.transform.position);
    }

在判断玩家进入巡逻兵视野之后调用这个函数即可让巡逻兵向玩家移动、同时面向玩家。

实现巡逻兵在自己区域内巡逻是在(PatrolAction.cs)的Gopatrol中:

void Gopatrol(){
        if (move_sign){
            switch (dirction){
                case Dirction.EAST:
                    pos_x -= move_length;
                    break;
                case Dirction.NORTH:
                    pos_z += move_length;
                    break;
                case Dirction.WEST:
                    pos_x += move_length;
                    break;
                case Dirction.SOUTH:
                    pos_z -= move_length;
                    break;
            }
            move_sign = false;
        }
        this.transform.LookAt(new Vector3(pos_x, 0, pos_z));
        float distance = Vector3.Distance(transform.position, new Vector3(pos_x, 0, pos_z));

        if (distance > 0.9){
            transform.position = Vector3.MoveTowards(this.transform.position, new Vector3(pos_x, 0, pos_z), move_speed * Time.deltaTime);
        }
        else{
            dirction = dirction + 1;
           
            if(dirction > Dirction.SOUTH)
            {
                dirction = Dirction.EAST;
            }
            move_sign = true;
        }
    }

通过定义一个direction来判断现在走到那个角,然后当到达角的时候根据情况修改前进方向,可以看到确实是以自己当前位置为原点计算相对坐标来确定下一位置的。

当玩家进入巡逻兵范围内实现追赶的实现如下(PlayerInDetection.cs):

void OnTriggerEnter(Collider collider){
        //玩家进入巡逻兵范围
        if (collider.gameObject.tag == "Player"){
            this.gameObject.transform.parent.GetComponent<PatrolData>().follow_player = true;
            this.gameObject.transform.parent.GetComponent<PatrolData>().player = collider.gameObject;
            
            //巡逻兵追踪玩家
            this.gameObject.transform.parent.GetComponent<Animator>().SetTrigger("shock");
        }
    }
    void OnTriggerExit(Collider collider){
        //玩家离开巡逻兵范围,停止追踪
        if (collider.gameObject.tag == "Player"){
            this.gameObject.transform.parent.GetComponent<PatrolData>().follow_player = false;
            this.gameObject.transform.parent.GetComponent<PatrolData>().player = null;
        }
    }

总控制(FirstController.cs)函数实现的功能如下:
加载资源,player、patrol还有地图plane,都是做成预置的;这里patrol使用了工厂类生产。

public void LoadResources(){
        player = _PatorlFactory.LoadPlayer();

        patrols = _PatorlFactory.LoadPatrol();
        for (int i = 0; i < patrols.Count; i++){
            _PatrolActionManager.GoPatrol(patrols[i]);
        }

        GameObject.Instantiate(Resources.Load("Prefabs/Plane"), new Vector3(0, 0, 0), Quaternion.identity);
    }

移动玩家:

public void MovePlayer(float translationX, float translationZ){
        if(!game_over){
            if (translationX != 0 || translationZ != 0){
                //run动画
                player.GetComponent<Animator>().SetBool("run", true);
            }
            else
            {
                player.GetComponent<Animator>().SetBool("run", false);
            }
            //通过键盘输入移动玩家。
            player.transform.Translate(translationX * player_speed * Time.deltaTime, 0, translationZ * player_speed * Time.deltaTime);

            if (player.transform.position.y != 0){
                player.transform.position = new Vector3(player.transform.position.x, 0, player.transform.position.z);
            }     
        }
    }

这里通过GUI中读取的translationX和Z控制玩家行走。
发布与订阅模式:

//发布与订阅模式
void OnEnable()
 {
     GameEventManager.ScoreChange += AddScore;
     GameEventManager.GameoverChange += Gameover;
 }
 void OnDisable()
 {
     GameEventManager.ScoreChange -= AddScore;
     GameEventManager.GameoverChange -= Gameover;
 }

GameEventManager.cs:

public class GameEventManager : MonoBehaviour{
    public delegate void ScoreEvent();
    public static event ScoreEvent ScoreChange;

    public delegate void GameoverEvent();
    public static event GameoverEvent GameoverChange;

    public void PlayerEscape(){
        //逃脱之后分数增加
        if (ScoreChange != null){
            ScoreChange();
        }
    }

    public void PlayerGameover(){
        //碰撞之后游戏结束
        if (GameoverChange != null){
            GameoverChange();
        }
    }
}

在enable中就将对应的分数增加函数赋给对应模式,然后通过他们调用。

除了这些以外,其他的像Singleton、UserGUI、ScoreRecorder、PatorlFactory、Interface等类的实现都是比较简单,或者和之前的作业中比较类似的,不再过多叙述。

效果

win:
在这里插入图片描述
lost:
在这里插入图片描述

源代码

GitHub
使用方法:将Asset拖入项目,找到Scene/myScene,点击后即可运行。

参考

Unity3d学习之路-简单巡逻兵

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值