如何正确地使用opengl转换的矩阵进行数学运算
How do I properly do math with matrices for opengl transformations?
我真的很困惑矩阵是如何工作的,以及如何尽可能地使用它们。作为一个测试,我试图让一个矩形在向一个角移动时逆时针旋转,而且这个矩形只有它的一半大。基本上,除了轮换之外,一切都不起作用。
如果你想知道,我没有使用GLM,我不想使用GLM。我觉得我需要尝试一下,但现在我真的陷入了困境。我试着重新安排转换的方式,但得到的结果是随机的。
这是main.cpp中的代码,它是一个自定义游戏引擎,所以这里只是的相关部分
Matrix4 transform = Matrix4(); // the matrix
float d = 5, g = 1;
void OnUserUpdate() override { // Called once every frame
transform.Translate(Vector3(d * DeltaTime, d * DeltaTime, 0.0f));
transform.Rotate(Vector3(0.0f, 0.0f, 1.0f), ToRadians(g));
transform.Scale(Vector3(0.5f, 0.5f, 0.5f));
g+=0.01;
Shaders.SetMat4(shader, "transform", transform);
}
这是完整的矩阵类。Vector3只是浮点x,y,z。只有一个构造函数。
template <typename Number> float ToRadians(Number deg) {
return deg * PI / 180;
}
struct Matrix4 {
private:
Matrix4& identity() {
m[0] = m[5] = m[10] = m[15] = 1.0f;
m[1] = m[2] = m[3] = m[4] = m[6] = m[7] = m[8] = m[9] = m[11] = m[12] = m[13] = m[14] = 0.0f;
return* this;
}
public:
float m[16];
Matrix4() {
identity();
}
Matrix4(const float src[16]) {
Set(src);
}
void Set(const float src[16]) {
for (int i = 0; i < 16; i++) {
m[i] = src[i];
}
}
void Projection(float fov, float aspectratio, float nearpane, float farpane) {
}
void Translate(Vector3 v) {
float x = v.x; float y = v.y; float z = v.y;
m[0] += m[3] * x; m[4] += m[7] * x; m[8] += m[11] * x; m[12] += m[15] * x;
m[1] += m[3] * y; m[5] += m[7] * y; m[9] += m[11] * y; m[13] += m[15] * y;
m[2] += m[3] * z; m[6] += m[7] * z; m[10] += m[11] * z; m[14] += m[15] * z;
}
void Rotate(Vector3 axis, float deg) {
m[0] = cosf(deg)+axis.x*axis.x*(1-cosf(deg));
m[1] = axis.y*axis.x*(1 - cosf(deg)) + axis.z*sinf(deg);
m[2] = axis.z*axis.x*(1 - cosf(deg)) - axis.y*sinf(deg);
m[4] = axis.x*axis.y*(1 - cosf(deg)) - axis.z*sinf(deg);
m[5] = cosf(deg)+axis.y*axis.y*(1-cosf(deg));
m[6] = axis.z*axis.y*(1 - cosf(deg)) + axis.x*sinf(deg);
m[8] = axis.x*axis.z*(1 - cosf(deg)) + axis.y*sinf(deg);
m[9] = axis.y*axis.z*(1 - cosf(deg)) - axis.x*sinf(deg);
m[10] = cosf(deg) + axis.z*axis.z*(1 - cosf(deg));;
m[15] = 1;
}
void Scale(Vector3 v) {
float x = v.x; float y = v.y; float z = v.y;
m[0] *= x; m[4] *= x; m[8] *= x; m[12] *= x;
m[1] *= y; m[5] *= y; m[9] *= y; m[13] *= y;
m[2] *= z; m[6] *= z; m[10] *= z; m[14] *= z;
}
};
正如我所说的,我试图让一个矩形在向一个角移动时逆时针旋转,而这个矩形只有它的一半大。我使用了一些来源,但老实说,我并没有完全理解
http://www.songho.ca/opengl/gl_matrix.html
https://learnopengl.com/Getting-started/Transformations
简化事情。
创建3种方法,初始化缩放、平移和旋转矩阵:
struct Matrix4 {
// ...
void SetScale( Vector3 v ) {
identity();
m[0] = v.x; m[5] = v.y; m[10] = v.z;
}
void SetTranslate( Vector3 v ) {
identity();
m[12] = v.x; m[13] = v.y; m[14] = v.z;
}
void SetRotate( Vector3 v, float rad ) {
identity();
float c = cosf(rad);
float s = sinf(rad);
float x = v.x, y = v.x, z = v.z;
m[0] = x*x*(1.0f-c)+c; m[1] = x*y*(1.0f-c)-z*s; m[2] = x*z*(1.0f-c)+y*s;
m[4] = y*x*(1.0f-c)+z*s; m[5] = y*y*(1.0f-c)+c; m[6] = y*z*(1.0f-c)-x*s;
m[8] = z*x*(1.0f-c)-y*s; m[9] = z*y*(1.0f-c)+x*s; m[10] = z*z*(1.0f-c)+c;
}
// ...
}
进一步创建一种将矩阵乘以当前矩阵的方法:
struct Matrix4 {
// ...
void Multiply( const Matrix4 &b ) {
Matrix4 a;
a.Set( m );
for ( int k = 0; k < 4; ++ k ) {
for ( int j = 0; j < 4; ++ j ) {
m[k*4+j] =
a.m[0*4+j] * b.m[k*4+0] +
a.m[1*4+j] * b.m[k*4+1] +
a.m[2*4+j] * b.m[k*4+2] +
a.m[3*4+j] * b.m[k*4+3];
}
}
}
// ...
}
基于这种方法,可以轻松地实现缩放、平移和旋转现有矩阵的方法:
struct Matrix4 {
// ...
void Scale(Vector3 v) {
Matrix4 s;
s.SetScale( v );
Multiply( s );
}
void Translate(Vector3 v) {
Matrix4 t;
t.SetTranslate( v );
Multiply( t );
}
void Rotate(Vector3 axis, float rad) {
Matrix4 r;
r.SetRotate( axis, rad );
Multiply( r );
}
// ...
}
进行此跑步。之后,您可以尝试对方法进行优化,但您将有一个工作基础来比较结果。
注意,在transform.Rotate(Vector3(0.0f, 0.0f, 1.0f), ToRadians(g));
中,角度从度转换为辐射角。CCD_ 2是一个非常微小的进步(在程度上)。也许您应该将其更改为g+=1.0
,以便在任何帧中都有明显的变化。
最后,在进行转换之前,您必须通过身份矩阵初始化调制解调器矩阵(transform
)。否则,新的转换将在前一帧的转换之上进行。这将导致网格迅速缩小,平移到无位置:
float g = 0, t = 0;
void OnUserUpdate() override {
transform = Matrix4();
transform.Translate(Vector3(t, t, 0.0f));
transform.Rotate(Vector3(0.0f, 0.0f, 1.0f), ToRadians(g));
transform.Scale(Vector3(0.5f, 0.5f, 0.5f));
t += 0.1 * DeltaTime;
g += 5;
Shaders.SetMat4(shader, "transform", transform);
}
- 如何正确地将分支添加到已存在的树中
- 如何正确地将带有指针的数组传递给函数
- 如何正确地推回然后遍历堆中对象的向量?
- 我是否不正确地集中了这些字符数组?
- 如何通过另一个对象中的命令正确地从一个对象返回数据
- 我怎样才能通过友谊正确地履行我的职能?
- C++:如何正确地将 deque::front() 返回的变量从函数中传递出去?
- 我应该如何正确地将packaged_task移动到 lambda?
- 如何正确地将SHA1CryptoServiceProvider转换为c ++?
- 如何正确地对 constexpr 函数进行单元测试
- 我的动态链接队列在同一输出流中调用时不正确地输出三个返回函数
- 如何正确地将boost::asio::buffers_iterator转换为InputIterator
- 如何正确地检查变量char中包含的字母
- 当试图交换可变模板类时,如何正确地重载operator=
- 如何正确地将 boost::optional<std::chrono::d uration> 作为函数参数?
- 如何在C++中正确地将我的语言字符输出到HTML文件中
- 如果我的类正确地管理了一个资源,那么拥有智能的poointer有什么意义
- 如何正确地编写这些条件,以便它们在控制台中工作
- 正确地编写一个类,并将pthread与vlc库和c++一起使用
- 如何正确地使用opengl转换的矩阵进行数学运算