UnityShader实例06:UV动画

2024-05-28 23:32
文章标签 实例 动画 06 uv unityshader

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

UV动画


UV动画,顾名思义,就是针对UV做的动画。在游戏中,一些动态水面,飞流直下的瀑布,流动的岩浆,跳动的火焰等等,很多都是通过操作UV做的动画。在unity中我可以实用挂载脚本或者直接针对UV key动画帧做动画操作,而在本文中将通过shader编写实现三个比较常见的UV动画方式:

1.UV位移动画

2.UV序列帧动画

3.UV旋转动画


先从UV位移动画开始


我们将做一个流动岩浆的效果,在开始前,我们需要介绍下Unity内置变量_Time

float4 _Time : Time (t/20, t, t*2, t*3)

这是个随时间变化的增量,从函数的定义我们可以知道这个变量的4个分量速率比 _Time=20*_Time.x=_Time.y=_Time.z/2=_Time.w/3.

为了方便控制位移动画的速率和方向我们定义一个变量

_ScrollingSpeed("Scrolling speed", Vector) = (0,0,0,0)

在顶点函数中 将uv坐标乘以 _Time 变量和_ScrollingSpeed,下面为关键代码

o.uvScroll = TRANSFORM_TEX((v.texcoord.xy+_Time.x*_ScrollingSpeed.xy), _Tex);// 这里只使用了xy上的两个分量对应uv

VF版本代码01:

Shader "PengLu/Self-Illumin/IlluminDiffuseScrollVF"   
{  Properties   {   _Color ("Main Color", Color) = (1,1,1,1)  _MainTex ("Base (RGB)", 2D) = "white" {}_Illum ("Illumin (A)", 2D) = "white" {}_Tex("Scroll Tex (RGB)", 2D)= "white" {}_ScrollingSpeed("Scrolling speed", Vector) = (0,0,0,0)}  SubShader   {  Tags { "RenderType"="Opaque" }LOD 200pass  {         	CGPROGRAM  #pragma vertex vert  #pragma fragment frag  #include "UnityCG.cginc"sampler2D _MainTex,_Illum,_Tex;  float4 _MainTex_ST,_Illum_ST,_Tex_ST,_Color,_ScrollingSpeed;struct appdata {  float4 vertex : POSITION;  float2 texcoord : TEXCOORD0;float2 texcoord1 : TEXCOORD1;  };  struct v2f  {  float4 pos : POSITION;  float2 uv_MainTex : TEXCOORD0;float2 uv_Illum : TEXCOORD1; float2 uvLM : TEXCOORD2;float2 uvScroll : TEXCOORD3;};  v2f vert (appdata v) {  v2f o;  o.pos = mul(UNITY_MATRIX_MVP,v.vertex)*_Color;  o.uv_MainTex = TRANSFORM_TEX(v.texcoord, _MainTex);o.uv_Illum = TRANSFORM_TEX(v.texcoord, _Illum);o.uvScroll = TRANSFORM_TEX((v.texcoord.xy+_Time.x*_ScrollingSpeed.xy), _Tex);o.uvLM = v.texcoord1.xy * unity_LightmapST.xy + unity_LightmapST.zw;    return o;  } float4 frag (v2f i) : COLOR  {  float4 texCol = tex2D(_MainTex, i.uv_MainTex);float4 IllumTex = tex2D(_Illum,i.uv_Illum);float3 lm = DecodeLightmap (UNITY_SAMPLE_TEX2D(unity_Lightmap, i.uvLM.xy));    IllumTex.rgb+=tex2D(_Tex,i.uvScroll).rgb;IllumTex*=IllumTex.a;texCol+=IllumTex;texCol.rgb*=lm;return texCol;  }  ENDCG  }  } FallBack "Diffuse" 
}  
VF版本代码01效果:



UV序列帧动画


序列帧动画是游戏比较常用的一种动画形式,在unity自带的粒子系统中就可以设置序列帧动画(下图),但是这个只能用于粒子系统的粒子效果,如果是自己做的模型就要使用序列帧动画就得自己写脚本或shader,下面我们将用shader实现序列帧动画效果。

首先得准备一张序列帧的贴图,如下图这样的:


然后我们需要声明三个变量:

_SizeX ("SizeX", Float) = 4//列数
_SizeY ("SizeY", Float) = 4//行数
_Speed ("Speed", Float) = 200//动画的播放速度

在frag函数里面处理动画

int indexX=fmod (_Time.x*_Speed,_SizeX);//获得列数的循环
int indexY=fmod ((_Time.x*_Speed)/_SizeX,_SizeY);//获得行数的循环

fmod函数是取余函数,int用来强制取整,当然也可以用floor函数来取整


为了直观解释,将用下面的列表列出上面变量的值的对应关系,假设_SizeX=_SizeY=4  


_Time.x*_Speed0123456789101112131415
indexX0123012301230123
indexY0000111122223333




_Time.x*_Speed 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
indexX 0 1 2 3 0 1 2 3 0 1 2 3 0 1 2 3
indexY 0 0 0 0 1 1 1 1 2 2 2 2 3 3 3 3






参考网络上的例子我还有写了另外一种获得循环动画的代码(我是孔乙己啦),和上面两句功能是一样的,貌似消耗稍微少一点
int index = floor(_Time .x * _Speed);
int indexY = index/_SizeX;
int indexX = index - indexY*_SizeX;
通过列表对应,可以看出获取循环数的差别


index 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
indexX 0 1 2 3 0 1 2 3 0 1 2 3 0 1 2 3
indexY 0 0 0 0 1 1 1 1 2 2 2 2 3 3 3 3



index 16 17 18 19 20 21 22 23 24 25 25 27 28 29 30 31
indexX 0 1 2 3 0 1 2 3 0 1 2 3 0 1 2 3
indexY 4 4 4 4 5 5 5 5 6 6 6 6 7 7 7 7





接下来将获得的循环数与uv贴图的UV进行操作

fixed2 seqUV = float2((i.texcoord.x) /_SizeX, (i.texcoord.y)/_SizeY);//将uv切分
seqUV.x += indexX/_SizeX;//U方向上循环
seqUV.y -= indexY/_SizeY;//V方向上循环

VF版本代码02:
Shader "PengLu/Particle/SequenceAdd" {
Properties {_TintColor ("Tint Color", Color) = (0.5,0.5,0.5,0.5)_MainTex ("Particle Texture", 2D) = "white" {}_InvFade ("Soft Particles Factor", Range(0.01,3.0)) = 1.0_SizeX ("SizeX", Float) = 4_SizeY ("SizeY", Float) = 4_Speed ("Speed", Float) = 200
}Category {Tags { "Queue"="Transparent" "IgnoreProjector"="True" "RenderType"="Transparent" }Blend SrcAlpha OneAlphaTest Greater .01ColorMask RGBCull Off Lighting Off ZWrite Off Fog { Color (0,0,0,0) }SubShader {Pass {CGPROGRAM#pragma vertex vert#pragma fragment frag#pragma multi_compile_particles#include "UnityCG.cginc"sampler2D _MainTex;fixed4 _TintColor;fixed _SizeX;fixed _SizeY;fixed _Speed;struct appdata_t {float4 vertex : POSITION;fixed4 color : COLOR;float2 texcoord : TEXCOORD0;};struct v2f {float4 vertex : POSITION;fixed4 color : COLOR;float2 texcoord : TEXCOORD0;#ifdef SOFTPARTICLES_ONfloat4 projPos : TEXCOORD1;#endif};float4 _MainTex_ST;v2f vert (appdata_t v){v2f o;o.vertex = mul(UNITY_MATRIX_MVP, v.vertex);#ifdef SOFTPARTICLES_ONo.projPos = ComputeScreenPos (o.vertex);COMPUTE_EYEDEPTH(o.projPos.z);#endifo.color = v.color;o.texcoord = TRANSFORM_TEX(v.texcoord,_MainTex);return o;}sampler2D _CameraDepthTexture;float _InvFade;fixed4 frag (v2f i) : COLOR{#ifdef SOFTPARTICLES_ONfloat sceneZ = LinearEyeDepth (UNITY_SAMPLE_DEPTH(tex2Dproj(_CameraDepthTexture, UNITY_PROJ_COORD(i.projPos))));float partZ = i.projPos.z;float fade = saturate (_InvFade * (sceneZ-partZ));i.color.a *= fade;#endifint indexX=fmod (_Time.x*_Speed,_SizeX);int indexY=fmod ((_Time.x*_Speed)/_SizeX,_SizeY);//				以下三行代码和之前两行代码功能一样 	
//				int index = floor(_Time .x * _Speed);
//    			int indexY = index/_SizeX;
//				int indexX = index-indexY*_SizeX;fixed2 seqUV = float2((i.texcoord.x) /_SizeX, (i.texcoord.y)/_SizeY);seqUV.x += indexX/_SizeX;seqUV.y -= indexY/_SizeY;return 2.0f * i.color * _TintColor * tex2D(_MainTex, seqUV);}ENDCG }}	
}
}

VF版本代码02效果:


UV旋转动画

UV旋转动画在游戏开发中用得相对比较少,特效师一般会采用其他方式代替,这里将用shader实现一个UV旋转的动画。 UV旋转实际上一个2D旋转,有关2D旋转的理论在 这里 。这边文章里讲得比较透彻,在这里我只需要拿到最终结果公式:

x' = r*cosα*cosθ - r*sinα*sinθ = x * cos θ – y * sin θ

y' = r*sinα*cosθ +r*cosα*sinθ = y * cos θ + x * sin θ


那么开始,同意需要声明一个变量来控制旋转方向和速度;

_Speed ("Speed", Float) = 200

接下来就要在frag函数里面操作UV,关键代码如下:

//将uv偏移0.5,使旋转中心到贴图中心
float2 uv=i.texcoord-0.5;
//定义一个二元变量,存储时间变量的正弦和余弦值
float2 rotate = float2(cos(_Speed*_Time.x),sin(_Speed*_Time.x));
//获得旋转后的uv坐标值
uv=float2((uv.x*rotate.x-uv.y*rotate.y),(uv.x*rotate.y+uv.y*rotate.x));
//将偏移的uv偏移回来
uv+=0.5;


VF版本代码03
Shader "PengLu/Particle/RotationAdd" {
Properties {_TintColor ("Tint Color", Color) = (0.5,0.5,0.5,0.5)_MainTex ("Particle Texture", 2D) = "white" {}_InvFade ("Soft Particles Factor", Range(0.01,3.0)) = 1.0_Speed ("Speed", Float) = 200
}Category {Tags { "Queue"="Transparent" "IgnoreProjector"="True" "RenderType"="Transparent" }Blend SrcAlpha OneAlphaTest Greater .01ColorMask RGBCull Off Lighting Off ZWrite Off Fog { Color (0,0,0,0) }SubShader {Pass {CGPROGRAM#pragma vertex vert#pragma fragment frag#pragma multi_compile_particles#include "UnityCG.cginc"sampler2D _MainTex;fixed4 _TintColor;fixed _Speed;struct appdata_t {float4 vertex : POSITION;fixed4 color : COLOR;float2 texcoord : TEXCOORD0;};struct v2f {float4 vertex : POSITION;fixed4 color : COLOR;float2 texcoord : TEXCOORD0;#ifdef SOFTPARTICLES_ONfloat4 projPos : TEXCOORD1;#endif};float4 _MainTex_ST;v2f vert (appdata_t v){v2f o;o.vertex = mul(UNITY_MATRIX_MVP, v.vertex);#ifdef SOFTPARTICLES_ONo.projPos = ComputeScreenPos (o.vertex);COMPUTE_EYEDEPTH(o.projPos.z);#endifo.color = v.color;o.texcoord = TRANSFORM_TEX(v.texcoord,_MainTex);return o;}sampler2D _CameraDepthTexture;float _InvFade;fixed4 frag (v2f i) : COLOR{#ifdef SOFTPARTICLES_ONfloat sceneZ = LinearEyeDepth (UNITY_SAMPLE_DEPTH(tex2Dproj(_CameraDepthTexture, UNITY_PROJ_COORD(i.projPos))));float partZ = i.projPos.z;float fade = saturate (_InvFade * (sceneZ-partZ));i.color.a *= fade;#endiffloat2 uv=i.texcoord-0.5;float2 rotate = float2(cos(_Speed*_Time.x),sin(_Speed*_Time.x));uv=float2((uv.x*rotate.x-uv.y*rotate.y),(uv.x*rotate.y+uv.y*rotate.x));uv+=0.5;return 2.0f * i.color * _TintColor * tex2D(_MainTex, uv);}ENDCG }}	
}
}

VF版本代码03效果:



PS:贴图的Wrap Mode要设置成Clamp,原因你懂的 ^_^



这篇关于UnityShader实例06:UV动画的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

Flutter 进阶:绘制加载动画

绘制加载动画:由小圆组成的大圆 1. 定义 LoadingScreen 类2. 实现 _LoadingScreenState 类3. 定义 LoadingPainter 类4. 总结 实现加载动画 我们需要定义两个类:LoadingScreen 和 LoadingPainter。LoadingScreen 负责控制动画的状态,而 LoadingPainter 则负责绘制动画。

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

实例:如何统计当前主机的连接状态和连接数

统计当前主机的连接状态和连接数 在 Linux 中,可使用 ss 命令来查看主机的网络连接状态。以下是统计当前主机连接状态和连接主机数量的具体操作。 1. 统计当前主机的连接状态 使用 ss 命令结合 grep、cut、sort 和 uniq 命令来统计当前主机的 TCP 连接状态。 ss -nta | grep -v '^State' | cut -d " " -f 1 | sort |

用Unity2D制作一个人物,实现移动、跳起、人物静止和动起来时的动画:中(人物移动、跳起、静止动作)

上回我们学到创建一个地形和一个人物,今天我们实现一下人物实现移动和跳起,依次点击,我们准备创建一个C#文件 创建好我们点击进去,就会跳转到我们的Vision Studio,然后输入这些代码 using UnityEngine;public class Move : MonoBehaviour // 定义一个名为Move的类,继承自MonoBehaviour{private Rigidbo

Java Websocket实例【服务端与客户端实现全双工通讯】

Java Websocket实例【服务端与客户端实现全双工通讯】 现很多网站为了实现即时通讯,所用的技术都是轮询(polling)。轮询是在特定的的时间间隔(如每1秒),由浏览器对服务器发 出HTTP request,然后由服务器返回最新的数据给客服端的浏览器。这种传统的HTTP request 的模式带来很明显的缺点 – 浏 览器需要不断的向服务器发出请求,然而HTTP

828华为云征文|华为云Flexus X实例docker部署rancher并构建k8s集群

828华为云征文|华为云Flexus X实例docker部署rancher并构建k8s集群 华为云最近正在举办828 B2B企业节,Flexus X实例的促销力度非常大,特别适合那些对算力性能有高要求的小伙伴。如果你有自建MySQL、Redis、Nginx等服务的需求,一定不要错过这个机会。赶紧去看看吧! 什么是华为云Flexus X实例 华为云Flexus X实例云服务是新一代开箱即用、体

前端-06-eslint9大变样后,如何生成旧版本的.eslintrc.cjs配置文件

目录 问题解决办法 问题 最近在写一个vue3+ts的项目,看了尚硅谷的视频,到了配置eslintrc.cjs的时候我犯了难,因为eslint从9.0之后重大更新,跟以前完全不一样,但是我还是想用和老师一样的eslintrc.cjs文件,该怎么做呢? 视频链接:尚硅谷Vue项目实战硅谷甄选,vue3项目+TypeScript前端项目一套通关 解决办法 首先 eslint 要