OpenGL

十、OpenGL中的向量、矩阵、矩阵堆栈

2020-07-17  本文已影响0人  小山93

1. 向量

1.1 向量相关概念
1.2 OpenGL 如何定义向量
1.3 向量运算
// 获取两个向量的夹角
float m3dDotProduct3(const M3DVector3f u,const M3DVector3f v);
// 获取两个向量的夹角的弧度
float m3dGetAngleBetweenVector3(const M3DVector3f u,const
M3DVector3f v);
// 获取两个向量的叉乘结果(法线)
void m3dCrossProduct3(M3DVector3f result,const M3DVector3f  u ,const
    M3DVector3f v);

2. 矩阵(Matrix)

2.1 矩阵的定义方式
typedef float M3DMatrix33f[9];   // 三维矩阵的声明
typedef float M3DMatrix44f[16];  // 四维矩阵的声明
// 初始化一个单元矩阵
void m3dLoadIdentity44f(M3DMatrix44f m);
2.2 为什么使用矩阵?

一个点(x,y,z)在坐标系中旋转后,我们想得到新的坐标,就需要使用到矩阵。因为新坐标的 x 值,不仅与旧坐标的x值及旋转参数有关系,甚至还和 y 值及 z 值有关系。一个物体的所有顶点都乘以矩阵,就能让整个物体变换到空间中给定的位置和方向。

2.3 使用矩阵的步骤

坐标系文章中,曾介绍过坐标系变换的步骤如下图所示:

坐标系变换的步骤.png
从逻辑上,在坐标系的变换过程中,依次使用了模型矩阵、视图矩阵、投影矩阵。不过在OpenGL计算变换后的顶点向量时,其计算顺序是相反的。 计算公式如下所示:
变换顶点向量 = M_pro * M_view * M_model * V_local
变换顶点向量 = 投影矩阵 ✖ 视图变换矩阵 ✖ 模型矩阵 ✖ 顶点

2.3.1. 视图变换:设置观察者的位置,以确定观察者坐标系。任何模型变换之前,都要先进行视图变换。因为一旦观察者坐标系改变后,所以的物体变换都要基于新的坐标系进行。
默认情况下,透视投影中,观察者处于原点(0,0,0),并沿着Z轴负方向看过去(看向显示器内部)。

2.3.2. 模型变换:物体进行平移、旋转、缩放

// 平移
void m3dTranslationMatrix44(M3DMatrix44f m, float x, float y, float z);
// 旋转
m3dRotationMatrix44(m3dDegToRad(45.0), float x, float y, float z);
// 缩放 (x/y/z参数传值-1时,可以实现物体围绕某一个轴的翻转)
void m3dScaleMatrix44(M3DMatrix44f m, float xScale, float yScale, float zScale);
// 综合变换, a 为先变换的矩阵、b 为后变换的矩阵
void m3dMatrixMultiply44(M3DMatrix44f product, const M3DMatrix44f a, const M3DMatrix44f b);

注:由于矩阵叉乘不符合交换律,矩阵交换后,矩阵相乘的结果不一样。所以物体有多种变换时,变换顺序不可交换。
例如“先平移后旋转”与“先旋转后平移”的效果是不一致的。如下图所示:


“先平移后旋转”与“先旋转后平移”的效果不一致.png
先平移再旋转.gif 先旋转再平移.gif

2.3.3. 投影变换:设置投影方式

// 平截头体设置透视投影
viewFrustum.SetPerspective(35.0f, float(nWidth)/float(nHeight), 1.0f, 100.0f);
// 平截头体设置正投影
void SetOrthographic(GLfloat xMin, GLfloat xMax, GLfloat yMin, GLfloat yMax, GLfloat zMin, GLfloat zMax)

3. 矩阵堆栈(GLMatrixStack)

///---- 矩阵加载、相乘、获取

//在堆栈顶部载⼊一个单元矩阵
void GLMatrixStack::LoadIdentity(void);
//在堆栈顶部载⼊任何矩阵 //参数:4*4矩阵
 void GLMatrixStack::LoadMatrix(const M3DMatrix44f m);
//矩阵乘以矩阵堆栈顶部矩阵,相乘结果存储到堆栈的顶部
void GLMatrixStack::MultMatrix(const M3DMatrix44f);
//获取矩阵堆栈顶部的值
void GLMatrixStack::GetMatrix(void);
void GLMatrixStack::GetMatrix(M3DMatrix44f mMatrix);

///---- 矩阵压栈、出栈

//将当前矩阵压⼊入堆栈(栈顶矩阵copy ⼀份到栈顶) 
void GLMatrixStack::PushMatrix(void);
//将 M3DMatrix44f 矩阵对象压入当前矩阵堆栈
void GLMatrixStack::PushMatrix(const M3DMatrix44f mMatrix);
//将GLFame 对象压⼊入矩阵对象
void GLMatrixStack::PushMatrix(GLFame &frame);
//出栈(出栈指的是移除顶部的矩阵对象) 
void GLMatrixStack::PopMatrix(void);

//将堆栈的顶部压⼊GLFrame角色帧
void GLMatrixStack::LoadMatrix(GLFrame &frame);
//GLFrame乘以矩阵堆栈顶部的矩阵。相乘结果存储在堆栈的顶部 
void GLMatrixStack::MultMatrix(GLFrame &frame);
//将当前的GLFrame压栈
void GLMatrixStack::PushMatrix(GLFrame &frame);

///---- 矩阵仿射变换(一般使用模型变换,很少直接通过堆栈进行放射变换)

// 旋转 angle是度数,⽽不是弧度
void MatrixStack::Rotate(GLfloat angle,GLfloat x,GLfloat y,GLfloat z);
// 平移
void MatrixStack::Translate(GLfloat x,GLfloat y,GLfloat z);
// 缩放
void MatrixStack::Scale(GLfloat x,GLfloat y,GLfloat z);
矩阵入栈、相乘、出栈.png

3.1 GLFrame 角色帧类

///---- 移动位置
// Move Forward (along Z axis)
inline void MoveForward(float fDelta);
// Move along Y axis
inline void MoveUp(float fDelta)
// Move along X axis
inline void MoveRight(float fDelta)

///---- 旋转
// Rotate in world coordinates...
void RotateWorld(float fAngle, float x, float y, float z)
// Rotate around a local axis
void RotateLocal(float fAngle, float x, float y, float z) 
上一篇下一篇

猜你喜欢

热点阅读