FBX格式mesh解析与加载(二)

2023-11-30 09:48
文章标签 加载 fbx 解析 格式 mesh

本文主要是介绍FBX格式mesh解析与加载(二),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

直接上代码:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace ImportFbx
{
public class FBXImporter
{
private static FBXImporter fbxImporter;
public static FBXImporter Instance
{
get
{
if (fbxImporter == null) fbxImporter = new FBXImporter();
return fbxImporter;
}
}

    private Vector3[] vertices;public Vector3[] Vertices {get {return vertices;}}private int[] vertexIndex;public int[] VertexIndex {get {return vertexIndex;}}private Vector3[] normal;public Vector3[] Normal {get {return normal;}}private Vector2[] uv;public Vector2[] UV{get{return uv;}}public void OpenFBXFile(string path){if (!System.IO.File.Exists(path)) return;System.IO.FileStream file = new System.IO.FileStream(path,System.IO.FileMode.Open);byte[] bytes = new byte[file.Length];file.Read(bytes, 0, (int)file.Length);string data =  System.Text.Encoding.UTF8.GetString(bytes);UnityEngine.Debug.Log(data);file.Dispose();file.Close();string[] ObjectStr = data.Split(new string[] { "Objects:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[] a = ObjectStr[1].Split(new string[] { "Vertices:" }, 2 ,StringSplitOptions.RemoveEmptyEntries);UnityEngine.Debug.Log(a[1]);string[] aa = a[1].Split(new string[] { "a:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[] aaa = aa[1].Split(new char[] { '}' }, 2, StringSplitOptions.RemoveEmptyEntries);string[] verticesStr = aaa[0].Split(new char[] { ',' });UnityEngine.Debug.Log(verticesStr.Length);string[]b = ObjectStr[1].Split(new string[] { "PolygonVertexIndex:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[]bb = b[1].Split(new string[] { "a:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[]bbb = bb[1].Split(new char[] { '}' }, 2, StringSplitOptions.RemoveEmptyEntries);string[] vertexIdStr = bbb[0].Split(new char[] { ',' });UnityEngine.Debug.Log(vertexIdStr.Length);string[]c = ObjectStr[1].Split(new string[] { "Normals:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[]cc = c[1].Split(new string[] { "a:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[]ccc = cc[1].Split(new char[] { '}' }, 2, StringSplitOptions.RemoveEmptyEntries);string[] normalStr = ccc[0].Split(new char[] { ',' });UnityEngine.Debug.Log(normalStr.Length);string[] d = ObjectStr[1].Split(new string[] { "UV:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[] dd = d[1].Split(new string[] { "a:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[] ddd = dd[1].Split(new char[] { '}' }, 2, StringSplitOptions.RemoveEmptyEntries);string[] uvStr = ddd[0].Split(new char[] { ',' });UnityEngine.Debug.Log(uvStr.Length);string[] e = ObjectStr[1].Split(new string[] { "UVIndex:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[] ee = e[1].Split(new string[] { "a:" }, 2, StringSplitOptions.RemoveEmptyEntries);string[] eee = ee[1].Split(new char[] { '}' }, 2, StringSplitOptions.RemoveEmptyEntries);string[] uvIndexStr = eee[0].Split(new char[] { ',' });UnityEngine.Debug.Log(uvIndexStr.Length);vertices = new Vector3[verticesStr.Length/3];for (int i = 0; i < verticesStr.Length; i+=3){float.TryParse(verticesStr[i], out vertices[i/3].x);float.TryParse(verticesStr[i + 1], out vertices[i/3].y);float.TryParse(verticesStr[i + 2], out vertices[i/3].z);}int[] vertexIndexTemp = new int[vertexIdStr.Length];int q = 0;bool isSquareMesh = false;for (int i = 0; i < vertexIdStr.Length; i++){int.TryParse(vertexIdStr[i], out vertexIndexTemp[i]);q++;if (vertexIndexTemp[i] < 0 && q == 3){q = 0;vertexIndexTemp[i] = -vertexIndexTemp[i] - 1;}else if (vertexIndexTemp[i] < 0 && q == 4){q = 0;vertexIndexTemp[i] = -vertexIndexTemp[i] - 1;isSquareMesh = true;}}if (isSquareMesh) {List<int> triangles = new List<int>();for (int i = 0; i < vertexIdStr.Length; i+=4){triangles.AddRange(new int[3] { vertexIndexTemp[i], vertexIndexTemp[i + 1], vertexIndexTemp[i + 3] });triangles.AddRange(new int[3] { vertexIndexTemp[i + 1], vertexIndexTemp[i + 2], vertexIndexTemp[i + 3] });}vertexIndex = triangles.ToArray();}else{vertexIndex = vertexIndexTemp;}normal = new Vector3[vertices.Length];Vector3[] normTemp = new Vector3[normalStr.Length / 3];for (int i = 0; i < normalStr.Length; i+=3){float.TryParse(normalStr[i], out normTemp[i/3].x);float.TryParse(normalStr[i + 1], out normTemp[i/3].y);float.TryParse(normalStr[i + 2], out normTemp[i/3].z);}for (int i = 0; i < vertices.Length; i++){for (int j = 0; j < vertexIndexTemp.Length; j++){if (vertexIndexTemp[j] == i){normal[i] = normTemp[j];break;}}}uv = new Vector2[vertices.Length];Vector2[] uvTemp = new Vector2[uvStr.Length / 2];//UVVector2[] uvIndexTemp = new Vector2[uvIndexStr.Length];//all UVfor (int i = 0; i < uvStr.Length; i += 2){float.TryParse(uvStr[i], out uvTemp[i / 2].x);float.TryParse(uvStr[i + 1], out uvTemp[i / 2].y);}for (int i = 0; i < uvIndexStr.Length; i++){int index;int.TryParse(uvIndexStr[i], out index);uvIndexTemp[i] = uvTemp[index];}for (int i = 0; i < vertices.Length; i++){for (int j = 0; j < vertexIndexTemp.Length; j++){if (vertexIndexTemp[j] == i){uv[i] = uvIndexTemp[j];break;}}}}
}

}

放在unity中运行:在这里插入图片描述
模型2是通过常规的导入方式导入场景中作为参考的模型。
模型1是运行时导入的结果,也就是代码运行的结果。

现在还有些问题为解决:
1.没有获取fbx中的transform,所以场景中的transform是错误的。
2.如果出现四边面,uv的计算是错误的,triangles的排序是错误的。
3.当前没有考虑模型有子模型的情况。

这篇关于FBX格式mesh解析与加载(二)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

IDEA如何将String类型转json格式

《IDEA如何将String类型转json格式》在Java中,字符串字面量中的转义字符会被自动转换,但通过网络获取的字符串可能不会自动转换,为了解决IDEA无法识别JSON字符串的问题,可以在本地对字... 目录问题描述问题原因解决方案总结问题描述最近做项目需要使用Ai生成json,可生成String类型

SpringBoot项目启动后自动加载系统配置的多种实现方式

《SpringBoot项目启动后自动加载系统配置的多种实现方式》:本文主要介绍SpringBoot项目启动后自动加载系统配置的多种实现方式,并通过代码示例讲解的非常详细,对大家的学习或工作有一定的... 目录1. 使用 CommandLineRunner实现方式:2. 使用 ApplicationRunne

使用Python实现批量访问URL并解析XML响应功能

《使用Python实现批量访问URL并解析XML响应功能》在现代Web开发和数据抓取中,批量访问URL并解析响应内容是一个常见的需求,本文将详细介绍如何使用Python实现批量访问URL并解析XML响... 目录引言1. 背景与需求2. 工具方法实现2.1 单URL访问与解析代码实现代码说明2.2 示例调用

SSID究竟是什么? WiFi网络名称及工作方式解析

《SSID究竟是什么?WiFi网络名称及工作方式解析》SID可以看作是无线网络的名称,类似于有线网络中的网络名称或者路由器的名称,在无线网络中,设备通过SSID来识别和连接到特定的无线网络... 当提到 Wi-Fi 网络时,就避不开「SSID」这个术语。简单来说,SSID 就是 Wi-Fi 网络的名称。比如

SpringBoot项目删除Bean或者不加载Bean的问题解决

《SpringBoot项目删除Bean或者不加载Bean的问题解决》文章介绍了在SpringBoot项目中如何使用@ComponentScan注解和自定义过滤器实现不加载某些Bean的方法,本文通过实... 使用@ComponentScan注解中的@ComponentScan.Filter标记不加载。@C

springboot 加载本地jar到maven的实现方法

《springboot加载本地jar到maven的实现方法》如何在SpringBoot项目中加载本地jar到Maven本地仓库,使用Maven的install-file目标来实现,本文结合实例代码给... 在Spring Boothttp://www.chinasem.cn项目中,如果你想要加载一个本地的ja

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)

《使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)》在现代软件开发中,处理JSON数据是一项非常常见的任务,无论是从API接口获取数据,还是将数据存储为JSON格式,解析... 目录1. 背景介绍1.1 jsON简介1.2 实际案例2. 准备工作2.1 环境搭建2.1.1 添加

最好用的WPF加载动画功能

《最好用的WPF加载动画功能》当开发应用程序时,提供良好的用户体验(UX)是至关重要的,加载动画作为一种有效的沟通工具,它不仅能告知用户系统正在工作,还能够通过视觉上的吸引力来增强整体用户体验,本文给... 目录前言需求分析高级用法综合案例总结最后前言当开发应用程序时,提供良好的用户体验(UX)是至关重要

在C#中合并和解析相对路径方式

《在C#中合并和解析相对路径方式》Path类提供了几个用于操作文件路径的静态方法,其中包括Combine方法和GetFullPath方法,Combine方法将两个路径合并在一起,但不会解析包含相对元素... 目录C#合并和解析相对路径System.IO.Path类幸运的是总结C#合并和解析相对路径对于 C