[OpenGL高级光照] 阴影改善

目录

一 阴影失真

二 阴影改善

2.1 减小片段深度值

2.2 降低纹理

2.3 注意事项

三 消除Repeat的问题

3.1 让裁剪矩阵的立方体变大

​3.2 利用采样范围重置

四 精度问题


本章节源码 点击此处

一 阴影失真

在上一篇中,实现了阴影效果之后,但是我们会发现阴影效果中地面上的阴影明显显示有问题。

效果

  • 我们放大显示不正确的地方后会发现它的线条是明暗交替的。

原因

  • 我们的阴影效果是根据当前顶点的深度值和深度缓冲中的深度值来做对比的,但要注意的是深度缓冲中的深度纹理是以像素为单位的,(可以理解为它是单通道的,采样返回的是一个向量,其中所有分量都相同,等于深度值) 
  • 但是当我们用当前片段的深度值去对比时,多个片段有可能就采样到同一个像素的纹理值。
  • 理解: 可以这样理解,在下面的图片中黑黄相间的就代表我们阴影中最后出现的黑白条纹,由于我们使用的是平行光(没有大小只有方向),我们假设一条黄色条纹和一条黑色条纹就是真实的片段,然后这样的一个片段刚好去采样一个纹理像素,但由于深度纹理像素的值(这时候要把这个理解为深度值)一般都是取最中心点的平均值,所以对于纹理来说无论怎样这个值都是固定的比如说0.5深度。但是对于真实的片段来说,黄色部分的条纹的实际深度值(与光源的距离)就要小于黑色部分的,而中间的值其实是刚好等于深度纹理中的深度值的。所以我们前面的就回变亮,因为黄色部分中的实际深度(也就是距离光线的距离)是小于深度纹理的,那么就说明深度纹理(也就是阴影)是不应该被显示的,反之黑色部分就会显示阴影。

二 阴影改善

  • 抬高和降低都是依据下面这个计算公式来处理的
  • curPepth是代表当前的片段的深度值
  • shadowDepth代表深度纹理中的深度值
 float shadow = curDepth  > shadowDepth ? 1.0 : 0.0;

2.1 减小片段深度值

  • 相当于减小片段表面的深度值,这样就会消除纹理多个片段采样到同一个纹理像素的问题。

  • 首先我们完全可以对片段的深度值减去一个很小的值比如说0.005,这个值是很巧妙的,但是有时候还是不能够消除,
  • 这时候我们需要引入点乘,根据表面和光线的夹角,比如在一个片段中,越光线距离越远,夹角越大,比如最大假设它夹角90度那么点乘就是0,此时用1减去0,再乘以0.05那么此时我们就使用0.05来进行片段的抬高。
   float bias = max(0.05 * ( 1.0 - dot(Normal,light.position - FragPos)),0.005);

缺点

  • 因为方式是用偏移量的,有点类似箱子被抬起来了,这里立方体角的光被漏出来了,按理来说这里应该是要有阴影的。
  • 这个也取决于上面偏移量这个值我们设置的是多少。

2.2 降低纹理

  • 相当于想办法让纹理上的深度值变小。
  • 我们可以完全在生成深度测试时,只需要生成背面而不生成正面。这样对于目前这个例子来说(三个箱子一个地面)箱子的正面以及整个地面都不会被绘制,那么在深度纹理中的地面的深度就永远都是1,那么最后计算时,地面的深度就不会大于纹理的深度就可以消除这种影响了。
glEnable(GL_CULL_FACE);
glCullFace(GL_FRONT);
    renderScene(&depthProgramObject);
glCullFace(GL_BACK);

缺点:

  • 我们可以看到这种方式,由于采用的原因中间的缝隙漏出了光。

2.3 注意事项

  • 改善纹理的方式有很多种,主要是根据你的场景和效果去改善,目前并没有完全最优的一种算法,这需要根据场景来尝试优化自己的算法。
  • 这两种方式不要重叠使用,除非你能保证混合算法下不会冲突。

三 消除Repeat的问题

  • 其实这里有两个问题,一个是纹理重复的问题,另一个是部分没有区域是黑色
  • 黑色区域:这是因为我们观察的立方体本身比较小,当坐标超出立方体的范围时,深度值都会变成1
  • 纹理重复:纹理重复是因为对于写入深度缓冲中的纹理环绕方式是Repeat,当纹理坐标大于1小于0时就会进行平复重复,因为我们获取深度值是采用的纹理 texture(depthMap,projCoords.xy).r;的方式获取的,所有当超出纹理范围时,这个深度纹理也会进行Repeat,就导致下面在别的地方也出现了阴影。

3.1 让裁剪矩阵的立方体变大

    float near_plane = 1.0f, far_plane = 25.5f;
    // 定义一个正交投影 矩阵的再x轴的值 和y轴的值
    lightProjection.ortho(-25.0f, 25.0f, -25.0f, 25.0f, near_plane, far_plane);
  • 这种方式效果还是不太好,你会发现他的锯齿特别严重,并且本身也不推荐这种。
  • 因为这样我们获取深度纹理时就不会出现超出纹理坐标的深度,也不会Repeat

3.2 利用采样范围重置

  • 当采样的纹理坐标超过0.0-1.0时,它会重复纹理,我们只需要利用代码将阴影设置为0即可
    if(projCoords.z > 1.0 || projCoords.x < 0.0 || projCoords.x > 1.0)
        shadow = 0.0;
  • 还有一种方法就是设置深度纹理的Repeat属性,当纹理超出范围时,我们设置边框为全白,相当于深度值全为1
  • 当然为了消除部分黑色区域我们还是要设置当纹理坐标Z大于0时的处理。
 if(projCoords.z > 1.0 )
    shadow = 0.0;   

glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_BORDER);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_BORDER);
float bordercolor[] = {1.0,1.0f,1.0f,1.0f};
glTexParameterfv(GL_TEXTURE_2D,GL_TEXTURE_BORDER_COLOR,bordercolor);

四 精度问题

  • 我们可以通过扩大采样范围,来提高精度,但是这会损失性能,当然后面有更好的处理方式,目前我们只需要知道这样可以处理即可
const unsigned int SHADOW_WIDTH = 10240, SHADOW_HEIGHT = 10240;
  • 我们可以看到这样处理,对阴影边缘的锯齿,有了很大的改善。

  • 14
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: OpenGL ES是一种用于在移动设备上进行图形渲染图形库。要实现纹理光照阴影效果,我们可以使用一系列的技术和算法。 首先,为了实现纹理光照,我们需要在场景中添加光源。可以使用不同的光照模型,如环境光、漫反射光和镜面光,以模拟不同光源的效果。为了计算这些光照效果,我们需要使用光照方程和法线向量来确定光照的强度和方向。在渲染每个物体时,将法线向量传递给着色器,并计算光照强度,再将其与纹理颜色相乘以获得最终的光照效果。 其次,要实现阴影效果,我们可以使用阴影映射技术。这种技术的基本思想是,在渲染场景之前,以光源为中心从不同角度渲染一个深度贴图。然后,在渲染场景时,将该深度贴图作为纹理应用到每个物体上。对于场景中的每个片元,我们可以使用片元的深度与深度贴图中的对应值进行比较,从而确定该片元是否处于阴影中。如果处于阴影中,片元的颜色将进行某种处理,以模拟阴影效果。 最后,为了在OpenGL ES中实现纹理光照阴影效果,我们需要使用着色器程序。通过编写顶点着色器和片元着色器代码,我们可以实现光照计算和阴影计算,并将其应用于每个顶点和片元。在顶点着色器中,我们可以对顶点和法线进行变换和计算。在片元着色器中,我们可以通过插值法获得光照强度,并使用法线向量和纹理颜色计算片元的最终颜色。 综上所述,要在OpenGL ES中实现纹理光照阴影,我们需要使用光照方程、法线向量、阴影映射技术和着色器程序来计算和应用光照效果。通过合理的设计和编码,可以实现出令人满意的纹理光照阴影效果。 ### 回答2: OpenGL ES是一种专门用于嵌入式系统的OpenGL API,并且支持在移动设备上实现图形渲染。在实现纹理光照阴影效果时,以下是一些关键步骤和方法。 1. 加载纹理:首先,需要加载纹理图像作为场景中的纹理贴图。可以使用OpenGL ES提供的纹理加载函数(如glTexImage2D)将图像数据加载到纹理对象中。 2. 设置光源:为了实现光照效果,需要设置光源的位置和光照属性(如光的颜色和强度)。可以使用OpenGL ES的光照函数(如glLightfv)来设置光源属性。 3. 设置材质:每个物体都有一个材质,它决定了物体如何反射光线。可以使用OpenGL ES的材质函数(如glMaterialfv)来设置物体的材质属性。 4. 配置阴影:要实现实时阴影效果,可以使用阴影贴图技术。首先,需要渲染场景的深度信息到一个特殊的贴图中,这可以通过使用帧缓冲对象(Framebuffer Object)实现。然后,可以使用阴影贴图来计算每个像素点是否受阴影影响,并在渲染时进行相应调整。 5. 渲染场景:根据设置的光源和材质属性,使用着色器对场景中的每个物体进行光照计算。可以使用OpenGL ES提供的着色器语言(如GLSL)编写光照效果的着色器程序。在绘制每个物体时,需要将纹理贴图应用到物体的表面。 通过以上步骤,可以实现纹理光照阴影效果。OpenGL ES提供了各种函数和工具来协助实现这些效果,但具体的实现方式可能因应用需求和设备性能而有所不同。因此,开发者需要结合具体情况进行调试和优化,以达到最佳的纹理光照阴影效果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值