【转】Alpha混合物体的深度排序

先说个题外话本来我想解答一下最近Creators Club论坛上经常出现的一个问题意外的是在网上竟然找不到什么全面的答案..

这是个有着复杂答案的简单问题:

为什么我的透明物体的绘制顺序不对或者有些不见了?”

当绘制一个3D场景时对图形进行深度排序是非常重要的这样离镜头近才画在远处物体的前面我们不会希望看到远处的山把近在眼前的建筑给挡住了!

如今有三种深度排序方法得到了广泛的应用:

不幸的是每种都有其局限性为了达到好的结果大多数游戏是把三种方法结合起来使用的.

 

深度缓冲

深度缓冲简单而有效结果也很完美但是对于透明的物体它就无能为力了!

这是因为深度缓冲只记录了当前已经绘制的最近像素对于不透明的物体这已经能够满足我们的需要了看一下这个绘制两个三角形的例子, AB:

 

如果我们先画B再画A, 深度缓冲会看到新的像素(A)比之前的(B)要近那么它就画在了前面如果我们用相反的顺序画(AB), 深度缓冲会看到B的像素比之前A已经画的要远所以就把它们给丢弃掉了无论哪种情况我们都会得到正确的结果: A在前面, B隐藏在后面.

但是当这些几何图形是透明的B透过A是部分可见的时会怎样呢如果我们先画B再画A的话是没有问题的但反过来就不行了在这种情况下深度缓冲会从B取一个像素同时注意到已经绘制了一个更近的像素(A), 然后它就没辙了唯一的选择是绘制B(这会得到一个错误的结果, B会画在A前面Aalpha 混合却没有起作用), 或者完全抛弃B. 不爽!

结论深度缓冲对于不透明的物体是很完美的但对于透明的物体却不实用.

 

油画家算法

深度缓冲没法应付以错误的顺序来绘制透明物体的情况这很好解决对吧保证它们按正确的顺序绘制就可以了如果对场景中的所有物体进行排序那我们就可以先画远处的再画近处的这样就可以确保前面例子中的B可以在A之前绘制.

不幸的是这说起来容易做起来难对物体进行排序在很多情况下并不适用AB相交的情况该怎么办?

如果A是个玻璃杯而B是它里面的一个玻璃球时就是这样现在我们就没法对它们进行排序了因为A的一部分比B而另一部分又比B.

甚至我们不需要两个不同的物体来复现这个问题组成玻璃杯的那些三角形会怎样要让它们显示正确需要在前面的绘制之前先绘制后面的所以只对物体进行排序是不够的我们要对每一个三角形进行排序.

问题是对每个三角形进行排序的代价太大就算我们能够承受这也不是在所有的场合下都能得到正确的结果的比如说两个透明的三角形相交时会怎样呢

没有方法对这样的三角形进行排序因为我们需要把B的上半部分画在A的前面, A的下半部分画在B的前面唯一的解决方案就是把三角形从相交处分割开来但是这样的消耗是不可承受的.

结论油画家算法需要你在选择排序的粒度好好权衡一下如果你仅仅对一些大的的物体进行排序速度很快但不是很精确如果你对一些小物体进行排序(包括三角形个体的极限情况), 速度会慢一些但更加精确.

 

背面剔除

一般不把背面剔除当成是一种排序技术但它确实是一种重要的方法它的局限性就是只适用于凸面体.

考虑一下一个简单的凸面体如一个球体或立方体无论你从哪个角度看每个屏幕上的像素都会被覆盖两遍一次是物体的前面一次是后面如果你用背面剔除丢弃了背面的三角形那就只剩前面了哈哈如果每个屏幕上的像素只进行一次判断那你就自动得到了一个完美的混合结果没有必要排序任何东西.

当然大多数的游戏不会只画球体或立方体J 所以只是背面剔除的话不是一个妥善的解决方案.

结论背面剔除对于凸面体是完美的但是对于其它的就无能为力了.

 

我该怎样让我的游戏看起来更好一些?

最常用的方法:

1.    设置DepthBufferEnable DepthBufferWriteEnable true

2.    绘制所有的不透明物体

3.    保持DepthBufferEnable true, 但是设置DepthBufferWriteEnablefalse

4.    alpha混合的物体按照与摄像机的距离进行排序然后从后到前画出来

这依赖于三种排序技术的结合:

  • 不透明的物体按深度缓冲排序
  • 透明物体和不透明物体仍然会被深度缓冲处理(所以你永远不会通过一个不透明物体看到一个透明的)
  • 油画家算法对透明的物体排序(两个透明物体相交时仍然会有排序错误)
  • 依赖背面剔除来对单个透明物体上的三角形排序(如果物体不是凸面体也会产生错误)

结果并不是非常完美但是非常高效易于实现对于大多数游戏来说也够用了.

当然还可以采取一些措施来改进排序的精确度:

避免alpha混合你的不透明物体越多排序就越容易也越精确仔细思考一下真得每个地方都需要alpha混合吗如果关卡设计师要在玻璃窗上再加一层那你应该考虑把设计改成更易于实现的方案如果你正使用alpha混合来绘制树木之类的图形那考虑用alpha测试来代替它只分完全透明和完全不透明这两种情况这样不透明的地方仍然可以通过深度缓冲来排序.

放松不用担心可能排序错误并不是很严重呢你可以试着调整一下你的图形(alpha通道更加柔和更加透明一些来让这个错误看起来没有那么显眼这个方法用在了我们的 Particle 3D sample它并不会对单独一个烟雾中的粒子进行排序而是选择了一个合适的粒子纹理让它看起来是好的如果你把烟雾的纹理换成更加不透明的那排序错误可能就比较容易觉察了.

如果你有透明物体不是凸面体或许你可以尝试让它们更加一些就算它们不是完全地凸面体那它们越”, 排序错误就越少还有就是考虑把复杂的模型分成多块这样它们就可以分开进行排序一个人体看起来一点也不像凸面体但你把它分成头胳膊驱干等几部分后每一块都接近凸面体了.

如果你有部分区域透明的纹理(如树叶), 并且图案边缘包含了一些半透明的像素用于反走样那你可以使用pass渲染技术:

  • Pass 1: 绘制不透明部分: alpha混合关闭, alpha测试只接受100%不透明的区域深度缓冲开启
  • Pass 2: 绘制边缘: alpha混合开启, alpha测试设置只接受alpha<1深度缓冲开启深度写入关闭

以每个物体渲染两次的代价为纹理中间完全不透明的部分提供了100%正确的深度缓冲排序和相对精确的半透明边缘排序这个方法为纹理裁剪的边缘进行了一些反走样同时也保证了不用对每一棵树或者草叶进行额外的排序在我们的 Billboard sample 中使用了这个技巧请阅读一下Billboard.fx中的pass和注释.

使用 z prepass当你需要淡出一个原来不透明的物体又不想透过它看到的是它自己的另一部分时这是一个好方法例如从右边看一个人类的身体如果它是玻璃做的你应该会希望透过右手臂看到躯干和左手臂但如果它是实心的(不透明)你会希望透过右手臂看到后面的背景而不应该是躯干和左手臂要达到这个目标需要这样做:

  • 设置 ColorWriteChannels=None, 开启深度缓冲
  • 绘制物体到深度缓冲(不影响颜色缓冲)
  • 设置ColorWriteChannels=All, DepthBufferFunction=Equal, 开启alpha混合
  • 再次绘制这个物体这样就只有最近的这一面与颜色缓冲进行混合了

Published Wednesday, February 18, 2009 1:47 PM by ShawnHargreaves

posted @ 2009-04-19 02:08  柒笑侠  阅读(1763)  评论(0编辑  收藏  举报