【Framework-Client系列】UIGenerate介绍

今天来介绍一下Framework-Client框架中的UIGenerate部分,UIGenerate是用来创建UI界面的自定义工具,可以快速创建出UI界面所需的Atlas、Prefab、Script、Texture等。

功能介绍

首先我们通过菜单栏来打开UIGenerate窗口,如下图。

第一个输入框用于输入面板的名称,Atlas、Prefab、Script、Texture会以面板名称创建对应的目录和资源。

**"是否为子面板"**勾选框用于标记创建的面板是否为子面板,子面板创建的资源会放置在主面板的目录下。勾选时则会显示主面板名称的输入框,反之则不显示。

第二个输入框用于输入主面板名称,只有在创建子面板时才需要输入主面板名称。

接下来四个勾选框是分别对应生成Atlas、Prefab、Script、Texture不同的资源,只有勾选时才会生成对应的资源。根据需要还可以增加默认创建的UI资源,采用勾选框的目的也是为了选择更加的灵活。

"全部勾选" 和**"全部取消"**两个按钮是为了方便勾选用的。

点击**"生成UI组件"**按钮则会创建勾选的对应资源。

这里我们使用UIGenerate创建了一个TestPanel面板。并生成了Atlas、Prefab、Script、Texture相关的资源和目录,如下图。

下图是**.spriteatlas**资源,并且给Atlas默认指定了Texture目录。

下图是**.prefab**资源,并且Prefab中默认创建了PanelMask对象。

下图是TestPanel所需的**.cs**文件,并且实现了基础结构的编写。

下图则是创建了放置TestPanel所需的Texture目录,由于会有子界面的存在,所以在Texture目录下又分别为主界面和子界面分别创建了对应的目录。

最后在创建成功后,会在控制台输出创建成功的日志,显示资源名称和资源所在位置。点击日志则可以直接跳转到对应的资源。

功能实现

由于UIGenerate涉及的功能比较多,所以分到了多个文件中进行实现。

UIGenerate窗口实现

UIGenerateEditor负责UIGenerate窗口的实现,在 **OnGUI()**方法里绘制输入框、勾选框、按钮等。这里主要说一下的是Atlas、Prefab、Script、Texture通过创建各自的UIGenerate类对象处理各自的生成逻辑,部分代码如下。

cs 复制代码
private void OnGUI()
{
    GUILayout.Label("请输入要创建面板的名称");
    mPanelName = GUILayout.TextField(mPanelName);

    mIsSubPanel = GUILayout.Toggle(mIsSubPanel, "是否为子面板");
    if (mIsSubPanel)
    {
        GUILayout.Label("输入主面板的名称");
        mMainPanelName = GUILayout.TextField(mMainPanelName);
    }
    else
    {
        mMainPanelName = mPanelName;
    }

    GUILayout.BeginHorizontal();
    mIsGenerateAtlas = GUILayout.Toggle(mIsGenerateAtlas, "生成 Atlas");
    mIsGeneratePrefab = GUILayout.Toggle(mIsGeneratePrefab, "生成 Prefab");
    GUILayout.EndHorizontal();

    GUILayout.BeginHorizontal();
    mIsGenerateScript = GUILayout.Toggle(mIsGenerateScript, "生成 Script");
    mIsGenerateTexture = GUILayout.Toggle(mIsGenerateTexture, "生成 Texture");
    GUILayout.EndHorizontal();

    GUILayout.BeginHorizontal();
    if (GUILayout.Button("全部勾选"))
    {
        mIsGenerateAtlas = true;
        mIsGeneratePrefab = true;
        mIsGenerateScript = true;
        mIsGenerateTexture = true;
    }

    if (GUILayout.Button("全部取消"))
    {
        mIsGenerateAtlas = false;
        mIsGeneratePrefab = false;
        mIsGenerateScript = false;
        mIsGenerateTexture = false;
    }
    GUILayout.EndHorizontal();

    if (GUILayout.Button("生成UI组件"))
    {
        if (CheckName())
        {
            if (mIsGeneratePrefab)
                GeneratePrefab();

            if (mIsGenerateScript)
                GenerateScript();

            if (mIsGenerateTexture)
                GenerateTexture();

            if (mIsGenerateAtlas)
                GenerateAtlas();
        }
    }
}

private void GeneratePrefab()
{
    UIGeneratePrefab uiGeneratePrefab = new UIGeneratePrefab(mPanelName, mIsSubPanel, mMainPanelName);
    uiGeneratePrefab.GeneratePrefab();
    AssetDatabase.Refresh();
}

private void GenerateScript()
{
    UIGenerateScript uiGenerateScript = new UIGenerateScript(mPanelName, mIsSubPanel, mMainPanelName);
    uiGenerateScript.GenerateScript();
    AssetDatabase.Refresh();
}

private void GenerateTexture()
{
    UIGenerateTexture uiGenerateTexture = new UIGenerateTexture(mPanelName, mIsSubPanel, mMainPanelName);
    uiGenerateTexture.GenerateTexture();
    AssetDatabase.Refresh();
}

private void GenerateAtlas()
{
    UIGenerateAtlas uiGenerateAtlas = new UIGenerateAtlas(mPanelName, mIsSubPanel, mMainPanelName);
    uiGenerateAtlas.GenerateAtlas();
    AssetDatabase.Refresh();
}

在这里值得一提的是,由于生成完成之后会调用 AssetDatabase.Refresh() 方法,因此Unity会将UIGenerate窗口进行刷新,此时 UIGenerateEditor 类会被重新创建,这也意味着记录Atlas 勾选框状态的 mIsGenerateAtlas变量会被初始化 。这也导致了每次创建UI完成后,勾选框会被重新勾选上。(Prefab、Script、Texture也有同样的问题)

解决的办法是用 PlayerPrefsOnDisable() 时记录下勾选框的状态,并在 OnEnable() 时重新读取状态。

cs 复制代码
private void OnEnable()
{
    mIsGenerateAtlas = PlayerPrefs.GetString("IsGenerateAtlas", "true") == "true";
    mIsGeneratePrefab = PlayerPrefs.GetString("IsGeneratePrefab", "true") == "true";
    mIsGenerateScript = PlayerPrefs.GetString("IsGenerateScript", "true") == "true";
    mIsGenerateTexture = PlayerPrefs.GetString("IsGenerateTexture", "true") == "true";
}

private void OnDisable()
{
    PlayerPrefs.SetString("IsGenerateAtlas", mIsGenerateAtlas ? "true" : "false");
    PlayerPrefs.SetString("IsGeneratePrefab", mIsGeneratePrefab ? "true" : "false");
    PlayerPrefs.SetString("IsGenerateScript", mIsGenerateScript ? "true" : "false");
    PlayerPrefs.SetString("IsGenerateTexture", mIsGenerateTexture ? "true" : "false");
}

UIGeneratePrefab实现

这里预先创建了UIPanel模版,通过加载模版,然后重新保存到指定目录,完成Panel面板的创建,部分代码如下。

cs 复制代码
public void GeneratePrefab()
{
    string fileName = $"{mPanelName}.prefab";
    string fileFullPath = $"{Application.dataPath}/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Prefabs/{fileName}";
    string fileRelativePath = $"Assets/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Prefabs/{fileName}";

    if (!File.Exists(fileFullPath))
    {
        string directoryPath = $"{Application.dataPath}/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Prefabs";
        CreateDirectory(directoryPath);

        GameObject uiPanel = GetUIPanel();
        GameObject panelPrefab = PrefabUtility.SaveAsPrefabAsset(uiPanel, fileRelativePath);

        DelayDebugLog($"<color=yellow>{fileName}</color> create succeed. Path is <color=yellow>{fileFullPath}</color>", panelPrefab);
    }
    else
    {
        GameObject panelPrefab = AssetDatabase.LoadAssetAtPath<GameObject>(fileRelativePath);
        Debug.Log($"<color=yellow>{fileName}</color> already exist. Path is <color=yellow>{fileFullPath}</color>", panelPrefab);
    }
}

private GameObject GetUIPanel()
{
    string uiPanelPath = $"Assets/{FrameworkDefine.UIComponentRootDirectoryPath}/UIPanel.prefab";
    GameObject uiPanelGameObject = AssetDatabase.LoadAssetAtPath<GameObject>(uiPanelPath);

    Transform panelMask = uiPanelGameObject.transform.Find("PanelMask");
    Image image = panelMask.GetComponent<Image>();
    image.sprite = AssetDatabase.LoadAssetAtPath<Sprite>($"Assets/{FrameworkDefine.UICommonPanelMaskPath}");

    return uiPanelGameObject;
}

UIGenerateScript实现

Script的创建主要是文件操作,将类文件模版替换类名,然后将编辑好的字符串模版保存为 .cs 文件,部分示例代码如下。

将Panel类的基础代码单独写在 GetPanelContent() 方法内,调用时字符串会自动替换面板名称。在 CreateFile() 文件内,使用 File.Create() 创建 FileStream对象,并将content内容写入其中,保存为 .cs 文件。Controller和View文件与之类似。

cs 复制代码
public void GenerateScript()
{
    string directoryPath = $"{Application.dataPath}/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Scripts";
    CreateDirectory(directoryPath);

    string fileName = $"{mPanelName}.cs";
    string content = GetPanelContent();
    CreateFile(fileName, content);
}

private void CreateFile(string pFileName, string pContentString)
{
    string fileFullPath = $"{Application.dataPath}/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Scripts/{pFileName}";
    string fileRelativePath = $"Assets/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Scripts/{pFileName}";

    if (!File.Exists(fileFullPath))
    {
        byte[] byteArray = UtilCollection.DataUtil.StringToByteArray(pContentString);
        FileStream fileStream = File.Create(fileFullPath);
        fileStream.Write(byteArray, 0, byteArray.Length);
        fileStream.Close();

        EditorApplication.delayCall += () =>
        {
            TextAsset textAsset = AssetDatabase.LoadAssetAtPath<TextAsset>(fileRelativePath);
            Debug.Log($"<color=yellow>{pFileName}</color> create succeed. Path is <color=yellow>{fileFullPath}</color>", textAsset);
        };
    }
    else
    {
        TextAsset textAsset = AssetDatabase.LoadAssetAtPath<TextAsset>(fileRelativePath);
        Debug.Log($"<color=yellow>{pFileName}</color> already exist. Path is <color=yellow>{fileFullPath}</color>", textAsset);
    }
}

private string GetPanelContent()
{
	return $@"
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Framework.UI;

namespace Game.UI
{{
    public class {mPanelName} : PanelBase<{mPanelName}, {mPanelName}Controller, {mPanelName}View>
    {{
        protected override void OnInit()
        {{
            Debug.LogError(""{mPanelName} Init"");
        }}
    }}
}}";
}

UIGenerateTexture实现

UIGenerateTexture比较简单,只是创建Texture目录,示例代码如下。

这里需要说明一下的是,为了输出日志可以点击跳转到目录,所以需要通过EditorApplication.delayCall延迟调用的方式,获取目录对象。因为在创建的目录的当前帧是无法获取到目录对象的。

cs 复制代码
public void GenerateTexture()
{
    string textureFullPath = $"{Application.dataPath}/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Textures/{mPanelName}";
    string textureRelativePath = $"Assets/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Textures/{mPanelName}";
    if (!Directory.Exists(textureFullPath))
    {
        Directory.CreateDirectory(textureFullPath);

        EditorApplication.delayCall += () =>
        {
            Object textureDirectory = AssetDatabase.LoadAssetAtPath<Object>(textureRelativePath);
            Debug.Log($"<color=yellow>{mPanelName}</color> Texture Directory create succeed. File path is:<color=yellow>{textureFullPath}</color>", textureDirectory);
        };
    }
    else
    {
        Object textureDirectory = AssetDatabase.LoadAssetAtPath<Object>(textureRelativePath);
        Debug.Log($"<color=yellow>{mPanelName}</color> Texture Directory already exist. File path is:<color=yellow>{textureFullPath}</color>", textureDirectory);
    }
}

UIGenerateAtlas实现

UIGenerateAtlas通过 SpriteAtlas 来创建 .spriteatlas 文件,并指定Texture目录,部分示例代码如下。

这里获取Texture目录没有用延迟调用,是因为UIGenerateEditor中,先调用的UIGenerateTexture,再调用的UIGenerateAtlas。如果顺序相反的话,则需要通过延迟调用来获取Texture目录

cs 复制代码
public void GenerateAtlas()
{
    string directoryPath = $"{Application.dataPath}/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Atlas";
    CreateDirectory(directoryPath);

    string fileName = $"Atlas_{mPanelName}.spriteatlas";
    string atlasFullPath = $"{Application.dataPath}/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Atlas/{fileName}";
    string atlasFileRelativePath = $"Assets/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Atlas/{fileName}";

    if (!File.Exists(atlasFullPath))
    {
        SpriteAtlas spriteAtlas = new SpriteAtlas();
        SpriteAtlasPackingSettings packSetting = new SpriteAtlasPackingSettings()
        {
            blockOffset = 1,
            enableRotation = false,
            enableTightPacking = false,
            padding = 4,
        };
        spriteAtlas.SetPackingSettings(packSetting);

        string textureRelativePath = $"Assets/{FrameworkDefine.UIRootDirectoryPath}/{mMainPanelName}/Textures/{mPanelName}";
        Object textureDirectory = AssetDatabase.LoadAssetAtPath<Object>(textureRelativePath);
        spriteAtlas.Add(new[] { textureDirectory });

        AssetDatabase.CreateAsset(spriteAtlas, atlasFileRelativePath);
        Object atlasObject = AssetDatabase.LoadAssetAtPath<Object>(atlasFileRelativePath);
        Debug.Log($"<color=yellow>{fileName}</color> atlas create succeed. File path is:<color=yellow>{atlasFullPath}</color>", atlasObject);
    }
    else
    {
        Object atlasObject = AssetDatabase.LoadAssetAtPath<Object>(atlasFileRelativePath);
        Debug.Log($"<color=yellow>{fileName}</color> atlas already exist. File path is:<color=yellow>{atlasFullPath}</color>", atlasObject);
    }
}

相关文档链接

Editor编写常用方法汇总:https://blog.csdn.net/huoyixian/article/details/129587964

AssetDatabase官方文档:https://docs.unity3d.com/cn/2022.2/ScriptReference/AssetDatabase.html

相关推荐
牙膏上的小苏打23332 小时前
Unity Surround开关后导致获取主显示器分辨率错误
unity·主屏幕
Unity大海4 小时前
诠视科技Unity SDK开发环境配置、项目设置、apk打包。
科技·unity·游戏引擎
双叶8366 小时前
(C语言)单链表(1.0)(单链表教程)(数据结构,指针)
c语言·开发语言·数据结构·算法·游戏
二狗哈9 小时前
go游戏后端开发21:处理nats消息
开发语言·游戏·golang
前端菜鸟日常10 小时前
HMTL+JS+CSS实现贪吃蛇游戏,包含有一般模式,困难模式,还有无敌模式
javascript·css·游戏
浅陌sss10 小时前
Unity中 粒子系统使用整理(一)
unity·游戏引擎
维度攻城狮14 小时前
实现在Unity3D中仿真汽车,而且还能使用ros2控制
python·unity·docker·汽车·ros2·rviz2
为你写首诗ge17 小时前
【Unity网络编程知识】FTP学习
网络·unity
神码编程20 小时前
【Unity】 HTFramework框架(六十四)SaveDataRuntime运行时保存组件参数、预制体
unity·编辑器·游戏引擎
菲fay21 小时前
Unity 单例模式写法
unity·单例模式