UStaticMesh几何数据相关(UE5.2)

2024-05-15 16:52

本文主要是介绍UStaticMesh几何数据相关(UE5.2),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

UStaticMesh相关类图

8f663eb4989e48579fd3d47252243c43.png

 

UStaticMesh的数据构成

 

UStaticMesh的FStaticMeshSourceModel

UStaticMesh的Mesh几何元数据来自于FStaticMeshSourceModel, 一级Lod就存在一个FStaticMeshSourceModel. FStaticMeshSourceModel几何数据大致包含以下几类:

Vertex(点), VertexInstance(顶点), Edge(边),   Triangle(三角形),   Polygon(多边形)

总体上概念和Houdini的一些几何概念相似,每种几何元素都可以附带属性,比如点的“Position”, 顶点的“Normal,UV”等等。

下面介绍下如何获取Mesh元数据和相关属性

void AMyActor::TestMeshInfo() const
{if(!Mesh)return;// SourceModelconst TArray<FStaticMeshSourceModel>& SourceModels = Mesh->GetSourceModels();for(int32 Index = 0; Index < SourceModels.Num(); Index++){FMeshDescription* MeshDesc = Mesh->GetMeshDescription(Index);if(MeshDesc){const FVertexArray& VertexArray =  MeshDesc->Vertices();// TVertexAttributesRef<FVector3f> VertexPositions = MeshDesc->VertexAttributes().GetAttributesRef<FVector3f>(MeshAttribute::Vertex::Position);TVertexAttributesRef<FVector3f> VertexPositions = MeshDesc->GetVertexPositions();//loop vertex postionfor(const FVertexID VertexID : VertexArray.GetElementIDs()){FVector3f& Pos = VertexPositions[VertexID];}UE_LOG(LogTemp, Warning, TEXT("Vertex num = %d"), VertexArray.Num());//loop vertex instance attributeconst FVertexInstanceArray& VertexInstanceArray = MeshDesc->VertexInstances();TVertexInstanceAttributesRef<FVector4f> VertexInstanceColors = MeshDesc->VertexInstanceAttributes().GetAttributesRef<FVector4f>(MeshAttribute::VertexInstance::Color);for(const FVertexInstanceID VertxInstanceId : VertexInstanceArray.GetElementIDs()){}UE_LOG(LogTemp, Warning, TEXT("VertexInstanceArray num = %d"), VertexInstanceArray.Num());//loop vertex edge attributeconst FEdgeArray& EdgeArray = MeshDesc->Edges();TEdgeAttributesRef<bool> EdgeHardness = MeshDesc->EdgeAttributes().GetAttributesRef<bool>(MeshAttribute::Edge::IsHard);for (FEdgeID EdgeID : EdgeArray.GetElementIDs()){}UE_LOG(LogTemp, Warning, TEXT("EdgeArray num = %d"), EdgeArray.Num());// loop triangle attributeconst FTriangleArray& TriangleArray = MeshDesc->Triangles();TTriangleAttributesRef<FVector3f> TriangleNormals = MeshDesc->TriangleAttributes().GetAttributesRef<FVector3f>(MeshAttribute::Triangle::Normal);for (FTriangleID TriangleID : TriangleArray.GetElementIDs()){}UE_LOG(LogTemp, Warning, TEXT("TriangleArray num = %d"), TriangleArray.Num());// loop polygonconst FPolygonArray& PolygonArray =  MeshDesc->Polygons();TPolygonAttributesRef<int32> PatchGroups = MeshDesc->PolygonAttributes().GetAttributesRef<int32>(MeshAttribute::Polygon::PolygonGroupIndex);UE_LOG(LogTemp, Warning, TEXT("PolygonArray num = %d"), PolygonArray.Num());}if(Mesh->GetRenderData()->LODResources.Num() > Index){TArray<FVector3f> Vertexs;auto& LodResource = Mesh->GetRenderData()->LODResources[Index];int32 VertexNum = LodResource.GetNumVertices();for(int32 I = 0; I < VertexNum; I++){Vertexs.Add(LodResource.VertexBuffers.PositionVertexBuffer.VertexPosition(I));}int32 TriangleNum = LodResource.GetNumTriangles();TArray<uint32> Indexs;for(int32 I = 0; I < TriangleNum; I++){Indexs.Add(LodResource.IndexBuffer.GetIndex(I * 3 + 0));Indexs.Add(LodResource.IndexBuffer.GetIndex(I * 3 + 1));Indexs.Add(LodResource.IndexBuffer.GetIndex(I * 3 + 2));}}}
}

 

 

如何填充数据生成一个UStaticMesh

主要是通过填充FMeshDescription的各种图元数据(Vertex, VertexInstance, Polygon, PolygonGroup等等)

void AMyActor::TestCreateStaticMesh()
{const FString ObjectName = "TestStaticMesh";const FString GameFoldPath = "/Game";const FString PackageName = FString::Printf(TEXT("%s/%s"), *GameFoldPath, *ObjectName);UPackage* Package = CreatePackage(*PackageName);UStaticMesh* MyMesh = NewObject<UStaticMesh>(Package, *ObjectName, RF_Standalone | RF_Public);FMeshDescription MeshDescription;FStaticMeshAttributes MeshAttribute(MeshDescription);MeshAttribute.Register();TArray<FDynamicMeshVertex> Vertices;Vertices.Add(FDynamicMeshVertex(FVector3f(500.0f, 500.0f, 200)));Vertices.Add(FDynamicMeshVertex(FVector3f(-500.0f, 500.0f, 200)));Vertices.Add(FDynamicMeshVertex(FVector3f(-500.0f, -500.0f, 200)));Vertices.Add(FDynamicMeshVertex(FVector3f(500.0f, -500.0f, 200)));TArray<int32> Indexs;Indexs.Add(0);Indexs.Add(3);Indexs.Add(1);Indexs.Add(1);Indexs.Add(3);Indexs.Add(2);MeshDescription.Empty();MeshDescription.ReserveNewVertices(Vertices.Num());MeshDescription.ReserveNewTriangles(Indexs.Num() / 3);MeshDescription.ReserveNewVertexInstances(Indexs.Num());TVertexAttributesRef<FVector3f> VertexPositions = MeshAttribute.GetVertexPositions();TVertexInstanceAttributesRef<FVector2f>	VertexInstanceUVs = MeshAttribute.GetVertexInstanceUVs();TVertexInstanceAttributesRef<FVector3f>	VertexInstanceNormals = MeshAttribute.GetVertexInstanceNormals();TVertexInstanceAttributesRef<FVector3f>	VertexInstanceTangents = MeshAttribute.GetVertexInstanceTangents();TPolygonGroupAttributesRef<FName> MaterialSlotNames = MeshAttribute.GetPolygonGroupMaterialSlotNames();TArray<UMaterialInterface*> MaterialInterfaces = {UMaterial::GetDefaultMaterial(MD_Surface)};for(int32 Index = 0; Index < MaterialInterfaces.Num(); Index++){FPolygonGroupID PolygonGroupID = MeshDescription.CreatePolygonGroup();FStaticMaterial StaticMaterial = MaterialInterfaces[Index];MaterialSlotNames[PolygonGroupID] = StaticMaterial.MaterialSlotName;}//create vertexfor(int32 Index = 0; Index < Vertices.Num(); Index++){FVertexID VertexId = MeshDescription.CreateVertex();VertexPositions.Set(VertexId, Vertices[Index].Position);}//create vertex instancefor(int32 Index = 0; Index < Indexs.Num(); Index++){FVertexInstanceID VertexInstanceId = FVertexInstanceID(Index);FVertexID VertexID = FVertexID(Indexs[Index]);MeshDescription.CreateVertexInstanceWithID(VertexInstanceId, VertexID);}// set vertex instance normal/tangent/uvFPolygonGroupID PolygonGroupID(0);TArray<FVertexInstanceID> InstanceIds;for(int32 Index = 0; Index < Indexs.Num(); Index++){FVertexInstanceID VertexInstanceId = FVertexInstanceID(Index);InstanceIds.Add(VertexInstanceId);const FDynamicMeshVertex& DynamicVertex = Vertices[Indexs[Index]];VertexInstanceUVs.Set(VertexInstanceId, DynamicVertex.TextureCoordinate[0]);VertexInstanceNormals.Set(VertexInstanceId, DynamicVertex.TangentX.ToFVector3f());VertexInstanceTangents.Set(VertexInstanceId, DynamicVertex.TangentZ.ToFVector3f());if(InstanceIds.Num() == 3){TArray<FEdgeID> EdgeIds;MeshDescription.CreatePolygon(PolygonGroupID, InstanceIds, &EdgeIds);InstanceIds.Empty();}}// create source model lod0 and commit mesh descconst int32 LodIndex = 0;if(!MyMesh->IsSourceModelValid(LodIndex)){FStaticMeshSourceModel& SourceModel = MyMesh->AddSourceModel();SourceModel.BuildSettings.bRecomputeNormals = false;SourceModel.BuildSettings.bRecomputeTangents = false;SourceModel.BuildSettings.bGenerateLightmapUVs = false;}MyMesh->CreateMeshDescription(LodIndex, MeshDescription);MyMesh->CommitMeshDescription(LodIndex);MyMesh->ImportVersion = EImportStaticMeshVersion::LastVersion;MyMesh->CreateBodySetup();MyMesh->GetBodySetup()->CollisionReponse = EBodyCollisionResponse::BodyCollision_Disabled;MyMesh->Build(true);MyMesh->PostEditChange();MyMesh->MarkPackageDirty();FAssetRegistryModule::AssetCreated(MyMesh);
}

4cf12657eaea4f30b908fbb25d410d6f.png

 

这篇关于UStaticMesh几何数据相关(UE5.2)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

关于数据埋点,你需要了解这些基本知识

产品汪每天都在和数据打交道,你知道数据来自哪里吗? 移动app端内的用户行为数据大多来自埋点,了解一些埋点知识,能和数据分析师、技术侃大山,参与到前期的数据采集,更重要是让最终的埋点数据能为我所用,否则可怜巴巴等上几个月是常有的事。   埋点类型 根据埋点方式,可以区分为: 手动埋点半自动埋点全自动埋点 秉承“任何事物都有两面性”的道理:自动程度高的,能解决通用统计,便于统一化管理,但个性化定

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

异构存储(冷热数据分离)

异构存储主要解决不同的数据,存储在不同类型的硬盘中,达到最佳性能的问题。 异构存储Shell操作 (1)查看当前有哪些存储策略可以用 [lytfly@hadoop102 hadoop-3.1.4]$ hdfs storagepolicies -listPolicies (2)为指定路径(数据存储目录)设置指定的存储策略 hdfs storagepolicies -setStoragePo

Hadoop集群数据均衡之磁盘间数据均衡

生产环境,由于硬盘空间不足,往往需要增加一块硬盘。刚加载的硬盘没有数据时,可以执行磁盘数据均衡命令。(Hadoop3.x新特性) plan后面带的节点的名字必须是已经存在的,并且是需要均衡的节点。 如果节点不存在,会报如下错误: 如果节点只有一个硬盘的话,不会创建均衡计划: (1)生成均衡计划 hdfs diskbalancer -plan hadoop102 (2)执行均衡计划 hd

sqlite3 相关知识

WAL 模式 VS 回滚模式 特性WAL 模式回滚模式(Rollback Journal)定义使用写前日志来记录变更。使用回滚日志来记录事务的所有修改。特点更高的并发性和性能;支持多读者和单写者。支持安全的事务回滚,但并发性较低。性能写入性能更好,尤其是读多写少的场景。写操作会造成较大的性能开销,尤其是在事务开始时。写入流程数据首先写入 WAL 文件,然后才从 WAL 刷新到主数据库。数据在开始

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

烟火目标检测数据集 7800张 烟火检测 带标注 voc yolo

一个包含7800张带标注图像的数据集,专门用于烟火目标检测,是一个非常有价值的资源,尤其对于那些致力于公共安全、事件管理和烟花表演监控等领域的人士而言。下面是对此数据集的一个详细介绍: 数据集名称:烟火目标检测数据集 数据集规模: 图片数量:7800张类别:主要包含烟火类目标,可能还包括其他相关类别,如烟火发射装置、背景等。格式:图像文件通常为JPEG或PNG格式;标注文件可能为X