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

相关文章

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

Flutter 进阶:绘制加载动画

绘制加载动画:由小圆组成的大圆 1. 定义 LoadingScreen 类2. 实现 _LoadingScreenState 类3. 定义 LoadingPainter 类4. 总结 实现加载动画 我们需要定义两个类:LoadingScreen 和 LoadingPainter。LoadingScreen 负责控制动画的状态,而 LoadingPainter 则负责绘制动画。

OWASP十大安全漏洞解析

OWASP(开放式Web应用程序安全项目)发布的“十大安全漏洞”列表是Web应用程序安全领域的权威指南,它总结了Web应用程序中最常见、最危险的安全隐患。以下是对OWASP十大安全漏洞的详细解析: 1. 注入漏洞(Injection) 描述:攻击者通过在应用程序的输入数据中插入恶意代码,从而控制应用程序的行为。常见的注入类型包括SQL注入、OS命令注入、LDAP注入等。 影响:可能导致数据泄

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

CSP 2023 提高级第一轮 CSP-S 2023初试题 完善程序第二题解析 未完

一、题目阅读 (最大值之和)给定整数序列 a0,⋯,an−1,求该序列所有非空连续子序列的最大值之和。上述参数满足 1≤n≤105 和 1≤ai≤108。 一个序列的非空连续子序列可以用两个下标 ll 和 rr(其中0≤l≤r<n0≤l≤r<n)表示,对应的序列为 al,al+1,⋯,ar​。两个非空连续子序列不同,当且仅当下标不同。 例如,当原序列为 [1,2,1,2] 时,要计算子序列 [

多线程解析报表

假如有这样一个需求,当我们需要解析一个Excel里多个sheet的数据时,可以考虑使用多线程,每个线程解析一个sheet里的数据,等到所有的sheet都解析完之后,程序需要提示解析完成。 Way1 join import java.time.LocalTime;public class Main {public static void main(String[] args) thro

easyui同时验证账户格式和ajax是否存在

accountName: {validator: function (value, param) {if (!/^[a-zA-Z][a-zA-Z0-9_]{3,15}$/i.test(value)) {$.fn.validatebox.defaults.rules.accountName.message = '账户名称不合法(字母开头,允许4-16字节,允许字母数字下划线)';return fal

ZooKeeper 中的 Curator 框架解析

Apache ZooKeeper 是一个为分布式应用提供一致性服务的软件。它提供了诸如配置管理、分布式同步、组服务等功能。在使用 ZooKeeper 时,Curator 是一个非常流行的客户端库,它简化了 ZooKeeper 的使用,提供了高级的抽象和丰富的工具。本文将详细介绍 Curator 框架,包括它的设计哲学、核心组件以及如何使用 Curator 来简化 ZooKeeper 的操作。 1