从NDC(归一化的设备坐标)坐标转换到世界坐标要点 参考资料
How to go from device coordinates back to worldspace
《Unity Shader入门精要》
产生运动模糊效果,一般来说有两种做法,第一种是将当前帧和下一帧或上一帧等等图像混合起来作为当前的屏幕图像,这样做法可以导致物体运动时会出现多个残影(因为多个帧混合起来了),可以产生运动模糊效果,但效率不高。
第二种做法是,在Shader里面利用View-Projection矩阵及其逆矩阵获得当前帧和上一帧的世界坐标,通过这两个坐标得到当前像素的运动速度及其方向,再根据这个速度,向这个像素速度方向的N个纹素进行取样并混合,从而产生模糊效果。
而这第二种做法,就是我遇到问题的地方。
总所周知,在Shader里面,可以根据深度图和当前uv坐标得到当前像素的NDC坐标,那么只要采用View-Projection(视图-裁剪)的逆矩阵就可以将NDC坐标变换到世界坐标下(正是我们所需要的值),按理说,这样应该就可以了,但是我在《Unity Shader入门精要》中看到的计算方法却是下面这样。
其中第9行是我最疑惑的内容,按说,你进行屏幕映射要除个w分量进行齐次除法我可以理解,但是你从NDC坐标变换到世界坐标还要除于w是个什么理喔?
百思不得其解之下,翻了下作者的GitHub,嗯,终于找到了答案(感谢作者及评论区的小伙伴!!!)。
在 中有详细的数学证明,下面我给翻译一下~~
数学推导过程 已知条件首先考虑从世界坐标是如何变换到NDC坐标下的。
很显然,首先世界坐标是通过VP矩阵(不用MVP,因为这里已经是世界坐标了,不用在从模型空间变换到世界空剑侠了)变换到了裁剪空间下,数学公式如下:
在裁剪空间下,通过进行齐次除法将坐标变换到NDC坐标中,公式如下:
最后,第三个已知条件是,世界坐标下的w分量在Unity中定义通常都是1,那么就是下面这样:
分析问题那么,我们的问题就变成了下面这样:
已知:
①
②
③
三个公式,给定一个NDC坐标和一个View-Projection矩阵,求得该NDC坐标所对应的世界坐标.
推导第一步,反转公式③,如下所示:
④
第二步,根据公式2可得:
⑤
同时,又因为clip=(clipxyz,clipw),所以可以根据公式⑤,可得下面的公式:
⑥
根据公式④和⑥,可得下面的公式:
⑦
下面有个解释不是很懂~,原文如下:
And note that since matrices are linear transforms, we can pull that clip_w in front of the matrix multiply:
大致意思是说,因为矩阵是线性变化的,所以可以把clipw放到矩阵的前面.(老实说clipw不是标量么,放哪里应该都可以把?)
总之,经过上面原文的解释,公式⑦变成下面这样.