之所以写这个插件呢,就是为了方便整理项目中的资源文件,我记得之前好像也用了这么一个插件,但是也没去找,还是自己动手写一个吧,需要什么功能就看自己的需求。

在项目的过程中呢,已经写了一个插件来管理材质,但是跟随模型导入的图片却越来越多,也是自己跟美术沟通不到位,导致根本没办法保证图片的命名规则或者是否有重复的图片,再加上不断的删除或者更新模型,项目中一些无用的图片也堆积了不少。我尝试了自己手动整理,但是没有命名规则的图片和一些靠缩略图甚至无法分辨是不是一样的图片,这样的工作,就相当无力了。

于是呢,就完成了这么一个插件:

1.点击Organize Resources/Editor Window打开插件,点击未使用资源按钮,这里就会罗列出Assets文件夹里所有未被引用的资源,当然当前未被引用肯定不代表他完全没用,还是得靠自己确认然后手动删除

2.点击全部资源,他就会输出所有的文件,并可以通过按钮输出他的GUID、应用和被引用的其他Assets路径,这里主要理清一些资源的关联关系

3.点击重复资源按钮,就可以分类显示当前重复有哪些文件,这里的文件重复判断后面有具体介绍,点击合并按钮,就可以清除掉多余的资源,如果其他本地资源引用了这个文件,合并的文件也会自动替换上去,不用担心其他文件引用会丢失

这里New Material材质引用了fish 2图片,然后我在合并界面随便点一个fish后面的合并按钮

然后,多余的图片已经被清除了,而本来引用fish 2的材质New Material,也变成了引用fish图片,这里就是我当前需要的主要功能

上面的呢,是大概介绍了一些插件的功能,接下来就展示一下主要的功能代码:

1.获取当前Object引用了其他哪些Object,这里在unity是有接口的,返回的就是引用了其他的Object的路径:

string[] _OtherPaths = AssetDatabase.GetDependencies(_PathValue);

2.获取当前Object被其他那些Object引用了,说着感觉有点绕,不过功能也就是根据上面的反正写一个查找功能,获取其他Obejc的引用路径,再与当前的的Object匹配,这里的主要判断依据靠的是Object的GUID

#region 获取其他引用Assets的路径
string[] GetUseAssetPaths(string _AssetPath)
{
List<string> _AssetPaths = new List<string>();
//使用GUID作为判断标准
string _AssetGUID = AssetDatabase.AssetPathToGUID(_AssetPath);
//遍历所有Assets
for (int i = 0; i < _AllAssetsPaths.Count; i++)
{
if (_AllAssetsPaths[i] == _AssetPath)
continue; string[] _OtherPaths = AssetDatabase.GetDependencies(_AllAssetsPaths[i]);
if (_OtherPaths.Length > 1)
{
for (int j = 0; j < _OtherPaths.Length; j++)
{
string _OtherGUID = AssetDatabase.AssetPathToGUID(_OtherPaths[j]);
if (_AssetGUID == _OtherGUID)
{
_AssetPaths.Add(_AllAssetsPaths[i]);
}
}
}
}
return _AssetPaths.ToArray();
}
#endregion

3.获取相同的文件,开始使用正常的文件MD5值的验证,发现了一个问题,就是对于外部资源(从外面导入的资源,比如Texture、fbx)的验证是正常的,但是对于unity内部资源(在unity内部创建的文件,比如prefab、material)就不对了,后来读取内部资源的Text发现,他在内部有Object的名称,而Unity里的文件又不能重名,于是文件就不一样了,后来我就直接将他有名称的那行数据先删掉再来获取MD5值,才成功判断相同文件

获取相同文件

string[] GetSameFilePaths(string _PathValue)
{
List<string> _AssetPaths = new List<string>(); string _AssetMD5 = GetFileMD5(_PathValue);
//遍历所有Assets
for (int i = 0; i < _AllAssetsPaths.Count; i++)
{
if (_AllAssetsPaths[i] == _PathValue)
continue;
if (_AssetMD5 == GetFileMD5(_AllAssetsPaths[i]))
_AssetPaths.Add(_AllAssetsPaths[i]); }
return _AssetPaths.ToArray();
}

获取文件的MD5值

#region 获取文件的MD5值
string GetFileMD5(string _PathValue)
{
//判断是否为本地资源 因为本地文件里有文件名称 但是在资源名称又不能重复 于是需要去掉名称 来检测md5值
Object _ObejctValue = AssetDatabase.LoadAssetAtPath<Object>(_PathValue);
bool _isNative =AssetDatabase.IsNativeAsset(_ObejctValue);
string _FileMD5 = "";
string _TemPath = Application.dataPath.Replace("Assets", ""); if (_isNative)
{
string _TempFileText = File.ReadAllText(_TemPath + _PathValue).Replace("m_Name: " + _ObejctValue.name,""); System.Security.Cryptography.MD5 md5 = new System.Security.Cryptography.MD5CryptoServiceProvider();
//将字符串转换为字节数组
byte[] fromData = System.Text.Encoding.Unicode.GetBytes(_TempFileText);
//计算字节数组的哈希值
byte[] toData = md5.ComputeHash(fromData);
_FileMD5 = "";
for (int i = 0; i < toData.Length; i++)
{
_FileMD5 += toData[i].ToString("x2");
}
}
else
{
try
{ FileStream fs = new FileStream(_TemPath + _PathValue, FileMode.Open); System.Security.Cryptography.MD5 md5 = new System.Security.Cryptography.MD5CryptoServiceProvider();
byte[] retVal = md5.ComputeHash(fs);
fs.Close();
_FileMD5 = "";
for (int i = 0; i < retVal.Length; i++)
{
_FileMD5 += retVal[i].ToString("x2");
}
}
catch (System.Exception ex)
{
Debug.Log(ex);
}
}
return _FileMD5;
}
#endregion

4.合并时,当前文件替换掉其他被清除的文件,这里也是比较简单的,其他Object也是通过文件的GUID来引用文件,这里主要将他的GUID替换掉就可以

 #region 合并
void OnRepeatMerge(string _PathValue,List<string> _ListValue)
{
string _FixedGUID= AssetDatabase.AssetPathToGUID(_PathValue);
string _AssetsPath = Application.dataPath.Replace("Assets", ""); for (int i = 0; i < _ListValue.Count; i++)
{
if (_PathValue == _ListValue[i])
continue;
string[] _OtherPaths = GetUseAssetPaths(_ListValue[i]); bool _isOtherNative = true;
string _OldGUI = AssetDatabase.AssetPathToGUID(_ListValue[i]);
for (int j = 0; j < _OtherPaths.Length; j++)
{
Object _OtherUseAsset = AssetDatabase.LoadAssetAtPath<Object>(_OtherPaths[j]);
if (AssetDatabase.IsNativeAsset(_OtherUseAsset))
{
string _RealAllText = File.ReadAllText(_AssetsPath+ _OtherPaths[j]).Replace(_OldGUI,_FixedGUID);
File.WriteAllText(_AssetsPath + _OtherPaths[j], _RealAllText);
}
else
_isOtherNative = false;
}
//如果没有外部资源引用他 就删除
if (_isOtherNative)
{
AssetDatabase.DeleteAsset(_ListValue[i]);
_ListValue.RemoveAt(i);
i--;
}
}
AssetDatabase.Refresh();
OnRepeatClick();
}
#endregion

5.最后总结,在相对较大的项目中,因为会遍历所有的文件,插件的反应速度就不那么尽如人意,想想也不会时时来整理资源,所以也暂时忽略掉了,然后,插件的功能也会在后面慢慢完善

工程链接:https://github.com/coding2233/OrganizeResources

Unity 资源管理插件的更多相关文章

  1. 开发unity DLL插件

    最近开发一款设备的SDK,想着要开发unity版本,怎么做呢?首先想到的就是在外部编写相关的驱动程序然后集成成几个dll作为unity的SDK使用了.So,我就开始了unity外部插件的研究之旅了. ...

  2. (转)Unity Cinemachine插件,实现单目标和多目标之间切换

    Unity Cinemachine插件学习笔记,实现单目标和多目标之间切换*版本要求Unity2017.1及以上. 参考资料: [官方] Unity 2017.1正式版发布 Cinemachine插件 ...

  3. 细谈unity资源管理的设计

    一.概要 本文主要说说Unity是如何管理的,基于何种方式,基于这种管理方式,又该如何规划资源管理,以及构建bundle,是后面需要详细讨论的. 二.Unity的资源管理方式 2.1 资源分类 uni ...

  4. 开源Unity小插件CheatConsole

    我们在开发游戏的过程中,通常都需要一些快捷的方式来进行一些非常规的测试,这些功能一般被称作控制台或者GM指令,比如虚幻竞技场中,可以使用~键呼出控制台,输入一些指令即可进行快捷设置,比如设置分辨率,全 ...

  5. 编写 Unity Editor 插件

    Editor Style Viewer 在开发过程中,我喜欢编写一些辅助的Editor插件,方便在游戏开发过程进行调试. 下面是摘自Asset Store的一个查看Unity 默认GUI样式的小工具 ...

  6. 【转】unity地形插件T4M使用帮助

    unity的地形系统在手机游戏中因为效率问题基本无法使用,只能通过T4M这个地形插件来进行优化制作.下面大概讲解一下使用流程及方法. 先中U3D里面用自带的地形系统刷出想要的地形和贴图.贴图可以大概刷 ...

  7. unity Dotween插件的简单介绍及示例代码

    unity里面做插值动画的插件有许多,比较常见的有itween.hotween.dotween.根据大家的反馈和实际体验来说,dotween插件在灵活性.稳定性.易用性上都十分突出.这里简单介绍下它的 ...

  8. Unity给力插件之ShaderForge(一)

    这是一个用来制作shader的插件,也是一个很好的学习shader的工具.这个插件上手很容易,但是要用它来制作理想的Shader,需要下点功夫. 这儿先列举出基础知识,以及我的一些实践.以后我还会继续 ...

  9. Unity给力插件之MeshBaker

    这是一个用来合并网格.材质.贴图的插件. 其实网上也有一些比较详细的使用说明,但是真实操作起来时,总是有一些搞不清bug.而且,作为功能比较全的插件,在Unity版本更新时,也难免会一些不兼容的地方. ...

随机推荐

  1. div中的内容超过容器宽度的问题

    问题描述: <div class="category">    <div class="column-a">排名</div> ...

  2. [LeetCode] Pour Water 倒水

    We are given an elevation map, heights[i] representing the height of the terrain at that index. The ...

  3. python re模块findall使用

    今天练习re模块时候出现了一个很奇怪的问题,同样的正则表达式用re.search()与用re.compile().findall()匹配出来的结果不一致. 很是奇怪,故此记录一下,防止以后碰到类似情况 ...

  4. [SDOI2008]Sue的小球

    题目描述 Sue和Sandy最近迷上了一个电脑游戏,这个游戏的故事发在美丽神秘并且充满刺激的大海上,Sue有一支轻便小巧的小船.然而,Sue的目标并不是当一个海盗,而是要收集空中漂浮的彩蛋,Sue有一 ...

  5. C++primer学习——左值和右值

    定义: 左值:用的是对象的身份 右值:用的是对象的值(内存) decltype: 当其作用于表达式时,如果求值结果是左值,那么返回一个引用 如果求值结果是右值,那么返回正常 int*p; declty ...

  6. ●BZOJ 1934 [Shoi2007]Vote 善意的投票

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=1934 题解: 题目有点迷. S向为1的点连边,为0的点向T连边,在有关系的两个点之间连双向边 ...

  7. hdu 5493 (树状数组)

    题意:在一个队列中,你知道一个人在他左边或者右边比他高的人的个数,求字典序最小的答案 思路:先将人按  矮-->高 排序,然后算出在每个人前面需要预留的位置.树状数组(也可以线段树)解决时,先二 ...

  8. [bzoj5016][Snoi2017]一个简单的询问

    来自FallDream的博客,未经允许,请勿转载,谢谢. 给你一个长度为N的序列ai,1≤i≤N和q组询问,每组询问读入l1,r1,l2,r2,需输出   get(l,r,x)表示计算区间[l,r]中 ...

  9. [51nod1238]最小公倍数之和V3

    来自FallDream的博客,未经允许,请勿转载,谢谢. ----------------------------------------------------------------------- ...

  10. glusterfs 4.0.1 rpc 分析笔记1

    Jimmy的文档:Glusterfs的rpc模块分析 第一节.rpc服务器端实现原理及代码分析 第二节.rpc客户端实现原理及代码分析 第三节.rpc通信过程分析 经过阅读源码对比之前提及的文档,我个 ...