如何手动读取 GLTF/GLB 文件

推荐:使用NSDT场景编辑器快速搭建3D应用场景

文件类型

GLTF文件有两种不同的主要文件类型:.gltf和.glb。

GLTF文件本质上只是一个重新命名的json文件,它们通常与包含顶点数据等内容的.bin文件相提并论,但这些内容也可以直接包含在json中。

GLB 文件类似于 GLTF 文件,但所有内容都包含在同一个文件中。它分为三个部分,一个小标头、json 字符串和二进制缓冲区。

图片来自官方gltf github

GLTF 格式

在 GLTF 中,与网格、动画和蒙皮相关的一切都存储在缓冲区中,虽然一开始,在没有库的情况下从原始二进制文件中读取似乎令人生畏,但实际上并不太难。我们将逐步进行。

在本文中,我们将介绍如何从 .gltf 和 .glb 文件中从单个网格读取顶点位置数据。

在我们获得实际代码之前,我们需要了解如何使用文件的 json 部分来查找我们想要的内容,因为我们必须跳来跳去才能找到任何东西。你可以从场景级别开始,如果需要,也可以逐步向下工作,但由于我计划只对单个网格使用格式,所以我将从图形的网格节点开始。

假设我们的 GLTF 文件看起来像这样(请注意,实际文件将包含更多数据):

{"accessors" : [{"bufferView": 0,"byteOffset": 0,"componentType": 5126,"count": 197,"max": [ -0.004780198, 0.0003038254, 0.007360002 ],"min": [ -0.008092392, -0.008303153, -0.007400591 ],"type": "VEC3"}],"buffers": [{"byteLength": 2460034,"uri": "example.bin"}],"bufferViews": [{"buffer": 0,"byteLength": 306642,"target": 34963,"byteOffset": 2153392},],"meshes": [{"name": "example mesh","primitives": [{"attributes": {"POSITION": 0,"NORMAL": 1,"TEXCOORD_0": 2,"TANGENT": 3},"indices": 4,"material": 0,"mode": 4}]}]
}

要查找网格的位置数据,我们首先需要访问索引 0 处的 “meshes” 键,然后访问第一个基元。(据我所知,基元本质上只是子网格。然后我们将检索“属性”->“位置”。这将为我们提供访问器的索引。插入它,我们可以从第一个访问器获取“bufferView”值。然后,这为我们提供了缓冲区视图的索引,我们最终可以使用它来获取缓冲区以从中检索数据。在这种情况下,缓冲区存储在外部文件“example.bin”中。打开该文件后,我们将转到访问器中“byteOffset”提供给我们的位置,最后读取缓冲区数据。

下面开始分别介绍如何从gltf/glb文件中读取数据 ,在此过程中你可以用GLTF编辑器对3D模型文件进行编辑和验证模型数据。

从 GLTF 文件读取

我将在我的示例代码中使用 c++ ,但对于任何其他语言,步骤应该大致相同。

// First define our filname, would probbably be better to prompt the user for one
const std::string& gltfFilename = "example.gltf"// open the gltf file
std::ifstream jsonFile(gltfFilename, std::ios::binary);// parse the json so we can use it later
Json::Value json;try{jsonFile >> json;
}catch(const std::exception& e){std::cerr << "Json parsing error: " << e.what() << std::endl;
}
jsonFile.close();// Extract the name of the bin file, for the sake of simplicity I'm assuming there's only one
std::string binFilename = json["buffers"][0]["uri"].asString();// Open it with the cursor at the end of the file so we can determine it's size,
// We could techincally read the filesize from the gltf file, but I trust the file itself more
std::ifstream binFile = std::ifstream(binFilename, std::ios::binary | std::ios::ate);// Read file length and then reset cursor
size_t binLength = binFile.tellg();
binFile.seekg(0);std::vector<char> bin(binLength);
binFile.read(bin.data(), binLength);
binFile.close();// Now that we have the files read out, let's actually do something with them
// This code prints out all the vertex positions for the first primitive// Get the primitve we want to print out: 
Json::Value& primitive = json["meshes"][0]["primitives"][0];// Get the accessor for position: 
Json::Value& positionAccessor = json["accessors"][primitive["attributes"]["POSITION"].asInt()];// Get the bufferView 
Json::Value& bufferView = json["bufferViews"][positionAccessor["bufferView"].asInt()];// Now get the start of the float3 array by adding the bufferView byte offset to the bin pointer
// It's a little sketchy to cast to a raw float array, but hey, it works.
float* buffer = (float*)(bin.data() + bufferView["byteOffset"].asInt());// Print out all the vertex positions 
for (int i = 0; i < positionAccessor["count"].asInt(); ++i)
{std::cout << "(" << buffer[i*3] << ", " << buffer[i*3 + 1] << ", " << buffer[i*3 + 2] << ")" << std::endl;
}// And as a cherry on top, let's print out the total number of verticies
std::cout << "vertices: " << positionAccessor["count"].asInt() << std::endl;

从 GLB 文件读取:

从 .glb 文件中读取有点困难,因为我们不能只是将其放入 JSON 解析器中,但它是可行的。在文件类型部分中参考上图,我们可以找到有关所需文件格式的所有信息:

std::ifstream binFile = std::ifstream(glbFilename, std::ios::binary); binFile.seekg(12); //Skip past the 12 byte header, to the json header
uint32_t jsonLength;
binFile.read((char*)&jsonLength, sizeof(uint32_t)); //Read the length of the json file from it's headerstd::string jsonStr;
jsonStr.resize(jsonLength);
binFile.seekg(20); // Skip the rest of the JSON header to the start of the string
binFile.read(jsonStr.data(), jsonLength); // Read out the json string// Parse the json
Json::Reader reader;
if(!reader.parse(jsonStr, _json))std::cerr << "Problem parsing assetData: " << jsonStr << std::endl;// After reading from the json, the file cusor will automatically be at the start of the binary headeruint32_t binLength;
binFile.read((char*)&binLength, sizeof(binLength)); // Read out the bin length from it's header
binFile.seekg(sizeof(uint32_t), std::ios_base::cur); // skip chunk typestd::vector<char> bin(binLength);
binFile.read(bin.data(), binLength);//Now you're free to use the data the same way we did above

总结

希望这对您有所帮助。我知道它在某些方面有点缺乏细节,所以一旦我了解更多,我可能会回来更新它,提供更多关于动画和皮肤的信息。但在那之前,再见。

原文链接:了解 glTF 2.0 格式

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/70235.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ElementUI浅尝辄止33:Form 表单

Form 表单&#xff1a;日常业务中很常见&#xff0c;由输入框、选择器、单选框、多选框等控件组成&#xff0c;用以收集、校验、提交数据&#xff0c;常见于表单请求、登录&#xff0c;数据校验等业务操作中 1.如何使用&#xff1f; 包括各种表单项&#xff0c;比如输入框、选…

分布式实时仿真系统-反射内存的应用

为了使分布式实时仿真系统(一个典型代表就行飞行模拟器)达到逼真的仿真效果&#xff0c;在系统内部&#xff0c;往往不仅需要对各种数据模型进行实时解算&#xff0c;而且需要一个延迟时间极低的确定性网络在系统之间传递数据&#xff0c;这样才能让各个子系统之间协调一致地工…

问道管理:分时高抛低吸策略?

分时高抛低吸是股市中的一种买卖战略&#xff0c;也是投资者经常运用的一种方法。这种战略经过剖析图表、股价和时刻&#xff0c;尽可能减少危险&#xff0c;添加收益。下面从多个视点对十二种分时高抛低吸进行剖析。 视点一&#xff1a;什么是分时高抛低吸&#xff1f; 分时高…

报错:为什么数组明明有内容但打印的length是0

文章目录 一、问题二、分析三、解决1.将异步改为同步2.设置延迟 一、问题 在日常开发中&#xff0c;for 循环遍历调用接口&#xff0c;并将接口返回的值进行拼接&#xff0c;即push到一个新的数组中&#xff0c;但是在for循环内部是可以拿到这个新的数组&#xff0c;而for循环…

Selenium 3和JUnit 5中的显示等待与隐式等待

目录 显示等待&#xff08;Explicit Wait&#xff09; 什么是显示等待&#xff1f; 如何实现显示等待&#xff1f; 显示等待条件 隐式等待&#xff08;Implicit Wait&#xff09; 什么是隐式等待&#xff1f; 如何实现隐式等待&#xff1f; 显示等待和隐式等待的区别 …

mapbox 导航记录(release-v2.15分支 纯kotlin)

一、简单使用示例 1. 初始化 MapboxNavigation 初始化时使用 NavigationOptions 设置一些参数&#xff0c;包括accessToken、appMetaData、LocationEngine等&#xff0c;其它还有很多&#xff0c;具体可以详看 NavigationOptions 类的内部。 下面示例中 LocationEngine 使用…

C++ 多线程编程教程:使用 std::thread 和 std::future 进行并发任务管理 ,处理线程超时

C 多线程编程教程&#xff1a;使用 std::thread 和 std::future 进行并发任务管理 引言 多线程编程是一种强大的工具&#xff0c;可以加速计算密集型任务的执行&#xff0c;提高应用程序的性能。C提供了多种多线程编程工具&#xff0c;包括std::thread和std::future&#xff…

【infiniband】用udaddy测试RDMA_CM API通过GID连接

1. 运行ibv_devinfo -vv 找到GID[ 0]。 2.udaddy用1.找到的GID[ 0]测试连接 服务器: rootdebian:~/infiniband/rdma-core-50mlnx1# build/bin/udaddy -f gid -b fe80:0000:0000:0000:e41d:2e03:0051:26d1 udaddy: starting server test.rai->ai_src_addr->sa_famil…

IDEA版SSM入门到实战(Maven+MyBatis+Spring+SpringMVC) -Maven使用前准备

一&#xff0e;Maven准备 注意&#xff1a;IDEA2019.1.x 最高支持Maven的3.6.0 下载地址&#xff1a;http://maven.apache.org/Maven底层使用Java语言编写的&#xff0c;所以需要配置JAVA_HOME环境变量及Path将Maven解压非中文无空格目录下配置MAVEN_HOME环境变量及Path输入【c…

基于AERMOD模型在大气环境影响评价中的实践

AERMOD模型是在美国EPA&#xff08;AMS/EPA&#xff09;在ISC3&#xff08;Industrial Source Complex Model&#xff09;基础上建立开发的高斯稳态扩散模型&#xff0c;是我国《环境影响评价技术导则 大气环境&#xff08;HJ 2.2-2018&#xff09;》技术导则推荐的大气污染物浓…

人工智能的优势:使用 GPT 和扩散模型生成图像

推荐&#xff1a;使用 NSDT场景编辑器快速搭建3D应用场景 世界被人工智能 &#xff08;AI&#xff09; 所吸引&#xff0c;尤其是自然语言处理 &#xff08;NLP&#xff09; 和生成 AI 的最新进展&#xff0c;这是有充分理由的。这些突破性技术有可能提高各种任务的日常生产力。…

Vue前端框架08 Vue框架简介、VueAPI风格、模板语法、事件处理、数组变化侦测

目录 一、Vue框架1.1渐进式框架1.2 Vue的版本 二、VueAPI的风格三、Vue开发准备工作四、模板语法文本插值属性绑定条件渲染列表渲染key管理状态 四、事件处理定义事件事件参数事件修饰符 五、数组变化侦测 一、Vue框架 渐进式JavaScript框架&#xff0c;易学易用&#xff0c;性…

Unity的GPUSkinning进一步介绍

大家好&#xff0c;我是阿赵。   在几年前&#xff0c;我曾经写过一篇介绍GPUSkinning的文章&#xff0c;这么多年之后&#xff0c;还是看到不停有朋友在翻看这篇旧文章。今天上去GitHub看了一下&#xff0c;GPUSkinning这个开源的插件已经很久没有更新过了&#xff0c;还是停…

SSH详解

文章目录 SSH简介SSH安装SSH秘钥秘钥生成公钥上传(免密登录) 基本用法命令行配置配置文件SSH代码动态转发本地转发远程转发搭建简易版的VPN SCP命令本地复制到远程远程复制到本地远程复制到远程 Rsync命令安装基本用法本地同步到远程远程同步到本地 SFTP命令 SSH简介 Secure Sh…

分享一个有意思的线程相关的程序运行题

翻开之前的代码&#xff0c;发现了一个有意思的代码&#xff0c;猜以下代码的运行结果&#xff1a; package thread;/*** author heyunlin* version 1.0*/ public class ThreadMethodExample {public static void main(String[] args) {Thread thread new Thread(new Runnabl…

云原生Kubernetes:kubectl管理命令

目录 一、理论 1.kubectl 管理命令 2.项目的生命周期 二、实验 1.kubectl 管理命令 2.项目的生命周期 三、总结 一、理论 1.kubectl 管理命令 &#xff08;1&#xff09;陈述式资源管理方法 kubernetes集群管理集群资源的唯一入口是通过相应的方法调用apiserver的接口…

复旦-华盛顿EMBA:AI时代掘金,科技进化里的挑战与机遇

如果从去年年底ChatGPT3.5发布算起&#xff0c;AI赛道的热度已经持续飙升了半年有余。      “AI的iPhone时刻”代表什么&#xff1f;AI驱动的商业时代已经到来&#xff1f;      我们能看到担忧、恐惧、憧憬&#xff0c;但唯独不缺狂飙突进、加速进化。人类制造AI&…

WordPress(4)关于网站的背景图片更换

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、更改的位置1. 红色区域是要更换的随机的图片二、替换图片位置三.开启随机数量四.结束前言 提示:这里可以添加本文要记录的大概内容: 例如:随着人工智能的不断发展,机器学习这门技术也…

Android 10.0 Launcher3桌面显示多个相同app图标的解决办法

1.前言 在10.0的系统ROM定制化开发中,在Launcher3的系统原生桌面中,在显示桌面的时候,在禁用和启用app的功能测试的时候,会发现有多个相同app的图标显示在桌面 这对Launcher3的体验效果不是很好,所以为了优化产品,需要解决这个bug,然后让产品更完善 2.桌面显示多个相同…

Hadoop的分布式文件存储系统HDFS组件的使用

Hadoop的第一个核心组件&#xff1a;HDFS&#xff08;分布式文件存储系统&#xff09; 一、HDFS的组成1、NameNode2、DataNode3、SecondaryNameNode4、客户端&#xff1a;命令行/Java API 二、HDFS的基本使用1、命令行操作2、Java API操作 三、HDFS的工作流程问题&#xff08;H…