Unity3D 动态创建的Mesh,导出为Obj模型文件,并生成Prefab文件

2024-04-16 18:08

本文主要是介绍Unity3D 动态创建的Mesh,导出为Obj模型文件,并生成Prefab文件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Unity运行时,动态创建的Mesh挂载到MeshFilter组件上,并不能保存到本地Prefab文件里。在运行的场景里,拖拽正确配置的MeshFilter对象到Unity资源管理器。生成的Prefab文件,里面的Mesh对象会missing。所以,我们需要在运行状态,导出Mesh到本地生成一个obj模型文件。


原理,就是根据obj文件的属性,把运行时Mesh的顶点,索引,贴图数据转化为固定格式流写入文件,生成obj模型文件。

		private string MeshToString(MeshFilter mf, Vector3 scale){Mesh          mesh            = mf.mesh;Material[]    sharedMaterials = mf.GetComponent<Renderer>().sharedMaterials;Vector2       textureOffset   = mf.GetComponent<Renderer>().material.GetTextureOffset("_MainTex");Vector2       textureScale    = mf.GetComponent<Renderer>().material.GetTextureScale ("_MainTex");StringBuilder stringBuilder   = new StringBuilder().Append("mtllib design.mtl").Append("\n").Append("g ").Append(mf.name).Append("\n");Vector3[] vertices = mesh.vertices;for (int i = 0; i < vertices.Length; i++){Vector3 vector = vertices[i];stringBuilder.Append(string.Format("v {0} {1} {2}\n", vector.x * scale.x, vector.y * scale.y, vector.z * scale.z));}stringBuilder.Append("\n");Dictionary<int, int> dictionary = new Dictionary<int, int>();if (mesh.subMeshCount > 1){int[] triangles = mesh.GetTriangles(1);for (int j = 0; j < triangles.Length; j += 3){if (!dictionary.ContainsKey(triangles[j])){dictionary.Add(triangles[j], 1);}if (!dictionary.ContainsKey(triangles[j + 1])){dictionary.Add(triangles[j + 1], 1);}if (!dictionary.ContainsKey(triangles[j + 2])){dictionary.Add(triangles[j + 2], 1);}}}for (int num = 0; num != mesh.uv.Length; num++){Vector2 vector2 = Vector2.Scale(mesh.uv[num], textureScale) + textureOffset;if (dictionary.ContainsKey(num)){stringBuilder.Append(string.Format("vt {0} {1}\n", mesh.uv[num].x, mesh.uv[num].y));}else{stringBuilder.Append(string.Format("vt {0} {1}\n", vector2.x, vector2.y));}}for (int k = 0; k < mesh.subMeshCount; k++){stringBuilder.Append("\n");if (k == 0){stringBuilder.Append("usemtl ").Append("Material_design").Append("\n");}if (k == 1){stringBuilder.Append("usemtl ").Append("Material_logo").Append("\n");}int[] triangles2 = mesh.GetTriangles(k);for (int l = 0; l < triangles2.Length; l += 3){stringBuilder.Append(string.Format("f {0}/{0} {1}/{1} {2}/{2}\n", triangles2[l] + 1, triangles2[l + 2] + 1, triangles2[l + 1] + 1));}}return stringBuilder.ToString();}
这段代码可以直接使用,把MeshFilter组件里Mesh数据变成一个固定格式的字符串流。写入到本地文件就是一个obj模型。这里有一个需要注意的地方,就是Unity加载obj文件的时候,顶点的X轴是翻转的。
			using (StreamWriter streamWriter = new StreamWriter(string.Format("{0}{1}.obj", datPath, this.meshGO.name))){streamWriter.Write(MeshToString(mf, new Vector3(-1f, 1f, 1f)));streamWriter.Close();}AssetDatabase.Refresh();
所以,在写入数据的时候,我们把scale.x设置为-1, 这样就翻转了X轴。并且正常情况下Mesh的顶点索引就是Triangles,需要逆时针才不会被摄像机剔除。当这里我们翻转了X顶点,同步我们需要在生成Mesh Triangles的时候,使用顺时针排列。这样,翻转X轴以后,对摄像机来说,顶点索引又是逆时针排列的了,就可以看见了。



第二部分,生成了obj模型文件以后,我们可以通过这个文件加载一个Mesh对象。动态生成一个Prefab到本地,把obj模型文件中的Mesh对象赋值给它,成为一个正确加载Mesh的Prefab。

			// create prefabMesh mesh 	= AssetDatabase.LoadAssetAtPath<Mesh>(string.Format("{0}{1}.obj", projectPath, this.meshGO.name));mf.mesh 	= mesh;PrefabUtility.CreatePrefab(string.Format("{0}{1}.prefab", projectPath, this.meshGO.name), this.meshGO);AssetDatabase.Refresh();

主要通过,AssetDatabase.LoadAssetAtPath的泛型方法来记载obj模型文件里的Mesh对象。然后动态创建一个Prefab这里需要注意Refresh一下,才能正确保存Mesh对象到Prefab上。


这篇关于Unity3D 动态创建的Mesh,导出为Obj模型文件,并生成Prefab文件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/909524

相关文章

C#使用Spire.XLS快速生成多表格Excel文件

《C#使用Spire.XLS快速生成多表格Excel文件》在日常开发中,我们经常需要将业务数据导出为结构清晰的Excel文件,本文将手把手教你使用Spire.XLS这个强大的.NET组件,只需几行C#... 目录一、Spire.XLS核心优势清单1.1 性能碾压:从3秒到0.5秒的质变1.2 批量操作的优雅

Python使用python-pptx自动化操作和生成PPT

《Python使用python-pptx自动化操作和生成PPT》这篇文章主要为大家详细介绍了如何使用python-pptx库实现PPT自动化,并提供实用的代码示例和应用场景,感兴趣的小伙伴可以跟随小编... 目录使用python-pptx操作PPT文档安装python-pptx基础概念创建新的PPT文档查看

在ASP.NET项目中如何使用C#生成二维码

《在ASP.NET项目中如何使用C#生成二维码》二维码(QRCode)已广泛应用于网址分享,支付链接等场景,本文将以ASP.NET为示例,演示如何实现输入文本/URL,生成二维码,在线显示与下载的完整... 目录创建前端页面(Index.cshtml)后端二维码生成逻辑(Index.cshtml.cs)总结

Python实现数据可视化图表生成(适合新手入门)

《Python实现数据可视化图表生成(适合新手入门)》在数据科学和数据分析的新时代,高效、直观的数据可视化工具显得尤为重要,下面:本文主要介绍Python实现数据可视化图表生成的相关资料,文中通过... 目录前言为什么需要数据可视化准备工作基本图表绘制折线图柱状图散点图使用Seaborn创建高级图表箱线图热

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

SQLServer中生成雪花ID(Snowflake ID)的实现方法

《SQLServer中生成雪花ID(SnowflakeID)的实现方法》:本文主要介绍在SQLServer中生成雪花ID(SnowflakeID)的实现方法,文中通过示例代码介绍的非常详细,... 目录前言认识雪花ID雪花ID的核心特点雪花ID的结构(64位)雪花ID的优势雪花ID的局限性雪花ID的应用场景

Qt中实现多线程导出数据功能的四种方式小结

《Qt中实现多线程导出数据功能的四种方式小结》在以往的项目开发中,在很多地方用到了多线程,本文将记录下在Qt开发中用到的多线程技术实现方法,以导出指定范围的数字到txt文件为例,展示多线程不同的实现方... 目录前言导出文件的示例工具类QThreadQObject的moveToThread方法实现多线程QC

SpringBoot集成EasyExcel实现百万级别的数据导入导出实践指南

《SpringBoot集成EasyExcel实现百万级别的数据导入导出实践指南》本文将基于开源项目springboot-easyexcel-batch进行解析与扩展,手把手教大家如何在SpringBo... 目录项目结构概览核心依赖百万级导出实战场景核心代码效果百万级导入实战场景监听器和Service(核心

Django HTTPResponse响应体中返回openpyxl生成的文件过程

《DjangoHTTPResponse响应体中返回openpyxl生成的文件过程》Django返回文件流时需通过Content-Disposition头指定编码后的文件名,使用openpyxl的sa... 目录Django返回文件流时使用指定文件名Django HTTPResponse响应体中返回openp

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3