FBX SDK的使用:基础知识

news2025/2/6 5:10:07

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;

// 将密码设置给lString

mFbxSettings->SetStringProp(EXP_FBX_PASSWORD, lString);
mFbxSettings->SetBoolProp(EXP_FBX_PASSWORD_ENABLE, true);

导入时填写密码

FbxString lString;

// 将密码设置给lString

mFbxSettings->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,  // mConvertRrsNodes
		true,	// mConvertLimits
		true,   // mConvertClusters
		true,	// mConvertLightIntensity
		true,   // mConvertPhotometricLProperties
		true,	// 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.coloradmin.cn/o/2293600.html

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

相关文章

计算机网络笔记再战——理解几个经典的协议6——TCP与UDP

目录 先说端口号 TCP 使用序号保证顺序性和应答来保证有效性 超时重传机制 TCP窗口机制 UDP 路由协议 协议分类&#xff1a;IGP和EGP 几个经典的路由算法 RIP OSPF 链路状态数据库&#xff08;LSDB&#xff09; LSA&#xff08;Link State Advertisement&#xff0…

【技海登峰】Kafka漫谈系列(二)Kafka高可用副本的数据同步与选主机制

【技海登峰】Kafka漫谈系列(二)Kafka高可用副本的数据同步与选主机制 一. 数据同步 在之前的学习中有了副本Replica的概念,解决了数据备份的问题。我们还需要面临一个设计难题即:如何处理分区中Leader与Follwer节点数据同步不匹配问题所带来的风险,这也是保证数据高可用的…

电商用户画像数据可视化分析

电商用户画像数据可视化分析 作者&#xff1a;i阿极 作者简介&#xff1a;Python领域新星作者、多项比赛获奖者&#xff1a;博主个人首页 &#x1f60a;&#x1f60a;&#x1f60a;如果觉得文章不错或能帮助到你学习&#xff0c;可以点赞&#x1f44d;收藏&#x1f4c1;评论&am…

Vue3.5常用特性整理

Vue3.5 发布已近半年&#xff0c;抽空整理下常用的新增/改动特性 响应式 Props 解构 Vue3.5 中 Props 正式支持解构了&#xff0c;并添加了响应式跟踪 设置默认值 使用 JavaScript 原生的默认值语法声明 props 默认值 以前 const props withDefaults(defineProps<{ co…

2024年12月 Scratch 图形化(一级)真题解析 中国电子学会全国青少年软件编程等级考试

202412 Scratch 图形化&#xff08;一级&#xff09;真题解析 中国电子学会全国青少年软件编程等级考试 一、单选题(共25题&#xff0c;共50分) 第 1 题 点击下列哪个按钮&#xff0c;可以将红框处的程序放大&#xff1f;&#xff08; &#xff09; A. B. C. D. 标…

游戏引擎学习第87天

当直接使用内存时&#xff0c;可能会发生一些奇怪的事情 在直接操作内存时&#xff0c;一些意外的情况可能会发生。由于内存实际上只是一个大块的空间&#xff0c;开发者可以完全控制它&#xff0c;而不像高级语言那样必须遵守许多规则&#xff0c;因此很容易发生错误。在一个…

【物联网】ARM核常用指令(详解):数据传送、计算、位运算、比较、跳转、内存访问、CPSR/SPSR

文章目录 指令格式&#xff08;重点&#xff09;1. 立即数2. 寄存器位移 一、数据传送指令1. MOV指令2. MVN指令3. LDR指令 二、数据计算指令1. ADD指令1. SUB指令1. MUL指令 三、位运算指令1. AND指令2. ORR指令3. EOR指令4. BIC指令 四、比较指令五、跳转指令1. B/BL指令2. l…

Qt展厅播放器/多媒体播放器/中控播放器/帧同步播放器/硬解播放器/监控播放器

一、前言说明 音视频开发除了应用在安防监控、视频网站、各种流媒体app开发之外&#xff0c;还有一个小众的市场&#xff0c;那就是多媒体展厅场景&#xff0c;这个场景目前处于垄断地位的软件是HirenderS3&#xff0c;做的非常早而且非常全面&#xff0c;都是通用的需求&…

html中的表格属性以及合并操作

表格用table定义&#xff0c;标签标题用caption标签定义&#xff1b;用tr定义表格的若干行&#xff1b;用td定义若干个单元格&#xff1b;&#xff08;当单元格是表头时&#xff0c;用th标签定义&#xff09;&#xff08;th标签会略粗于td标签&#xff09; table的整体外观取决…

html的字符实体和颜色表示

在HTML中&#xff0c;颜色可以通过以下几种方式表示&#xff0c;以下是具体的示例&#xff1a; 1. 十六进制颜色代码 十六进制颜色代码以#开头&#xff0c;后面跟随6个字符&#xff0c;每两个字符分别表示红色、绿色和蓝色的强度。例如&#xff1a; • #FF0000&#xff1a;纯红…

unordered_map/set的哈希封装

【C笔记】unordered_map/set的哈希封装 &#x1f525;个人主页&#xff1a;大白的编程日记 &#x1f525;专栏&#xff1a;C笔记 文章目录 【C笔记】unordered_map/set的哈希封装前言一. 源码及框架分析二.迭代器三.operator[]四.使用哈希表封装unordered_map/set后言 前言 哈…

idea中git的简单使用

提交&#xff0c;推送直接合并 合到哪个分支就到先切到哪个分支

Fastdds学习分享_xtpes_发布订阅模式及rpc模式

在之前的博客中我们介绍了dds的大致功能&#xff0c;与组成结构。本篇博文主要介绍的是xtypes.分为理论和实际运用两部分.理论主要用于梳理hzy大佬的知识&#xff0c;对于某些一带而过的部分作出更为详细的阐释&#xff0c;并在之后通过实际案例便于理解。案例分为普通发布订阅…

开发板上Qt运行的环境变量的三条设置语句的详解

在终端中运行下面三句命令用于配置开发板上Qt运行的环境变量&#xff1a; export QT_QPA_GENERIC_PLUGINStslib:/dev/input/event1 export QT_QPA_PLATFORMlinuxfb:fb/dev/fb0 export QT_QPA_FONTDIR/usr/lib/fonts/设置成功后可以用下面的语句检查设置成功没有 echo $QT_QPA…

语言月赛 202412【顽强拼搏奖的四种发法】题解(AC)

》》》点我查看「视频」详解》》》 [语言月赛 202412] 顽强拼搏奖的四种发法 题目描述 在 XCPC 竞赛里&#xff0c;会有若干道题目&#xff0c;一支队伍可以对每道题目提交若干次。我们称一支队伍对一道题目的一次提交是有效的&#xff0c;当且仅当&#xff1a; 在本次提交…

自定义数据集 使用scikit-learn中svm的包实现svm分类

引入必要的库 import numpy as np from sklearn.datasets import make_classification from sklearn.model_selection import train_test_split from sklearn.svm import SVC from sklearn.metrics import accuracy_score, classification_report 生成自定义数据集 X, y ma…

有用的sql链接

『SQL』常考面试题&#xff08;2——窗口函数&#xff09;_sql的窗口函数面试题-CSDN博客 史上最强sql计算用户次日留存率详解&#xff08;通用版&#xff09;及相关常用函数 -2020.06.10 - 知乎 (zhihu.com) 1280. 学生们参加各科测试的次数 - 力扣&#xff08;LeetCode&…

【Numpy核心编程攻略:Python数据处理、分析详解与科学计算】2.27 NumPy+Pandas:高性能数据处理的黄金组合

2.27 NumPyPandas&#xff1a;高性能数据处理的黄金组合 目录 #mermaid-svg-x3ndEE4hrhO6WR6H {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-x3ndEE4hrhO6WR6H .error-icon{fill:#552222;}#mermaid-svg-x3ndEE4hr…

第一个3D程序!

运行效果 CPP #include <iostream> #include <fstream> #include <string> #include <cmath>#include <GL/glew.h> #include <GLFW/glfw3.h> #include <glm/glm.hpp> #include <glm/gtc/type_ptr.hpp> #include <glm/gtc/…

NeuralCF 模型:神经网络协同过滤模型

实验和完整代码 完整代码实现和jupyter运行&#xff1a;https://github.com/Myolive-Lin/RecSys--deep-learning-recommendation-system/tree/main 引言 NeuralCF 模型由新加坡国立大学研究人员于 2017 年提出&#xff0c;其核心思想在于将传统协同过滤方法与深度学习技术相结…