UnityShader实例11:积雪材质

2024-05-28 23:32

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

积雪材质



概述


积雪材质是我自己给这个材质取的名字,既然是积雪,那顾名思义,雪是从天而降的,因此积雪都是在物体朝上的表面;不管你的模型怎么摆放 ,雪都保证是积在物体朝上(在unity里就是y轴正方向)的表面,如下图所示:



实现原理



要保证向上的面有积雪,其实就是模型表面的法线方向与世界坐标空间的Y轴正方向保持一致积雪多,否则积雪就少雪,所以将模型的法线方向normal从模型空间转化到世界空间,然后与y轴正方向float3(0,1,0)做点积,根据结果来确认法相的朝向和Y轴的夹角大小,从而确认是否积雪;这个和之前的边缘光材质是一样的道理,不同的是,那个是考虑模型法线与视线方向的夹角;


Shader代码实现


终于到代码实现了,想想有点小激动,嘿嘿。鉴于vf写灯光比较复杂,本例的材质是没有实现灯光支持的,废话不多说,先从属性定义开始,至少需要两张贴图,然后定义一个参数控制雪的数量:

Properties {_MainTex ("Base (RGB)", 2D) = "white" {}_SnowTex ("SnowTexture(RGB)",2D) = "white"{}_SnowCount ("SnowCount", Range (0.0, 1)) = 0.078
}

定义输入结构体和输出结构体;

			struct appdata_t {float4 vertex : POSITION;float2 texcoord : TEXCOORD0;float3 normal : NORMAL;};struct v2f {float4 vertex : SV_POSITION;half2 texcoord : TEXCOORD0;half2 snowUV : TEXCOORD1  ;fixed4 snow : COLOR;//定义一个四元数用来传递将积雪位置到frag函数UNITY_FOG_COORDS(1)};

接下来是比较关键的vert函数部分,在这里获得模型的法线,并将法线方向转换到世界空间与Y方向轴做点积,确定积雪的面,关键代码如下:

float3 worldNormal = mul((float3x3)_Object2World ,v.normal );//将法线转换到世界空间
float rim = 1-saturate(dot(float3(0,1,0),worldNormal ));//将世界空间的法线方向与Y轴做点积运算
o.texcoord = TRANSFORM_TEX(v.texcoord, _MainTex);
o.snowUV = TRANSFORM_TEX(v.texcoord , _SnowTex);
o.snow = pow(rim,_SnowCount*64);//控制积雪的量,这个值将传入到frag函数里面用来混合两张贴图

Frag函数比较简单,用vert传入的值i.snow将两张贴图混合,做出积雪的效果。下面是完整代码

VF版本代码01
Shader "PengLu/Unlit/SnowVF" {
Properties {_MainTex ("Base (RGB)", 2D) = "white" {}_SnowTex ("SnowTexture(RGB)",2D) = "white"{}_SnowCount ("SnowCount", Range (0.0, 1)) = 0.078
}SubShader {Tags { "RenderType"="Opaque" }LOD 100Pass {  CGPROGRAM#pragma vertex vert#pragma fragment frag#pragma multi_compile_fog#include "UnityCG.cginc"struct appdata_t {float4 vertex : POSITION;float2 texcoord : TEXCOORD0;float3 normal : NORMAL;};struct v2f {float4 vertex : SV_POSITION;half2 texcoord : TEXCOORD0;half2 snowUV : TEXCOORD1  ;fixed4 snow : COLOR;UNITY_FOG_COORDS(1)};sampler2D _MainTex;sampler2D _SnowTex;float4 _MainTex_ST,_SnowTex_ST;float _SnowCount;v2f vert (appdata_t v){v2f o;o.vertex = mul(UNITY_MATRIX_MVP, v.vertex);float3 worldNormal = mul((float3x3)_Object2World ,v.normal );float rim = 1-saturate(dot(float3(0,1,0),worldNormal ));o.texcoord = TRANSFORM_TEX(v.texcoord, _MainTex);o.snowUV = TRANSFORM_TEX(v.texcoord , _SnowTex);o.snow = pow(rim,_SnowCount*64);								UNITY_TRANSFER_FOG(o,o.vertex);return o;}fixed4 frag (v2f i) : SV_Target{fixed4 col = tex2D(_MainTex, i.texcoord);fixed4 snow = tex2D(_SnowTex,i.snowUV);col = lerp(snow,col,saturate(i.snow));UNITY_APPLY_FOG(i.fogCoord, col);UNITY_OPAQUE_ALPHA(col.a);return col;}ENDCG}
}}


带法线贴图效果的实现


上面的代码由于是直接由顶点的法线计算,所以可以看到积雪的效果还是有点粗糙,因此可以考虑用法线贴图来事的积雪的效果更加精确和细腻一些。同样这个shader并没有实现光照的效果,还是一个Unlit材质;原理差不多,因此不做过多的解析,完整代码即效果如下:

VF版本代码02:


Shader "PengLu/Unlit/SnowBumpVF" {Properties {_MainTex ("Base (RGB)", 2D) = "white" {}_BumpMap ("BaseBump", 2D) = "bump" {}_SnowTex ("SnowTexture(RGB)",2D) = "white"{}_SnowCount ("SnowCount", Range (0.01, 1)) = 0.078
}SubShader {Tags { "RenderType"="Opaque" }LOD 200Pass {  CGPROGRAM#pragma vertex vert#pragma fragment frag#pragma multi_compile_fog#include "UnityCG.cginc"struct v2f {float4 vertex : SV_POSITION;half2 texcoord : TEXCOORD0;half2 snowUV : TEXCOORD1  ;// fixed4 snow : COLOR;float3 tangent:TEXCOORD2;float3 binormal : TEXCOORD3;float3 normal : TEXCOORD4;UNITY_FOG_COORDS(1)};sampler2D _MainTex,_BumpMap;sampler2D _SnowTex;float4 _MainTex_ST,_SnowTex_ST;float _SnowCount;v2f vert (appdata_full v){v2f o;o.vertex = mul(UNITY_MATRIX_MVP, v.vertex);o.tangent = v.tangent.xyz;o.normal = v.normal;o.binormal=cross(v.normal,v.tangent.xyz)*v.tangent.w;o.texcoord = TRANSFORM_TEX(v.texcoord, _MainTex);o.snowUV = TRANSFORM_TEX(v.texcoord , _SnowTex);UNITY_TRANSFER_FOG(o,o.vertex);return o;}fixed4 frag (v2f i) : SV_Target{fixed4 col = tex2D(_MainTex, i.texcoord);fixed4 snow = tex2D(_SnowTex,i.snowUV);float3x3 rotation=float3x3 (i.tangent.xyz,i.binormal,i.normal);float3 N = UnpackNormal(tex2D(_BumpMap,i.texcoord));N=normalize(mul(N,rotation));N = mul((float3x3)_Object2World ,N );float rim = 1-saturate(dot(float3(0,1,0),N));float4 lerpsnow = pow(rim,_SnowCount*16);col = lerp(snow,col,saturate(lerpsnow));UNITY_APPLY_FOG(i.fogCoord, col);UNITY_OPAQUE_ALPHA(col.a);return col;}ENDCG}
}}


VF版本代码02效果:




这篇关于UnityShader实例11:积雪材质的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

【机器学习】高斯过程的基本概念和应用领域以及在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

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 |

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实例云服务是新一代开箱即用、体

LLVM入门2:如何基于自己的代码生成IR-LLVM IR code generation实例介绍

概述 本节将通过一个简单的例子来介绍如何生成llvm IR,以Kaleidoscope IR中的例子为例,我们基于LLVM接口构建一个简单的编译器,实现简单的语句解析并转化为LLVM IR,生成对应的LLVM IR部分,代码如下,文件名为toy.cpp,先给出代码,后面会详细介绍每一步分代码: #include "llvm/ADT/APFloat.h"#include "llvm/ADT/S

OpenStack离线Train版安装系列—11.5实例使用-Cinder存储服务组件

本系列文章包含从OpenStack离线源制作到完成OpenStack安装的全部过程。 在本系列教程中使用的OpenStack的安装版本为第20个版本Train(简称T版本),2020年5月13日,OpenStack社区发布了第21个版本Ussuri(简称U版本)。 OpenStack部署系列文章 OpenStack Victoria版 安装部署系列教程 OpenStack Ussuri版

OpenStack实例操作选项解释:启动和停止instance实例

关于启动和停止OpenStack实例 如果你想要启动和停止OpenStack实例时,有四种方法可以考虑。 管理员可以暂停、挂起、搁置、停止OpenStack 的计算实例。但是这些方法之间有什么不同之处? 目录 关于启动和停止OpenStack实例1.暂停和取消暂停实例2.挂起和恢复实例3.搁置(废弃)实例和取消废弃实例4.停止(删除)实例 1.暂停和取消暂停实例

Cmake之3.0版本重要特性及用法实例(十三)

简介: CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布:《Android系统多媒体进阶实战》🚀 优质专栏: Audio工程师进阶系列【原创干货持续更新中……】🚀 优质专栏: 多媒体系统工程师系列【原创干货持续更新中……】🚀 优质视频课程:AAOS车载系统+AOSP14系统攻城狮入门视频实战课 🚀 人生格言: 人生从来没有捷径,只有行动才是治疗恐惧

实例demo理解面向接口思想

浅显的理解面向接口编程 Android开发的语言是java,至少目前是,所以理解面向接口的思想是有必要的。下面通过一个简单的例子来理解。具体的概括我也不知道怎么说。 例子: 现在我们要开发一个应用,模拟移动存储设备的读写,即计算机与U盘、MP3、移动硬盘等设备进行数据交换。已知要实现U盘、MP3播放器、移动硬盘三种移动存储设备,要求计算机能同这三种设备进行数据交换,并且以后可能会有新的第三方的