服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - C#使用Shader实现夜幕降临倒计时的效果

C#使用Shader实现夜幕降临倒计时的效果

2021-12-08 13:19凯奥斯 C#

这篇文章主要介绍了C#使用Shader实现夜幕降临倒计时的效果,非常不错具有参考借鉴价值,需要的朋友可以参考下

最近火爆全球的pc游戏battlerite(战争仪式)在倒计时的会生成一种类似夜幕降临的效果,会以战场中心为圆心,某个长度为半径的范围外是暗的,而这个半径会逐渐缩小,而圆之外的阴暗部分是附着地形的,本文就尝试使用屏幕后处理的手段来实现这种效果。

(暂时缺少battlerite的截图,稍后会补上)

首先看效果图:

C#使用Shader实现夜幕降临倒计时的效果

注:本文参考了tasharen fog of war插件

创建一个c#脚本,命名为nightfall.cs,为nightfall类创建一些公共变量(nightcolor,center和radius),另外还需要一个nightfall.shader。

首先,我们要确定这个效果是在场景渲染之后还未送到屏幕显示之前的实现的,所以,nightfall脚本是要挂载到主camera上的(添加特性[requirecomponent(typeof(camera))]),并要实现onrenderimage方法。

其次,在onrenderimage方法里,我们最终需要调用graphics.blit方法,而这个方法的第三个参数是material类型,所以我们需要在代码里创建一个临时材质,这个材质使用了nightfall.shader。

再次,我们需要在shader里面将屏幕坐标转换为世界坐标,来计算与世界中心的坐标,所以我们需要mvp的逆矩阵(参考shader山下(十六)坐标空间与转换矩阵)。

最后,为了附着地形,我们需要在shader计算深度,也就是坐标点与摄像机的相对距离,所以需要摄像机的位置。

c#的代码:

using unityengine;

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
[requirecomponent(typeof(camera))]
public class nightfall : monobehaviour
{
public shader shader;
public color nightcolor = new color(0.05f, 0.05f, 0.05f, 0.5f);
public vector3 center = vector3.zero;
public float radius = 10;
camera mcam;
matrix4x4 minversemvp;
material mmat;
/// the camera we're working with needs depth.
void onenable ()
{
mcam = getcomponent<camera>();
mcam.depthtexturemode = depthtexturemode.depth;
if (shader == null) shader = shader.find("image effects/nightfall");
}
/// destroy the material when disabled.
void ondisable () { if (mmat) destroyimmediate(mmat); }
/// automatically disable the effect if the shaders don't support it.
void start ()
{
if (!systeminfo.supportsimageeffects || !shader || !shader.issupported)
{
enabled = false;
}
}
// called by camera to apply image effect
void onrenderimage (rendertexture source, rendertexture destination)
{
print (nightcolor);
print (destination);
// calculate the inverse modelview-projection matrix to convert screen coordinates to world coordinates
minversemvp = (mcam.projectionmatrix * mcam.worldtocameramatrix).inverse;
if (mmat == null)
{
mmat = new material(shader);
mmat.hideflags = hideflags.hideanddontsave;
}
vector4 campos = mcam.transform.position;
// this accounts for anti-aliasing on windows flipping the depth uv coordinates.
// despite the official documentation, the following approach simply doesn't work:
// http://docs.unity3d.com/documentation/components/sl-platformdifferences.html
if (qualitysettings.antialiasing > 0)
{
runtimeplatform pl = application.platform;
if (pl == runtimeplatform.windowseditor ||
pl == runtimeplatform.windowsplayer ||
pl == runtimeplatform.windowswebplayer)
{
campos.w = 1f;
}
}
mmat.setvector("_campos", campos);
mmat.setmatrix("_inversemvp", minversemvp);
mmat.setcolor("_nightcolor", nightcolor);
mmat.setvector ("_center", center);
mmat.setfloat ("_radius", radius);
graphics.blit(source, destination, mmat);
}
}

shader代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
shader "image effects/nightfall"
{
properties
{
_nightcolor ("night color", color) = (0.05, 0.05, 0.05, 0.05)
_center ("center", vector) = (0,0,0,0)
_radius ("radius", float) = 10
}
subshader
{
pass
{
ztest always
cull off
zwrite off
fog { mode off }
blend srcalpha oneminussrcalpha
cgprogram
#pragma vertex vert_img
#pragma fragment frag vertex:vert
#pragma fragmentoption arb_precision_hint_fastest
#include "unitycg.cginc"
sampler2d _cameradepthtexture;
uniform float4x4 _inversemvp;
uniform float4 _campos;
uniform half4 _nightcolor;
uniform half4 _center;
uniform half _radius;
struct input
{
float4 position : position;
float2 uv : texcoord0;
};
void vert (inout appdata_full v, out input o)
{
o.position = mul(unity_matrix_mvp, v.vertex);
o.uv = v.texcoord.xy;
}
float3 camtoworld (in float2 uv, in float depth)
{
float4 pos = float4(uv.x, uv.y, depth, 1.0);
pos.xyz = pos.xyz * 2.0 - 1.0;
pos = mul(_inversemvp, pos);
return pos.xyz / pos.w;
}
fixed4 frag (input i) : color
{
#if shader_api_d3d9 || shader_api_d3d11
float2 depthuv = i.uv;
depthuv.y = lerp(depthuv.y, 1.0 - depthuv.y, _campos.w);
float depth = unity_sample_depth(tex2d(_cameradepthtexture, depthuv));
float3 pos = camtoworld(depthuv, depth);
#else
float depth = unity_sample_depth(tex2d(_cameradepthtexture, i.uv));
float3 pos = camtoworld(i.uv, depth);
#endif
// limit to sea level
if (pos.y < 0.0)
{
// this is a simplified version of the ray-plane intersection formula: t = -( n.o + d ) / ( n.d )
float3 dir = normalize(pos - _campos.xyz);
pos = _campos.xyz - dir * (_campos.y / dir.y);
}
half4 col;
float dis = length(pos.xz - _center.xz);
if (dis < _radius)
{
col = fixed4(0,0,0,0);
}
else
{
col = _nightcolor;
}
return col;
}
endcg
}
}
fallback off
}

需要说明的几个点:

1、因为平台差异性,为了兼容direct3d,所以在c#和shader里通过campos(_campos)的w分量来调整uv坐标。

2、这里虽然没有声明_maintex,但是_maintex实际上就是即将成像的屏幕图像,所以这里的i.uv也就是指屏幕图像的纹理坐标。

3、_cameradepthtexture是摄像机的深度纹理,通过unity_sample_depth方法获取深度。

4、camtoworld里面,先是根据uv坐标和深度depth创建了一个float4的坐标值pos,然后对pos乘2减1是将这个坐标范围从[0,1]转换到了[-1,1],对应世界坐标。然后使用传入的mvp逆矩阵_inversemvp乘以这个坐标值,就得到了屏幕点的世界坐标。最后将pos的xyz分量除以w分量,这里w分量表示因为远近而产生的缩放值。

5、在计算过世界坐标之后,对于y小于0的坐标要做一下处理,将效果限制在海平面(sea level)之上,使用射线平面相交方程(ray-plane intersection formula)的简化版本来处理。

6、最后根据距离返回色彩值。

如果要实现夜幕降临倒计时的效果,只需要在控制脚本(c#)中获取camera上的nightfall组件,根据时间修改radius变量即可。

以上所述是小编给大家介绍的c#使用shader实现夜幕降临倒计时的效果,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对服务器之家网站的支持!

原文链接:http://blog.csdn.net/ecidevilin/article/details/52930701

延伸 · 阅读

精彩推荐
  • C#C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    这篇文章主要介绍了C#设计模式之Strategy策略模式解决007大破密码危机问题,简单描述了策略模式的定义并结合加密解密算法实例分析了C#策略模式的具体使用...

    GhostRider10972022-01-21
  • C#深入理解C#的数组

    深入理解C#的数组

    本篇文章主要介绍了C#的数组,数组是一种数据结构,详细的介绍了数组的声明和访问等,有兴趣的可以了解一下。...

    佳园9492021-12-10
  • C#利用C#实现网络爬虫

    利用C#实现网络爬虫

    这篇文章主要介绍了利用C#实现网络爬虫,完整的介绍了C#实现网络爬虫详细过程,感兴趣的小伙伴们可以参考一下...

    C#教程网11852021-11-16
  • C#SQLite在C#中的安装与操作技巧

    SQLite在C#中的安装与操作技巧

    SQLite,是一款轻型的数据库,用于本地的数据储存。其优点有很多,下面通过本文给大家介绍SQLite在C#中的安装与操作技巧,感兴趣的的朋友参考下吧...

    蓝曈魅11162022-01-20
  • C#如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    这篇文章主要给大家介绍了关于如何使用C#将Tensorflow训练的.pb文件用在生产环境的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴...

    bbird201811792022-03-05
  • C#三十分钟快速掌握C# 6.0知识点

    三十分钟快速掌握C# 6.0知识点

    这篇文章主要介绍了C# 6.0的相关知识点,文中介绍的非常详细,通过这篇文字可以让大家在三十分钟内快速的掌握C# 6.0,需要的朋友可以参考借鉴,下面来...

    雨夜潇湘8272021-12-28
  • C#C#微信公众号与订阅号接口开发示例代码

    C#微信公众号与订阅号接口开发示例代码

    这篇文章主要介绍了C#微信公众号与订阅号接口开发示例代码,结合实例形式简单分析了C#针对微信接口的调用与处理技巧,需要的朋友可以参考下...

    smartsmile20127762021-11-25
  • C#VS2012 程序打包部署图文详解

    VS2012 程序打包部署图文详解

    VS2012虽然没有集成打包工具,但它为我们提供了下载的端口,需要我们手动安装一个插件InstallShield。网上有很多第三方的打包工具,但为什么偏要使用微软...

    张信秀7712021-12-15