请选择 进入手机版 | 继续访问电脑版

查看: 14780|回复: 422

[原创] Unity教程 | 从零开始制作一款3D炸弹超人游戏

  [复制链接]

910

主题

1570

帖子

2万

贡献

管理员

Rank: 9Rank: 9Rank: 9

积分
20185
QQ
发表于 2017-6-3 09:28:52 | 显示全部楼层 |阅读模式
说起炸弹超人,相信很多朋友都玩过类似的游戏,其中最为人熟知的莫过于《泡泡堂》。该类型游戏需要玩家在地图中一边跑动一边放置炸弹,同时还要躲避敌方炸弹保护自己。最初的炸弹超人游戏都是2D的,今天这篇文章将教大家在Unity中实现一款3D的炸弹超人游戏。

温馨提示,本教程需要大家了解Unity的基本操作与脚本概念。

下载初始项目资源(请回复本帖哦~):
游客,如果您要查看本帖隐藏内容请回复


准备工作

将项目初始资源导入Unity项目,资源目录如下:

0.png


其中分别包含要用于游戏的动画、材质、模型、背景音乐、物理材质、预制件、场景、脚本、音效及图片资源。

放置炸弹

打开项目中的Game场景并运行。

1.png


可以通过WASD键或方向键来操作所有角色进行移动。下面来让角色可以放置炸弹。角色1(红色)通过按下空格键来放置炸弹,角色2(蓝色)则通过按下回车键进行同样的操作。

打开Player脚本,该脚本负责角色所有的移动及动画逻辑。找到DropBomb函数,添加代码如下:


[C#] 纯文本查看 复制代码
    /// <summary>
    /// Drops a bomb beneath the player
    /// </summary>
    private void DropBomb() {
        if (bombPrefab) { //Check if bomb prefab is assigned first
            // Create new bomb and snap it to a tile
            Instantiate(bombPrefab,
                new Vector3(Mathf.RoundToInt(myTransform.position.x), bombPrefab.transform.position.y, Mathf.RoundToInt(myTransform.position.z)),
                bombPrefab.transform.rotation);
        }
    }


其中RoundToInt函数用于对炸弹的坐标参数四舍五入,以避免炸弹放置位置偏离出地块中心。

3.png


运行场景,效果如下:

4.gif


创建爆炸效果

在Scripts文件夹下新建C#脚本命名为Bomb:

5.gif


找到Prefabs文件夹下的Bomb预制件,将Bomb脚本绑定到该游戏对象上。然后打开Bomb脚本,添加代码如下:


[C#] 纯文本查看 复制代码
using UnityEngine;
using System.Collections;
using System.Runtime.CompilerServices;

public class Bomb : MonoBehaviour {
    public AudioClip explosionSound;
    public GameObject explosionPrefab; 
    public LayerMask levelMask; // This LayerMask makes sure the rays cast to check for free spaces only hits the blocks in the level
    private bool exploded = false;

    // Use this for initialization
    void Start() {
        Invoke("Explode", 3f); //Call Explode in 3 seconds
    }

    void Explode() {
        //Explosion sound
        AudioSource.PlayClipAtPoint(explosionSound,transform.position);

        //Create a first explosion at the bomb position
        Instantiate(explosionPrefab, transform.position, Quaternion.identity);

        //For every direction, start a chain of explosions
        StartCoroutine(CreateExplosions(Vector3.forward));
        StartCoroutine(CreateExplosions(Vector3.right));
        StartCoroutine(CreateExplosions(Vector3.back));
        StartCoroutine(CreateExplosions(Vector3.left));

        GetComponent<MeshRenderer>().enabled = false; //Disable mesh
        exploded = true; 
        transform.FindChild("Collider").gameObject.SetActive(false); //Disable the collider
        Destroy(gameObject,.3f); //Destroy the actual bomb in 0.3 seconds, after all coroutines have finished
    }

    public void OnTriggerEnter(Collider other) {
        if (!exploded && other.CompareTag("Explosion")) { //If not exploded yet and this bomb is hit by an explosion...
            CancelInvoke("Explode"); //Cancel the already called Explode, else the bomb might explode twice 
            Explode(); //Finally, explode!
        }
    }

    private IEnumerator CreateExplosions(Vector3 direction) {
        for (int i = 1; i < 3; i++) { //The 3 here dictates how far the raycasts will check, in this case 3 tiles far
            RaycastHit hit; //Holds all information about what the raycast hits

            Physics.Raycast(transform.position + new Vector3(0,.5f,0), direction, out hit, i, levelMask); //Raycast in the specified direction at i distance, because of the layer mask it'll only hit blocks, not players or bombs

            if (!hit.collider) { // Free space, make a new explosion
                Instantiate(explosionPrefab, transform.position + (i * direction), explosionPrefab.transform.rotation);
            }
            else { //Hit a block, stop spawning in this direction
                break;
            }

            yield return new WaitForSeconds(.05f); //Wait 50 milliseconds before checking the next location
        }

    }
}



在检视面板中,将Bomb预制件赋值给脚本的Explosion Prefab属性,该属性用于定义需要生成爆炸效果的对象。Bomb脚本使用了协程来实现爆炸的效果,StartCoroutine函数将朝着4个方向调用CreateExplosions函数,该函数用于生成爆炸效果,在For循环内遍历炸弹能够炸到的所有单元,然后为能够被炸弹影响的各个单元生成爆炸特效,炸弹对墙壁是没有伤害的。最后,在进入下一次循环前等待0.05秒。

代码作用类似下图:

7.gif


红线就是Raycast,它会检测炸弹周围的单元是否为空,如果是,则朝着该方向生成爆炸效果。如果碰撞到墙,则不生成爆炸并停止检测该方向。所以前面需要让炸弹在地块中心生成,负责就会出现不太理想的效果:

8.png


Bomb代码中定义的LayerMask用于剔除射线对地块的检测,这里还需要在检视面板中编辑层,并新增用户层命名为“Blocks”,然后将层级视图中Blocks游戏对象的Layer设置为“Blocks”。

9.png


更改Blocks对象的层级时会跳出提示框,询问是否更改子节点,选择是即可:

10.png


然后选中Bomb对象,在检视面板中将Bomb脚本的Level Mask设为“Blocks”:

11.png


连锁反应

如果炸弹炸到了另一个炸弹,那么被炸到的炸弹也会爆炸。Bomb脚本中的OnTriggerEnter
函数是MonoBehaviour预定义的函数,会在触发器与Rigidbody碰撞之前调用。这里OnTriggerEnter会检测被碰撞的炸弹是否是被炸弹特效所碰撞,如果是,则该炸弹也要爆炸。

现在运行场景,效果如下:

12.gif


判定游戏结果

打开Player脚本,添加下面的代码:


[C#] 纯文本查看 复制代码
  //Manager
    public GlobalStateManager GlobalManager;

    //Player parameters
    [Range(1, 2)] //Enables a nifty slider in the editor
    public int playerNumber = 1; //Indicates what player this is: P1 or P2
    public float moveSpeed = 5f;
    public bool canDropBombs = true; //Can the player drop bombs?
    public bool canMove = true; //Can the player move?
    public bool dead = false; //Is this player dead?


其中GlobalManager是GlobalStateManager脚本的引用,该脚本用于通知玩家获胜或死亡的消息。dead则用于标志玩家是否死亡。

更改OnTriggerEnter函数代码如下:


[C#] 纯文本查看 复制代码
    public void OnTriggerEnter(Collider other) {
        if (!dead && other.CompareTag("Explosion")) { //Not dead & hit by explosion
            Debug.Log("P" + playerNumber + " hit by explosion!");

            dead = true;
            GlobalManager.PlayerDied(playerNumber); //Notify global state manager that this player died
            Destroy(gameObject);
        }
    }


该函数作用为设置dead变量来通知玩家死亡,并告知全局状态管理器玩家的死亡信息,然后销毁玩家对象。

在检视面板中选中两个玩家对象,将Global State Manager游戏对象赋值给Player脚本的Global Manger字段。

15.gif


再次运行场景,效果如下:

16.gif


打开GlobalStateManager脚本,添加以下代码:


[C#] 纯文本查看 复制代码
   public List<GameObject> Players = new List<GameObject>();

    private int deadPlayers = 0;
    private int deadPlayerNumber = -1;

    public void PlayerDied(int playerNumber) {
        deadPlayers++;

        if (deadPlayers == 1) {
            deadPlayerNumber = playerNumber;
            Invoke("CheckPlayersDeath", .3f);
        }
    }


其中deadPlayers表示死亡的玩家数量,deadPlayerNumber则用于记录死亡玩家的编号。PlayerDied函数用于添加死亡玩家,并设置deadPlayerNumber属性,在0.3秒后检测另一位玩家是否也死亡。

然后在脚本中添加CheckPlayersDeath函数,代码如下:


[C#] 纯文本查看 复制代码
   void CheckPlayersDeath() {
        if (deadPlayers == 1) { //Single dead player, he's the winner

            if (deadPlayerNumber == 1) { //P1 dead, P2 is the winner
                Debug.Log("Player 2 is the winner!");
            }
            else { //P2 dead, P1 is the winner
                Debug.Log("Player 1 is the winner!");
            }
        }
        else {  //Multiple dead players, it's a draw
            Debug.Log("The game ended in a draw!");
        }
    }


以上代码用于判断哪位玩家获得胜利,如果两位玩家均死亡,则打成平局。

运行场景,效果如下:

19.gif


总结

到此本篇教程就结束了,大家还可以在此基础上对该项目进行扩展,例如添加“推箱子”功能,将位于自己脚边的炸弹推给敌方,或是限制能够放置的炸弹数量,添加快速重新开始游戏的界面,设置可以被炸弹炸毁的障碍物,设置一些道具用于获得炸弹或者增加生命值,还可以增加多人对战模式与朋友一起变身炸弹超人等等。大家都来发挥自己的创意吧!

下载完整示例工程(请回复本帖哦~):
游客,如果您要查看本帖隐藏内容请回复



原文链接:https://www.raywenderlich.com/125559/make-game-like-bomberman
原作者:Eric Van de Kerckhove
转载请注明来自Unity官方中文社区

0

主题

3

帖子

50

贡献

初级UU族—2级

Rank: 2

积分
50
发表于 2017-6-3 10:23:07 | 显示全部楼层
hao.......

0

主题

4

帖子

70

贡献

初级UU族—2级

Rank: 2

积分
70
发表于 2017-6-3 11:32:55 | 显示全部楼层
下载下来看看 希望很棒

0

主题

2

帖子

25

贡献

初级UU族—1级

Rank: 1

积分
25
发表于 2017-6-3 11:37:47 | 显示全部楼层
学习学习。

0

主题

5

帖子

75

贡献

初级UU族—2级

Rank: 2

积分
75
发表于 2017-6-3 12:33:32 | 显示全部楼层
11111111ksnflknf

1

主题

9

帖子

190

贡献

初级UU族—2级

Rank: 2

积分
190
发表于 2017-6-3 13:03:14 | 显示全部楼层
o goodaaaaa

0

主题

6

帖子

135

贡献

初级UU族—2级

Rank: 2

积分
135
发表于 2017-6-3 13:14:58 | 显示全部楼层
新手前来报到,希望可以学习学习

0

主题

2

帖子

20

贡献

初级UU族—1级

Rank: 1

积分
20
发表于 2017-6-3 13:21:05 | 显示全部楼层
新手刚学正需要了

0

主题

1

帖子

10

贡献

初级UU族—1级

Rank: 1

积分
10
发表于 2017-6-3 13:32:05 | 显示全部楼层
ganxiefenx

0

主题

2

帖子

15

贡献

初级UU族—1级

Rank: 1

积分
10
发表于 2017-6-4 01:46:05 | 显示全部楼层
又来回复了

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表