深入URP之Shader篇12: 深度值专题(3)

这篇具有很好参考价值的文章主要介绍了深入URP之Shader篇12: 深度值专题(3)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

深度图及其应用

前两篇介绍了深度相关的一些知识,和URP Shader中提供的函数。而我们处理的深度一般会来自于深度图,本篇介绍URP Shader中深度图的使用。

深度图的生成

URP中有两种方式可以生成深度图。

  • 在情况允许的情况下,URP会在渲染完不透明物体之后使用一个Copy Depth Pass将场景的深度copy到深度图_CameaDepthTexture中。
  • 如果情况不允许直接copy,则会增加一个额外的PreDepthPass,这个不是就一个pass,而是一组pass,对于所有shader中包含DepthOnlyPass的物体都使用DepthOnlyPass绘制一遍深度到_CameraDepthTexture中。
    看起来,PreDetphPass麻烦很多,虽然每次绘制只会更新depth texture,但是毕竟增加了很多draw call。那么什么情况不允许直接Copy Depth呢?这个之前分析Depth Only Pass时已经说过了。

Shader中定义深度图

深度图的定义在Packages\com.unity.render-pipelines.universal\ShaderLibrary\DeclareDepthTexture.hlsl中:

TEXTURE2D_X_FLOAT(_CameraDepthTexture);
SAMPLER(sampler_CameraDepthTexture);

float SampleSceneDepth(float2 uv)
{
    return SAMPLE_TEXTURE2D_X(_CameraDepthTexture, sampler_CameraDepthTexture, UnityStereoTransformScreenSpaceTex(uv)).r;
}

深度图的采样

使用上面的SampleSceneDepth函数,内部使用了一个宏SAMPLE_TEXTURE2D_X,这个宏封装了VR的使用,对于正常渲染就是SAMPLE_TEXTURE2D,传入texture, sampler和uv坐标,得到该uv处的texel。由于现在的Unity版本已经全平台支持深度纹理格式了,所以深度图里面存的就是深度,不需要再decode了。(之前有些平台不支持深度纹理,会将深度值encode到普通纹理的颜色通道中,因此采样后需要decode)

深度图的应用

软粒子

普通的粒子面片和背景的交叉没有过渡,会显得很硬。比如早期CS游戏中的烟雾弹,烟雾扩散时可以看到半透明的片插入地面和墙面。而软粒子效果是指根据粒子的深度和背景上已有像素的深度计算出一个alpha值,用来做alpha混合,这样粒子效果就比较柔和了。
看一下URP中实现软粒子的shader代码:

// Soft particles - returns alpha value for fading particles based on the depth to the background pixel
float SoftParticles(float near, float far, float4 projection)
{
    float fade = 1;
    if (near > 0.0 || far > 0.0)
    {
        float sceneZ = LinearEyeDepth(SAMPLE_TEXTURE2D_X(_CameraDepthTexture, sampler_CameraDepthTexture, UnityStereoTransformScreenSpaceTex(projection.xy / projection.w)).r, _ZBufferParams);
        float thisZ = LinearEyeDepth(projection.z / projection.w, _ZBufferParams);
        fade = saturate(far * ((sceneZ - near) - thisZ));
    }
    return fade;
}
  • 大部分都是我们看过的,先使用SAMPLE_TEXTURE2D_X采样_CameraDepthTexture获取到场景图中当前位置的深度,采样使用的uv坐标是projection.xy / projection.w,因为projection是clip space坐标,除w后变成NDC坐标,Unity内部处理为0~1之间,这就是屏幕空间坐标了。然后使用LinearEyeDepth转成view space的线性深度值。
  • 然后计算当前这个粒子的片段的线性深度值,使用projection.z / projection.w得到NDC的深度,然后转成view space的线性深度。
  • 这之后使用sceneZthisZ和near,far计算出alpha值。那么这是什么意思?near和far又是什么?找一下调用的地方看下:
 #if defined(_SOFTPARTICLES_ON)
     ALBEDO_MUL *= SoftParticles(SOFT_PARTICLE_NEAR_FADE, SOFT_PARTICLE_INV_FADE_DISTANCE, projectedPosition);
 #endif

找一下这两个宏的定义:

#define SOFT_PARTICLE_NEAR_FADE _SoftParticleFadeParams.x
#define SOFT_PARTICLE_INV_FADE_DISTANCE _SoftParticleFadeParams.y

再找一个_SoftParticleFadeParams赋值的地方:

            // Soft particles
            var useSoftParticles = false;
            if (material.HasProperty("_SoftParticlesEnabled"))
            {
                useSoftParticles = (material.GetFloat("_SoftParticlesEnabled") > 0.0f && isTransparent);
                if (useSoftParticles)
                {
                    var softParticlesNearFadeDistance = material.GetFloat("_SoftParticlesNearFadeDistance");
                    var softParticlesFarFadeDistance = material.GetFloat("_SoftParticlesFarFadeDistance");
                    // clamp values
                    if (softParticlesNearFadeDistance < 0.0f)
                    {
                        softParticlesNearFadeDistance = 0.0f;
                        material.SetFloat("_SoftParticlesNearFadeDistance", 0.0f);
                    }

                    if (softParticlesFarFadeDistance < 0.0f)
                    {
                        softParticlesFarFadeDistance = 0.0f;
                        material.SetFloat("_SoftParticlesFarFadeDistance", 0.0f);
                    }
                    // set keywords
                    material.SetVector("_SoftParticleFadeParams",
                        new Vector4(softParticlesNearFadeDistance,
                            1.0f / (softParticlesFarFadeDistance - softParticlesNearFadeDistance), 0.0f, 0.0f));
                }
                else
                {
                    material.SetVector("_SoftParticleFadeParams", new Vector4(0.0f, 0.0f, 0.0f, 0.0f));
                }
                CoreUtils.SetKeyword(material, "_SOFTPARTICLES_ON", useSoftParticles);
            }

这个代码位于Packages\com.unity.render-pipelines.universal\Editor\ShaderGUI\ParticleGUI.cs中,和我们一开始研究unlit shader时属性处理的BaseShaderGUI一样,都是在编辑器里面给材质设置属性和关键字用的。从这儿的逻辑看,如果开启了软粒子,SoftParticles的参数near就是softParticlesNearFadeDistance,即软粒子开始fade的深度值;参数far就是1.0f / (softParticlesFarFadeDistance - softParticlesNearFadeDistance) 即软粒子fade范围深度的倒数。在编辑器UI中,指定的是fade的范围,实际的near distance和far distance,然后转换成函数需要使用的参数。不得不说,这儿的far参数很有迷惑性,不看源码真不知道是啥。好了现在可以分析这个fade是怎么计算的了,转成基于nearDistance和farDistance的表达式:

fade = saturate( (sceneZ - thisZ - nearDis ) / (farDis - nearDis) );

首先我们使用的都是view space的线性深度,sceneZ - thisZ说明我们需要粒子在场景背景的前面(如果这个差是负数就会被saturate clamp为0,也就不会显示出来了),此时这个差值表示粒子在场景背景前面多远。当这个差值大于nearDis的时候就需要过渡了,过渡的范围就是nearDis到farDis之间,得到一个0~1之间的fade值。最终fade值再乘到粒子的alpha上进行混合。

本篇总结

本篇中分析了深度图的生成,定义和采样,结合软粒子效果看了一下深度图和线性深度的应用。文章来源地址https://www.toymoban.com/news/detail-625634.html

到了这里,关于深入URP之Shader篇12: 深度值专题(3)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Unity URP Shader “Redefinition of _Time“ error

    强烈建议先尝试阅读本文之后自行解决:https://zhuanlan.zhihu.com/p/360566324 我这里记录一下我的思路: 首先检查URP升级是否正确,主要看Asset是否设置,ShaderGraph表现是否正常 尝试排除是否是未定义宏导致的问题,主要是对比ShaderGraph自动生成的代码 确认自己的代码是否使用了

    2024年02月15日
    浏览(29)
  • unity URP 利用particle system制作简单的shader交互

    首先这里制作了一个简单交互,使用shader grapgh,根据计算距离和变化数值的差实现交互后的扩散,同时计算消散遮罩让它逐渐谈去。 将他赋予材质物体,根据脚本传入位置和逐渐变化的大小后,呈现这样的效果。 但是,shader graph这样的工具,在做这种效果非常快的同时,也

    2024年02月07日
    浏览(31)
  • unity build-in 渲染管线升级urp渲染 shader篇

            由于工作原因需要对项目进行升级,从build-in渲染管线升级到urp渲染管线,我自己对应的unity版本是2018.版本升级到2021.3.2版本,由于最近几年unity版本升级比较快,个体版本差异有所不同,如遇与版本不一致问题敬请谅解。以下是根据官网等系列网站整理的内容

    2023年04月16日
    浏览(44)
  • 《Unity的URP项目中使用自定义shader导致材质消失的解决办法》

            在Unity中使用URP时,会有需求使用自定义的一些shader来实现特殊效果,这时如果我们直接使用新建材质与无光照着色器(Unlit shader),可能会发生一个对于新手而言意料之外的问题—— 物体!消失了!         打开你正在使用的的 通用渲染器(Universal Rendere

    2024年02月06日
    浏览(46)
  • Unity Shader 学习笔记(4)URP渲染管线带阴影PBR-Shader模板 -- 新增可自定义阴影颜色

    材质面板截图 功能实现(URP渲染管线下): 1、进一步优化Shader结构和算法; 2、包含PBR材质; 3、投射和接收阴影,并升级 支持自定义阴影颜色 ; 4、支持点光源照射(但不支持点光源阴影)。 通用渲染截图 自定义阴影颜色截图 完整代码: 写在最后: 1、在我的上一篇文

    2024年02月12日
    浏览(29)
  • 【Unity Shader Graph URP渲染管线下的自定义半透明效果_半透明案例分享】

    URP的渲染管线下 在项目设置里找到“Graphic” 找到URP Asset文件 索引到Renderer List文件——“ForwardRenderer” 在这个“ForwardRenderer”文件里找到“Add Renderer Feature” 添加一个渲染对象,类似下图:Render Object (Experimental) 如图设置,将“Event”设置成 AfterRenderingSkybox ,然后“Layer M

    2024年02月09日
    浏览(46)
  • Unity中URP下实现深度贴花

    在游戏中,有很多用到贴画的地方。比如:地面污渍、地面喷漆、地面血迹、魔法阵、地裂等效果。 我们在这篇文章中,来用深度图实现一下贴画的效果。 使用之前的棋盘格设置一个场景,且在场景中,增加一些物体,来给贴花吸附。 然后,我们创建一个面片用于承载贴花

    2024年01月21日
    浏览(33)
  • 【unity】URP的shader开发中支持多光源,_ADDITIONAL_LIGHTS_VERTEX 和 _ADDITIONAL_LIGHTS 区别

    项目里有一个其他同事实现的shader,美术那边希望能支持多个光源, 我一看代码里面, frag 函数里已经实现了   代码也加了:             #pragma multi_compile _ _ADDITIONAL_LIGHTS_VERTEX _ADDITIONAL_LIGHTS 材质里加了这个keyword还是没起作用,   若宏控制注了有效。  一开始没搞明白

    2024年02月11日
    浏览(38)
  • 深入URP之Shader篇3: Unlit Shader分析[下]

    上篇中我们分析了Unlit shader的Properties在ShaderGUI中的处理,接下来看Sub Shader。 unlit shader以及其他URP shader包含两个SubShader,分别是针对ShaderModel4.5和2.0。由于unlit shader本身很简单,这两个SubShader几乎一样,唯一的差别是ShaderModel 4.5的SubShader会定义 #pragma multi_compile _ DOTS_INSTANCIN

    2024年02月10日
    浏览(29)
  • 深入URP之Shader篇6: SimpleLit Shader分析(2) Vertex Shader

    看看在顶点shader中都计算了什么 计算顶点坐标 这个和之前一样: VertexPositionInputs vertexInput = GetVertexPositionInputs(input.positionOS.xyz); 再复习一下,这个函数位于 ShaderVariablesFunctions.hlsl 中。 计算法线和切线 VertexNormalInputs normalInput = GetVertexNormalInputs(input.normalOS, input.tangentOS); 输入的

    2024年02月02日
    浏览(25)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包