最近在Unity在ForwardPath的情况下,实现一个DeferRender的效果。 其中在DeferRender的时候会用到世界坐标。 世界坐标有三个值x,y,z,如果要记录三个值就需要三个通道来记录世界坐标。
第一部分:根据深度值世界坐标
其实在DeferRender的时候我们对每一个像素值都是可以知道屏幕坐标的xy值的, 那么只要知道了深度值, 就可以反向推出世界坐标了。
图1-1 模型空间到屏幕空间的过程
在经过透视除法得到的结果xyz 都是在-1 -- 1 (opengl) 或者0-1(directx)。 我们只要获取到这个深度值。 在PC中, vertex的 shader写法如下:
图1-2 在vertshader中获取深度值
记录的深度值就可以在后处理过程当中使用了。 当然可以通过另外一种方法获取深度值就是设置Camera的depthTextureMode 为depth 或者depthnormal。 这样在shader中只要做如下声明就可以获得深度贴图了。
图1-3 在sheder中声明 _CameraDepthTexture
现在已经可以获得深度值了。 如何构建屏幕空间的坐标呢? 拿到屏幕上的坐标,组合深度值就是了。我们在使用Graphics.Blit的时候UV值就是 屏幕空间的归一化坐标了。
现在只要获得两个矩阵就可以获得世界坐标。 一个是投影矩阵MatriProjection, 一个是MatrixWToV。 求出这两个矩阵的逆矩阵就可以轻松通过矩阵成向量获得世界坐标。
图1-4 根据屏幕坐标和深度值求世界坐标
图1-4中 有个矩阵_ViewProjectionInverseMatrix 是投影矩阵和世界坐标转观察空间矩阵相乘求逆得到的。通过Material.setMatrix设置的。
图1-5 在PreRender中设置_ViewProjectionInverseMatrix
第二部分:如何测试你的结果是否正确
在Shader中如何测试我们计算的结果是多少,或者是否正确? 有几种方式都可以测试。 一种是使用帧调试器, 在调试的过程当中可以清楚的看到Shader计算的值。不论是参数还是计算结果。 Unity5已经自带帧调试器了。 而Unity5以前的版本 则可以使用IntelGPA, 高通的AdrenoProfiler 打出一个exe的包用Visual Studio的帧调试器。 当然还有其他一些工具。
另外一种方式则是直接输出颜色值, 把你的值压缩在0-1范围内。这里可以直接使用第二种方法。因为我们在正常的渲染情况下渲染一遍物体,把深度值渲染到一张RenderTexture上,然后在后处理中反推出世界坐标。 那么只要我们把世界坐标的值压缩在0-1赋值给这个物体的颜色即可。 然后在后处理中同样把计算的世界坐标压缩到0-1直接。 只要编辑器下Scene里的物体颜色和Game里的物体颜色一致,结果就对了。否则就是错的。
图2-1 在渲染物体的输出世界坐标坐标作为颜色值
图2-2 后处理中计算的直接坐标值和实际的世界坐标值一致的结果
在实际测试过程当中,会出现物体拖得太远在camera里就看不见了, 这是因为z-fighting的原因, 我们计算出来的z值和原坐标的z并非是线性关系,而是一条曲线值。 在某些地方就会出现精度问题。 z-fighting可以阅读文章:https://www.scratchapixel.com/lessons/3d-basic-rendering/perspective-and-orthographic-projection-matrix/opengl-perspective-projection-matrix
第三部分:仍然有疑问的地方
1、回到图1-1当中, 在透视除法之后理论上xyz的值都应该是-1--1 或者0-1。 而我使用的是OpenGL的做法即 -1---1。 但我在图1-2中取深度值直接取
z = o.position.z / o.postion.w
而不是
z =( o.position.z / o.postion.w + 1 ) * 0.5
测试也是前哪一种方法是对的, 而后一种方法是错的。
2、我是否可以观察空间下的z 除以 farplane的值就可以了。 因为如果z超过了farplane 就被裁剪掉了。 然后在后处理过程当中直接拿到view空间下的坐标替换z值,然后乘上 UNITY_MATRIX_MV 的逆矩阵 和 _Object2World 就可以得到世界坐标了呢?
3、在实际的不同平台上裁剪空间上区别还没有验证。
附件是我在Unity4.6下测试代码, 在unity5中会报错,只要修改其中的
GenerateDepthAndShowWoldPos.shader即可运行了。