同樣的溜哮,實(shí)現(xiàn)過程只記大致思路與新知識點(diǎn)谊路,不記細(xì)節(jié)赊锚。
github: https://github.com/robotluo/2DRoguelike
大致實(shí)現(xiàn)功能:
1.創(chuàng)建主角和敵人成箫,創(chuàng)建地圖
2.隨機(jī)生成敵人和食物
3.獲取食物格郁、碰撞檢測
4.與敵人碰撞時(shí)狀態(tài)機(jī)的變換
5.血量控制腹殿、關(guān)卡轉(zhuǎn)換
6.添加音效
2DRoguelike 是官方的案例之一独悴,與之前所用的有所區(qū)別在于使用了狀態(tài)機(jī)來控制人物動(dòng)畫。
狀態(tài)機(jī)保存了許多不同的狀態(tài)锣尉,事件觸發(fā)可以改變狀態(tài)機(jī)的狀態(tài)刻炒,實(shí)現(xiàn)效果的轉(zhuǎn)換。
創(chuàng)建地圖:
將素材裝入一個(gè)數(shù)組自沧,隨機(jī)一個(gè)索引并創(chuàng)建實(shí)例落蝙。
// outWalls 為外墻數(shù)組
// Map 為一個(gè)空 GameOnject 的 Transform,用于裝載地圖暂幼,比較整潔
int outWallIndex = Random.Range (0, outWalls.Length);
GameObject tmp = GameObject.Instantiate (outWalls [outWallIndex], new Vector3 (row, col, 0), Quaternion.identity) as GameObject;
tmp.transform.SetParent (Map);
當(dāng)我們有許多實(shí)例需要隨機(jī)生成時(shí)筏勒,例如,敵人旺嬉、食物管行、障礙物等都需要隨機(jī)生成在地圖中,因此可以寫一個(gè)專門的方法邪媳。
先將地圖上能存實(shí)例的坐標(biāo)存入 mapList:
private List<Vector2> mapList = new List<Vector2>();
//將地圖上能放置物品的坐標(biāo)存入 mapList
mapList.Clear ();
for (int row = 2; row <= rowMax - 2; row++) {
for(int col =2 ;col <= colMax - 2; col++) {
mapList.Add (new Vector2 (row, col));
}
}
專門用于創(chuàng)建的方法:
void creatElement(int count, GameObject[] array){
for (int i = 0; i < count; i++) {
int listIndex = Random.Range (0, mapList.Count ); //隨機(jī)取得放置點(diǎn)
int arrIndex = Random.Range (0, array.Length); //隨機(jī)取得對象
Vector2 pos = mapList [listIndex]; //取出放置點(diǎn)
mapList.Remove(pos); // list 中去掉該位置
GameObject tmp = GameObject.Instantiate (array[arrIndex],pos,Quaternion.identity) as GameObject;
tmp.transform.SetParent (Map);
}
}
調(diào)用時(shí)只需要傳入數(shù)量和數(shù)組即可捐顷。
狀態(tài)機(jī)的創(chuàng)建:
將素材多選直接拖入Hierarchy 中,會(huì)自動(dòng)生成兩個(gè)文件雨效。一個(gè)是動(dòng)畫迅涮,這里命名為 Animation ,另一個(gè)是控制器徽龟,這里命名為AnimatorController叮姑。狀態(tài)機(jī)的邏輯非常簡單,將每個(gè)狀態(tài)的具體實(shí)現(xiàn)封裝起來据悔,當(dāng)滿足某個(gè)條件传透,將狀態(tài)從當(dāng)前狀態(tài)轉(zhuǎn)換到另一狀態(tài),狀態(tài)內(nèi)部與狀態(tài)轉(zhuǎn)換之間不互相影響极颓。
如果一個(gè)主角有多個(gè)狀態(tài)朱盐,例如有靜止、移動(dòng)菠隆、攻擊兵琳、受傷等等,只需要將其全部拖到同一個(gè)物體上骇径,如果直接拖到 Hierarchy 會(huì)產(chǎn)生多個(gè)狀態(tài)機(jī)躯肌。
共用狀態(tài)機(jī):
狀態(tài)機(jī)分為兩部分,動(dòng)畫和控制既峡。在實(shí)際中我們可能需要多個(gè)角色共用同一狀態(tài)機(jī)羡榴。例如碧查,兩種不同的怪物运敢,他們的動(dòng)畫不同校仑,但是處理狀態(tài)的邏輯是完全一樣的,這時(shí)候如果分別使用兩個(gè)狀態(tài)機(jī)就有點(diǎn)多余传惠,所以需要共用迄沫。
我們可以這么做,如果我們有了一個(gè) Controller_1 想要重用,首先在同一文件夾下右鍵 -> create -> Animator Override Controller卦方,命名為 Controller_2 羊瘩,點(diǎn)擊 Controller_2 會(huì)看到 Inspector 視圖下有一個(gè) Controller 需要賦值,將 Controller_1 賦值到此處盼砍,再將新的動(dòng)畫進(jìn)行對應(yīng)賦值尘吗。這樣,Controller_1 與 Controller_2 就共用一套邏輯浇坐。
狀態(tài)切換:
雙擊一個(gè)控制器睬捶,選中一個(gè)狀態(tài),右鍵 Make Transition近刘,會(huì)出現(xiàn)一個(gè)箭頭擒贸,此時(shí)連接到需要轉(zhuǎn)換的狀態(tài)。
例如:
連線代表能夠轉(zhuǎn)換觉渴,現(xiàn)在需要自定義轉(zhuǎn)換的條件介劫。如下圖,添加兩個(gè) Trigger
選中連線案淋,再在 Inspector 視圖下 Conditions 添加 所需Trigger座韵,改變Has Exit Time 勾選,Has Exit Time表示播放完變換踢京,當(dāng)我們攻擊時(shí)應(yīng)該是立刻做出反應(yīng)回右,所以不需要勾選。而從攻擊狀態(tài)變?yōu)槠胀顟B(tài)只需要播放完即可漱挚,不需要Trigger翔烁。
主角移動(dòng):
添加 Rigidbody ,用 MovePosition 方法移動(dòng)
myRigidbody = GetComponent<Rigidbody2D> ();
myRigidbody.MovePosition (Vector2.Lerp(transform.position, movePosition, moveSpeed * Time.deltaTime));
//(當(dāng)前位置旨涝,目標(biāo)位置蹬屹,移動(dòng)速度)
碰撞檢測:
這里采用射線的方式進(jìn)行碰撞檢測,向目標(biāo)位置發(fā)射射線白华,判斷所碰撞物體種類再進(jìn)行相關(guān)處理慨默。
myCollider.enabled = false; //去掉當(dāng)前碰撞器,防止檢測到自身
RaycastHit2D hit = Physics2D.Linecast (movePosition,movePosition + new Vector2(hor,ver));
myCollider.enabled = true; //重新啟用碰撞器
對 hit 進(jìn)行判斷和相關(guān)處理即可弧腥。
例如:攻擊墻體厦取,需要向其發(fā)送一個(gè)信息,以及播放攻擊動(dòng)畫
private Animator myAnimator;
myAnimator = GetComponent<Animator> ();
myAnimator.SetTrigger ("Attack"); //改變狀態(tài)機(jī)
hit.collider.SendMessage ("Damager");//調(diào)用墻體的 Damager fangfa
怪物控制:
怪物與主角的控制方式不同管搪,主角是按鍵控制虾攻,而怪物是自己移動(dòng)铡买。
在一個(gè)單例 GameManager 中用一個(gè) List 儲(chǔ)存所有敵人,當(dāng)主角移動(dòng)時(shí)霎箍,調(diào)用 GameManager 中的 相應(yīng)方法奇钞,遍歷 List 通知所有敵人移動(dòng)。
Text 顯示信息:
我們需要幾個(gè) UI Text 來顯示信息漂坏,需要顯示當(dāng)前食物的剩余景埃、提示當(dāng)前關(guān)卡、游戲結(jié)束時(shí)顯示 Game Over顶别」柔悖
例如顯示食物數(shù)量:
private Text foodText; //
foodText = GameObject.Find ("FoodText").GetComponent<Text> ();//按名稱查找
foodText.text = "Food : " + Hp;
關(guān)卡變換:
關(guān)卡的變換首先要判斷是否到達(dá)這種,這里有一個(gè)細(xì)節(jié)驯绎。我們判斷是否到達(dá)終點(diǎn)需要知道主角的位置蒂胞,所以在 GameManager 中要先獲得主角的引用。但若是在 Awake 中獲取条篷,MapManager 可能并沒有把主角創(chuàng)建出來骗随,此時(shí)會(huì)報(bào)錯(cuò)。所以地圖的創(chuàng)建一定要在 MapMahager 中的 Awake赴叹,而主角的獲取需要在 GameManager 中的 Start鸿染。
當(dāng)然,還有一種方式就是在 GameManage 中統(tǒng)一控制乞巧,先創(chuàng)建地圖再創(chuàng)建 UI涨椒。
關(guān)卡變換涉及到一個(gè)場景的切換(新知識),關(guān)卡變換地圖需要重新生成绽媒,但血量蚕冬,關(guān)卡數(shù)需要保留。
為了保留血量和關(guān)卡是辕,我們在進(jìn)入新的關(guān)卡時(shí)不能銷毀當(dāng)前 GameManager 囤热。
DontDestroyOnLoad (this.gameObject);//在 GameManager 中使用此方法,重新加載時(shí)不銷毀 GamaManager
void OnLevelWasLoaded(int scenesLevel) {
//每次加載關(guān)卡時(shí)都會(huì)調(diào)用获三,可以在其中寫一些需要更新的信息旁蔼,具體可以查看API
level++;
initGame (); //初始化場景
}
此時(shí)會(huì)發(fā)現(xiàn)疙教,雖然當(dāng)前 GameManager 沒有被銷毀棺聊,但是出現(xiàn)了兩個(gè) GamaManager 。所以我們需要將 GameManager 做成 Prefabs 贞谓,不讓它存在場景中限佩,而是用創(chuàng)建的方式。