嗯……要开始尝试做自己的事了
一开始策划给我列了这些
工程仓库搭建 工程程序框架 第三人称相机实现 主角基础移动实现 主角闪避实现 主角战斗实现 主角能量条实现 近战敌人战斗实现 远程敌人战斗实现 UI实现02 用户设置实现02
开始开始
我觉得主角基础移动是首先要做的
那就先看什么时候要开始控制主角吧
那自然是从流程开始看
ProcedureLaunch 中做语言、资源包、音乐设置 一帧之后跳转 然后到 ProcedureSplash 这个是游戏开始前的闪屏,比如展示公司标志 闪屏任务结束后跳转 然后到 ProcedurePreload 加载设置、数据表、字典、字体 订阅加载函数成功事件,设置 flag 为真。所有 flag 都为真,说明所有加载项都加载完毕,可以跳转 NextSceneId 是 Assets\GameMain\Configs\DefaultConfig.txt 中的 Scene.Menu 然后到 ProcedureChangeScene,进入时 停止所有声音,隐藏所有实体,卸载所有场景,还原游戏速度 这其实也是基于游戏的,比如还原游戏速度,是因为游戏中陨石被打碎的时候会让游戏速度先减慢后恢复,游戏有可能在陨石被击碎,游戏速度还没恢复正常的时候退出,这个时候就要恢复正常了 但总归来说,也是一个通用的中间流程
接下来自然到了这一句
IDataTable<DRScene> dtScene = GameEntry.DataTable.GetDataTable<DRScene>();
感觉很神奇,因为实际上我需要的是一个 scene 表,但是我是向 api 输入了一个泛型 那么就有一个问题是,为什么我能够得到一个表 如果对 GetDataTable 溯源的话,就会找到一个 DataTable 中的接口,其中找不到对 GetDataTable 接口的实现
这个时候没有头绪的话,我就会想,是不是我直接在 Assets\GameMain\DataTables 中创建好了 txt 格式的表格,就能直接通过 GameEntry.DataTable.GetDataTable<DR表格名>() 获得表格 毕竟因为我看不到内容嘛 但总归是不确定,所以就很烦躁
之后找别的教程才知道所有数据表已经 load 过了 在 ProcedurePreload 的 OnEnter 就开始加载资源,加载资源的函数里面包含了加载数据表,这个加载数据表是对一个名称列表遍历,把名称输入到加载数据表函数里面,加载数据表函数根据这个数据表名称找到资源名,名称和资源名一起输入到 DataTableExtension 的加载函数里面,在那里面再给文件名前面加上一个 DR,看有没有定义过 DR+数据表名 的类,有的话就新建一个表然后读数据进去
Assets\GameMain\Scripts\Procedure\ProcedurePreload.cs
protected override void OnEnter(ProcedureOwner procedureOwner)
{
PreloadResources();
}
private void PreloadResources()
{
LoadConfig("DefaultConfig");
foreach (string dataTableName in DataTableNames)
{
LoadDataTable(dataTableName);
}
Assets\GameMain\Scripts\Procedure\ProcedurePreload.cs
namespace OaksMayFall
{
public class ProcedurePreload : ProcedureBase
{
public static readonly string[] DataTableNames = new string[]
{
"Aircraft",
"Armor",
"Asteroid",
"Entity",
"Music",
"Scene",
"Sound",
"Thruster",
"UIForm",
"UISound",
"Weapon",
};
Assets\GameMain\Scripts\Procedure\ProcedurePreload.cs
private void LoadDataTable(string dataTableName)
{
string dataTableAssetName = AssetUtility.GetDataTableAsset(dataTableName, false);
m_LoadedFlag.Add(dataTableAssetName, false);
GameEntry.DataTable.LoadDataTable(dataTableName, dataTableAssetName, this);
}
Assets\GameMain\Scripts\DataTable\DataTableExtension.cs
public static void LoadDataTable(this DataTableComponent dataTableComponent, string dataTableName, string dataTableAssetName, object userData)
{
if (string.IsNullOrEmpty(dataTableName))
{
Log.Warning("Data table name is invalid.");
return;
}
string[] splitedNames = dataTableName.Split('_');
if (splitedNames.Length > 2)
{
Log.Warning("Data table name is invalid.");
return;
}
string dataRowClassName = DataRowClassPrefixName + splitedNames[0];
Type dataRowType = Type.GetType(dataRowClassName);
if (dataRowType == null)
{
Log.Warning("Can not get data row type with class name '{0}'.", dataRowClassName);
return;
}
string name = splitedNames.Length > 1 ? splitedNames[1] : null;
DataTableBase dataTable = dataTableComponent.CreateDataTable(dataRowType, name);
dataTable.ReadData(dataTableAssetName, Constant.AssetPriority.DataTableAsset, userData);
}
那么流程应该就是
1.使用 Excel 创建数据表 2.保存为 UTF-8 的 txt 文件 3.创建 DR+数据表名 的继承 DataRowBase 的类 4.在 ProcedurePreload 中的 DataTableNames 中添加数据表名
好吧,之后要去跑流程了
一开始卡在 ProcedureSplash 了,因为运行时 GameEntry.Base 为 null 难以理解…… GameEntry 的 组件都是在 Assets\GameMain\Scripts\Base\GameEntry.BuiltIn.cs 中初始化的,这个初始化函数是在 InitBuiltinComponents()
所以加了个测试
Assets\GameMain\Scripts\Base\GameEntry.cs
public partial class GameEntry : MonoBehaviour
{
private void Start()
{
InitBuiltinComponents();
InitCustomComponents();
GameFramework.GameFrameworkLog.Debug(11111);
}
}
呃……真没打印出来 好奇怪 哦好吧,是我忘记加 Game Entry 了hhh
加了之后,继续调试,ProcedureLaunch 有错
Console
NullReferenceException: Object reference not set to an instance of an object
OaksMayFall.ProcedureLaunch.OnEnter (GameFramework.Fsm.IFsm`1[T] procedureOwner) (at Assets/GameMain/Scripts/Procedure/ProcedureLaunch.cs:30)
GameFramework.Fsm.Fsm`1[T].Start (System.Type stateType) (at <ca3005014ab44423a8eae080265b9678>:0)
GameFramework.Procedure.ProcedureManager.StartProcedure (System.Type procedureType) (at <ca3005014ab44423a8eae080265b9678>:0)
UnityGameFramework.Runtime.ProcedureComponent+<Start>d__9.MoveNext () (at Assets/GameFramework/UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5/Scripts/Runtime/Procedure/ProcedureComponent.cs:105)
UnityEngine.SetupCoroutine.InvokeMoveNext (System.Collections.IEnumerator enumerator, System.IntPtr returnValueAddress) (at <deb0ce7c66da485ead1d2ea5a2182103>:0)
UnityEngine.GUIUtility:ProcessEvent(Int32, IntPtr, Boolean&)
第三十行是
GameEntry.BuiltinData.InitBuildInfo();
哦……是需要在 custom 里面新增一个 Buildin 难受,是我抄 StarForce 没抄全
加了 custom 里的 Buildin 和 HP Bar 之后,又是 HP Bar 出现问题
Assets\GameMain\Scripts\HPBar\HPBarComponent.cs
for (int i = m_ActiveHPBarItems.Count - 1; i >= 0; i--)
这个出现问题,但是 m_ActiveHPBarItems 应该早就被初始化了才对啊
Assets\GameMain\Scripts\HPBar\HPBarComponent.cs
m_ActiveHPBarItems = new List<HPBarItem>();
后面发现应该是在前面返回了
private void Start()
{
if (m_HPBarInstanceRoot == null)
{
Log.Error("You must set HP bar instance root first.");
return;
}
m_CachedCanvas = m_HPBarInstanceRoot.GetComponent<Canvas>();
m_HPBarItemObjectPool = GameEntry.ObjectPool.CreateSingleSpawnObjectPool<HPBarItemObject>("HPBarItem", m_InstancePoolCapacity);
m_ActiveHPBarItems = new List<HPBarItem>();
原来这个 m_HPBarInstanceRoot 是要设置的东西 好吧,那就连着那个原型一起复制粘贴过来吧
然后是
m_HPBarItemObjectPool = GameEntry.ObjectPool.CreateSingleSpawnObjectPool<HPBarItemObject>("HPBarItem", m_InstancePoolCapacity);
中的 GameEntry.ObjectPool 为 null 好怪,我去 StarForce 试了一下,确实是 GameEntry.ObjectPool 的初始化在 HPBarComponent 之前的 但是在我这里,我反而是 HPBarComponent 在 GameEntry.ObjectPool 之前 嗯……看了一下 StarForce 的初始化,我是在 Assets\GameMain\Scripts\Base\GameEntry.Custom.cs 中缺了 HPBarComponent 的部分 补上补上
哦,补上了之后还是不行
知道了,原来在项目设置里面可以设置脚本之间的执行顺序啊……是我菜了 之前我还以为 GF 有控制 Component 的初始化顺序啥的
Game.Entry 的 Order 设置为 -100
诶……我原来是要干啥来着 哦,原来是我加了一个血量条就出现了很多错误,然后就一直在做这个 哦,再往前是 ProcedureSplash 没跑通 那就继续跑 ProcedureSplash
哦,之前卡在 ProcedureSplash 好像也是因为 GameEntry.Base 没初始化的问题 现在碰巧解决了问题hhhh 终究要遇到的问题在很后面的地方被我遇到了hhhh
然后可以到 ProcedurePreload 缺本地化,那就复制 Assets\GameMain\Localization 过去 之后会卡在 ProcedurePreload
测试代码
Assets\GameMain\Scripts\Procedure\ProcedurePreload.cs
protected override void OnUpdate(ProcedureOwner procedureOwner, float elapseSeconds, float realElapseSeconds)
{
base.OnUpdate(procedureOwner, elapseSeconds, realElapseSeconds);
foreach (KeyValuePair<string, bool> loadedFlag in m_LoadedFlag)
{
if (!loadedFlag.Value)
{
GameFramework.GameFrameworkLog.Debug(loadedFlag.Key);
return;
}
}
procedureOwner.SetData<VarInt32>("NextSceneId", GameEntry.Config.GetInt("Scene.Menu"));
ChangeState<ProcedureChangeScene>(procedureOwner);
}
Console
Assets/GameMain/Localization/ChineseSimplified/Dictionaries/Default.xml
UnityEngine.Debug:Log (object)
UnityGameFramework.Runtime.DefaultLogHelper:Log (GameFramework.GameFrameworkLogLevel,object) (at Assets/GameFramework/UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5/Scripts/Runtime/Utility/DefaultLogHelper.cs:28)
GameFramework.GameFrameworkLog:Debug (string)
OaksMayFall.ProcedurePreload:OnUpdate (GameFramework.Fsm.IFsm`1<GameFramework.Procedure.IProcedureManager>,single,single) (at Assets/GameMain/Scripts/Procedure/ProcedurePreload.cs:70)
GameFramework.Fsm.Fsm`1<GameFramework.Procedure.IProcedureManager>:Update (single,single)
GameFramework.Fsm.FsmManager:Update (single,single)
GameFramework.GameFrameworkEntry:Update (single,single)
UnityGameFramework.Runtime.BaseComponent:Update () (at Assets/GameFramework/UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5/Scripts/Runtime/Base/BaseComponent.cs:231)
太怪了,我明明复制了 Assets/GameMain/Localization/ChineseSimplified/Dictionaries/Default.xml,但是这个资源还是没有加载成功
Assets\GameMain\Scripts\Procedure\ProcedurePreload.cs
private void OnLoadDictionaryFailure(object sender, GameEventArgs e)
{
LoadDictionaryFailureEventArgs ne = (LoadDictionaryFailureEventArgs)e;
if (ne.UserData != this)
{
return;
}
GameFramework.GameFrameworkLog.Debug("Can not load dictionary '{0}' from '{1}' with error message '{2}'.", ne.DictionaryAssetName, ne.DictionaryAssetName, ne.ErrorMessage);
}
不知道为啥,原来的 Log.Error 不会打印东西
Console
Can not load dictionary 'Assets/GameMain/Localization/ChineseSimplified/Dictionaries/Default.xml' from 'Assets/GameMain/Localization/ChineseSimplified/Dictionaries/Default.xml' with error message 'GameFramework.GameFrameworkException: Load data failure in data provider helper, data asset name 'Assets/GameMain/Localization/ChineseSimplified/Dictionaries/Default.xml'.
at GameFramework.DataProvider`1[T].LoadAssetSuccessCallback (System.String dataAssetName, System.Object dataAsset, System.Single duration, System.Object userData) [0x00027] in <ca3005014ab44423a8eae080265b9678>:0 '.
UnityEngine.Debug:Log (object)
UnityGameFramework.Runtime.DefaultLogHelper:Log (GameFramework.GameFrameworkLogLevel,object) (at Assets/GameFramework/UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5/Scripts/Runtime/Utility/DefaultLogHelper.cs:28)
GameFramework.GameFrameworkLog:Debug<string, string, string> (string,string,string,string)
OaksMayFall.ProcedurePreload:OnLoadDictionaryFailure (object,GameFramework.Event.GameEventArgs) (at Assets/GameMain/Scripts/Procedure/ProcedurePreload.cs:202)
GameFramework.EventPool`1<GameFramework.Event.GameEventArgs>:HandleEvent (object,GameFramework.Event.GameEventArgs)
GameFramework.EventPool`1<GameFramework.Event.GameEventArgs>:Update (single,single)
GameFramework.Event.EventManager:Update (single,single)
GameFramework.GameFrameworkEntry:Update (single,single)
UnityGameFramework.Runtime.BaseComponent:Update () (at Assets/GameFramework/UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5/Scripts/Runtime/Base/BaseComponent.cs:231)
好吧……那就是 data provider helper 读不了 Assets/GameMain/Localization/ChineseSimplified/Dictionaries/Default.xml 为啥呢……我明明都是复制的
看到了这个警告
或许我应该找找是谁接受的 应该是接受字典的那边也应该和 xml 对应起来
好吧,最后发现是 Hierarchy - GameFramework - Builtin - Localization 中的 Localization Helper 要设置成 项目名+XmlLocalizationHelper 我看到这个警告还以为是文件格式啥的问题
接下来到了 ProcedureChangeScene,太不容易了
这里面有东西为空
DRScene drScene = dtScene.GetDataRow(sceneId);
测试出来确实是 dtScene 为空
那就是还要把 Assets\GameMain\DataTables 中的 txt 复制过去,然后得到生成数据表
二进制数据表 和 数据表内容的类都是自动生成的 然后新建的类默认的是 StarForce 命名空间hhh我得改过来
Assets\GameMain\Scripts\Editor\DataTableGenerator\DataTableGenerator.cs
private static void DataTableCodeGenerator(DataTableProcessor dataTableProcessor, StringBuilder codeContent, object userData)
{
string dataTableName = (string)userData;
codeContent.Replace("__DATA_TABLE_CREATE_TIME__", DateTime.UtcNow.ToLocalTime().ToString("yyyy-MM-dd HH:mm:ss.fff"));
codeContent.Replace("__DATA_TABLE_NAME_SPACE__", "OaksMayFall");
之后还有一个报错 我把 Entity.txt 改成了 UEntity.txt 之后,他还是要求 Entity
Assets\GameMain\Scripts\Editor\DataTableGenerator\DataTableGeneratorMenu.cs
foreach (string dataTableName in ProcedurePreload.DataTableNames)
{
DataTableProcessor dataTableProcessor = DataTableGenerator.CreateDataTableProcessor(dataTableName);
这个生成的名称是从 ProcedurePreload 的 DataTableNames 中取的,这个我还没改,是我忘了
那之前那个是我写错了,应该是
数据表使用方法 1.使用 Excel 创建数据表 2.保存为 UTF - 8 的 txt 文件 3.在 ProcedurePreload 中的 DataTableNames 中添加数据表名 4.点击菜单栏中的生成工具,自动生成二进制文件的数据表和 DR+数据表名 类
唉,他这个数据表生成器在工程代码报错的时候好像就不能重新生成了 比如我把 ProcedurePreload 的 DataTableNames 改动了,但是我接下来又把 DR+数据表名 的类都删了,结果代码里面用到这些类的地方都报错了,然后我就想着,如果数据表能够正确生成的话我就不用管了,结果我点生成之后,他还是遵循着没有改动的 ProcedurePreload 的 DataTableNames 来的。然后我重启了 unity,这次工具栏中的生成按钮都不见了。正常来说,就是要整个工程都没有报错,才能有正确的显示嘛,那就是有正确的 DR 类才能正确构建嘛。但是我这个时候都已经删了 DR 类了,我又需要数据表生成工具创建正确的 DR 类,这不就死循环了hh 要不就是删了用了 DR 类的代码,这样的话要复原就太麻烦了 还有一种办法是我直接复制 StarForce 里面的 DR 类过来hhh
继续跑 ProcedureChangeScene
还要把 Main 和 Menu 场景加入 build setting
啊……直到这里就可以进入 menu 了 心累,今天只做了一件事,跑通资源加载流程
|