廣義的的說,和戰斗結算相關的內容都算技能系統,包括技能信息管理、技能調用接口、技能目標查找、技能表現、技能結算、技能創生體(buff/法術場/彈道)管理,此外還涉及的模塊包括:AI模塊(技能調用者)、動作模塊、尋路/移動模塊以及人物屬性和傷害數值結算等。
先說下技能模塊每個部分的職責和原理:
- 技能信息管理:管理unit所擁有的技能以及技能的等級、cd等。在我們游戲中,這里還需要負責管理符文,符文會對技能信息進行修改。
- 技能調用接口:AI或者UI操作觸發技能,觸發技能時可能選擇了一個目標(AI),也可能并沒有目標。
- 技能流程管理:一個技能可能由多個子技能以移動的執行模式組合而成,而每一個最終執行的技能執行過程也存在一個流程,一般包括:前搖過程-結算點-后搖過程。技能在前搖結束時進入技能真正的結算流程,結算流程可能創建子彈,也可能觸發buf或者創建法術場。
- 技能目標查找:若技能觸發時已經設置了技能目標unit(如怪物AI釋放技能),則直接將其作為目標unit,否則需要根據一定的策略選擇一個目標。此外,技能釋放的時候還需要釋放方向和釋放位置等信息,也通過這個模塊獲取。
- 技能表現:技能釋放過程中,需要創建相應的特效以及執行相應的動作。
- 技能創生體(buf/彈道/法術場)管理:buf掛在unit身上,可能影響unit的一些行為和狀態;法術場一般由場景管理,影響場景中某范圍內的unit;彈道就是技能創建的一個子彈,這個子彈可能以不同的路線移動(直線/拋物線/直接命中等)
1、技能表
首先說下實現技能的基本思路。實現技能的基本思路就是通過策劃填寫表格,來配制成某些技能,在執行某個技能的時候,分別去根據這些表格中的內容,確定技能如何表現。基本的邏輯是:
- if skillTable.get("技能動作"):
- paly 動作
- if skillTable.get("特效"):
- 播放特效
- if skillTable.get("法術場"):
- 創建法術場
- ....
復制代碼
2、技能信息管理
unit創建時,此模塊管理unit可使用哪些技能,比如游戲中玩家可以選擇使用哪些技能。
游戲中技能的升級、技能加點、技能池管理都在這個模塊。
此模塊還需要管理技能等級/符文/裝備等外部模塊對技能參數的修改。
3、技能調用接口
提供技能調用的接口供AI或玩家操作調用,調用時可以提供一個目標unit,也可以不提供讓技能自己查找。
提供三個接口:
- 技能開始skill_enter:開始執行技能,若技能不循環進行,則技能可以自動結束。
- 技能結束skill_exit:有的技能不能自己結束,比如某些循環技能,對于循環技能玩家可以按住按鈕一直釋放。當玩家松開按鈕,調用技能結束接口,告訴當前技能使其結束,此時技能到達后搖點時,技能不再繼續執行。
- 技能停止skill_stop:當技能被強制打斷時,如被攻擊、暈眩、藍不足等,技能會被強制停止。
此外,當前一個技能正在執行時新的技能調用啟動,此時新的技能調用信息會被保存。一般來說,并不會把所有新的技能調用信息保存下來,那樣就成了一個技能執行的序列。我們游戲僅保存一個新的技能調用信息。
總的來說,技能模塊提供盡量少的接口供AI/UI等上層邏輯使用,這樣可以有效的與AI和UI進行解耦。
4、技能流程管理
技能流程這里分兩點討論:
(1)一個技能可能由多個子技能以一定的模式組合起來。
一個技能常常由多個子技能以一定的模式組合而成,比如三段擊、比如沖鋒斬(先沖鋒、后斬)等,甚至還存在根據不同的環境選擇執行不同的子技能。分析策劃需求發現,技能可以分成一個樹形結構,這個樹形結構非常類似行為樹,同樣可以將節點分為控制節點和執行節點,甚至可以包括condition節點。為此,我們項目引入一個技能樹概念來描述這種數據結構。
(2)一個具體的技能(技能樹執行節點)也有一個固定的執行流程。這個流程一般為:前搖過程、前搖過程結束=技能結算時間點、后搖時間點。
4.1 技能樹
技能樹參考傳統行為樹的設計,使用樹形結構控制技能的執行流程。
技能樹和行為樹在結構上比較類似,但是在運行邏輯上有很大的不同。
首先,技能樹的重點并不是根據上下文選擇一個合適的節點執行,而是以一定的策略將技能樹從頭到尾遍歷執行一遍。
其次,技能樹沒有tick的概念,而是基于回調的,比如一個順序節點,順序節點中一個子節點執行完畢后,馬上通知順序節點,順序節點執行下一個子節點,直至順序節點的最后一個子節點執行完畢,順序節點就會通知父節點(如果有)它已經執行完畢。
此外,為了完成技能的一些需求,控制節點往往存儲更多的控制信息來控制子節點的執行流程。具體的信息根據策劃需求設置,比如順序結點包括原子屬性和循環屬性。如果一個順序節點具有原子屬性,則這個順樹節點在執行的過程中并不會被end,只有全部子節點執行結束才可以end。
以我們游戲中戰士普攻三段擊為例:
三段擊本身是一個順序節點,當技能開始時,此節點順序執行三個子節點。對于第一個子節點,它依然是一個順序節點,首先沖鋒至目標單位身前,然后對目標單位進行揮砍。但是沖鋒節點還包括了一個condition,若和目標的距離很近,則跳過沖鋒節點,直接揮砍。
普攻是一個循環技能,這個技能只要玩家點著按鈕不放開,技能就會一直執行,因此根節點(普攻)是一個具有循環屬性的順序節點。而對于子技能1(控制節點),他是一個具有原子屬性的順序技能,即當單位正在沖鋒時,玩家松開按鈕,單位也會執行完揮砍后才會推出技能。
!關于技能樹的使用和思考
技能樹開始的設計思路是,有些技能的執行流程和行為樹類似,比如以一定的順序執行一系列子技能,比如根據不同的上下文確定技能的執行流程。簡單的說,技能樹的引入有以下好處:1.使技能模塊可以獲得部分AI的能力,從而將和技能強相關的AI邏輯放在技能模塊使技能模塊和AI模塊降低耦合,2.可以清晰的描述技能流程,3.使用樹增加拓展性,策劃可以設計出各種各樣復雜的技能。
關于好處1,舉個例子:屠夫boss的勾子技能可以將玩家拉過來,若成功的拉過來,boss會執行一個攻擊子技能,否則不執行。通過這樣可以將勾人和攻擊作為兩個子技能構成技能樹,攻擊子技能有一個condition過程,即判斷上一個子技能是否成功。
技能樹在使用后慢慢發現一些問題,首先,技能樹的同步要求每個樹節點都進行同步,增加同步負擔,其次,技能本身并不會有太復雜的控制結構。
為此,后來我們對技能樹進行了優化:
簡化同步信息,不再同步所有節點的enter/exit信息(具體參考文章《技能模塊的同步》)。
取消并行節點,通過拓展表頭實現一個技能同時執行多件事情。
最終的技能樹基本上是只有順序/隨機兩種控制類型節點,節點擁有較輕度的condition功能。
4.2 執行節點的技能流程
一般來說,技能的執行流程包括:
前搖時間:技能開始,但是技能真正的結算流程還沒開始。技能開始以后,機能相關的特效和動作就開始播放。
前搖時間結束:技能前搖結束時技能開始真正的釋放以及結算,等技能前搖結束以后,技能真正的釋放并結算。釋放包括創建相應的彈道/法術場和buff。
技能后搖點:技能播放到后搖點時間時,技能真正的結束。這時,技能對應的特效以及人物動作可能還會繼續播放,但是技能流程已經正式結束了。也就是說,下一個技能可以執行。
5、技能目標查找
技能釋放時,目標可能已經由AI傳給了技能模塊,也有可能沒有一個目標,如玩家控制單位。
技能在釋放法術場、彈道的時候,重要的是技能的方向而不是技能目標一般來說,技能獲得一個目標對象以后,技能的方向就是釋法者到目標的方向。
此外,技能方向可能需要一些配置,如前搖鎖定(前搖過程中目標移動,技能方向不變),UI可控制(技能釋放過程中,玩家可以通過控制UI控制技能的釋放方向)。
6、技能表現
技能的表現包括動作、特效、shader、音效等。其中,特效比較復雜,需要配置的內容也比較多。比如,有些特效掛在模型上,有的特效掛在場景里。對于法術場的特效,分別可以分為法術場開始、結算、結束特效,分別在法術場開始時、結算時、結束時顯示。對于buff也類似。
7、彈道、法術場和buff等技能創生體
狹義的來說,技能只是負責技能的執行流程(技能樹管理以及技能流程管理),而技能真正的結算主要是由其創生體結算的。當技能前搖結束開始生效時,技能創建相應的彈道和法術場,法術場彈道擊中敵人時又有可能產生相應的buff。
一般來說,法術場是一個場景的某塊檢測區域,每隔一段時間法術場檢測此區域的敵人,并對其攻擊結算。
彈道是一類子彈移動路徑的抽象,創建一個彈道就表示一個子彈特效沿這個彈道移動并檢測路徑上的敵人。
buff就是掛在單位身上的一個具有持續時間的狀態,狀態對單位產生一些正面或者負面的影響,并且在此段時間內,每隔一段時間進行一次傷害結算 。
對于技能、法術場、buff之間的功能界定并不是很固定,比如技能能否直接對單位造成傷害,法術場能否對單位造成傷害,甚至技能只能創建法術場,法術場只能檢測目標不能造成傷害,只能掛buff,而所有的傷害都是通過buff來結算。當然,這樣并不一定好,一般來說,技能和法術場都可以對單位造成傷害。
總之,創生體功能的界定需要根據策劃需求、效率考慮等因素調整。
7.1 Buff狀態
Buff就是掛在單位身上持續一定時間的有益或者有害的狀態,這里狀態=buff。
Buff模塊有個需要注意的是Buff之間的相互關系,如排斥(A狀態在,B狀態掛不上去),清除(A狀態掛上去同時導致B狀態消失)等。
為了實現以上功能,最簡單的方式是在狀態A中直接填寫狀態關系狀態字段,如狀態A排斥狀態B/C/D/E...,A狀態清除狀態X/Y/Z...。
以上的實現方式有個問題,等游戲做到后期,我們有成千上萬個buff狀態,那么一個魔法免疫狀態,策劃需要填表的排斥狀態可能成千上萬。
為了解決這個問題,可以使用分類的思想解決。定義某類狀態和另一類狀態之間的規則。
基于以上思想,引入一個叫buff原子狀態的概念,原子狀態表示一類狀態,如減速、禁魔、魔免、懸空、暈眩、變羊等等等。
在給單位掛一個新的buff的之前,查詢此buff持有的原子狀態和單位身上已經有的原子狀態之間的關系,根據單位身上已有的原子狀態判定新的原子狀態應該使用何種行為處理。
此處的何種行為,代表的就是原子狀態之間的規則,如排斥等。這些規則可以讓策劃填一個名字叫“原子狀態關系”的表,此表是一個n*n的二維數組,n為游戲中所有的原子狀態的數量。
原子狀態的數量遠遠小于buff的數量,所以可以很容易的定義這些規則。
7.2 法術場
法術場描述對一塊區域的影響,這塊區域可以每隔一段時間進行一次檢測,檢測這塊區域內的單位并且對單位進行結算。
法術場需要注意一個問題,就是一個法術場每次結算可能使用不同的參數進行結算,比如一個技能,第一次結算對每個單位進行暈眩,第二次結算對單位進行傷害。
解決這種問題比較直接的方式是技能直接創建兩個法術場,每個法術場結算一次,第二個法術場創建具有延遲時間。但是這種方式有個問題,有可能策劃需求做一個結算十次而且每次結算的參數都不同的法術場。那么,一個技能以一定的時間間隔創建是個法術場,同時法術場的管理具有一定的成本,從而導致效率的降低。
為解決這個問題,我們優化了法術場結算的實現機制,增加了一種新的法術場:序列法術場。這類法術場策劃可以配置法術場每次結算之間的時間間隔以及每次結算所使用的法術場參數。