UnityShader 地面全息扫描遮挡物体效果(屏幕空间)

文章介绍了如何在第三人称游戏中实现类似《使命召唤11》中的全息扫描效果,包括从_CameraDepthTexture获取世界坐标,计算距离,渲染扫描区域,以及使用CommandBuffer处理持续显示的目标。通过Shader技术,对屏幕空间进行栅格化和采样,最终叠加生成扫描效果。
摘要由CSDN通过智能技术生成


前段时间刚玩《使命召唤11》的时候发现里面新增了一种很高科技的扫描手榴弹,可以产生一圈类似全息效果的扫描圈,并显示出墙后的敌人,类似这样:

 

最近打算实现一个用在第三人称中类似的效果,如下:

实现方案:

1.根据_CameraDepthTexture计算屏幕空间像素点的世界坐标

2.将扫描发起位置的世界坐标传入shader

3.计算屏幕空间世界坐标到扫描发起位置世界坐标的距离

4.根据相关参数渲染出扫描区域


1._CameraDepthTexture中记录了投影空间的深度信息,通过如下方式可以得到世界坐标:

fixed depth = tex2D(_CameraDepthTexture, i.uv).r;
fixed4 projPos = fixed4(i.uv.x * 2 - 1, i.uv.y * 2 - 1, -depth * 2 + 1, 1);
worldPos = mul(internalCameraToWorld, worldPos);
worldPos /= worldPos.w;

2.计算传入的初始位置和屏幕空间世界坐标距离:

fixed dis = length(internalCentPos.xyz - worldPos.xyz);
 
fixed a = 1 - saturate((abs(dis - internalArg.x) - internalArg.y) / internalArg.z);
a = a * internalFade.x + c * internalFade.y;

最终可以得到如下效果:

 
3.保存上一步的渲染结果,使用CommandBuffer,将需要标记为持续显示的目标(例如敌人)也渲染到该纹理,注意需要判断目标是否在摄像机内,效果如下:

public static void CallRender(Vector3 worldPosition, Renderer[] renderer)
    {
        if (!IsInitialized())
            return;
        if (instance.m_IsShowingEffect)
        {
            if (renderer == null)
                return;
            Vector3 pjpos = instance.m_Camera.worldToCameraMatrix.MultiplyPoint(worldPosition);
            pjpos = instance.m_Camera.projectionMatrix.MultiplyPoint(pjpos);
            if (pjpos.x < -1 || pjpos.x > 1 || pjpos.y < -1 || pjpos.y > 1 || pjpos.z < -1 || pjpos.z > 1)
                return;
            for (int i = 0; i < renderer.Length; i++)
            {
                instance.m_CommandBuffer.DrawRenderer(renderer[i], instance.m_ReplaceMaterial);
            }
        }
    }

 

4.根据屏幕uv信息将屏幕uv栅格化,并计算每个格子中采样到的颜色值,可以得到如下结果:

 

float2 fl = floor(i.uv * _EffectScale);
float dp = tex2D(_PreTex, (fl + float2(0.5, 0.5)) / _EffectScale);
                 
float4 led = tex2D(_EffectTex, i.uv * _EffectScale - fl);
                 
col.rgb += led.rgb*dp;

5.同样根据刚刚栅格的结果,可以计算出每一小格的uv,根据该uv来采样用于作为全息扫描效果的纹理,得到如下结果:

 


6.叠加最终结果:

 


————————————————
原文链接:https://blog.csdn.net/mobilebbki399/article/details/78379758

实现全息投影效果Unity Shader可以分为以下几个步骤: 1. 在场景中创建一个Plane作为投影面,将其设置为不渲染,并且禁用光照。 2. 在Shader中定义一个包含纹理和透明度的材质,用于渲染全息投影。 3. 使用Shader中的顶点着色器将投影面上的顶点位置转换为世界空间中的位置。 4. 在片段着色器中,使用透明度值来控制投影的强度。可以使用噪声纹理来模拟全息投影的扭曲效果。 5. 为了增加真实感,可以在场景中添加适当的光源和阴影。可以使用Unity内置的光源或者自定义的光源。 以下是一个简单的Unity全息投影Shader示例: ``` Shader "Custom/Hologram" { Properties { _MainTex ("Texture", 2D) = "white" {} _Color ("Color", Color) = (1,1,1,1) _Distortion ("Distortion", Range(0.0, 1.0)) = 0.1 } SubShader { Tags {"Queue"="Transparent" "RenderType"="Opaque"} LOD 100 Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag #include "UnityCG.cginc" struct appdata { float4 vertex : POSITION; }; struct v2f { float4 vertex : SV_POSITION; float2 uv : TEXCOORD0; }; sampler2D _MainTex; float4 _MainTex_ST; float4 _Color; float _Distortion; v2f vert (appdata v) { v2f o; o.vertex = UnityObjectToClipPos(v.vertex); o.uv = TRANSFORM_TEX(v.vertex.xy, _MainTex); return o; } fixed4 frag (v2f i) : SV_Target { float4 color = tex2D(_MainTex, i.uv) * _Color; float distance = length(i.vertex); float distortion = _Distortion * (1.0 / distance); float2 distortionUV = i.uv + (distortion * tex2D(_MainTex, i.uv * 10.0).rg - distortion * 0.5); float4 distortionColor = tex2D(_MainTex, distortionUV) * _Color; color = lerp(color, distortionColor, 0.5); color.a = pow(color.a, 2.0); return color; } ENDCG } } FallBack "Diffuse" } ``` 在这个Shader中,我们使用了一个包含纹理和透明度的材质,通过顶点着色器将投影面上的顶点位置转换为世界空间中的位置,然后在片段着色器中使用透明度值来控制投影的强度,并使用噪声纹理来模拟全息投影的扭曲效果。最后,我们使用lerp函数将原始颜色和扭曲颜色混合在一起,并使用pow函数来调整透明度的强度。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值