FBX SDK的使用:基础知识

Windows环境配置

FBX SDK安装后,目录下有三个文件夹:

  • include 头文件
  • lib 编译的二进制库,根据你项目的配置去包含相应的库
  • samples 官方使用案列

动态链接

libfbxsdk.dll, libfbxsdk.lib是动态库,需要在配置属性->C/C++->预处理器->预处理器定义中添加FBXSDK_SHARED;

静态链接

libfbxsdk-md.lib, libfbxsdk-mt.lib 是两种静态库,不同的是运行库选项,在配置属性->C/C++->代码生成->运行库中设置/MD/MT/MDd/MTd时debug模式。

  • mt 编译时,将LIBCMT.lib编译到obj文件中,连接器通过它处理外部符号,会将引用的外部库集成到生成的库里面;
  • md 编译时,将MSVCRT.lib编译到obj文件中,链接器会链接到MSVCRT.dll,因此生成的库会比mt的小;

如果时debug模式的静态链接,还需要在配置属性->链接器->输入->忽略特定默认库项添加LIBCMT配置属性->链接器->输入->附加依赖项项添加wininet.lib

注意:FBX SDK可以使用多线程,但是不保证线程安全。

FBX SDK基础知识

内存管理

FbxManager类用来管理FBX SDK对象的创建和销毁,且每一个程序里面只能有一个FbxManager的实例。

// 创建FbxManager的实例
FbxManager* lSdkManager = FbxManager::Create();
// 使用FbxManager来创建对象
FbxScene* lScene = FbxScene::Create(lSdkManager, "Scene Name");
// 释放所有FbxManager分配的内存
lSdkManager->Destroy();

导入导出设置

FbxIOSettings ioSettings = FbxIOSettings::Create(lSdkManager, IOSROOT);  // 创建FbxIOSettings
mFbxSettings->SetBoolProp(IMP_FBX_MATERIAL, true);                       // 设置材质导入
lSdkManager->SetIOSettings(ioSettings);									 // 应用设置

设置导入导出对象,所有设置的默认值为true

mFbxSettings->SetBoolProp(IMP_FBX_MATERIAL, true)	// 输入设置,以“IMP_”为前缀
mFbxSettings->SetBoolProp(EXP_FBX_MATERIAL, true)	// 输出设置,以“EXP_”为前缀

导出文件格式设置

EXP_FBX 导出二进制FBX文件,能嵌入媒体文件
EXP_ASCIIFBX 导出ascii FBX文件
EXP_DXF 导出DXF文件
EXP_COLLADA 导出collada文件
EXP_OBJ 导出OBJ文件

嵌入媒体文件

只有binary fbx文件才能嵌入媒体文件。当导入有媒体文件嵌入的fbx文件(myExportFile.fbx)时,将提取嵌入的媒体文件到当前目录下的myExportFile.fbm/subdirectory文件夹中。

mFbxSettings->SetBoolProp(EXP_FBX_EMBEDDED, true);		// 导出Fbx binary文件时,媒体文件嵌入到导出文件中

设置密码保护

导出时设置密码

FbxString lString;// 将密码设置给lStringmFbxSettings->SetStringProp(EXP_FBX_PASSWORD, lString);
mFbxSettings->SetBoolProp(EXP_FBX_PASSWORD_ENABLE, true);

导入时填写密码

FbxString lString;// 将密码设置给lStringmFbxSettings->SetStringProp(IMP_FBX_PASSWORD, lString);
mFbxSettings->SetBoolProp(IMP_FBX_PASSWORD_ENABLE, true);

打印支持的写入和读取文件格式

void PrintWriterFormatAndReaderFormat(FbxManager* fbxManager)
{FbxIOPluginRegistry* ioPlugin = fbxManager->GetIOPluginRegistry();int readerFormatCnt = ioPlugin->GetReaderFormatCount();printf("read format count = %d\n", readerFormatCnt);for (int i = 0; i < readerFormatCnt; i++){const char* extension = ioPlugin->GetReaderFormatExtension(i);const char* desc = ioPlugin->GetReaderFormatDescription(i);printf("extension = %s   description = %s\n", extension, desc);}int writerFormatCnt = ioPlugin->GetWriterFormatCount();printf("writer format count = %d\n", writerFormatCnt);for (int i = 0; i < writerFormatCnt; i++){const char* extension = ioPlugin->GetWriterFormatExtension(i);const char* desc = ioPlugin->GetWriterFormatDescription(i);printf("extension = %s   description = %s\n", extension, desc);}
}

导入导出场景

导入场景

FbxImporter* lImporter = FbxImporter::Create(lSdkManager, "");
bool lImportStatus = lImporter->Initialize(lFileName, -1, mFbxSettings);	// -1 表示让fbxsdk根据文件的扩展名去判断文件的格式
if(!lImportStatus)
{printf("Call to FbxImporter::Initialize() failed.\n");printf("Error returned: %s\n\n", lImporter->GetStatus().GetErrorString());exit(-1);
}// 将文件中的内容导入到scene对象中
FbxScene* scene = FbxScene::Create(lSdkManager, "");
lImporter->Import(scene);// 导入完成后就可以安全销毁,减少内存占用
lImporter->Destroy();

索引非嵌入的多媒体文件

当ASCII FBX或其他不包含嵌入媒体的文件格式定位引用的媒体文件时,遵循下面两个步骤:

  1. 使用绝对路径来检查引用的多媒体文件是否存在,这个绝对路径是导出场景到Fbx文件时指定的 FbxFileTexture::SetFileName()

    // The resource file is in the application's directory.
    FbxString lTexPath = gAppPath + "\\Crate.jpg";// Create a texture object.
    FbxFileTexture* lTexture = FbxFileTexture::Create(pScene,"Crate Texture");// Set the texture's absolute file path.
    lTexture->SetFileName(lTexPath.Buffer())
    
  2. 如果绝对路径不存在多媒体文件,则使用相对路径,相对路径在导出场景到FBX文件时会自动保存

获取导入文件的版本

int lFileMajor, lFileMinor, lFileRevision;
lImporter->GetFileVersion(lFileMajor, lFileMinor, lFileRevision);

导出场景

FbxExporter* lExporter = FbxExporter::Create(lSdkmanager, "");
const char* lFilename = "file.fbx";
bool lExportStatus = lExporter->Initialize(lFilename, -1, lSdkManager->GetIOSettings());
if(!lExportStatus) {printf("Call to FbxExporter::Initialize() failed.\n");printf("Error returned: %s\n\n", lExporter->GetStatus().GetErrorString());return false;
}FbxScene* scene = FbxScene::Create(lSdkManager, "myScene");
lExporter->Export(lScene);
lExporter->Destroy();

坐标系和单位转换

我们读取FBX文件的数据到自己的应用中,需要的坐标系和单位可能和FBX文件的不一样,这时需要去修改这些设置。使用FBX SDK创建的对象是右手坐标系,Y轴向上。

// 转换为OpenGL坐标系
FbxGlobalSettings& globalSettings = mScene->GetGlobalSettings();
if (globalSettings.GetAxisSystem() != FbxAxisSystem::OpenGL)
{FbxAxisSystem::OpenGL.ConvertScene(mScene);
}// 转换为m作为单位
FbxSystemUnit systemUnit = globalSettings.GetSystemUnit();
if (globalSettings.GetSystemUnit() != FbxSystemUnit::m)
{const FbxSystemUnit::ConversionOptions options = {false,  // mConvertRrsNodestrue,	// mConvertLimitstrue,   // mConvertClusterstrue,	// mConvertLightIntensitytrue,   // mConvertPhotometricLPropertiestrue,	// mConvertCameraClipPlanes};FbxSystemUnit::m.ConvertScene(mScene, options);
}

注意:坐标系转换只会作用到节点Transform的pre-rotaion和动画,单位转换只会作用到节点Transform的scale和动画,并不会作用到顶点的值,比如位置,法线和UV。因此顶点值的转换需要自己去处理。

OpenGL和DirectX坐标系的转换

OpenGL坐标系是Y轴向上,X轴向右,Z轴向屏幕外,DirectX坐标系是Y轴向上,X轴向右,Z轴向屏幕内。如果将DirectX坐标系的顶点转换为OpenGL坐标系的顶点,我们以OpenGL的坐标系为世界坐标系,三个轴的基向量为 x ⃗ 0 ( 1 , 0 , 0 ) , y ⃗ 0 ( 0 , 1 , 0 ) , z ⃗ 0 ( 0 , 0 , 1 ) \vec x_0(1,0,0),\vec y_0(0,1,0),\vec z_0(0,0,1) x 0(1,0,0)y 0(0,1,0)z 0(0,0,1),则DirectX坐标系的三个轴的基向量为 x ⃗ d ( 1 , 0 , 0 ) , y ⃗ d ( 0 , 1 , 0 ) , z ⃗ d ( 0 , 0 , − 1 ) \vec x_d(1,0,0),\vec y_d(0,1,0),\vec z_d(0,0,-1) x d(1,0,0)y d(0,1,0)z d(0,0,1)。若点 P ( x p , y p , z p ) P(x_p,y_p,z_p) P(xp,yp,zp)为DirectX坐标系中的一点,将其转换为OpenGL坐标系为 ( ( x p , 0 , 0 ) ⋅ x ⃗ d , ( 0 , y p , 0 ) ⋅ y ⃗ d , ( 0 , 0 , z p ) ⋅ z ⃗ d ) = ( x p , y p , − z p ) ((x_p,0,0) \cdot \vec x_d, (0,y_p,0) \cdot \vec y_d, (0,0,z_p) \cdot \vec z_d)=(x_p,y_p,-z_p) ((xp,0,0)x d,(0,yp,0)y d,(0,0,zp)z d)=(xp,yp,zp),所以x,y坐标保持不变,z坐标取反。顶点的位置,法线可以按照这个规则处理。

因为OpenGL是右手坐标系,DirectX是左手坐标系,它们三角面的顶点的顺序是相反,所以 ( v 1 , v 2 , v 3 ) (v_1,v_2,v_3) (v1,v2,v3)要变成 ( v 1 , v 3 , v 2 ) (v_1,v_3,v_2) (v1,v3,v2)

顶点的UV, V n e w = 1 − V V_{new}=1-V Vnew=1V,怎么得出来的,目前不清楚。

FBX场景

在这里插入图片描述

每个FBX文件是一个FbxSceneFbxSceneFbxNode以树状层级结构组成。

一个场景由许多元素组成,包含:meshes, lights, cameras, skeletons, NURBS等,这些元素继承FbxNodeAttributeFbxNode是一个或多个场景元素的容器。

FbxNode* lRootNode = lScene->GetRootNode();
if(lRootNode) {for(int i = 0; i < lRootNode->GetChildCount(); i++)PrintNode(lRootNode->GetChild(i));
}
// Print a node, its attributes, and all its children recursively.
void PrintNode(FbxNode* pNode) {const char* nodeName = pNode->GetName();FbxDouble3 translation = pNode->LclTranslation.Get();FbxDouble3 rotation = pNode->LclRotation.Get();FbxDouble3 scaling = pNode->LclScaling.Get();// Print the node's attributes.for(int i = 0; i < pNode->GetNodeAttributeCount(); i++)PrintAttribute(pNode->GetNodeAttributeByIndex(i));// Recursively print the children.for(int j = 0; j < pNode->GetChildCount(); j++)PrintNode(pNode->GetChild(j));
}

FBX对象,属性,特性

FbxObject

FBX对象都继承于FbxObject,比如FbxSceneFbxNodeFbxImporter, FbxExporter, FbxCollection等。

FbxCollection是FBX object的容器,FbxAnimLayer, FbxAnimStack, FbxScene都是继承于它。

遍历场景的动画

int numStacks = mScene->GetSrcObjectCount<FbxAnimStack>();
for (int i = 0; i < numStacks; i++)
{FbxAnimStack* animStack = FbxCast<FbxAnimStack>(mScene->GetSrcObject(i));int numAnimLayers = animStack->GetMemberCount<FbxAnimLayer>();for (int layerIdx = 0; layerIdx < numAnimLayers; layerIdx++){FbxAnimLayer* animLayer = animStack->GetMember<FbxAnimLayer>(layerIdx);···}
}

FbxProperty

FbxProperty目前我知道的就是获取FbxNode的变换数据。

FbxDouble* lTranslation = lNode->LclTranslation.Get().mData;
FbxDouble* lRotation    = lNode->LclRotation.Get().mData;
FbxDouble* lScaling     = lNode->LclScaling.Get().mData;

属性的创建和销毁

// 创建FBX Property
FbxProperty p = FbxProperty::Create(pScene, DTDouble3, "Vector3Property");
FbxSet<FbxDouble3>(p, FbxDouble3(1.1, 2.2, 3.3);// ... initialize FbxNode* lNode ...
FbxDouble3 translation = lNode->LclTranslation.Get();
FbxDouble3 rotation = lNode->LclRotation.Get();
FbxDouble3 scaling = lNode->LclScaling.Get();                 

FbxNodeAttribute

FbxNodeAttribute定义了一个场景元素,比如FbxMesh,FbxSkeleton。

遍历FbxNode的FbxNodeAttribute

FbxNode* pNode;
int attrCount = pNode->GetNodeAttributeCount();
for (int i = 0; i < attrCount; i++)
{FbxNodeAttribute* attribute = pNode->GetNodeAttributeByIndex(i);FbxNodeAttribute::EType type = attribute->GetAttributeType();switch (type){case fbxsdk::FbxNodeAttribute::eSkeleton:mSkeletonNodes.push_back(pNode);break;case fbxsdk::FbxNodeAttribute::eMesh:mFbxMeshes.push_back((FbxMesh*)attribute);break;default:break;}
}

对象和属性的关系

FbxObjec和FbxObject,FbxProperty和FbxProperty,FbxObject和FbxProperty直接可以建立父子关系,FBX SDK里面叫Connection。

在这里插入图片描述

GetSrcXXX可以理解为获取子对象,GetDstXX可以理解为获取父对象

  • object-property FbxObject::GetSrcProperty()获取节点的属性,FbxProperty::GetDstObject()获取属性绑定的节点
  • object-object FbxObject::GetSrcObject()获取子节点,FbxProperty::GetDstObject()获取父节点
  • property-property FbxProperty::GetSrcProperty()获取子属性,FbxProperty::GetDstObject()获取父属性

参考:

  • FBX SDK文档
  • 官方案列

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

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

相关文章

【单层神经网络】基于MXNet库简化实现线性回归

写在前面 同最开始的两篇文章 完整程序及注释 导入使用的库# 基本 from mxnet import autograd, nd, gluon # 模型、网络 from mxnet.gluon import nn from mxnet import init # 学习 from mxnet.gluon import loss as gloss # 数据集 from mxnet.gluon…

【爬虫】JS逆向解决某药的商品价格加密

⭐️⭐️⭐️⭐️⭐️欢迎来到我的博客⭐️⭐️⭐️⭐️⭐️ 🐴作者:秋无之地 🐴简介:CSDN爬虫、后端、大数据领域创作者。目前从事python爬虫、后端和大数据等相关工作,主要擅长领域有:爬虫、后端、大数据开发、数据分析等。 🐴欢迎小伙伴们点赞👍🏻、收藏⭐️、…

OpenAI开源战略反思:中国力量推动AI产业变革

在周五的Reddit问答会上&#xff0c;OpenAI首席执行官Sam Altman罕见承认公司正面临来自中国科技企业的强劲挑战。这位向来强硬的硅谷领军者坦言&#xff0c;以深度求索&#xff08;DeepSeek&#xff09;为代表的中国AI公司正在改写行业游戏规则。 这场历时三小时的对话揭示了…

一文讲解HashMap线程安全相关问题(上)

HashMap不是线程安全的&#xff0c;主要有以下几个问题&#xff1a; ①、多线程下扩容会死循环。JDK1.7 中的 HashMap 使用的是头插法插入元素&#xff0c;在多线程的环境下&#xff0c;扩容的时候就有可能导致出现环形链表&#xff0c;造成死循环。 JDK 8 时已经修复了这个问…

android java系统弹窗的基础模板

1、资源文件 app\src\main\res\layout下增加custom_pop_layout.xml 定义弹窗的控件资源。 <?xml version"1.0" encoding"utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android"http://schemas.android.com/apk/…

python学习——常用的内置函数汇总

文章目录 类型转换函数数学函数常用的迭代器操作函数常用的其他内置函数 类型转换函数 数学函数 常用的迭代器操作函数 实操&#xff1a; from cv2.gapi import descr_oflst [55, 42, 37, 2, 66, 23, 18, 99]# (1) 排序操作 asc_lst sorted(lst) # 升序 desc_lst sorted(l…

《解锁AI黑科技:数据分类聚类与可视化》

在当今数字化时代&#xff0c;数据如潮水般涌来&#xff0c;如何从海量数据中提取有价值的信息&#xff0c;成为了众多领域面临的关键挑战。人工智能&#xff08;AI&#xff09;技术的崛起&#xff0c;为解决这一难题提供了强大的工具。其中&#xff0c;能够实现数据分类与聚类…

MySQL数据库环境搭建

下载MySQL 官网&#xff1a;https://downloads.mysql.com/archives/installer/ 下载社区版就行了。 安装流程 看b站大佬的视频吧&#xff1a;https://www.bilibili.com/video/BV12q4y1477i/?spm_id_from333.337.search-card.all.click&vd_source37dfd298d2133f3e1f3e3c…

AI学习指南HuggingFace篇-Tokenizers 与文本处理

一、引言 在自然语言处理(NLP)中,文本数据的预处理是至关重要的一步。分词器(Tokenizers)是将文本分割成单词、短语或其他单元的工具,是文本处理的基础。Hugging Face的Tokenizers库提供了高效且灵活的分词工具,支持多种预训练模型的分词需求。本文将深入讲解Tokenizer…

如何用微信小程序写春联

​ 生活没有模板,只需心灯一盏。 如果笑能让你释然,那就开怀一笑;如果哭能让你减压,那就让泪水流下来。如果沉默是金,那就不用解释;如果放下能更好地前行,就别再扛着。 一、引入 Vant UI 1、通过 npm 安装 npm i @vant/weapp -S --production​​ 2、修改 app.json …

[SAP ABAP] 静态断点的使用

在 ABAP 编程环境中&#xff0c;静态断点通过关键字BREAK-POINT实现&#xff0c;当程序执行到这一语句时&#xff0c;会触发调试器中断程序的运行&#xff0c;允许开发人员检查当前状态并逐步跟踪后续代码逻辑 通常情况下&#xff0c;在代码的关键位置插入静态断点可以帮助开发…

96,【4】 buuctf web [BJDCTF2020]EzPHP

进入靶场 查看源代码 GFXEIM3YFZYGQ4A 一看就是编码后的 1nD3x.php 访问 得到源代码 <?php // 高亮显示当前 PHP 文件的源代码&#xff0c;用于调试或展示代码结构 highlight_file(__FILE__); // 关闭所有 PHP 错误报告&#xff0c;防止错误信息泄露可能的安全漏洞 erro…

基于深度学习的输电线路缺陷检测算法研究(论文+源码)

输电线路关键部件的缺陷检测对于电网安全运行至关重要&#xff0c;传统方法存在效率低、准确性不高等问题。本研究探讨了利用深度学习技术进行输电线路关键组件的缺陷检测&#xff0c;目的是提升检测的效率与准确度。选用了YOLOv8模型作为基础&#xff0c;并通过加入CA注意力机…

3、从langchain到rag

文章目录 本文介绍向量和向量数据库向量向量数据库 索引开始动手实现rag加载文档数据并建立索引将向量存放到向量数据库中检索生成构成一条链 本文介绍 从本节开始&#xff0c;有了上一节的langchain基础学习&#xff0c;接下来使用langchain实现一个rag应用&#xff0c;并稍微…

DeepSeek-R1大模型本地化部署

前言 Ollama作为一个轻量级、易上手的工具&#xff0c;可以帮助你在自己的电脑上快速部署和运行大型语言模型&#xff0c;无需依赖云端服务。通过加载各种开源模型&#xff0c;比如LLaMA、GPT-J等&#xff0c;并通过简单的命令行操作进行模型推理和测试。 此小结主要介绍使用…

【小白学AI系列】NLP 核心知识点(五)Transformer介绍

Transformer Transformer 是一种基于自注意力机制&#xff08;Self-Attention Mechanism&#xff09;的深度学习模型&#xff0c;首次由 Vaswani 等人于 2017 年在论文《Attention is All You Need》中提出。与 RNN 和 LSTM 不同&#xff0c;Transformer 不需要依靠序列顺序进…

【高级篇 / IPv6】(7.6) ❀ 03. 宽带IPv6 - ADSL拨号宽带上网配置 ❀ FortiGate 防火墙

【简介】大部分ADSL拨号宽带都支持IPv6&#xff0c;这里以ADSL拨号宽带为例&#xff0c;演示在FortiGate防火墙上的配置方法。 准备工作 同上篇文章一样&#xff0c;为了兼顾不熟悉FortiGate防火墙的朋友&#xff0c;我们从基础操作进行演示&#xff0c;熟练的朋友可以跳过这一…

【Elasticsearch】_all 查询

在 Elasticsearch 中&#xff0c;_all 查询是一种特殊的查询方式&#xff0c;用于在多个索引或数据流中执行搜索操作&#xff0c;而无需显式指定每个目标索引或数据流的名称。以下是关于 _all 查询的详细说明&#xff1a; _all 查询概述 用途&#xff1a;_all 查询允许您在多个…

Linux第104步_基于AP3216C之I2C实验

Linux之I2C实验是在AP3216C的基础上实现的&#xff0c;进一步熟悉修改设备树和编译设备树&#xff0c;以及学习如何编写I2C驱动和APP测试程序。 1、AP3216C的原理图 AP3216C集成了一个光强传感器ALS&#xff0c;一个接近传感器PS和一个红外LED&#xff0c;为三合一的环境传感…

基于单片机的盲人智能水杯系统(论文+源码)

1 总体方案设计 本次基于单片机的盲人智能水杯设计&#xff0c;采用的是DS18B20实现杯中水温的检测&#xff0c;采用HX711及应力片实现杯中水里的检测&#xff0c;采用DS1302实现时钟计时功能&#xff0c;采用TTS语音模块实现语音播报的功能&#xff0c;并结合STC89C52单片机作…