快要一个月没有更新了,这段时间也没闲着。今天就把这段时间学到的东西稍微总结一下。

前段时间开始看一些实战开发的内容和教程。一些是来自于麦子学院的视频,一些来自于蛮牛教育,还有一部分官方教学视频。最终选择试着跟着官方的教程,尝试敲一次这个叫做拾荒者的官方示例。

过程中遇到了许多之前不明白的知识,所以虽然代码量不是很大,但是仍然消耗了大量的时间用于查询其中一些使用的方法和技巧。

最终当然是仿照着实例成功地将游戏做了出来(虽然其中大部分东西只要跟着敲都问题不大。)

以下我将一些自己的见解以注释的形式写入了代码中。

首先是一个叫做BoardMannager的脚本,这个脚本用于生成每张地图。由于游戏是设计为一个Roguelike类型的游戏的,那么随机生成的地图一定是必不可少的。

using UnityEngine;
using System; //为了使用其中的Serializable
using System.Collections.Generic;
using Random = UnityEngine.Random;

public class BoardMannager : MonoBehaviour {

    [Serializable]//序列化,<span style="font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; line-height: 25.2px;"><span style="font-size:10px;">使得变量可以被Inspector界面获得</span></span>
    public class Count //之后用于存储随机生成最小数量和最大数量的值
    {
        public int minimum;
        public int maximum;
        public Count(int min,int max)
        {
            minimum = min;
            maximum = max;
        }
    }

    public int columns = 8;
    public int rows = 8;
    public Count wallCount = new Count(5, 9);
    public Count foodCount = new Count(1, 5);
    public GameObject exit;
    public GameObject[] floorTiles;
    public GameObject[] wallTiles;
    public GameObject[] foodTiles;
    public GameObject[] enemyTiles;
    public GameObject[] outerWallTiles;

    private Transform boardHolder;//将所有通过代码生成的对象整合在boardHolder中
    private List<Vector3> gridPositions = new List<Vector3>();//存储网格位置信息

    void InitialiseList()
    {
        gridPositions.Clear();

        for (int x = 1; x < columns-1; x++)
        {
            for (int y = 1; y < rows-1; y++)
            {
                gridPositions.Add(new Vector3(x, y, 0f));//通过二重循环获得整个网格的位置
            }
        }
    }

    void BoardSetup()//生成外墙和场景内的地面
    {
        boardHolder = new GameObject("Board").transform;
        for (int x = -1; x < columns + 1; x++)
        {
            for (int y = -1; y < rows + 1; y++)
            {
                GameObject toInstantiate = floorTiles[Random.Range(0,floorTiles.Length)];
                if (x==-1|| x==columns||y==-1||y==rows)//判断是否是外墙位置
                {
                    toInstantiate=outerWallTiles[Random.Range(0,outerWallTiles.Length)];
                }
                GameObject instance = Instantiate(toInstantiate, new Vector3(x, y, 0f), Quaternion.identity) as GameObject;//prefab实例化
                instance.transform.SetParent(boardHolder); //将实例化的对象放到board中
            }
        }
    }

    Vector3 RandomPosition() //随机获取一个地图内的位置
    {
        int randomIndex = Random.Range(0, gridPositions.Count);
        Vector3 randomPosition = gridPositions[randomIndex];
        gridPositions.RemoveAt(randomIndex);
        return randomPosition;
    }


    void LayoutObjectAtRandom(GameObject[] tileArray, int minimum, int maxmum)  //将物品在地图中实例化
    {
        int objectCount = Random.Range(minimum,maxmum+1);

        for (int i = 0; i < objectCount; i++)
        {
            Vector3 randomPosition = RandomPosition();
            GameObject tileChoice = tileArray[Random.Range(0, tileArray.Length)];
            Instantiate(tileChoice,randomPosition,Quaternion.identity);
        }
    }

    public void SetupScene(int level) //调用之前的函数,完成一张地图的生成
    {
        BoardSetup();
        InitialiseList();
        LayoutObjectAtRandom(wallTiles, wallCount.minimum, wallCount.maximum);
        LayoutObjectAtRandom(foodTiles, foodCount.minimum, foodCount.maximum) ;
        int enemyCount = (int)Mathf.Log(level,2f);
        LayoutObjectAtRandom(enemyTiles, enemyCount, enemyCount);
        Instantiate(exit, new Vector3(columns - 1, rows - 1, 0f), Quaternion.identity);
        
    }
}
using UnityEngine;
using System.Collections;
using System.Collections.Generic;
using UnityEngine.UI;


public class GameManager : MonoBehaviour {
//GameManager脚本主要用于管理整个的逻辑运行
    public float levelStartDelay = 2f;
    public float turnDelay = .1f;
    public static GameManager instance = null;//单例设计,保证在游戏运行中永远只有一个GameManager
    public BoardMannager boardScript;
    public int platerFoodPoints = 100;
    [HideInInspector]public bool playersTurn = true;

    private Text levelText;
    private GameObject levelImage;
    private int level = 1;
    private List<Enemy> enemies;
    private bool enemiesMoving;
    private bool doingSetup;


    void Awake() {
        if (instance==null)
        {
            instance = this;
        }
        else if (instance!=null)
	            Destroy(gameObject);//单例设计部分
        DontDestroyOnLoad(gameObject);
        enemies=new List<Enemy>();
        boardScript=GetComponent<BoardMannager>();//获取boardMannager脚本
        InitGame();    
    }

    private void OnLevelWasLoaded(int index)
    {
        level++;
        InitGame();
    }

    void InitGame() {
        doingSetup = true;
        levelImage = GameObject.Find("LevelImage");
        levelText = GameObject.Find("LevelText").GetComponent<Text>();
        levelText.text = "Day " + level;
        levelImage.SetActive(true);
        Invoke("HideLevelImage", levelStartDelay); //UI部分,显示第几日、显示黑色的背景等等

        enemies.Clear();
        boardScript.SetupScene(level); //调用BoardMannager生成新地图
    
    }

    private void HideLevelImage()
    {
        levelImage.SetActive(false);
        doingSetup = false;
    }
    IEnumerator MoveEnemies( )
    {
        enemiesMoving = true;
        yield return new WaitForSeconds(turnDelay);
        if (enemies.Count==0)
        {
            yield return new WaitForSeconds(turnDelay);
        }
        for (int i = 0; i < enemies.Count; i++)
        {
            enemies[i].MoveEnemy();
            yield return new WaitForSeconds(enemies[i].moveTime);
        }
        playersTurn = true;
        enemiesMoving = false;
    }

    public void AddEnemyToList(Enemy script)
    {
        enemies.Add(script);
    }


    public void GameOver()
    {
        levelText.text = "After " + level + " days,you starved.";
        levelImage.SetActive(true);
        enabled = false; //游戏结束时显示的UI
    }
	
	void Update () {
        if (playersTurn || enemiesMoving||doingSetup)
            return;
        StartCoroutine(MoveEnemies());
	
	}
}

通过以上两段脚本,能够实现地图的生成、游戏关卡的管理、以及一些简单的GUI的显示与关闭,接下来则是用于控制角色与敌人的脚本。

首先是用于可移动物体的基类MovingObject:

using UnityEngine;
using System.Collections;

public abstract class MovingObject : MonoBehaviour {
    public float moveTime = .1f;
    public LayerMask blockingLayer;
    
    public BoxCollider2D boxCollider;
    private Rigidbody2D rb2D;
    private float inverseMoveTime;
	
	protected virtual void Start () {
        boxCollider = GetComponent<BoxCollider2D>();
        rb2D = GetComponent<Rigidbody2D>();
        inverseMoveTime = 1f / moveTime;
	}
    protected bool Move(int xDir, int yDir, out RaycastHit2D hit)//通过传入两个坐标值与碰撞检测来返回能否进行移动
    {<span style="white-space:pre">	</span>
        Vector2 start = transform.position;
        Vector2 end = start + new Vector2(xDir, yDir);

        boxCollider.enabled = false;

        hit = Physics2D.Linecast(start,end,blockingLayer);//使用射线碰撞检测
        boxCollider.enabled = true;
        if (hit.transform==null)
        {
            StartCoroutine(SmoothMovement(end));
            return true;
        }
        return false;
    }

    protected IEnumerator SmoothMovement(Vector3 end)
    {
        float sqrRemainingDistance = (transform.position - end).sqrMagnitude;
        //sqrMagnitude用于计算向量长度的平方,其效率比计算长度高
        while (sqrRemainingDistance>float.Epsilon)//Epsilon表示一个极小的数
        {
            Vector3 newPosition = Vector3.MoveTowards(rb2D.position, end, inverseMoveTime * Time.deltaTime);
            rb2D.MovePosition(newPosition);
            sqrRemainingDistance = (transform.position - end).sqrMagnitude;
            yield return null;
        }
    }

    protected virtual void AttemptMove<T>(int xDir, int yDir) //使用泛型T,在子类中可以指定不同的对象
        where T : Component
    {
        RaycastHit2D hit;
        bool canMove = Move(xDir,yDir,out hit);
        if (hit.transform==null)
        {
            return;
        }
        T hitComponent = hit.transform.GetComponent<T>();//使用泛型T作为一个公共的方法
        if (!canMove && hitComponent != null)
            OnCantMove(hitComponent);
    }

    protected abstract void OnCantMove<T>(T component)
        where T : Component;

}

接下来是继承自MovingObject的子类Player和Enemy

using UnityEngine;
using System.Collections;
using UnityEngine.UI;

public class Player : MovingObject{

    public int wallDamage = 1;
    public int pointsPerFood = 10;
    public int pointsPerSoda = 20;
    public float restartLevelDelay = 1f;
    public Text foodText;
    public AudioClip moveSound1;
    public AudioClip moveSound2;
    public AudioClip eatSound1;
    public AudioClip eatSound2;
    public AudioClip drinkSound1;
    public AudioClip drinkSound2;
    public AudioClip gameOverSound;

    private Animator animator;
    private int food;
    private Vector2 touchOrigin = -Vector2.one;
	protected override void Start () {
        animator = GetComponent<Animator>();//获取Animator以便更改Trigger
        food = GameManager.instance.platerFoodPoints;//获取食物可增加的点数
        foodText.text = "Food: " + food;//更新UI
        base.Start();

	}
    private void OnDisable()
    {
        GameManager.instance.platerFoodPoints = food;
    }
    protected override void AttemptMove<T>(int xDir, int yDir)
    {//重写父类中的移动方法
        food--;//每移动一次食物值减一
        foodText.text = "Food: " + food;//更新UI
        base.AttemptMove<T>(xDir, yDir);//调用父类中的移动方法
        RaycastHit2D hit;
        if (Move(xDir,yDir,out hit))
        {
            SoundManager.instance.RandomizeSfx(moveSound1, moveSound2);//如果移动成功,播放移动音效
        }
        CheckIfGameover();
        GameManager.instance.playersTurn = false;
    }

    private void CheckIfGameover()
    {//检测游戏是否结束,如果food小于等于0则结束,显示UI播放音效
        if (food <= 0)
        {
            SoundManager.instance.PlaySingle(gameOverSound);
            SoundManager.instance.musicSource.Stop();
            GameManager.instance.GameOver();
        }

            
    }
    protected override void OnCantMove<T>(T component)
    {//重写父类中的OnCantMove,可以对墙造成伤害,后见Wall类脚本
        Wall hitWall = component as Wall;
        hitWall.DamageWall(wallDamage);
        animator.SetTrigger("PlayerChop");//通过SetTrigger播放Chop的动画
    }
    private void Restart()
    {
        Application.LoadLevel(Application.loadedLevel);
    }
    public void LoseFood(int loss)
    {//受到攻击时减少food值,播放音效,更新UI
        animator.SetTrigger("PlayerHit");
        food -= loss;
        foodText.text = "-" + loss + "Food: " + food;
        CheckIfGameover();
    }
    private void OnTriggerEnter2D(Collider2D other)
    {//碰撞物体发生的事件,如果是Exit进入下一关,如果是Food或者Soda,增加food值,播放音效更新UI
        if (other.tag == "Exit")
        {
            Invoke("Restart", restartLevelDelay);
            enabled = false; 
        }
        else if (other.tag == "Food")
        {
            food += pointsPerFood;
            SoundManager.instance.RandomizeSfx(eatSound1, eatSound2);
            foodText.text = "+" + pointsPerFood + "Food: " + food;
            other.gameObject.SetActive(false);
        }
        else if (other.tag == "Soda")
        {
            food += pointsPerSoda;
            SoundManager.instance.RandomizeSfx(drinkSound1, drinkSound2);
            foodText.text = "+" + pointsPerSoda + "Food: " + food;
            other.gameObject.SetActive(false);
        }
        
    }
	
	void Update () {
        if (!GameManager.instance.playersTurn) return;
        int horizontal = 0;
        int vertical = 0;
    #if UNITY_STANDALONE||UNITY_WEBPLAYER //多平台操控,如果是PC或者WEBPALYER可以直接使用键盘WSAD或者方向键操作
        horizontal = (int)Input.GetAxisRaw("Horizontal");
        vertical = (int)Input.GetAxisRaw("Vertical");
        if (horizontal != 0)
            vertical=0;
        
    #else
        if (Input.touchCount>0) //如果是手机端则可以使用滑动触摸屏的方法进行操作
	    {
		 Touch myTouch=Input.touches[0];
            if (myTouch.phase==TouchPhase.Began)
	    {
		    touchOrigin=myTouch.position;
	    }
            else if (myTouch.phase==TouchPhase.Ended&&touchOrigin.x>=0)
	    {
		     Vector2 touchEnd=myTouch.position;
                float x=touchEnd.x-touchOrigin.x;
                float y=touchEnd.y-touchOrigin.y;
                touchOrigin.x=-1;
                if(Mathf.Abs(x)>Mathf.Abs(y))
                    horizontal=x>0?-1:1;
                else
                    vertical=y>0?-1:1;
	    }
	}
#endif
        if (horizontal != 0 || vertical != 0)
            AttemptMove<Wall>(horizontal,vertical);
	}
}
using UnityEngine;
using System.Collections;

public class Enemy : MovingObject {
//同样继承自MovingObject类
    public int playerDamage;
    private Animator animator;
    private Transform target;
    private bool skipMove;
    public AudioClip enemyAttack1;
    public AudioClip enemyAttack2;


	protected override void Start () {
        GameManager.instance.AddEnemyToList(this);
        animator = GetComponent<Animator>();
        target=GameObject.FindGameObjectWithTag("Player").transform;
        base.Start();
    }

    protected override void AttemptMove<T>(int xDir, int yDir)
    {
        if (skipMove)
        {
            skipMove = false;
            return;
        }
        base.AttemptMove<T>(xDir, yDir);
        skipMove = true;

    }

    public void MoveEnemy()
    {
        int xDir = 0;
        int yDir = 0;
        if (Mathf.Abs(target.position.x - transform.position.x) < float.Epsilon)//先判断敌人与玩家X轴之间是否相差,如果相差先移动X方向
        {
            yDir = target.position.y > transform.position.y ? 1 : -1; //判断移动向玩家的方向

        }
        else
            xDir = target.position.x > transform.position.x ? 1 : -1; //同理的判断Y轴移动的方向
        AttemptMove<Player>(xDir, yDir); //传入泛型<Player>
    }

    protected override void OnCantMove<T>(T component)
    {
        Player hitPlayer = component as Player;
        animator.SetTrigger("enemyAttack");//播放Attack动画
        hitPlayer.LoseFood(playerDamage); //攻击使得玩家减少food值
        SoundManager.instance.RandomizeSfx(enemyAttack1, enemyAttack2);
    }
}

接下来是墙的脚本Wall


using UnityEngine;
using System.Collections;

public class Wall : MonoBehaviour {

    public Sprite dmgSprite;
    public int hp = 4;
    public AudioClip chopSound1;
    public AudioClip chopSound2;
    private SpriteRenderer spriteRenderer;
	
	void Awake () 
    {   
        spriteRenderer=GetComponent<SpriteRenderer>();
	
	}

    public void DamageWall(int loss)
    {
        SoundManager.instance.RandomizeSfx(chopSound1, chopSound2);
        spriteRenderer.sprite = dmgSprite;
        hp -= loss;
        if (hp <= 0)//每堵墙有4点生命值,当生命值为0时就会被设置为Disable
            gameObject.SetActive(false);
    }
}
using UnityEngine;
using System.Collections;
 //SoundManager类用于管理游戏中会出现的背景音乐以及音效
public class SoundManager : MonoBehaviour {
    public AudioSource efxSource;
    public AudioSource musicSource;
    public static SoundManager instance = null;

    public float lowPitchRange = .95f;
    public float highPitchRange = 1.05f;
	// Use this for initialization
	void Awake () {
        if (instance == null)
            instance = this;
        else if (instance != this)
            Destroy(gameObject);
        DontDestroyOnLoad(gameObject);
	
	}
    public void PlaySingle(AudioClip clip)
    {
        efxSource.clip = clip;
        efxSource.Play();
    }

    public void RandomizeSfx(params AudioClip[] clips)
    {
        int randomIndex = Random.Range(0, clips.Length);
        float randomPitch = Random.Range(lowPitchRange, highPitchRange);
        efxSource.pitch = randomPitch;
        efxSource.clip = clips[randomIndex];
        efxSource.Play(); 
    }

	
}
using UnityEngine;
using System.Collections;
//最后将Loader脚本放在MainCamera上,实现开始游戏
public class Loader : MonoBehaviour {
    public GameObject gameManager;
    void Awake()
    {
        if (GameManager.instance == null)
            Instantiate( gameManager);
    }
}