UE引擎 LandscapeGrass 实现机制分析(UE5.2)

news2025/2/1 1:39:34

前言

随着电脑和手机硬件性能越来越高,游戏越来越追求大世界,而大世界非常核心的一环是植被,目前UE5引擎提供给植被生成的主流两种方式为 手刷植被LandscapeGrass(WeightMap程序化植被)。当然UE5.3推出新一代PCGFramework 节点程序化生成框架,也可以用于植被生成。(本文暂不涉及到PCGFramework), 主要分析LandscapeGrass方案

手刷植被

手刷植被是比较传统的植被生成方式,基本各种引擎都会提供植被编辑器让美术人力刷植被,这里倒是没什么好分析的。UE5提供了手刷植被模式FoliageMode, 工具比较齐全,比较易用,能让场景美术高效率手工刷植被。FoliageMode生成的植被是用HISM组件管理,不明白的可以看看我的另外一篇博客(UE4 4.27) UHierarchicalInstancedStaticMesh(HISM)原理分析

LandscapeGrass(WeightMap程序化植被)

背景

weightmap程序化植被, 顾名思义: 由地形weightmap程序化生成的植被。首先地形WeightMap的概念,说白了就是地形材质混合的权重图,不懂的可以看下UE地形系统材质混合实现和Shader生成分析(UE5 5.2)。场景美术在地形上频繁刷了各种材质,比如草原或者沙漠,然后需要再相应的地貌上种植相应植被,比如草原种绿草,沙漠种仙人掌,因为地形的材质层是频繁改动,每次刷完草或者沙漠后,又得用FoliageMode 在相应地貌刷植被,两重工作量,而且容易遗忘,这时候就非常需要一种工具:  在相应的WeightMap上程序化铺满一种或许多种植被,每次刷完WeightMap后地形可以自动生成。UE引擎 就有这样一套方案,叫LandscapeGrass

LandscapeGrass使用简介

配置LandscapeGrassType

材质Grass采样层配置

显示结果

LandscapeGrass生成分析

LandscapeGrass UML图

LandscapeGrass执行流程图

这里重点介绍RenderGrassMap,GrassInstancedStaticMeshComponent, FAsyncGrassTask.

这里的三个过程都是逐组件的,针对单独一个Landscapecomponent进行

RenderGrassMap(Per LandscapeComponent)

GrassMap本质就是一张图集(TextureAltas)

这里是将LandscapeComponent的HeightMap和Grass用到的所有WeightMap(也就是上面的L2,L3,L4,L5)合并到一张图集里,格式为PF_B8G8R8A8,也就是所谓的RenderGrassMap。

假设一个Landscapecomponent 分辨率为128,一张HeightMap占16位,一张WeightMap占8位, 由于上面使用了L2,L3,L4,L5四个草权重层,则 (2 + 4)/ 4 = 1.5, 得使用两张128 * 128,合起来(128 * 2)* 128 = 256 * 128,这就是RenderGrassMap图集。

GrassMap创建

Render GrassMap

通过DrawLandscapeComponentMesh,设置Ortho正交相机, 进行多次RenderPass, 将HeightMap和WeightMap写入到GrassMap图集里。一次RenderPass写入4个8位通道,6个8位通道就是需要两次RenderPass. 

核心代码:

FLandscapeGrassWeightVS,FLandscapeGrassWeightPS, FLandscapeGrassWeightMeshProcessor等文件

grass.CaptureNextGrassUpdate 1 可以对RenderGrassMap进行Renderdoc抓帧

GrassMap fetch成cpu数据

上一步生成了GrassMap 图集,里面存储了LandscapeComponent地块的HeightMap数据和所有的Grass相关的WeightMap数据。最终直接对GrassMap进行CPU ReadBack,读取数据出来,合并到FLandscapeComponentGrassData的 TArray<uint8> HeightWeightData里。HeightWeightData按顺序分段存储了高度数据和所有的Grass各层权重数据。

为什么需要Render GrassMap

回过头来看RenderGrassMap存在的意义是因为UE引擎地形的各个LandscapeComponent共用了HeightMap和WeightMap, RenderGrassMap就是把每个组件的HeightData和GrassWeightData从公用纹理抽离出来,存到LandscapeComponent里,方便每个LandscapeComponent生成自己的GrassInstancedStaticMeshComponent.

GrassInstancedStaticMeshComponent

GrassInstancedStaticMeshComponent继承于HISM组件,两者基本没什么区别。这里需要强调的是一个GSMC组件针对一个LandcapeComponent的LandscapeGrassType的GrassVariety来创建的。

FAsyncGrassTask(随机撒点和HISM构建)

LandscapeComponent已经获取高度和权重数据,下面就是进行随机撒点。FAsyncGrassTask

随机撒点

撒点密度由LandscapeComponent地块大小和GrassVariety的密度决定

FAsyncGrassBuilder封装了随机撒点的核心算法, 目前一共提供了两种随机撒点算法: JitterGrid和 Halto序列

勾选了UseGrid就是使用了JitterGrid随机撒点,反之使用Halto随机撒点。

JitterGrid

JitterGrid也就是根据密度把整个地形块均分切成多个块,每个块内随机生成一个二维点,并且加上一定的旋转和位置偏移,最终得到一个随机点。具体算法参考我之前一篇博客: 程序化物件放置(procedural placement)之泊松硬盘采样(poisson disk sampling)

一个随机点虽然生成出来了,但是是否存活得满足一定条件:

可以看出当权重为0,百分百不通过。权重为1时,通过率是百分百,如果权重为其它,那就看随机的运气了。

Halto序列

Halto随机数生成参考 应用halton序列生成均匀散点图

Halto序列生成的随机点存活条件判断和JitterGrid是一样的。比较注意的是,JitterGrid和Halto序列生成的随机点都是二维的,都需要进行一次SampleHeight来得到高度位置Z,Sample方法是 Bilinear interpolates。

撒完点后最终进行HISM构建。

FoliageModeLandscapeGrass的区别

未完待续

参考文章

[1] UE地形系统材质混合实现和Shader生成分析(UE5 5.2)

[2] (UE4 4.27) UHierarchicalInstancedStaticMesh(HISM)原理分析

[3] 应用halton序列生成均匀散点图

[4] 源码文件:LandscapeGrass.cpp

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

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

相关文章

办公word-从不是第一页添加页码

总结 实际需要注意的是&#xff0c;分隔符、分节符和分页符并不是一个含义 分隔符包含其他两个&#xff1b;分页符&#xff1a;是增加一页&#xff1b;分节符&#xff1a;指将文档分为几部分。 从不是第一页插入页码1步骤 1&#xff0c;插入默认页码 自己可以测试时通过**…

一文掌握Ascend C孪生调试

1 What&#xff0c;什么是孪生调试 Ascend C提供孪生调试方法&#xff0c;即CPU域模拟NPU域的行为&#xff0c;相同的算子代码可以在CPU域调试精度&#xff0c;NPU域调试性能。孪生调试的整体方案如下&#xff1a;开发者通过调用Ascend C类库编写Ascend C算子kernel侧源码&am…

784. 字母大小写全排列

字母大小写全排列 描述 : 给定一个字符串 s &#xff0c;通过将字符串 s 中的每个字母转变大小写&#xff0c;我们可以获得一个新的字符串。 返回 所有可能得到的字符串集合 。以 任意顺序 返回输出。 题目 : LeetCode 784.字母大小写全排列 : 784. 字母大小写全排列 分析…

JavaScript常用技巧专题二

文章目录 一、前言二、生成随机字符串三、转义HTML特殊字符四、单词首字母大写五、将字符串转换为小驼峰六、删除数组中的重复值七、移除数组中的假值八、获取两个数字之间的随机数九、将数字截断到固定的小数点十、日期10.1、计算两个日期之间天数10.2、从日期中获取是一年中的…

fd信息查看

一、/proc/pid/fdinfo和/proc/pid/fd的含义 二、实例 ref&#xff1a; linux下的/proc/pid/fdinfo和/proc/pid/fd_proc/417/fdinfo/0-CSDN博客 linux select read阻塞_图解Linux的IO模型和相关技术-CSDN博客

实现加盐加密方法以及java nio中基于MappedByteBuffer操作大文件

自己实现 传统MD5可通过彩虹表暴力破解&#xff0c; 加盐加密算法是一种常用的密码保护方法&#xff0c;它将一个随机字符串&#xff08;盐&#xff09;添加到原始密码中&#xff0c;然后再进行加密处理。 1. 每次调用方法产生一个唯一盐值&#xff08;UUID &#xff09;密码…

生产问题: 利用线程Thread预加载数据缓存,其它类全局变量获取缓存偶发加载不到

生产问题: 利用线程Thread预加载数据缓存偶发加载不到 先上代码 public class ThreadTest {//本地缓存Map<String, Object> map new HashMap<String, Object>();class ThreadA implements Runnable{Overridepublic void run() {System.out.println("Thread…

【崩坏:星穹铁道】卡芙卡狂喜!1.5版本新遗器值不值得刷?

《崩坏&#xff1a;星穹铁道》在1.5版本上线了4套新的遗器套装&#xff0c;照例还是两套外圈遗器两套内圈遗器&#xff0c;分别可以通过幽冥之径侵蚀隧洞副本和模拟宇宙第八世界刷取。 那么本期闲游盒小盒子就简单解析一下这4套新遗器&#xff0c;它们的属性适合哪些角色&#…

2021版吴恩达深度学习课程Deeplearning.ai 05序列模型 12.5

学习内容 05.序列模型 1.1 为什么用序列模型 1.序列模型常见的应用 1.2 注释 notation 1.*T_x(i)表示训练样本x(i)的序列长度&#xff0c;T_y(i)表示target(i)的序列长度2.训练集表示单词的方式*构建字典的方式*在训练集中查找出现频率最高的单词*网络搜集常用字典3.如果遇…

微服务——服务保护Sentinel

雪崩问题 在单体项目里面&#xff0c;如果某一个模块出问题会导致整个项目都有问题。 在微服务项目里面&#xff0c;单独一个服务出问题理论上是不会影响别的服务的。 但是如果有别的业务需要调用这一个模块的话还是会有问题。 问题产生原因和解决思路 最初那只是一个小小…

GEE影像升尺度(10m->250m)

GEE影像升尺度&#xff08;10m->250m&#xff09; 代码 var ext /* color: #d63000 *//* shown: false *//* displayProperties: [{"type": "rectangle"}] */ee.Geometry.Polygon([[[108.74625980473367, 28.562445155322063],[108.74625980473367, …

云计算在计算机领域的应用与发展

云计算在计算机领域的应用与发展 一、引言 随着科技的不断发展&#xff0c;计算机领域已经成为当今社会最为活跃和创新的领域之一。云计算作为一种新兴的计算模式&#xff0c;已经在计算机领域中得到了广泛的应用&#xff0c;并且正在不断地推动着计算机领域的发展。本文将探…

令牌桶算法理解学习(限流算法)

令牌桶算法是网络流量整形&#xff08;Traffic Shaping&#xff09;和速率限制&#xff08;Rate Limiting&#xff09;中最常使用的一种算法。典型情况下&#xff0c;令牌桶算法用来控制发送到网络上的数据的数目&#xff0c;并允许突发数据的发送。 用简单的话语来说就是限制…

基于MyBatis二级缓存深入装饰器模式

视频地址 学习文档 文章目录 一、示意代码二、装饰器三、经典案例—MyBatis二级缓存1、Cache 标准定义2、PerpetualCache 基础实现3、增强实现3-1、ScheduledCache3-2、LruCache 先来说说我对装饰器理解&#xff1a;当你有一个基础功能的代码&#xff0c;但你想在不改变原来代…

轻量封装WebGPU渲染系统示例<46>- 材质组装管线(MaterialPipeline)灯光、阴影、雾以及多Pass(源码)

当前示例源码github地址: https://github.com/vilyLei/voxwebgpu/blob/feature/material/src/voxgpu/sample/MaterialPipelineMultiPasses.ts 当前示例运行效果: 此示例基于此渲染系统实现&#xff0c;当前示例TypeScript源码如下&#xff1a; export class MaterialPipelin…

airserver mac 7.27官方破解版2024最新安装激活图文教程

airserver mac 7.27官方破解版是一款好用的airplay投屏工具&#xff0c;可以轻松将ios荧幕镜像&#xff08;airplay&#xff09;至mac上&#xff0c;在mac平台上实现视频、音频、幻灯片等文件资源的接收及投放演示操作&#xff0c;解决iphone或ipad的屏幕录像问题&#xff0c;满…

Leetcode1466. 重新规划路线

Every day a Leetcode 题目来源&#xff1a;1466. 重新规划路线 解法1&#xff1a;深度优先搜索 n 座城市&#xff0c;从 0 到 n-1 编号&#xff0c;其间共有 n-1 条路线。 因此&#xff0c;要想在两座不同城市之间旅行只有唯一一条路线可供选择&#xff08;路线网形成一颗…

实战1-python爬取安全客新闻

一般步骤&#xff1a;确定网站--搭建关系--发送请求--接受响应--筛选数据--保存本地 1.拿到网站首先要查看我们要爬取的目录是否被允许 一般网站都会议/robots.txt目录&#xff0c;告诉你哪些地址可爬&#xff0c;哪些不可爬&#xff0c;以安全客为例子 2. 首先测试在不登录的…

高项备考葵花宝典-项目进度管理核心概念加强记忆

项目进度管理的核心目标是使项目按时完成。 目录 一、待办事项列表 二、看板方法 三、在制品 四、进度计划模型 五、活动清单 六、里程碑清单 七、前导图法 八、资源日历 九、活动历时估算方法 一、待办事项列表 如上图所示&#xff0c;实际工作中需求往往不是一次性全…

【Linux】find . -perm 644 -exec ls -l {} \;

find . -perm 644 -exec ls -l {} ; find 命令使用 -perm 644 条件来查找文件权限为644的文件&#xff0c;然后通过 -exec ls -l {} \; 将这些文件传递给 ls -l 命令来显示详细的文件列表。 find . -perm 644&#xff1a;在当前目录及其子目录中查找文件权限为644的文件。 -e…