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

相关文章

使用EasyPoi快速导出Word文档功能的实现步骤

《使用EasyPoi快速导出Word文档功能的实现步骤》EasyPoi是一个基于ApachePOI的开源Java工具库,旨在简化Excel和Word文档的操作,本文将详细介绍如何使用EasyPoi快速... 目录一、准备工作1、引入依赖二、准备好一个word模版文件三、编写导出方法的工具类四、在Export

前端导出Excel文件出现乱码或文件损坏问题的解决办法

《前端导出Excel文件出现乱码或文件损坏问题的解决办法》在现代网页应用程序中,前端有时需要与后端进行数据交互,包括下载文件,:本文主要介绍前端导出Excel文件出现乱码或文件损坏问题的解决办法,... 目录1. 检查后端返回的数据格式2. 前端正确处理二进制数据方案 1:直接下载(推荐)方案 2:手动构造

k8s admin用户生成token方式

《k8sadmin用户生成token方式》用户使用Kubernetes1.28创建admin命名空间并部署,通过ClusterRoleBinding为jenkins用户授权集群级权限,生成并获取其t... 目录k8s admin用户生成token创建一个admin的命名空间查看k8s namespace 的

Linux五种IO模型的使用解读

《Linux五种IO模型的使用解读》文章系统解析了Linux的五种IO模型(阻塞、非阻塞、IO复用、信号驱动、异步),重点区分同步与异步IO的本质差异,强调同步由用户发起,异步由内核触发,通过对比各模... 目录1.IO模型简介2.五种IO模型2.1 IO模型分析方法2.2 阻塞IO2.3 非阻塞IO2.4

JAVA实现亿级千万级数据顺序导出的示例代码

《JAVA实现亿级千万级数据顺序导出的示例代码》本文主要介绍了JAVA实现亿级千万级数据顺序导出的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 前提:主要考虑控制内存占用空间,避免出现同时导出,导致主程序OOM问题。实现思路:A.启用线程池

Vue3 如何通过json配置生成查询表单

《Vue3如何通过json配置生成查询表单》本文给大家介绍Vue3如何通过json配置生成查询表单,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录功能实现背景项目代码案例功能实现背景通过vue3实现后台管理项目一定含有表格功能,通常离不开表单

springboot集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

oracle 11g导入\导出(expdp impdp)之导入过程

《oracle11g导入导出(expdpimpdp)之导入过程》导出需使用SEC.DMP格式,无分号;建立expdir目录(E:/exp)并确保存在;导入在cmd下执行,需sys用户权限;若需修... 目录准备文件导入(impdp)1、建立directory2、导入语句 3、更改密码总结上一个环节,我们讲了

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

Python从Word文档中提取图片并生成PPT的操作代码

《Python从Word文档中提取图片并生成PPT的操作代码》在日常办公场景中,我们经常需要从Word文档中提取图片,并将这些图片整理到PowerPoint幻灯片中,手动完成这一任务既耗时又容易出错,... 目录引言背景与需求解决方案概述代码解析代码核心逻辑说明总结引言在日常办公场景中,我们经常需要从 W