GAMES202——作业1 实时阴影(ShadowMap,PCF,PCSS)

2024-08-28 22:12

本文主要是介绍GAMES202——作业1 实时阴影(ShadowMap,PCF,PCSS),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

任务

        ShadowMap

        PCF

        PCSS

实现 

       ShadowMap

        useShadowMap      

        PCF

        findBlocker

        PCSS

结果


任务

        ShadowMap
        1.在 ShadowMaterial.js 中需要向 Shader 传递正确的 uLightMVP 矩阵,该矩阵参与了第一步从光源处渲染场景从而构造ShadowMap 的过程。 你需要完成 DirectionalLight 中的CalcLightMVP(translate, scale) 函数,它会在 ShadowMaterial 中被调用,并将返回光源处的 MVP 矩阵绑定从而完成参数传递过程。
        2.需要完善 phongFragment.glsl 中的 useShadowMap(sampler2D shadowMap,vec4 shadowCoord) 函数。该函数负责查询当前着色点在 ShadowMap 上记录的深度值,并与转换到 light space 的深度值比较后返回 visibility 项(请注意,使用的查询坐标需要先转换到 NDC 标准空间 [0,1])。
        PCF
1.需要完善 phongFragment.glsl 中的 PCF(sampler2D shadowMap, vec4 shadowCoord, float filterSize) 函数。使用作业框架提供的圆盘滤波核,框架中提供了泊松圆盘采样和均匀圆盘采样
两种采样函数。
        PCSS

        1.需要完善 phongFragment.glsl 中的 findBlocker(sampler2D shadowMap, vec2 uv, float zReceiver)。findBlocker 函数中需要完成对遮挡物平均深度的计算。

        2.完善PCSS(sampler2D shadowMap, vec4 shadowCoord) 函数。

        

实现 

       ShadowMap

        使用shadow map实现硬阴影,需要实现经典的 Two Pass Shadow Map 方法,第一次先以光源为视点位置,将光源能看见的所有物体进行光栅化,并将以深度值进行绘制的结果保存到帧缓冲区,帧缓冲区的内容写入纹理。第二次绘制,根据纹理和一个将世界坐标下的点转化到以光源为视点的坐标系的矩阵,判断是否被遮挡,从而实现硬阴影。

        

        第一个任务是求世界坐标转化为光源坐标的矩阵。

        在这里,采用glMatrix库的相关API。

        以光源为视点,计算方法就是刚开始学图形学的时候学的MVP矩阵。对于模型矩阵modelMatrix,该函数传入了两个向量,一个是位移translate一个是缩放scale,使用这两个向量,计算出模型矩阵。对于视图矩阵,直接用该类自带的光源属性的值传入API得到。对于投影矩阵,使用正交投影,因为能够方便判断深度值。zNear值最好不要设置为0,因为zNear为0在后面的PCSS中会带来不必要的麻烦。

//DirectionalLight.jsCalcLightMVP(translate, scale) {let lightMVP = mat4.create();let modelMatrix = mat4.create();let viewMatrix = mat4.create();let projectionMatrix = mat4.create();// Model transformmat4.translate(modelMatrix, modelMatrix, translate);mat4.scale(modelMatrix, modelMatrix, scale);// View transformmat4.lookAt(viewMatrix, this.lightPos, this.focalPoint, this.lightUp);// Projection transformmat4.ortho(projectionMatrix, -100, 100, -100, 100, 0.01, 500);mat4.multiply(lightMVP, projectionMatrix, viewMatrix);mat4.multiply(lightMVP, lightMVP, modelMatrix);return lightMVP;}
        useShadowMap      
第二个任务是完善useShadowMap函数。
         在vertexShader中,已经预先帮我们计算好了每一个片元在以光源为视点的坐标的位置vPositionFromLight,因此在fragmentShader里便能专注实现功能。
        要使用阴影贴图,需在fragmentShader的main函数中,先将[-1,1]空间转化为[0,1]空间。
  vec3 shadowCoord = vPositionFromLight.xyz / vPositionFromLight.w;shadowCoord = (shadowCoord + 1.0) / 2.0;

        转换完后,[0,1]空间的xy坐标就能刚好对应贴图的uv坐标。在阴影贴图中使用uv坐标查询相应位置的深度。如果查到的深度比现在的点的深度小,说明现在的点被遮挡。

float useShadowMap(sampler2D shadowMap, vec4 shadowCoord){vec4 shadowColor = texture2D(shadowMap,shadowCoord.xy);float depth = unpack(shadowColor);float z = shadowCoord.z;if(z > depth + EPS ){return 0.0;}return 1.0;
}
        PCF

        第三个任务是实现PCF的功能。其实就是在原来ShadowMap的基础上,对周围进行采样,并将结果求平均。

        先预定义一些常量以方便下面的PCF和PCSS的运算

#define SHADOW_MAP_SIZE 2048    //阴影贴图大小
#define NEAR 0.01               //之前在计算正交投影矩阵的时候的zNear
#define LIGHT_SIZE 10.0         //光源在世界的大小
#define LIGHT_UV_SIZE 0.15      //光源在贴图上的大小

        开始先调用作业框架里自带的获取采样点的函数。获取对周围的随机采样方向,乘以采样半径大小再除以贴图大小即可求得采样点在贴图中的位置。在这里定义一个阻挡值来记录积累阻挡的数量。作业原代码,参数是没有sampleRadious的,这里是另外加上去的,方便控制采样区域大小。

float PCF(sampler2D shadowMap, vec4 coords ,float sampleRadious ) {poissonDiskSamples(coords.xy);float block = 0.0;for(int i =0;i<NUM_SAMPLES;i++){vec4 shadowColor = texture2D(shadowMap,coords.xy + poissonDisk[i] * sampleRadious/ float(SHADOW_MAP_SIZE) );float depth = unpack(shadowColor);float z = coords.z;if(z > depth + EPS ){block = block + 1.0;}}return 1.0 - block / float(NUM_SAMPLES);}
        findBlocker

        该函数实现的功能是计算遮挡物的平均深度。

        要根据点到光源的距离来决定采样区域的大小,也就是W_{Penumbra},采用相似三角形来计算。得到采样区域的大小后,进行采样。因为是计算遮挡物的平均深度,所以没有遮挡物的话,返回-1处理,有遮挡物返回遮挡物的平均深度,而不是返回整个采样区域的平均深度。

float findBlocker( sampler2D shadowMap,  vec2 uv, float zReceiver ) {poissonDiskSamples(uv);int blockCnt = 0;float blockDepth = 0.0;float sampleSize = LIGHT_UV_SIZE * (vPositionFromLight.z - NEAR ) / vPositionFromLight.z;for(int i =0;i<NUM_SAMPLES;i++){vec4 shadowColor = texture2D(shadowMap,uv + poissonDisk[i] * 10.0 / float(SHADOW_MAP_SIZE) );float depth = unpack(shadowColor);if(zReceiver > depth + EPS ){blockCnt++;blockDepth = blockDepth + depth;}}if(blockCnt == 0)return -1.0;return blockDepth /float(blockCnt);}
        PCSS

        有了上面的findBlolcker后,实现起来就非常简单,使用上面findblocker的数据,调用PCF即可。注意当avgDepth为-1时,即无遮挡,为特殊情况,直接返回1.0。

float PCSS(sampler2D shadowMap, vec4 coords){vec2 uv = coords.xy;float zReceiver = coords.z;// STEP 1: avgblocker depthfloat avgDepth = findBlocker(shadowMap,uv,zReceiver);if(avgDepth < 0.0)return 1.0;// STEP 2: penumbra sizefloat penumbra = (zReceiver - avgDepth) / avgDepth * LIGHT_SIZE;// STEP 3: filteringreturn PCF(shadowMap,coords,penumbra);
}
void main(void) {vec3 shadowCoord = vPositionFromLight.xyz / vPositionFromLight.w;shadowCoord = (shadowCoord + 1.0) / 2.0;float visibility;//visibility = useShadowMap(uShadowMap, vec4(shadowCoord, 1.0));//visibility = PCF(uShadowMap, vec4(shadowCoord, 1.0) , 10.0);visibility = PCSS(uShadowMap, vec4(shadowCoord, 1.0));vec3 phongColor = blinnPhong();gl_FragColor = vec4(phongColor * visibility, 1.0);//gl_FragColor = vec4(phongColor, 1.0);
}

结果

ShadowMap硬阴影

PCF实现软阴影

PCSS实现软阴影

这篇关于GAMES202——作业1 实时阴影(ShadowMap,PCF,PCSS)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Nginx实现前端灰度发布

《Nginx实现前端灰度发布》灰度发布是一种重要的策略,它允许我们在不影响所有用户的情况下,逐步推出新功能或更新,通过灰度发布,我们可以测试新版本的稳定性和性能,下面就来介绍一下前端灰度发布的使用,感... 目录前言一、基于权重的流量分配二、基于 Cookie 的分流三、基于请求头的分流四、基于请求参数的分

基于Canvas的Html5多时区动态时钟实战代码

《基于Canvas的Html5多时区动态时钟实战代码》:本文主要介绍了如何使用Canvas在HTML5上实现一个多时区动态时钟的web展示,通过Canvas的API,可以绘制出6个不同城市的时钟,并且这些时钟可以动态转动,每个时钟上都会标注出对应的24小时制时间,详细内容请阅读本文,希望能对你有所帮助...

HTML5 data-*自定义数据属性的示例代码

《HTML5data-*自定义数据属性的示例代码》HTML5的自定义数据属性(data-*)提供了一种标准化的方法在HTML元素上存储额外信息,可以通过JavaScript访问、修改和在CSS中使用... 目录引言基本概念使用自定义数据属性1. 在 html 中定义2. 通过 JavaScript 访问3.

CSS模拟 html 的 title 属性(鼠标悬浮显示提示文字效果)

《CSS模拟html的title属性(鼠标悬浮显示提示文字效果)》:本文主要介绍了如何使用CSS模拟HTML的title属性,通过鼠标悬浮显示提示文字效果,通过设置`.tipBox`和`.tipBox.tipContent`的样式,实现了提示内容的隐藏和显示,详细内容请阅读本文,希望能对你有所帮助... 效

前端bug调试的方法技巧及常见错误

《前端bug调试的方法技巧及常见错误》:本文主要介绍编程中常见的报错和Bug,以及调试的重要性,调试的基本流程是通过缩小范围来定位问题,并给出了推测法、删除代码法、console调试和debugg... 目录调试基本流程调试方法排查bug的两大技巧如何看控制台报错前端常见错误取值调用报错资源引入错误解析错误

Vue中动态权限到按钮的完整实现方案详解

《Vue中动态权限到按钮的完整实现方案详解》这篇文章主要为大家详细介绍了Vue如何在现有方案的基础上加入对路由的增、删、改、查权限控制,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、数据库设计扩展1.1 修改路由表(routes)1.2 修改角色与路由权限表(role_routes)二、后端接口设计

Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)

《Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)》文章介绍了如何使用dhtmlx-gantt组件来实现公司的甘特图需求,并提供了一个简单的Vue组件示例,文章还分享了一... 目录一、首先 npm 安装插件二、创建一个vue组件三、业务页面内 引用自定义组件:四、dhtmlx

Vue ElementUI中Upload组件批量上传的实现代码

《VueElementUI中Upload组件批量上传的实现代码》ElementUI中Upload组件批量上传通过获取upload组件的DOM、文件、上传地址和数据,封装uploadFiles方法,使... ElementUI中Upload组件如何批量上传首先就是upload组件 <el-upl

前端知识点之Javascript选择输入框confirm用法

《前端知识点之Javascript选择输入框confirm用法》:本文主要介绍JavaScript中的confirm方法的基本用法、功能特点、注意事项及常见用途,文中通过代码介绍的非常详细,对大家... 目录1. 基本用法2. 功能特点①阻塞行为:confirm 对话框会阻塞脚本的执行,直到用户作出选择。②

如何使用CSS3实现波浪式图片墙

《如何使用CSS3实现波浪式图片墙》:本文主要介绍了如何使用CSS3的transform属性和动画技巧实现波浪式图片墙,通过设置图片的垂直偏移量,并使用动画使其周期性地改变位置,可以创建出动态且具有波浪效果的图片墙,同时,还强调了响应式设计的重要性,以确保图片墙在不同设备上都能良好显示,详细内容请阅读本文,希望能对你有所帮助...