2.8Flowmap的实现

2024-06-01 09:12
文章标签 实现 2.8 flowmap

本文主要是介绍2.8Flowmap的实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、Flowmap 是什么

半条命2中水的流动

求生之路2中的水的流动

这种方式原理简单,容易实现,运算量少,如今也还在使用

1.flowmap的实质

Flow map(流向图) ,一张记录了2D向量信息的纹理,Flow map上的颜色(通常为RG通道)记录该处向量场的方向,让模型上某一点表现出定量流动的特征。

通过在shader中偏移uv再对纹理进行采样,来模拟流动效果。

右边向量场的方向于中间Flowmap和它的方向是不一致的,这是为什么呢?

2.前置了解:UV映射

UV坐标用于查找纹理的颜色值:

UV贴图(用于理解):使用(R,G)颜色通道表示坐标:

黑色(0,0),绿色处(0,1),红色处(1,0),黄色处(1,1)

如果我们要对一个矩形进行纹理查找的话,我们首先会用到它的UV坐标

左边为unity中的UV坐标 与我们熟知的xy坐标轴相似

用该UV坐标去查找右边这张贴图的颜色值,我们将会得到和原贴图一摸一样的结果

如果让更改UV坐标,让每一列都拥有相同的UV值再去采样,则会得到右图这样的条纹图结果

若使整个采样的时候用的UV值是同一个的话,如图我们将该UV坐标的UV值全都变为黑色,那么结果就会对左下角进行采样

UV贴图上颜色相同的地方就意味着它在采样纹理的时候使用了同一个位置

Flowmap则是通过它上面所带有的向量场的信息对UV进行一个偏移之后,去干扰我们采样纹理时候的过程,如上图

注意在UE4中,UV坐标是不同的,与unity相比是将绿通道反转了。所以使用的Flowmap也会发生变化,要根据引擎进行调整

偏转后的UV坐标去采样右图后得到下面扭曲贴图

2.为什么要使用flowmap?

原因:

类似UV动画,而非顶点动画。换言之,无需对模型顶点进行操作,易实现,运算开销小。

不仅仅是水面,任何和流动相关的效果都可以采用flowmap。

flowmap不仅仅被用于制作一些侵蚀效果或表面的流动效果,还被用于制作流动的天空球中

二、Flowmap shader

1. 借助Shader Graph与desmos理解flowmap

  1. 采样Flow map获取向量场信息
  2. 用向量场信息,使采样贴图时的UV随时间变化
  3. 对同一贴图以半个周期的相位差采集两次,并线性插值,使贴图流动连续

2. 在shader中实现flowmap

  • 目标:移。根据flowmap上的值,使纹理随时间偏
  • 最简单的随时间偏移:UV - time
  • 为什么是相减:
  1. 先来看看 uv+time 的情况
  2. (u,v) + (time,0) :模型上某个点: 随着time增加,采样到的像素越远,
  3. 视觉上可以形容为:更远距离的像素偏移向该点,视觉效果和我们直观认识到的运算法则是相反的。
  4. UV值作为向量(u,v),自然也遵循向量的运算法则。但UV偏移时,改变的不是顶点的位置。  

  5. 单方向运动X

    由flow map获取流动方向√

  6. flow map不能直接使用

    将flow map上的色值从[0,1]的范围映射到方向向量的范围[-1,1]即乘2减1

从flowmap获取需要的流动方向,再乘Time就可以让某个点去根据flowmap进行流动

调整采样时的UV为: adjust_uv = uv - flowDir * time

在这里我们用FlowSpeed来控制向量场的强度

但随着时间的增加,扭曲程度会越来越大,所以需要一个Fraction函数把时间变成[0,1]的循环 把它变成一个三角波函数

为了解决这一问题,我们需要构造两层采样相差半个周期的采样,再对他们进行插值混合

随着时间进行,变形越来越夸张,为了把偏移控制在一定范围内:

解决frac产生的跳变,把上面改成:

我们希望的流动:无缝循环

用相位差半个周期的两层采样进行加权混合,使纹理流动一个周期重新开始时的不自然情况被另一层采样覆盖

用flowmap修改法线贴图

三、Flowmap的制作

这篇关于2.8Flowmap的实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot实现多环境配置文件切换

《SpringBoot实现多环境配置文件切换》这篇文章主要为大家详细介绍了如何使用SpringBoot实现多环境配置文件切换功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 示例代码结构2. pom文件3. application文件4. application-dev文

Python FastAPI实现JWT校验的完整指南

《PythonFastAPI实现JWT校验的完整指南》在现代Web开发中,构建安全的API接口是开发者必须面对的核心挑战之一,本文将深入探讨如何基于FastAPI实现JWT(JSONWebToken... 目录一、JWT认证的核心原理二、项目初始化与环境配置三、安全密码处理机制四、JWT令牌的生成与验证五、

Python使用Turtle实现精确计时工具

《Python使用Turtle实现精确计时工具》这篇文章主要为大家详细介绍了Python如何使用Turtle实现精确计时工具,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录功能特点使用方法程序架构设计代码详解窗口和画笔创建时间和状态显示更新计时器控制逻辑计时器重置功能事件

Linux给磁盘扩容(LVM方式)的方法实现

《Linux给磁盘扩容(LVM方式)的方法实现》本文主要介绍了Linux给磁盘扩容(LVM方式)的方法实现,涵盖PV/VG/LV概念及操作步骤,具有一定的参考价值,感兴趣的可以了解一下... 目录1 概念2 实战2.1 相关基础命令2.2 开始给LVM扩容2.3 总结最近测试性能,在本地打数据时,发现磁盘空

Golang实现Redis分布式锁(Lua脚本+可重入+自动续期)

《Golang实现Redis分布式锁(Lua脚本+可重入+自动续期)》本文主要介绍了Golang分布式锁实现,采用Redis+Lua脚本确保原子性,持可重入和自动续期,用于防止超卖及重复下单,具有一定... 目录1 概念应用场景分布式锁必备特性2 思路分析宕机与过期防止误删keyLua保证原子性可重入锁自动

golang 对象池sync.Pool的实现

《golang对象池sync.Pool的实现》:本文主要介绍golang对象池sync.Pool的实现,用于缓存和复用临时对象,以减少内存分配和垃圾回收的压力,下面就来介绍一下,感兴趣的可以了解... 目录sync.Pool的用法原理sync.Pool 的使用示例sync.Pool 的使用场景注意sync.

IDEA实现回退提交的git代码(四种常见场景)

《IDEA实现回退提交的git代码(四种常见场景)》:本文主要介绍IDEA实现回退提交的git代码(四种常见场景),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.已提交commit,还未push到远端(Undo Commit)2.已提交commit并push到

Kotlin Compose Button 实现长按监听并实现动画效果(完整代码)

《KotlinComposeButton实现长按监听并实现动画效果(完整代码)》想要实现长按按钮开始录音,松开发送的功能,因此为了实现这些功能就需要自己写一个Button来解决问题,下面小编给大... 目录Button 实现原理1. Surface 的作用(关键)2. InteractionSource3.

java对接第三方接口的三种实现方式

《java对接第三方接口的三种实现方式》:本文主要介绍java对接第三方接口的三种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录HttpURLConnection调用方法CloseableHttpClient调用RestTemplate调用总结在日常工作

golang中slice扩容的具体实现

《golang中slice扩容的具体实现》Go语言中的切片扩容机制是Go运行时的一个关键部分,它确保切片在动态增加元素时能够高效地管理内存,本文主要介绍了golang中slice扩容的具体实现,感兴趣... 目录1. 切片扩容的触发append 函数的实现2. runtime.growslice 函数gro