基于Unity引擎的RPG3D项目开发笔录
RPG游戲開發筆錄
文章目錄
- RPG游戲開發筆錄
- 1.將普通3D項目升級為RPG渲染管線
- 2.導入素材(人物,場景,天空盒)
- 3.第三人稱自由視角與移動
- 4.切換鼠標指針
- 5.遮擋剔除實現
- 6.敵人的創建,站崗,追逐
- 7.人物基本數值實現
- 8.攻擊功能的實現(重難)
- 9.泛型單例模式以及怪物獲勝通知
- 10.模板生成更多Enemy
- 11.拓展方法實現怪物的攻擊范圍限制
- 12.血條UI的設計
- 13.玩家升級系統
- 14.玩家的血條UI
- 15.傳送門(切換關卡)
- 16.保存數據
- 17.主菜單的制作
- 18.場景轉場
1.將普通3D項目升級為RPG渲染管線
- 1.Package Manager 搜索 Universal RP進行安裝
- 2.創建通用渲染管線 Rendering--->URP Assets(with Universal Renderer)
- 3.進入Project Settings--->Graphics,選中剛創建的渲染管線
- 4.進入Quality中同樣上述操作
- 5.進入渲染管線,修改Shadows-->Max Distance,以減小渲染時帶給顯卡的壓力
- 6.修改默認渲染方式為GPU,并可修改GPU類型和烘焙渲染:
2.導入素材(人物,場景,天空盒)
更新渲染:
Windows--->Rendering-->Render Pipeline Converter--->Built-in to URP--->全部勾選--->Initialize Converters--->Convert Assets
3.第三人稱自由視角與移動
- 1.Input Manager–>復制粘貼 Horizontal和Vertical 并修改其為第四,第五坐標軸,命名為 Camera Rate X & Camera Rate Y
- 2.給攝像機設置父節點 Photographer,使其能繞著父物體移動
相機腳本:
public class Photographer : MonoBehaviour {//相機抬升(繞X軸)public float Pitch { get; private set; }//相機水平角度(繞Y軸)public float Yaw { get; private set; }//鼠標靈敏度public float mouseSensitivity = 5;//攝像機旋轉速度public float cameraRotatingSpeed = 80;public float cameraYSpeed = 5;//相機跟隨目標private Transform followTarget;public void InitCamera(Transform target){followTarget = target;transform.position = target.position;}private void Update(){UpdateRotation();UpdatePosition();}private void UpdateRotation(){Yaw += Input.GetAxis("Mouse X") * mouseSensitivity;Yaw += Input.GetAxis("CameraRateX") * cameraRotatingSpeed * Time.deltaTime ;//Debug.Log(Yaw);Pitch += Input.GetAxis("Mouse Y") * mouseSensitivity;Pitch += Input.GetAxis("CameraRateY") * cameraRotatingSpeed * Time.deltaTime;//限制抬升角度Pitch = Mathf.Clamp(Pitch, -90, 90);//Debug.Log(Pitch);transform.rotation = Quaternion.Euler(Pitch, Yaw, 0);}private void UpdatePosition(){Vector3 position = followTarget.position;float newY = Mathf.Lerp(transform.position.y, position.y, Time.deltaTime * cameraYSpeed);transform.position = new Vector3(position.x,newY,position.z);} }人物移動腳本(通用):
[RequireComponent(typeof(Rigidbody))] public class CharacterMove : MonoBehaviour {//剛體組件[Header("組件")]private Rigidbody rb;public Vector3 CurrentInput { get; private set; }public float MaxWalkSpeed = 5;void Awake(){rb = GetComponent<Rigidbody>();}private void FixedUpdate(){rb.MovePosition(rb.position + CurrentInput * MaxWalkSpeed * Time.fixedDeltaTime);//rb.MoveRotation(Quaternion.LookRotation(CurrentInput * MaxWalkSpeed * Time.fixedDeltaTime));//rb.rotation = Quaternion.LookRotation(CurrentInput * MaxWalkSpeed * Time.fixedDeltaTime);//目標旋轉角度Quaternion quaternion = Quaternion.LookRotation(CurrentInput * MaxWalkSpeed * Time.fixedDeltaTime);//平滑過渡,deltaTime為每幀渲染的時間transform.localRotation = Quaternion.Lerp(transform.localRotation, quaternion, Time.deltaTime * 10f);}public void SetMovementInput(Vector3 input){CurrentInput = Vector3.ClampMagnitude(input, 1);}玩家移動腳本:
[RequireComponent(typeof(Rigidbody))] public class PlayerLogic : MonoBehaviour {[Header("組件")]//剛體組件private Rigidbody rb;//動畫組件private Animator anim;//移動組件private CharacterMove characterMove;//攝像機組件public Photographer photographer;//攝像機的根位置public Transform followTarget;//附加項public float jumpForce = 10f;[Header("人物信息")]//人物移動方向private bool isForward, isBack, isLeft, isRight,isFall;void Awake(){rb = GetComponent<Rigidbody>();anim = GetComponent<Animator>();characterMove = GetComponent<CharacterMove>();photographer.InitCamera(followTarget);}void Update(){UpdateMovementInput();//Jump();AttackAnim();SwitchAnim();}//動畫變量同步private void SwitchAnim(){anim.SetBool("Forward", isForward);anim.SetBool("Back", isBack);anim.SetBool("Left", isLeft);anim.SetBool("Right", isRight);anim.SetBool("Fall", isFall);}#region 玩家移動//人物移動函數private void UpdateMovementInput(){/** TODO:添加人物死亡條件限制*/float ad = Input.GetAxis("Horizontal");//Debug.Log("ad值為:" + ad);float ws = Input.GetAxis("Vertical");//Debug.Log("ws值為:" + ws);Quaternion rot = Quaternion.Euler(0, photographer.Yaw, 0);characterMove.SetMovementInput(rot * Vector3.forward * ws + rot * Vector3.right * ad);if (ad != 0 || ws != 0){/*//目標旋轉角度Quaternion quaternion = Quaternion.LookRotation(new Vector3(ad, 0, ws));//平滑過渡,deltaTime為每幀渲染的時間transform.localRotation = Quaternion.Lerp(transform.localRotation, quaternion, Time.deltaTime * 10f);*///動畫狀態切換if (ad > 0.1){isRight = true;}else if (ad < -0.1){isLeft = true;}else{isRight = false;isLeft = false;}if (ws > 0.1){isForward = true;}else if (ws < -0.1){isBack = true;}else{isForward = false;isBack = false;}}//添加速度//rb.velocity = new Vector3(ad * moveSpeed, 0, ws * moveSpeed);}#endregion #region 玩家跳躍//跳躍函數private void Jump(){if (Input.GetKeyDown(KeyCode.Space) && isFall){anim.SetTrigger("Jump");rb.velocity = new Vector3(rb.velocity.x, jumpForce,rb.velocity.z);isFall = false;} }private void OnTriggerEnter(Collider other){if (other.gameObject.CompareTag("Ground")){Debug.Log("觸發到地面了");isFall = true;}}#endregion#region 玩家攻擊private void AttackAnim(){this.transform.LookAt(this.transform);if (Input.GetMouseButtonDown(0)){anim.SetTrigger("Attack");}else if (Input.GetMouseButtonDown(1)){anim.SetTrigger("Ability");}}#endregion }4.切換鼠標指針
切換鼠標指針:
using System.Collections; using System.Collections.Generic; using UnityEngine; using UnityEngine.Events;[RequireComponent(typeof(Rigidbody))] public class CharacterMove : MonoBehaviour { [Header("人物攻擊模塊")]//鼠標圖標public Texture2D target, attack;RaycastHit hitInfo;void Awake(){rb = GetComponent<Rigidbody>();}void Update(){SetCursorTexture(); }//設置鼠標指針void SetCursorTexture(){Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);if (Physics.Raycast(ray,out hitInfo)){switch (hitInfo.collider.gameObject.tag){case "Ground":Cursor.SetCursor(target, new Vector2(16, 16), CursorMode.Auto);break;case "Enemy":Cursor.SetCursor(attack, new Vector2(16, 16), CursorMode.Auto);break;}}}5.遮擋剔除實現
- 1.create-->Shader Graph--> URP--->Unit Shader Graph,并起名為 Occlusion Shader。
- 2.基于它創建材質Occlusion放回meterials文件夾
- 編輯Occlusion Shader為如下參考:
材質參數參考:
URP參數參考:
遮擋剔除實現完成!
6.敵人的創建,站崗,追逐
- 1.導入素材,設置基本狀態信息(Idle,Chase,Guard,Dead)
- 2.敵人的狀態切換,追逐玩家以及切換動畫:
7.人物基本數值實現
- 1.給文件夾分好類,分別創建 MonoBehavior和ScriptableObject文件夾。
- 2.創建第一個ScriptableObject腳本文件,命名為 CharacterData_SO。
- 3.編寫人物應有的通用屬性,并在專門的數據文件夾下創建出數據文件。
數據模板寫法:
[CreateAssetMenu(fileName ="New Data",menuName = "Character Stats/Data")] public class CharacterData_SO : ScriptableObject {[Header("Stats Info")]//最大生命值public int maxHealth;//當前生命值public int currentHealth;//基礎防御力public int baseDefence;//當前防御力public int currentDefence; }數據操作寫法:
public class CharacterStats : MonoBehaviour {public CharacterData_SO characterData;#region Read from Data_SOpublic int MaxHealth {get{return characterData != null ? characterData.maxHealth : 0;}set{characterData.maxHealth = value;}}public int CurrentHealth{get{return characterData != null ? characterData.currentHealth : 0;}set{characterData.currentHealth = value;}}public int BaseDefence{get{return characterData != null ? characterData.baseDefence : 0;}set{characterData.baseDefence = value;}}public int CurrentDefence{get{return characterData != null ? characterData.currentDefence : 0;}set{characterData.currentDefence = value;}}#endregion }- 4.將數據操作腳本 CharacterStats掛載到Player和Enemy上,并拖入對應數據文件。
- 5.在玩家控制腳本中獲取到數據文件,并獲取到操作該數據的方法。
- 6.同樣方式實現攻擊數值的基本書寫:
8.攻擊功能的實現(重難)
- 1.創建攻擊數據腳本:包括玩家和怪物
給出參考:Player Attack Data_SO
玩家攻擊腳本面板參考:
- 2.在角色數據操作腳本中引入攻擊腳本:并寫出傷害計算邏輯
由于這里攻擊函數是玩家和怪物通用的,所以后續只需要傳入攻擊者和受擊者即可正常完成傷害計算并扣血。其次,怪物的進攻邏輯是AI操控,只需要在追蹤玩家時,吧玩家對象傳入怪物的攻擊目標變量即可。但玩家的攻擊目標選擇卻成了問題,這里采用類似怪物發現玩家的方式,采用一個圓形探測范圍(該范圍閾值即為玩家的攻擊范圍),當玩家進行攻擊操作時,只需要檢測怪物是否在范圍檢測距離內,如果在距離內,則正常調用TakeDamage(),如果不在,則視為空刀。缺點是實際攻擊范圍(以玩家為圓心,以攻擊距離為半徑的圓形內)與動畫攻擊范圍(人物的前方扇形區域)不符合。后續有待改進…此外,暴擊功能較為繁瑣,目前未實現。以下為判斷暴擊的常用方法之一,僅供參考:
//設置全局變量,判斷是否暴擊 characterStats.isCritical = UnityEngine.Random.value < characterStats.attackData.criticalChance;- 3.怪物的攻擊邏輯:(將其掛載到怪物攻擊動畫的某一幀上)
- 4.玩家的攻擊邏輯
- 5.死亡的判斷
思路:死亡判斷直接在Update()函數中逐幀判斷當前生命值是否為0即可:
玩家死亡:
怪物死亡:
void Update(){if (characterStats.CurrentHealth != 0){SwitchStates();}else{isDead = true;enemyStates = EnemyStates.DEAD;}SwitchAnimation();}9.泛型單例模式以及怪物獲勝通知
- 1.寫一個管理單例模式的基類( Tools/Singleton ):
- 2.寫一個發放廣播的接口 IEndGameObserver
- 3.寫一個游戲管理類繼承管理基類 ( Manages/GameManager )
- 4.在玩家邏輯中添加注冊信息到管理類
PlayerLogic:
- 5.在怪物邏輯中添加 注冊和銷毀通知,以及通知內容的具體實現
EnemyController:
報錯總結:此處在GameManager的實例化階段一直報 空引用異常,后來發現自己沒有創建 Game Manager 的結點,并掛載GameManager的腳本。此外,還需注意 Onable函數是場景初始化時被創建,此處不涉及場景切換,故,需將其暫時寫在Start方法中,后續再更改。
10.模板生成更多Enemy
- 1.由于之前寫法會使多個復制出來的怪物屬性共享,這顯然不是我們需要的功能,故我們需要創建一個屬性模板,讓其新生成的怪物以該模板生成對應的數值。
Character Stats中:
- 2.引入新怪物的美術模型資源,添加必要組件EnemyController,NavMesh Agent以及基本碰撞體,別忘了將其標簽改為Enemy,并創建對應的數據文件。
- 3.在重置新類型敵人的動畫信息時,可直接創建一個 Override Animators,可十分便利的重置怪物動畫。
11.拓展方法實現怪物的攻擊范圍限制
由于之前只要怪物發動攻擊動畫,玩家必掉血。這是極度不合理的,我們希望玩家有一定的閃避空間或容錯,故我們采用 拓展方法 Extension Method 來對其攻擊范圍做一個限制。
- 1.寫一個 Extension Method腳本:
- 2.并在怪物的攻擊動畫事件中加上限制條件:
EnemyController:
- 3.給Boss的基礎攻擊增加擊退效果
但暫時有點問題,無法正常擊退玩家。后續再改BUG。
public class Boss_Rock : EnemyController {//擊飛玩家的力public float kickForce = 25f;public void KickOff(){if (AttackTarget != null && transform.IsFacingTarget(AttackTarget.transform)){Debug.Log("被踢開了");//拿到受害者的屬性信息var targetStats = AttackTarget.GetComponent<CharacterStats>();//計算擊飛方向(問題)//FIXME:有待修改Vector3 direction = (AttackTarget.transform.position - transform.position).normalized;AttackTarget.GetComponent<Rigidbody>().velocity = direction * kickForce;//造成傷害targetStats.TakeDamage(characterStats, targetStats);}} }- 4.設置石頭人可投擲物
– 1.拖入石頭的素材,添加必要組件:RigidBody,Mesh Collider(勾選第一項),腳本Rock.cs
public class Rock : MonoBehaviour {private Rigidbody rb;[Header("Basic Settings")]public float force;public GameObject target;private Vector3 direction;void Start(){rb = GetComponent<Rigidbody>();FlyToTarget(); }public void FlyToTarget(){//預防石頭生成瞬間玩家脫離范圍if (target == null){target = FindObjectOfType<PlayerController>().gameObject;}direction = (target.transform.position - transform.position + Vector3.up).normalized;rb.AddForce(direction * force, ForceMode.Impulse);} }– 2.修改石頭人生成石頭代碼:
public class Boss_Rock : EnemyController {//扔出的石頭的預制體public GameObject rockPrefab;//出手點的坐標public Transform handPos;//投擲石頭的邏輯public void ThrowRock(){if (AttackTarget != null){var rock = Instantiate(rockPrefab,handPos.position,Quaternion.identify);rock.GetComponent<Rock>().target = AttackTarget;}} }– 3.將ThrowRock方法添加到動畫對應幀數上。
– 4.設置石頭的狀態:在被投擲出的時候能對敵人以及玩家造成傷害,但落地以后無法對玩家或敵人造成傷害。
public class Rock : MonoBehaviour {public enum RockStates { HitPlayer,HitEnemy,HitNothing };//石頭的傷害值public int damage = 8;//石頭的狀態public RockStates rockStates;void Start(){rb = GetComponent<Rigidbody>();//為了防止石頭剛一出來就被判斷為hitNothingrb.velocity = Vector3.one;//初始化石頭的狀態rockStates = RockStates.HitPlayer;//石頭被生成的時候就自動飛向目標FlyToTarget();//石頭扔出三秒后延遲銷毀Destroy(this.gameObject, 3);}//逐幀判斷,當石頭幾乎靜止時變得不再有威脅void FixedUpdate(){Debug.Log(rb.velocity.sqrMagnitude);if (rb.velocity.sqrMagnitude < 1){rockStates = RockStates.HitNothing;} }void OnCollisionEnter(Collision collision){switch (rockStates){case RockStates.HitPlayer:if (collision.gameObject.CompareTag("Player")){CharacterStats characterStats = collision.gameObject.GetComponent<CharacterStats>();//碰到玩家了,造成傷害,并對玩家播放受擊動畫(TakeDamage的函數重載)characterStats.TakeDamage(damage,characterStats);collision.gameObject.GetComponent<Animator>().SetTrigger("Hit");rockStates = RockStates.HitNothing;}break;case RockStates.HitEnemy:if (collision.gameObject.CompareTag("Enemy")){var EnemyStats = collision.gameObject.GetComponent<CharacterStats>();EnemyStats.TakeDamage(damage, EnemyStats);//攻擊到敵人以后,也將其設為無危脅狀態}break;} } }– 5.函數重載TakeDamage()方法,讓石頭也能造成傷害:
public void TakeDamage(int damage,CharacterStats defencer){int finalDamage = Mathf.Max(damage - defencer.CurrentDefence, 1);CurrentHealth = Mathf.Max(CurrentHealth - finalDamage, 0);}– 6.修改玩家的攻擊邏輯,使其攻擊石頭也具有一定邏輯:
//范圍檢測private bool AttackRangeTest(){//Debug.Log(characterStats.AttackData.attackRange);//拿到檢測到對應范圍內的所有碰撞體Collider[] hitColliders = Physics.OverlapSphere(transform.position, characterStats.AttackData.attackRange);foreach (Collider collider in hitColliders){//有限判斷敵人,如果不存在敵人,則判斷是否有可攻擊物if (collider.gameObject.CompareTag("Enemy") || collider.gameObject.CompareTag("Attackable")){AttackTarget = collider.gameObject;return true;}}return false;}//玩家的出傷害邏輯void Hit(){if (AttackRangeTest()){if (AttackTarget.CompareTag("Attackable")){//進一步判斷是石頭if (AttackTarget.GetComponent<Rock>()){AttackTarget.GetComponent<Rock>().rockStates = Rock.RockStates.HitEnemy;AttackTarget.GetComponent<Rigidbody>().velocity = Vector3.one;AttackTarget.GetComponent<Rigidbody>().AddForce(transform.forward * 20, ForceMode.Impulse);}}else if (AttackTarget.CompareTag("Enemy")){CharacterStats targetStats = AttackTarget.GetComponent<CharacterStats>();targetStats.TakeDamage(characterStats, targetStats);}}}12.血條UI的設計
-
1.創建一個Canvas命名為 HealthBarCanvas,修改Canvas的 UI Scale Mode改為 World Space,并設置相機Camera。創建一個子物體UI image,命名 Bar Holder。
-
2.對UI界面的位置信息進行調整,修改長3和高0.25(參考值)。
-
3.在Package Manager中引入 3D sprite,創建一個2D Object–>Square,找到其基礎的文件,復制一份圖片另存起來。將其拖入到Bar Holder的Source Image中,并可修改其顏色(血條底色)。
-
4.繼續創建Bar Health的子節點Image,尺寸參數與父節點保持一致,拖入Source Image,修改顏色(血條上層色),并將其改為滑動條的形式進行顯示。
-
5.寫UI腳本操控血條的變化。
總結:該套程序采用逐幀檢測血條變化,相對來說效率較差,優化可以讓角色攻擊時計算一次血條變化。后續會優化!!!
13.玩家升級系統
- 1.先添加一下玩家的屬性
Character_SO:
-
暫時為了學習升級邏輯的構思以及數值的模范書寫,后續功能有待優化!
-
2.在造成傷害界面 添加擊殺增加經驗的判斷:
CharacterStats:
14.玩家的血條UI
- 1.同樣優先創建一個Canvas,設置參考如下:
做出如下效果的UI界面:
并在其父節點下嵌入以下腳本以控制血條的變化:
PlayerHealthUI:
15.傳送門(切換關卡)
場景內傳送:
- 1.在之前Shader Graph目錄下繼續創建Shader,參數參考如下:
- 2.利用創建的Shader的基礎上創建一個Meterial,并調參數。
- 3.在層級窗口中創建一個Quad,并添加以上材質,在其下創建一個子節點 DestinationPoint 作為被傳送點。
- 4.創建 TransitionPoint.cs腳本控制傳送點:(掛載在傳送門父類上)
- 5.創建傳送目標點腳本 TransitionDestination.cs:
- 6.寫場景控制腳本實現同場景傳送邏輯:SceneController.cs:
以上代碼遇到的問題:
- 1.切換到新場景后,管理類代碼全部消失,導致游戲無法正常運行,解決方法:在Manage相關代碼前都加上重寫的Awake()方法即可:
- 2.注意在切換至新場景之前,一定要在新場景安置 自己寫的 PhotoGrapher 結點,并且在PlayerLogic初始化的時候找到攝像機別賦值完整:
- 3.這樣改完以后從第二場景重新返回第一場景時,會出現兩個人物,并且人物在半空中。 (埋個伏筆)
16.保存數據
- 1.使用JSON保存游戲數據,在切換場景時調用保存函數與讀取函數:
- 2.新建一個結點 SaveManager并創建腳本 SaveManager.cs掛載到結點上:
- 3.在切換場景時調用這個兩個函數即可完成對玩家數據的保存:
SceneController.cs:
- 4.切記勿忘在新場景創建角色UI。
17.主菜單的制作
- 1.制作UI,按鈕,標題,背景等等
- 2.分別實現退出游戲,新的游戲和繼續游戲的腳本功能:
– 1.退出游戲
– 2.新的游戲:
1).清除之前的數據
2).在游戲控制中添加尋找全圖起點的方法:
GameManager.cs:
2).切換到第一場景的某點(在場景控制中使用攜程切換場景)
public void TransitionToFirstLevel(){StartCoroutine(LoadLevel("City"));}IEnumerator LoadLevel(string scene){if (scene != ""){yield return SceneManager.LoadSceneAsync(scene);yield return player = Instantiate(playerPrefab, GameManager.Instance.GetEntrance().position, GameManager.Instance.GetEntrance().rotation).transform;//保存數據SaveManager.Instance.SavePlayerData();yield break;}}– 3.繼續游戲的實現:
1).在保存函數里加入保存當前地圖邏輯
2).在場景控制里,調用協程:
SceneController.cs
3).在玩家生成的生成的時候,就讀取一遍玩家數據
PlayerController.cs
4).給予玩家回到Main的方式:
SceneController.cs:
5).添加補全繼續游戲的調用函數
MainMenu.cs:
18.場景轉場
- 1.引入TimeLine窗口:Windows-->Sequencing-->Timeline
總結
以上是生活随笔為你收集整理的基于Unity引擎的RPG3D项目开发笔录的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Java 深copy 浅copy 引用c
- 下一篇: docker log 文件 清理