Unity之一键创建自定义Package包

news2025/1/15 18:52:07

内容将会持续更新,有错误的地方欢迎指正,谢谢!
 

Unity之一键创建自定义Package包
     
TechX 坚持将创新的科技带给世界!

拥有更好的学习体验 —— 不断努力,不断进步,不断探索
TechX —— 心探索、心进取!

助力快速掌握 自定义Package 快速创建

为初学者节省宝贵的学习时间,避免困惑!


前言:

  在之前的文章从零开始创建Unity自定义包Package:一步一步实现您的功能,我已经详细的介绍了该如何一步一步的去创建自己的自定义Package包,但是我相信有的小伙伴还是觉得文章很长不想看,或者觉得创建一个自定义包很麻烦,又要创建包,又要导入包的。

  今天小伙伴们可以轻松的去做这件事了,你可以直接一键的去生成包和导入包到Unity,真正的做到了省心、省时、省力,接下来就让我们一起来看怎么实现的吧。

TechX 教程效果:

在这里插入图片描述


文章目录

  • 一、绘制package创建器面板
  • 二、创建包
    • 1、创建文件夹和Assembly Definition
    • 2、创建包清单文件 (package.json)
    • 3、创建CHANGELOG.md
    • 4、创建README.md
  • 三、安装包
  • 四、项目地址


一、绘制package创建器面板


创建器面板的绘制基本上是package.json文件中的内容,包含大多数的包清单中的内容,包含:包名,包版本,包展示名,包描述,Unity版本,Unity发布版本号,文档地址,改变日志地址,证书地址,依赖列表,关键词列表,作者信息,案例列表等。

其中还包含了创建包时的选项,包含是否创建Scripts文件夹,是否创建Tests文件夹,是否创建CHANGELOG文件,是否创建ReadMe文件。

public class PackageCreatorWindow : EditorWindow
{
	private bool isCreteScriptsFolder = true;
	private bool isCreteTestsFolder = false;
	private bool isCreateChangeLog = true;
	private bool isCreateReadme = true;
	
	private string packageName = "";
	private string version = "";
	private string displayName = "";
	private string description = "";
	private string unity = "";
	private string unityRelease = "";
	private string documentationUrl = "";
	private string changelogUrl = "";
	private string licensesUrl = "";
	private List<Dependency> dependencies = new List<Dependency>();
	private List<string> keywords = new List<string>();
	private Author author = new Author();
	private List<Sample> samples = new List<Sample>();
	
	private ReorderableList keywordsReorderableList;
	private ReorderableList samplesReorderableList;
	private ReorderableList dependenciesReorderableList;
	
	private void OnGUI()
	{
	    DrawGUI();
	}

	private void DrawGUI()
	{
	    GUILayout.Space(5f);
	
	    GUILayout.Label("Package Config", new GUIStyle { fontSize = 20, padding = new RectOffset() { left = 5 }, fontStyle = FontStyle.Bold, normal = { textColor = Color.white } }); ;
	
	    GUILayout.Space(5f);
	
	    scrollPos = EditorGUILayout.BeginScrollView(scrollPos,EditorStyles.helpBox);
	
	    isCreteScriptsFolder = EditorGUILayout.ToggleLeft("Is Crete Scripts Folder", isCreteScriptsFolder);
	 	isCreteTestsFolder = EditorGUILayout.ToggleLeft("Is Crete Tests Folder", isCreteTestsFolder);
		isCreateChangeLog = EditorGUILayout.ToggleLeft("Is Create ChangeLog File", isCreateChangeLog);
		isCreateReadme = EditorGUILayout.ToggleLeft("Is Create ReadMe File", isCreateReadme);
		packageName = EditorGUILayout.TextField("Package Name *", packageName);
		version = EditorGUILayout.TextField("Version *", version);
		displayName = EditorGUILayout.TextField("Display Name *", displayName);
		description = EditorGUILayout.TextArea(description, GUILayout.Height(50));
		unity = EditorGUILayout.TextField("Unity Version *", unity);
		unityRelease = EditorGUILayout.TextField("Unity Release *", unityRelease);
		documentationUrl = EditorGUILayout.TextField("Documentation URL", documentationUrl);
		changelogUrl = EditorGUILayout.TextField("Changelog URL", changelogUrl);
		licensesUrl = EditorGUILayout.TextField("Licenses URL", licensesUrl);
		dependenciesReorderableList.DoLayoutList();
		keywordsReorderableList.DoLayoutList();
		autorToogleGroup = EditorGUILayout.BeginFoldoutHeaderGroup(autorToogleGroup, "Author");
		if (autorToogleGroup)
		{
		    EditorGUI.indentLevel += 1;
		    author.name = EditorGUILayout.TextField("Name", author.name);
		    author.email = EditorGUILayout.TextField("Email", author.email);
		    author.url = EditorGUILayout.TextField("Url", author.url);
		    EditorGUI.indentLevel -= 1;
		}
		EditorGUILayout.EndFoldoutHeaderGroup();
		samplesReorderableList.DoLayoutList();
	    EditorGUILayout.EndScrollView();
	
	    //创建之前要验证面板信息
	    if (GUILayout.Button("Create Package"))
	    {
	        string selectPath = EditorUtility.OpenFolderPanel("Select Folder for New Package", "", "");
	        bool isSuccess = CreateNewPackage(selectPath, packageName, out string packagePath);
	        if (isSuccess == false) return;
	        InstallPackage(selectPath, packagePath);
	    }
	}
}


二、创建包


1、创建文件夹和Assembly Definition


为创建的包创建一些需要的文件夹,比如Scripts/Editor、Scripts/Runtime、Tests/Editor、Tests/Runtime文件夹。
Scripts/Editor文件夹包含编辑器使用的功能,放置的脚本只在编辑器环境中使用。
Scripts/Runtime文件夹包含游戏运行时使用的功能,放置的脚本将在游戏运行时执行。
Tests/Editor用于编辑器测试,Tests/Runtime用于运行时测试。

/// <summary>
/// 创建包文件夹
/// </summary>
private void CreatePackageFolder(string pacakgePath, string packageName)
{
	//TODO 创建Scripts文件夹。。。。
    string scriptsFolderPath = Path.Combine(pacakgePath, "Scripts");
    Directory.CreateDirectory(scriptsFolderPath);

    //创建Editor和Runtime文件夹
    string editorFolderPath = Path.Combine(scriptsFolderPath, "Editor");
    string runtimeFolderPath = Path.Combine(scriptsFolderPath, "Runtime");

    Directory.CreateDirectory(editorFolderPath);
    Directory.CreateDirectory(runtimeFolderPath);

    // Create .asmdef files
    //这里的asmdef的文件的包名是否首字母大写??
    string editorAsmdefPath = Path.Combine(editorFolderPath, $"{packageName}.Editor.asmdef");
    string runtimeAsmdefPath = Path.Combine(runtimeFolderPath, $"{packageName}.Runtime.asmdef");

    AsmdefConfigProcess.CreateAsmdefContent(editorAsmdefPath, true);
    AsmdefConfigProcess.CreateAsmdefContent(runtimeAsmdefPath, false);
    
    //TODO 创建Tests文件夹。。。。
    string testsFolderPath = Path.Combine(pacakgePath, "Tests");
    Directory.CreateDirectory(testsFolderPath);

    string testseEditorFolderPath = Path.Combine(testsFolderPath, "Editor");
    string testseRuntimeFolderPath = Path.Combine(testsFolderPath, "Runtime");

    Directory.CreateDirectory(testseEditorFolderPath);
    Directory.CreateDirectory(testseRuntimeFolderPath);

    string testseEditorAsmdefPath = Path.Combine(testseEditorFolderPath, $"{packageName}.Editor.Tests.asmdef");
    string testseRuntimeAsmdefPath = Path.Combine(testseRuntimeFolderPath, $"{packageName}.Runtime.Tests.asmdef");

    AsmdefConfigProcess.CreateAsmdefContent(testseEditorAsmdefPath, true);
    AsmdefConfigProcess.CreateAsmdefContent(testseRuntimeAsmdefPath, false);
}

在每个文件夹创建完成之后,我们都需要在文件夹中定义一个程序集文件,注意在不同的文件夹中,程序集文件的名称是不一样的,同时也要注意,对于运行时和编辑器下的程序集平台也是不一样的,运行时的平台一般是Any Platform,而编辑器的平台是Editor。

public class AsmdefConfig
{
    public string name;
    public string rootNamespace;
    public List<string> references;
    public List<string> includePlatforms;
    public List<string> excludePlatforms;
    public bool allowUnsafeCode;
    public bool overrideReferences;
    public List<string> precompiledReferences;
    public bool autoReferenced;
    public List<string> defineConstraints;
    public List<string> versionDefines;
    public bool noEngineReferences;
}

public class AsmdefConfigProcess
{
    /// <summary>
    /// 创建.asmdef
    /// </summary>
    /// <param name="path">创建的位置</param>
    /// <param name="isEditor">适用于运行时还是编辑器下</param>
    public static void CreateAsmdefContent(string filePath, bool isEditor)
    {
        string fileName = Path.GetFileNameWithoutExtension(filePath) ;
        AsmdefConfig asmdefClass = new AsmdefConfig();
        asmdefClass.name = fileName;
        asmdefClass.rootNamespace = "";
        asmdefClass.references = new List<string>();
        asmdefClass.includePlatforms = isEditor ? new List<string> { "Editor" } : new List<string>();
        asmdefClass.excludePlatforms = new List<string>();
        asmdefClass.allowUnsafeCode = false;
        asmdefClass.overrideReferences = false;
        asmdefClass.precompiledReferences = new List<string>();
        asmdefClass.autoReferenced = true;
        asmdefClass.defineConstraints = new List<string>();
        asmdefClass.versionDefines = new List<string>();
        asmdefClass.noEngineReferences = false;

        JObject asmdefJson = JObject.FromObject(asmdefClass);

        File.WriteAllText(filePath, asmdefJson.ToString());
    }
}

2、创建包清单文件 (package.json)


每个Unity Package都必须包含一个名为package.json的清单文件。这个文件包含了有关包的元信息,如名称、版本、依赖项等。

/// <summary>
/// 创建package.json文件
/// </summary>
private void CreatePackageFile(string pacakgePath)
{
    // Create package.json
    PackageConfigProcess.CreatePackageJson(pacakgePath, GetPackageJsonContent());
}

这里定义了包清单的一些相关类型,当需要创建package.json文件时,从面板上获取到包相关信息,并生成PackageConfig实例,将该实例转换成JSON字符串写入到json文件中。

[System.Serializable]
public class Author
{
    public string name = "";
    public string email = "";
    public string url = "";
}

[System.Serializable]
public class Dependency
{ 
    public string packageName;
    public string version;
}

[System.Serializable]
public class Sample
{
    public string displayName;
    public string description;
    public string path;
}

public class PackageConfig
{
    public string name = "";
    public string version = "";
    public string displayName = "";
    public string description = "";
    public string unity = "";
    public string unityRelease = "";
    public string documentationUrl = "";
    public string changelogUrl = "";
    public string licensesUrl = "";
    public JObject dependencies = new JObject();
    public List<string> keywords = new List<string>();
    public Author author = new Author();
    public List<Sample> samples = new List<Sample>();
}

public class PackageConfigProcess
{
    public static void CreatePackageJson(string path, PackageConfig packageConfig)
    {
        JObject packageConfigJson = JObject.FromObject(packageConfig);

        string fullPath = Path.Combine(path, $"package.json");

        File.WriteAllText(fullPath, packageConfigJson.ToString());
    }
}

3、创建CHANGELOG.md


CHANGELOG.md文件是版本信息改变的日志文件,建议在每次发布新版本时更新CHANGELOG.md文件。在文件中记录新增功能、改进和错误修复。

/// <summary>
/// 创建CHANGELOG.md文件
/// </summary>
private void CreateChangeLogFile(string pacakgePath)
{
    // Create CHANGELOG.md
    string changeLogPath = Path.Combine(pacakgePath, "CHANGELOG.md");
    File.WriteAllText(changeLogPath, "# Changelog\nAll notable changes to this package will be documented in this file.\n\n");
}

4、创建README.md


README.md文件是关于该插件的介绍和如何使用的。

/// <summary>
/// 创建README.md文件
/// </summary>
private void CreateReadMeFile(string pacakgePath)
{
    // Create README.md
    string readmePath = Path.Combine(pacakgePath, "README.md");
    File.WriteAllText(readmePath, $"# {Path.GetFileName(pacakgePath)}\n\n");
}


三、安装包


在包创建完成后,会把包创建到本地的某个文件夹中,但是这个时候并没有把包安装到Unity中,按照一般的方法,在创建完成包后,可以通过Package Manager中的Add package form disk选项去文件夹中选择包的package.json文件来安装本地包。

但是在这里我们在创建完成包后,直接根据包的路径来,直接使用代码来模拟Add package form disk选择package.json文件来安装包。

注意:当我们把包直接创建到工程的Packages文件中时,那么我们就不需要通过代码来添加包到工程中,因为Unity会自动完成这一步。

/// <summary>
/// 安装包
/// </summary>
/// <param name="creteResult"></param>
/// <param name="selectPath"></param>
/// <param name="packagePath"></param>
private void InstallPackage(string selectPath, string packagePath)
{
    //如果直接创建到了Packages文件加中,那么就不需要手动添加到工程中,Unity会自动添加
    if (File.Exists(Path.Combine(selectPath, "manifest.json"))) return;

    //TODO 添加包到工程中
    PackageInstaller.InstallPackageFromDisk(packagePath);
}

从本地安装包的方式有两种,一种是从文件夹安装,要求包含package.json文件,并且符合包清单的条件,一种是通过.tgz压缩包的方式安装包。

这两种包在安装时都需要在路径前"file:"才能正确的安装包。

public class PackageInstaller
{
    /// <summary>
    /// 从本地路径安装包
    /// 1、可以通过本地文件夹安装包,格式file:pathtopackagefolder,如file:E:/UPMProject/UPM/com.fxb.test
    /// 2、从.tgz文件安装包,格式file:pathtopackage.tgz,如file:E:/UPMProject/UPM/com.fxb.test.tgz
    /// </summary>
    /// <param name="packagePath"></param>
    public static void InstallPackageFromDisk(string packagePath)
    {
        //从本地文件夹安装包
        if (Directory.Exists(packagePath))
        {
            // 构造package.json文件的完整路径
            string packageJsonPath = Path.Combine(packagePath, "package.json");

            // 检查package.json文件是否存在
            if (!File.Exists(packageJsonPath))
            {
                Debug.LogError("The provided folder does not contain a valid 'package.json' file and is not a valid Unity package.");
                return;
            }
        }
        //从.tgz文件安装包
        else if (File.Exists(packagePath))
        {
            if (!packagePath.EndsWith(".tgz"))
            {
                Debug.LogError($"{packagePath} file is not a valid Unity package.");
                return;
            }
        }
        else
        {
            Debug.LogError($"The package at path {packagePath} does not exist.");
            return;
        }

        // 构建正确的标识符
        string identifier = $"file:{packagePath}";
        // 如果存在package.json,那么这是一个有效的包,可以继续安装
        AddRequest request = Client.Add(identifier);
        CallbackFunction onAddUpdate = null;
        onAddUpdate = () =>
        {
            if (request.IsCompleted)
            {
                if (request.Status == StatusCode.Success)
                    Debug.Log($"Package from {packagePath} installed successfully.");
                else
                    Debug.LogError($"Failed to install package from {packagePath}: {request.Error.message}");

                // 移除更新回调
                EditorApplication.update -= onAddUpdate;
            }
        };

        // 注册更新事件
        EditorApplication.update += onAddUpdate;
    }
}


四、项目地址


以下是项目地址,已经整理成了Package包,有需要的小伙伴门可以自取:

https://gitcode.com/CTLittleNewbie/com.fxb.unitypackagecreator_v1.0.0/overview





TechX —— 心探索、心进取!

每一次跌倒都是一次成长

每一次努力都是一次进步


END
感谢您阅读本篇博客!希望这篇内容对您有所帮助。如果您有任何问题或意见,或者想要了解更多关于本主题的信息,欢迎在评论区留言与我交流。我会非常乐意与大家讨论和分享更多有趣的内容。
如果您喜欢本博客,请点赞和分享给更多的朋友,让更多人受益。同时,您也可以关注我的博客,以便及时获取最新的更新和文章。
在未来的写作中,我将继续努力,分享更多有趣、实用的内容。再次感谢大家的支持和鼓励,期待与您在下一篇博客再见!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2250807.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

E2、UML类图顺序图状态图实训

一、实验目的 在面向对象的设计里面&#xff0c;可维护性复用都是以面向对象设计原则为基础的&#xff0c;这些设计原则首先都是复用的原则&#xff0c;遵循这些设计原则可以有效地提高系统的复用性&#xff0c;同时提高系统的可维护性。在掌握面向对象七个设计原则基础上&…

【Java基础入门篇】一、变量、数据类型和运算符

Java基础入门篇 一、变量、数据类型和运算符 1.1 变量 计算机中的数据表示方式是&#xff1a;“二进制(0/1)”&#xff0c;但是同时也可以兼容其他进制&#xff0c;例如八进制、十进制、十六进制等。 Java变量的本质是&#xff1a;存储在固定空间的内容&#xff0c;变量名是…

前端学习笔记之文件下载(1.0)

因为要用到这样一个场景&#xff0c;需要下载系统的使用教程&#xff0c;所以在前端项目中就提供了一个能够下载系统教程的一个按钮&#xff0c;供使用者进行下载。 所以就试着写一下这个功能&#xff0c;以一个demo的形式进行演示&#xff0c;在学习的过程中也发现了中文路径…

【力扣】387.字符串中的第一个唯一字符

问题描述 思路解析 对于这种个数有限的问题&#xff0c;我的第一想法是使用桶排序来解决因为s中只有小写英文单词&#xff0c;所以我只需要一个26个单位的数组就好了通过 charAt() 来提取单个字符&#xff0c;然后通过 -‘a’ 来将其映射到 0~25&#xff0c;不然数组长度就被超…

数据采集中,除了IP池的IP被封,还有哪些常见问题?

在数据采集的过程中&#xff0c;代理IP池的使用无疑为我们打开了一扇通往信息宝库的大门。然而&#xff0c;除了IP被封禁这一常见问题外&#xff0c;还有许多其他问题可能影响数据采集的效果。本文将探讨在数据采集中&#xff0c;除了IP被封之外&#xff0c;还可能遇到的一些常…

【笔记】自动驾驶预测与决策规划_Part8_数据驱动的规划方法

文章目录 0. 前言1.生成模型1.1 Diffusion-ES1. Diffusion-ES算法介绍2. Diffusion-ES算法具体流程Diffusion Model 是什么&#xff1f;Diffusion-ES: Evolutionary StrategiesDiffusion-ES MethodDiffusion-ES Mapping Language instructions to reward functions with LLM pr…

里氏替换原则:Java面向对象设计的基石

在面向对象编程&#xff08;OOP&#xff09;中&#xff0c;继承是一个强大的工具&#xff0c;它允许我们创建新的类&#xff08;子类&#xff09;来复用和扩展现有类&#xff08;父类&#xff09;的功能。然而&#xff0c;继承也带来了复杂性&#xff0c;特别是在确保子类能够正…

C++笔记之单例模式与静态方法的使用辨析及代码规范

C++笔记之单例模式与静态方法的使用辨析及代码规范 code review! 文章目录 C++笔记之单例模式与静态方法的使用辨析及代码规范一.示例代码二.讲解2.1.代码规范2.1.1.单例模式实现2.1.2.静态方法实现2.1.3.单例模式结合静态方法2.2.总结一.示例代码 // 使用 set 方法设置值(通…

18:(标准库)DMA二:DMA+串口收发数据

DMA串口收发数据 1、DMA串口发送数据2、DMA中断串口接收定长数据包3、串口空闲中断DMA接收不定长数据包4、串口空闲中断DMA接收不定长数据包DMA发送数据包 1、DMA串口发送数据 当串口的波特率大于115200时&#xff0c;可以通过DMA1进行数据搬运&#xff0c;以防止数据的丢失。如…

加载不同本地gltf模型,模型内容不更新的解决方案

相关链接 http://mars3d.cn/editor-vue.html?keyex_6_2_2&idlayer-graphic/draw/draw-model 问题内容 加载本地gltf模型的时候&#xff0c;不clear图层&#xff0c;再打开其他本地gltf&#xff0c;gltf的内容就不更新 重现步骤 进入官网示例&#xff0c;贴入以下代码…

可视化建模以及UML期末复习篇----相关软件安装

作为一个过来人&#xff0c;我的建议是别过来。 一、可视化建模 <1>定义: 官方&#xff1a;一种使用图形符号来表示系统结构和行为的建模技术。 我&#xff1a;其实说白了就是把工作流程用图形画出来。懂不&#xff1f; <2>作用: 提高理解和分析复杂系统的能力。促…

AI开发 - GPT之魂 用Python 演示chatGPT的自注意力机制 - 机器学习

自注意力机制&#xff08;Self-Attention&#xff09;就是让模型在处理每个词时&#xff0c;学会“关注重点”&#xff0c;而不是平均地对每个词一视同仁。这种机制让 GPT 能更聪明地理解句子的上下文和语义之间的关系。 自注意力机制是 GPT 的核心&#xff0c;它帮助模型在理解…

Web 表单开发全解析:从基础到高级掌握 HTML 表单设计

文章目录 前言一、什么是 Web 表单?二、表单元素详解总结前言 在现代 Web 开发中,表单 是用户与后端服务交互的重要桥梁。无论是用户登录、注册、搜索,还是提交反馈,表单都无处不在。在本文中,我们将从基础入手,全面解析表单的核心知识点,并通过示例带你轻松掌握表单开…

HCIE:详解OSPF,从基础到高级特性再到深入研究

目录 前言 一、OSPF协议基本原理 简介 基本原理 OSPF路由器类型 OSPF网络类型 OSPF报文类型和封装 OSPF邻居的建立的维护 DR和BDR的选举 伪节点 LSDB的更新 OSPF的配置 二、OSPF的高级特性 虚连接&#xff08;Virtual-Link&#xff09; OSPF的LSA和路由选择 OSPF…

C#读取本地图像的方法总结

前言&#xff1a; 大家好&#xff0c;我是上位机马工&#xff0c;硕士毕业4年年入40万&#xff0c;目前在一家自动化公司担任软件经理&#xff0c;从事C#上位机软件开发8年以上&#xff01;我们在C#开发C#程序的时候&#xff0c;有时候需要读取本地图像&#xff0c;下面进行详…

scrapy爬虫框架小案例

豆瓣案例 一、scrapy安装二、scrapy的基本使用&#xff08;爬虫项目创建->爬虫文件创建->运行 爬虫项目结构 response的属性和方法&#x1f31f;&#xff09;1、创建项目2、创建爬虫文件3、scrapy项目的结构4、运行爬虫文件5、response的属性和方法&#xff08;爬虫的处…

服务器实现ssh证书登录

1.生成公钥和私钥 ssh-keygen -t rsa 提示默认生成位置为/root/.ssh/id_rsa ,直接回车。(也可以自己修改) 提示输入证书的密码&#xff0c;可以留空&#xff0c;建议输入&#xff0c;如果输入了&#xff0c;则需要再次确认&#xff0c;记住这个证书密码&#xff08;证书再加…

css:转换

转换 移动 /* transform: translate(100px, 200px); */transform: translateX(100px);transform: translateY(100px); /*一个意思*/ 如果后面跟百分数的意思是移动盒子自身x/y方向长度的百分比&#xff0c;可以用作子绝父相控制盒子水平居中垂直居中 translate里的xy值是相对…

音视频相关的一些基本概念

音视频相关的一些基本概念 文章目录 音视频相关的一些基本概念RTTH264profile & levelI帧 vs IDRMP4 封装格式AAC封装格式TS封装格式Reference RTT TCP中的RTT指的是“往返时延”&#xff08;Round-Trip Time&#xff09;&#xff0c;即从发送方发送数据开始&#xff0c;到…

Flink--API 之Transformation-转换算子的使用解析

目录 一、常用转换算子详解 &#xff08;一&#xff09;map 算子 &#xff08;二&#xff09;flatMap 算子 &#xff08;三&#xff09;filter 算子 &#xff08;四&#xff09;keyBy 算子 元组类型 POJO &#xff08;五&#xff09;reduce 算子 二、合并与连接操作 …