【Unity】Camera.WorldToViewportPoint 计算原理及实现

具体原理我是参考的这篇文章:

详解MVP矩阵之ProjectionMatrix-腾讯游戏学堂

本地具体实现实现方法(不调用Camera的API)如下:

1、获取相机的VP矩阵:

        模型顶点坐标转换成屏幕坐标是用MVP矩阵,但是因为我们这边本来就使用的世界坐标进行转换,所以只需要VP矩阵即可。获取VP矩阵的方法如下:        

    private Matrix4x4 Matix_VP_LastFrame;

    private void LateUpdate()
    {
        Matix_VP_LastFrame= mainCamera.projectionMatrix * mainCamera.worldToCameraMatrix;
    }

        之所以放在LateUpdate是因为这个方法消耗挺大的,缓存起来每帧只计算一次即可。

        Camera中还有个属性是:Camera.previousViewProjectionMatrix ,看起来就是算好的值。但是我测试时取出来的值是算出来是不对的,所以就没有使用这个属性。

2、计算世界坐标的屏幕位置:

        计算原理在上述的文章连接中提到了,这里就提供代码即可。

        简单地说就是直接用VP矩阵乘世界坐标就可以了,不过这个乘出来的值并不直接是viewPos,还需要进一步转换:


    /// <summary>
    /// 传入世界坐标,获取屏幕坐标:
    /// </summary>
    private Vector3 WorldToViewportPoint(Vector3 wolrdPos)
    {
        Vector4 pos = new Vector4(wolrdPos.x, wolrdPos.y, wolrdPos.z, 1);
        Vector4 clipPos = Matix_VP_LastFrame * pos;

        float x = 0.5f + 0.5f * clipPos.x / clipPos.w;
        float y = 0.5f + 0.5f * clipPos.y / clipPos.w;

        Vector3 screenPos = new Vector3(x, y, clipPos.w);
        return screenPos;
    }

        这里返回的结果中,x、y就是在屏幕中的比例,左下角为(0,0),右上角为(1,1);z轴代表深度(正数在相机前面,负数在相机后面)。

        这个计算出来和Unity自己的API会有细微的误差(都在小数点后面好几位),但是不影响使用。

PS:

        虽然实现了算法,但是建议大家尽可能使用Unity原生的API。因为这个算法的性能消耗是大于Camera.WorldToViewportPoint 的(具体瓶颈在矩阵乘法那里,从Profile上看到了大量的String.memcpy,占用了大量的CPU消耗)。

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值