Space Shooter

根据Unity官网的SpaceShooter教程视频和参考资料的博客完成了这个小游戏,下边简单的记录一下。
参考资料:
http://www.jianshu.com/p/8cc3a2109d3b
https://unity3d.com/cn/learn/tutorials/projects/space-shooter-tutorial


开发步骤

  • 1.首先在Unity的 Asset Store下载游戏的资源包,包含了游戏的背景,模型,音频等。

  • 2.将飞机模型Player添加到新建的场景_Scenes中,调整相机位置,新建灯光,调整灯光位置。

  • 3..为Player添加脚本PlayerController,控制Player移动。


 void FixedUpdate()
    {
        float moveHorizontal = Input.GetAxis("Horizontal");
        float moveVertical  = Input.GetAxis("Vertical");

        //moveHorizontal和moveVertical记录玩家输入的方向数据
        Vector3 movement = new Vector3(moveHorizontal, 0.0f, moveVertical);
        //刚体根据矢量方向movement移动,位移为speed
        GetComponent<Rigidbody>().velocity = movement * speed;

        //移动时倾斜,以Z为中心轴旋转
        GetComponent<Rigidbody>().rotation = Quaternion.Euler(0.0f, 0.0f, GetComponent<Rigidbody>().velocity.x * - tilt);

        //首先序列化Boundary类,保存飞船在X和Z轴上的范围
        //Mathf.Clamp将飞船的x和z值锁定在屏幕范围内
        GetComponent<Rigidbody>().position = new Vector3(
            Mathf.Clamp(GetComponent<Rigidbody>().position.x, boundary.xMin, boundary.xMax),
            0.0f,
            Mathf.Clamp(GetComponent<Rigidbody>().position.z, boundary.zMin, boundary.zMax)
            );
    }
  • 4.添加游戏背景,制作子弹,为子弹添加碰撞:添加Rigidbody(diselect use gravity), 添加碰撞Mesh Collider(select is Trigger),为子弹编写脚本Mover,使子弹发射,将子弹设为Prefab
   public float speed;
    void Start () {
       //transform.forward:The blue axis of the transform in world space.
        GetComponent<Rigidbody>().velocity = transform.forward * speed;
    }
    
  • 5.设置火箭发射子弹。设置子弹发射点,新建Shot Spawn,把Shot Spawn拖入到Player下面,并调整Shot Spawn的位置到合适的点,在PlayerController编写脚本
    private float nextFire;  //记录下一发子弹的时间
    public float fireRate;    //控制子弹发射的间隔时间
    public GameObject shot;    //保存我们发射的子弹Prefab
    public Transform shotSqawn;   //保存子弹的发射点

    void Update () {
        //按下发射键(ctrl或鼠标左键)并且到达下一发子弹发射时间
        if (Input.GetButton("Fire1") && Time.time > nextFire)
        {
            nextFire = Time.time + fireRate;//重置下一发子弹发射时间
           //Instantiate方法生成一发子弹
           //3个参数,一个是发射的对象(复制出一个新的),一个是发射对象的position,一个是发射对象的rotation
            Instantiate(shot, shotSqawn.position, shotSqawn.rotation);
            GetComponent<AudioSource>().Play();
        }
    }
  • 6.回收子弹,飞船每发射一颗子弹,就会生成一个Bolt对象。创建一个盒子,这个盒子大小和我们的屏幕差不多,然后我们可以判断如何子弹飞出了这个盒子,就意味着子弹飞出了屏幕,这个时候我们就可以通知程序来回收子弹。创建cube,命名Boundary,将盒子调整到屏幕大小,删除 mesh renderer,添加Box Colider,is Trigger set on,为盒子添加脚本DestroyByBoundary
Paste_Image.png
     //DestroyByBoundary中添加碰撞的触发方法。即当子弹碰到时,子弹被回收
    void OnTriggerExit(Collider other)
    {
        //注意,避免将飞船回收
        if(other.tag != "Player")
        {
            Destroy(other.gameObject);
        }
    }
  • 7.创建障碍,创建GameObject,命名为Asteroid,为Asteroid添加美术效果,我们在Assets->Models文件夹中找到prop_asteroid_01文件,然后将其拖入Asteroid做其的子对象,为Asteroid添加Rigidibody,diselect use gravity,将阻力drag 和angular drag设为0,添加capsule collider,编写脚本RandomRotator,使其自动旋转
    public float tumble;
    //tumble来设置障碍翻滚的幅度
    void Start () {
        GetComponent<Rigidbody>().angularVelocity = Random.insideUnitSphere * tumble;
    }

  • 8.首先我们需要检测子弹和障碍是否碰撞,如果碰撞了,我们需要创建一个爆炸效果来给予反馈,最后我们还需要将子弹和障碍删除掉。再为子弹编写一个脚本DestroyByContact
    public GameObject explosion;    //爆炸效果
    public GameObject playerExplosion;
    void OnTriggerEnter(Collider other)
    {
        //初始时障碍物会与盒子boundary发生碰撞触发此函数,要规避此种触发
        if (other.tag == "Boundary")
        {
            return;
        }
        //生成一个爆炸效果
        Instantiate(explosion, transform.position, transform.rotation);
        if(other.tag == "Player")
        {
            Instantiate(playerExplosion, other.transform.position, other.transform.rotation);
            gameController.GameOver();
        }

        gameController.AddScore(scoreValue);
        //Debug.Log(other.name);
        Destroy(other.gameObject);
        Destroy(gameObject);
    }

回到Unity编辑器,接着在Assets->Prefabs->VFX->Explosions文件夹中找到explosion_asteroid文件,然后拖入到我们定义的explosion变量中。

  • 8. 使障碍物向下移动,随机生成障碍。mover是之前为子弹添加的脚本,同样可以用到障碍物上,只需将speed设置为负数,即可实现向下移动。新建一个GameObject, 命名为GameController,实现这些逻辑
    public GameObject hazard;   //设置生成的障碍对象
    public Vector3 spawnValue;  //设置障碍生成的位置
    public int hazardCount;     //每次生成的障碍的数量
    public float spawnWait;     //每个障碍生成的间隔时间
    public float startWait;     //游戏开始生成障碍的准备时间
    public float waveWait;      //设置每一波的间隔时间


    IEnumerator SpawnWaves()
    {
        //让代码等待一定时间,这是个协同程序,可以让游戏不暂停的同时,让代码暂停
        yield return new WaitForSeconds(startWait);
        while (true)
        {
            for (int i = 0; i < hazardCount; i++)
            {
                //随机生成障碍位置
                Vector3 spawnPosition = new Vector3(Random.Range(-spawnValue.x, spawnValue.x), spawnValue.y, spawnValue.z);
                //Rotation属性如何确定,Rotation属性是一个Quaternion类型的变量,
                //所以我们定义了一个spawnRotation的变量,用来记录障碍的Rotation属性,在这里我们不需要让障碍一开始带着任何的旋转
                //所以我们直接赋值Quaternion.identity
                Quaternion spawnRotation = Quaternion.identity;

                Instantiate(hazard, spawnPosition, spawnRotation);
                
                yield return new WaitForSeconds(spawnWait);
            }
            yield return new WaitForSeconds(waveWait);

            if(gameOver == true)
            {
                restartText.text = "Press R for Restart";
                restart = true;
                break;
            }
        }
        
    }
    
        void Start () {
            //因为SpawnWaves中有协同程序,StartCoroutine为启动一个协同程序
           StartCoroutine(SpawnWaves());
    }


  • 9. 回收爆炸效果,新建脚本DestroyByTime,lifetime为存活时间,将此脚本添加到perfab下的爆炸效果中,并设置lifetime。
    public float lifeTime;

    // Use this for initialization
    void Start () {
        Destroy(gameObject,lifeTime);
    }
  • 10.添加背景音乐,Assets->Audio文件夹中找到背景音乐music_background直接拖入GameController对象,同理,障碍爆炸和飞船爆炸音效,也是将其音频文件直接拖到对应的perfab文件中,这些音频文件是默认play on awake的,其中还要讲背景音乐选中loop,子弹音效不能设为play on wake,只需要在发射时有声音,需要在脚本中设置,只需在PlayController中添加
    void Update () {
        //按下发射键(ctrl或鼠标左键)并且到达下一发子弹发射时间
        if (Input.GetButton("Fire1") && Time.time > nextFire)
        {
            nextFire = Time.time + fireRate;
            //Instantiate方法生成一发子弹,他有3个参数
           //一个是发射的对象(复制出一个新的),一个是发射对象的position,一个是发射对象的rotation
            Instantiate(shot, shotSqawn.position, shotSqawn.rotation);

            //当子弹发射时开启音效
            GetComponent<AudioSource>().Play();
        }
    }
  • 11.游戏计分按钮等,新建 scoreText,restartText,gameOverText,并调整其在屏幕上的位置。在GameController脚本设置。
    public Text scoreText;
    public Text restartText;
    public Text gameOverText;
    private bool gameOver;
    private bool restart;
    private int score;

    IEnumerator SpawnWaves()
    {
        //让代码等待一定时间,这是个协同程序,可以让游戏不暂停的同时,让代码暂停
        yield return new WaitForSeconds(startWait);
        while (true)
        {
            for (int i = 0; i < hazardCount; i++)
            {
                //随机生成障碍位置
                Vector3 spawnPosition = new Vector3(Random.Range(-spawnValue.x, spawnValue.x), spawnValue.y, spawnValue.z);
                //Rotation属性如何确定,Rotation属性是一个Quaternion类型的变量,
                //所以我们定义了一个spawnRotation的变量,用来记录障碍的Rotation属性,在这里我们不需要让障碍一开始带着任何的旋转
                //所以我们直接赋值Quaternion.identity
                Quaternion spawnRotation = Quaternion.identity;

                Instantiate(hazard, spawnPosition, spawnRotation);
                
                yield return new WaitForSeconds(spawnWait);
            }
            yield return new WaitForSeconds(waveWait);

            if(gameOver == true)
            {
                restartText.text = "Press R for Restart";
                restart = true;
                break;
            }
        }
        
    }

    void UpdateScore()
    {
        scoreText.text = "Score :" + score; 
    }

    public void  AddScore(int newScoreValue)
    {
        score += newScoreValue;
        UpdateScore();
    }

    public void GameOver ()
    {
        gameOverText.text = "Game Over !";
        gameOver = true;

    }
    void Start () {
        gameOver = false;
        restart = false;
        restartText.text = "";
        gameOverText.text = "";
        score = 0;
        UpdateScore();
        //因为SpawnWaves中有协同程序,StartCoroutine为启动一个协同程序
        StartCoroutine(SpawnWaves());
    }
    
    void Update () {
        if (restart)
        {
            if (Input.GetKeyDown(KeyCode.R))
            { 
                //重新加载
                Application.LoadLevel(Application.loadedLevel);
            }
        }
    }

在GameController脚本中设置计分的相关方法,在DestoryByContact的脚本中调用这些方法

using UnityEngine;
using System.Collections;

public class DestroyByContact : MonoBehaviour {

    public GameObject explosion;
    public GameObject playerExplosion;


    
    public int scoreValue;//分值
    private GameController gameController; //gameController脚本类

    // Use this for initialization
    void Start () {
        //获取GameController的实例
        //首先找到GameController的GameObject,然后在通过GetComponent方法来获取GameController脚本的实例
        GameObject gameControllerObject = GameObject.FindWithTag("GameController");
        if (gameControllerObject != null)
        {
            gameController = gameControllerObject.GetComponent<GameController>();
        }
        if (gameController == null)
        {
            Debug.Log("Cannot find 'GameController' script");
        }
    }
    
    // Update is called once per frame
    void Update () {
    
    }

    void OnTriggerEnter(Collider other)
    {
        //初始时障碍物会与盒子boundary发生碰撞触发此函数,要规避此种触发
        if (other.tag == "Boundary")
        {
            return;
        }
        //生成一个爆炸效果
        Instantiate(explosion, transform.position, transform.rotation);

        //飞船碰到障碍物时爆炸效果
        if(other.tag == "Player")
        {
            Instantiate(playerExplosion, other.transform.position, other.transform.rotation);

            //飞船爆炸时调用gameover方法
            gameController.GameOver();  
        }

        gameController.AddScore(scoreValue);
        //Debug.Log(other.name);
        Destroy(other.gameObject);
        Destroy(gameObject);
    }
}

总结:在实际操作中,对unity的一些构建,基本的操作不熟悉,还有场景的设置即位置等。了解到了脚本的功能,控制GameObject,触发事件,不同的脚本之间相互调用。这个游戏,通过脚本新建的对象如子弹障碍物是不会自动回收的,需要在最外围设置一个框来检测物体碰撞触发事件进行回收,同时爆炸时Instantiate方法产生的爆炸效果,也不会被自动回收,需要对这些效果写脚本DestoryByTime定时进行Destory。


画一个非常丑的图。。。

  • PlayerController:飞船的移动,发射子弹。
  • Mover:设置速度,自动控制移动。
  • RandomRotator: 自动控制障碍物旋转。
  • DestoryByContact:碰撞时触发的方法,物体消失,爆炸特效,计分,飞船爆炸时调用GameController的GameOver方法。
  • DestoryByBoundary:子弹或障碍物出边界,即与Boundary相碰撞触发方法Destory对象。
  • DestoryByTime:特效对象Destory。
  • GameController: 产生障碍物,计分,Restart等方法实现。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容