Unity——自動化程式碼生成
自動化程式碼生成
一.前言
由於之前寫過關於UI框架的文章,這篇基於之前的基礎,添加了自動生成程式碼的功能;
如果學習過程有困惑可以跳轉到之前的文章《Unity——基於UGUI的UI框架》;
二.效果展示
三.打包UIFrame
我使用的是Rider編輯器,用其他的也可以;
目的就是將之前寫好的UIFrame框架的幾個公共類打包成dll供調用,只有打包成dll才可以在Plugins和Scripts文件夾中都能使用;
操作步驟如下:
1.創建新的classLibrary解決方案:
2.導入寫好的類,目前只需要到這以下這三個類即可;
這幾個類要添加統一的命名空間,在其他項目使用只需要using這個命名空間即可;
3.點擊BuildSolution打包;
由於這幾個類繼承了MonoBehaviour,需要添加UnityEngine依賴,根據錯誤提示添加;
另外.Net的版本過高也會導致打包出錯,修改一下版本即可;
四.自動生成程式碼
主要方法類UGUITool;由於是編輯器拓展功能,必須放在plugins文件夾下;
欄位:
uiDirPath:所有面板載入路徑;
classPath:自動生成的程式碼存放路徑;
classTemp:自動化程式碼模板字元串;
方法:
CreateCode():核心方法;
思路:
1.根據面板載入路徑,讀取所有面板的prefab資訊;
2.根據預製體的名稱,自動生成@Init和@UIType部分的程式碼字元串;
3.字元串全部生成完成後替換對應部分,保存在存放路徑;
注意:
我這裡所有面板都是放在Resources中的,如果正式項目肯定不會Resources載入,所以需要做SteamingAssets路徑判斷;
程式碼部分:
public class UGUITool
{
public static string uiDirPath = "Resources/Prefabs/UIPanel/";
public static string classPath = "Scripts/UIFrame/UIManagerA.cs";
private static string classTemp =
@"
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using UIFrame;
public partial class UIManager : MonoSingle<UIManager>
{
private void InitPath()
{
dicPath = new Dictionary<string, string>();
//@Init
}
}
public class UIType
{
//@UIType
}
";
[MenuItem("LittlePerilla/UGUI/CreateCode")]
public static void CreateCode()
{
//Resources可以替換
string path = Path.Combine(Application.dataPath, uiDirPath);
Debug.Log("生成程式碼" + path);
StringBuilder pt = new StringBuilder();
StringBuilder tp = new StringBuilder();
GameObject go = null;
string[] fileList = Directory.GetFiles(path, "*.prefab", SearchOption.AllDirectories);
int index = 0;
string s = "Resources";
foreach (var filePath in fileList)
{
string newPath = string.Empty;
newPath = filePath.Replace("\\", "/");
if (newPath.Contains(s))
{
newPath = newPath.Substring(newPath.IndexOf(s) + s.Length + 1);
newPath = newPath.Substring(0, newPath.IndexOf("."));
go = Resources.Load<GameObject>(newPath);
}
else
{
//newPath = newPath.Substring(0, newPath.IndexOf(".")); //其他路徑必須加後綴名
go = AssetDatabase.LoadAssetAtPath<GameObject>(newPath) as GameObject;
}
if (go == null)
{
Debug.LogError(newPath);
continue;
}
if (go.GetComponent<UIBase>())
{
string name = go.name.ToLower();
string uiName = $"UI{name.Substring(3, 1).ToUpper()}{name.Substring(4, name.Length-4)}";
pt.AppendLine($"dicPath[\"{name}\"] = \"{newPath}\";");
tp.AppendLine($"public const string {uiName} = \"{name}\";");
index++;
}
}
string codeClass = classTemp.Replace("//@Init", pt.ToString());
codeClass = codeClass.Replace("//@UIType", tp.ToString());
path = Path.Combine(Application.dataPath, classPath);
File.WriteAllText(path, codeClass, Encoding.UTF8);
Debug.Log($"{codeClass}生成完畢");
AssetDatabase.Refresh();
}
}
五.使用須知
路徑必須設置正常,並且之後所有UI面板都必須放在對應的文件夾下才會自動生成對應的程式碼;
自動化程式碼是提高工作效率的重要手段,如果有修改意見請與作者聯繫;