美文网首页
七、OpenGL中观察方式与矩阵的关系

七、OpenGL中观察方式与矩阵的关系

作者: Style_月月 | 来源:发表于2020-07-12 16:29 被阅读0次

    目前在OpenGL中,矩阵的变换主要涉及两种观察方式:

    • 观察者不动,物体动
    • 观察者动,物体不动

    两种方式涉及步骤大致总结如下:

    • ChangeSize函数
      设置投影方式,得到投影矩阵,并往矩阵堆栈中压入一个单元矩阵(单元矩阵的压入可省略)
        //创建投影矩阵,并将它载入投影矩阵堆栈中
        viewFrustum.SetPerspective(35.0f, float(w)/float(h), 1.0f, 500.0f);
        projectionMatrix.LoadMatrix(viewFrustum.GetProjectionMatrix());
        //调用顶部载入单元矩阵
        modelViewMatrix.LoadIdentity();
    
    • SetupRC函数
      设置观察者的位置
    • SpecialKeys函数
      决定物体/观察者围绕旋转的坐标轴
    • RenderScene函数
      完成矩阵的相应变换:入栈、相乘、出栈

    下面我们来分别说说两种方式的区别

    观察者不动,物体动

    观察者不动,物体动

    05_OpenGL 点/线/线段/线环/金字塔/六边形/圆柱的副本(观察者不动,物体动-objectFrame)的代码中

    • 设置观察者的位置
      ==> GLFrame中默认的方向是z轴的负方向 -- (0.0f, 0.0f, -1.0f)
      ==> 参数:表示离屏幕之间的距离。 负数,是往屏幕后面移动;正数,往屏幕前面移动
    cameraFrame.MoveForward(-15.0f);
    
    • 在SpecialKeys函数中,进行旋转是物体

    例如,按上键位,物体围绕x轴旋转 -5度

    objectFrame.RotateWorld(m3dDegToRad(-5.0f), 1.0f, 0.0f, 0.0f);
    
    • RenderScene函数中,同时涉及观察者矩阵和物体矩阵的变换
      因为观察者设置位置时,往屏幕后面平移了15个像素点,所以观察者发生了变化
      在我们选中特殊键位移动图形时,物体进行了旋转,所以物体也发生了变化
      因此最后所需的mvp矩阵,需要按照压栈cameraFrame- 压栈objectFrame - 投影矩阵这个顺序进行矩阵变换,这个顺序是OpenGL规定的,且顺序是不能交换的,因为矩阵相乘不满足交换律, 即 矩阵A * 矩阵B != 矩阵B * 矩阵A
        //压栈
        modelViewMatrix.PushMatrix();
        M3DMatrix44f mCamera;
        cameraFrame.GetCameraMatrix(mCamera);
    
        //矩阵乘以矩阵堆栈的顶部矩阵,相乘的结果随后简存储在堆栈的顶部
        modelViewMatrix.MultMatrix(mCamera);
        
        M3DMatrix44f mObjectFrame;
        //只要使用 GetMatrix 函数就可以获取矩阵堆栈顶部的值,这个函数可以进行2次重载。用来使用GLShaderManager 的使用。或者是获取顶部矩阵的顶点副本数据
        objectFrame.GetMatrix(mObjectFrame);
        
        //矩阵乘以矩阵堆栈的顶部矩阵,相乘的结果随后简存储在堆栈的顶部
        modelViewMatrix.MultMatrix(mObjectFrame);
        
        shaderManager.UseStockShader(GLT_SHADER_FLAT, transformPipeline.GetModelViewProjectionMatrix(), vBlack);
    

    其中矩阵堆栈的变化过程如下:


    矩阵堆栈的变化过程
    • PushMatrix()函数,根据源码分析可知:在没有任何参数的情况下,矩阵堆栈会将栈顶单元矩阵拷贝一份,并将拷贝的单元矩阵入栈


      PushMatrix函数源码

    观察者动,物体不动

    观察者动,物体不动

    在Demo05_OpenGL 点/线/线段/线环/金字塔/六边形/圆柱的副本(观察者动,物体不动-cameraFrame)代码中

    • 设置观察者的位置
      此时的objectFrame是指观察者,并不是物体
    objectFrame.MoveForward(15.0f);
    
    • 在SpecialKeys函数中,进行旋转是观察者
    objectFrame.RotateWorld(m3dDegToRad(-5.0f), 1.0f, 0.0f, 0.0f);
    
    • RenderScene函数中,仅涉及观察者的矩阵变换
      ==> 设置观察者位置时,观察者平移了15个像素点
      ==> 特殊键位触发时,观察者绕着相应的轴旋转了一定的弧度
      ==> 所以objectFrame不仅记录了往前移动的状态,还记录了旋转的转台,所以只需要将objectFrame载入矩阵堆栈即可, 即 把往前的变化、旋转的变化都 放入 模型矩阵堆栈中
    modelViewMatrix.PushMatrix(objectFrame);
    shaderManager.UseStockShader(GLT_SHADER_FLAT, transformPipeline.GetModelViewProjectionMatrix(), vBlack);
    

    其中矩阵堆栈的变化过程如下:


    矩阵堆栈的变化过程

    观察者矩阵中包含了观察者的所有变换,会直接在变换管道中完成矩阵相乘,最后通过变换管道得到mvp矩阵。

    总结
    其实选择哪种观察方式,还是需要结合需求,个人建议,哪种方便用哪个

    • 在观察者不动,物体动时,观察者也并非是一动不动,需要调整一个好的观察位置
    • 观察者动,主要动的是V_local,其实就是通过moveForward方法设置的观察者位置
    • 物体动,主要动的是mvp

    疑问

    • 在观察者不动,物体动时,为什么不能直接这样写modelViewMatrix.PushMatrix(cameraFrame)

    我们从PushMatrix(cameraFrame)的源码分析可知,如图所示,传入push函数的frame在 PushMatrix方法中获取的是GetMatrix,而我们需要的是GetCameraMatrix,所以不能直接将cameraFrame直接push到矩阵堆栈中

    源码分析

    ``

    相关文章

      网友评论

          本文标题:七、OpenGL中观察方式与矩阵的关系

          本文链接:https://www.haomeiwen.com/subject/uzsicktx.html