IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> [Unity] GameFramework 学习记录 2 -> 正文阅读

[游戏开发][Unity] GameFramework 学习记录 2

嗯……要开始尝试做自己的事了

一开始策划给我列了这些

工程仓库搭建
工程程序框架
第三人称相机实现
主角基础移动实现
主角闪避实现
主角战斗实现
主角能量条实现
近战敌人战斗实现
远程敌人战斗实现
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()
        {
            // Preload configs
            LoadConfig("DefaultConfig");

            // Preload data tables
            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 之前的
但是在我这里,我反而是 HPBarComponentGameEntry.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 了
心累,今天只做了一件事,跑通资源加载流程

  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2022-03-15 22:58:26  更:2022-03-15 23:01:14 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 17:40:40-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码