一杯清酒邀明月
天下本无事,庸人扰之而烦耳。
在这个教程里,我们将在Qt Creator环境中创建OpenGL对象,它将显示一个空的OpenGL窗口,可以在窗口和全屏模式下切换,按ESC退出,它将是我们后面应用程序的基础框架。
Qt中写OpenGL与在VC上还是有不少差别的,对Qt机制不熟悉的朋友,请先大致了解下Qt的机制,再往下看教程。
程序运行时效果如下:
下面进入教程:
新建空的Qt项目,项目名称为myOpenGL,然后往项目中添加新的C++类,类名为MyGLWidget,基类为QGLWidget,类型信息选择“继承自QWidget”。添加完成后,打开项目文件myOpenGL.pro,将代码补全如下:
 1 TARGET = myOpenGL
 2 TEMPLATE = app
 3  
 4 HEADERS += \
 5     myglwidget.h
 6  
 7 SOURCES += \
 8     main.cpp \
 9     myglwidget.cpp
10  
11 QT       += core gui
12  
13 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
14  
15 QT       += opengl

然后保存该文件。下面打开myglwidget.h文件,将类声明补全如下:

 1 #ifndef MYGLWIDGET_H
 2 #define MYGLWIDGET_H
 3  
 4 #include <QWidget>
 5 #include <QGLWidget>
 6  
 7 class MyGLWidget : public QGLWidget
 8 {
 9     Q_OBJECT
10 public:
11     explicit MyGLWidget(QWidget *parent = 0);
12     ~MyGLWidget();
13     
14 protected:
15     //对3个纯虚函数的重定义
16     void initializeGL();
17     void resizeGL(int w, int h);
18     void paintGL();
19     
20     void keyPressEvent(QKeyEvent *event);           //处理键盘按下事件
21     
22 private:
23     bool fullscreen;                                //是否全屏显示
24 };
25  
26 #endif // MYGLWIDGET_H

再到myglwidget.cpp文件中先包含#include<GL/glu.h>,#include<QKeyEvent>头文件,然后添加类中函数的定义:

 1 MyGLWidget::MyGLWidget(QWidget *parent) :
 2     QGLWidget(parent)
 3 {
 4     fullscreen = false;
 5 }
 6  
 7 MyGLWidget::~MyGLWidget()
 8 {
 9     
10 }
构造函数中只需对fullscreen初始化,析构函数暂时并不需要做什么。
下面是initializeGL()的定义:
 1 void MyGLWidget::initializeGL()                         //此处开始对OpenGL进行所以设置
 2 {
 3     glClearColor(0.0, 0.0, 0.0, 0.0);                   //黑色背景
 4     glShadeModel(GL_SMOOTH);                            //启用阴影平滑
 5     
 6     glClearDepth(1.0);                                  //设置深度缓存
 7     glEnable(GL_DEPTH_TEST);                            //启用深度测试
 8     glDepthFunc(GL_LEQUAL);                             //所作深度测试的类型
 9     glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST);  //告诉系统对透视进行修正
10 }
glClearColor()函数用来设置清除屏幕时使用的颜色,4个参数分别用来设置红、绿、蓝颜色分量和Alpha值,它们的取值范围都是0.0~1.0,这里4个参数都为0.0,表示纯黑色。然后设置了阴影平滑,这样可以使色彩和光照更加精细。
接下来的三行必须做的是关于depth buffer(深度缓存)的。将深度缓存设想为屏幕后面的层。深度缓存不断地对物体进入屏幕内部有多深进行跟踪。我们本节的程序其实没有真正的使用深度缓存,但几乎所有在屏幕上显示3D场景OpenGL程序都使用深度缓存。它的排序决定那个物体先画。这样就不会将一个圆形后面的正方形画到圆形前面来。深度缓存是OpenGL十分重要的部分。最后我们希望进行最好的透视修正。这会十分轻微的影响性能,但使得透视图看起来好一点。
下面是resizeGL()的定义:
 1 void MyGLWidget::resizeGL(int w, int h)                 //重置OpenGL窗口的大小
 2 {
 3     glViewport(0, 0, (GLint)w, (GLint)h);               //重置当前的视口
 4     glMatrixMode(GL_PROJECTION);                        //选择投影矩阵
 5     glLoadIdentity();                                   //重置投影矩阵
 6     //设置视口的大小
 7     gluPerspective(45.0, (GLfloat)w/(GLfloat)h, 0.1, 100.0);
 8     glMatrixMode(GL_MODELVIEW);                         //选择模型观察矩阵
 9     glLoadIdentity();                                   //重置模型观察矩阵
10 }
glViewport()函数用来设置视口的大小。使用glMatrixMode()设置了投影矩阵,投影矩阵用来为场景增加透视,后面使用了glLoadIdentity()重置投影矩阵,这样可以将投影矩阵恢复到初始状态。gluPerspective()用来设置透视投影矩阵,这里设置视角为45°,纵横比为窗口的纵横比,最近的位置为0.1,最远的位置为100,这两个值是场景中所能绘制的深度的临界值。可以想象,离我们眼睛比较近的东西看起来比较大,而比较远的东西看起来就比较小。最后设置并重置了模型视图矩阵。
下面是paintGL()的定义:
1 void MyGLWidget::paintGL()                              //从这里开始进行所以的绘制
2 {
3     glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); //清除屏幕和深度缓存
4     glLoadIdentity();                                   //重置当前的模型观察矩阵
5 }
paintGL()函数包含了所以的绘图代码,任何想在屏幕上显示的东西都将在此段代码中出现。以后每个教程中都会在这个函数增加代码,已达到绘图目的。
最后是键盘事件处理函数KeyPressEvent()的定义,由于这与OpenGL关系不大,不做过多解释:
 1 void MyGLWidget::keyPressEvent(QKeyEvent *event)
 2 {
 3     switch (event->key()) 
 4     {
 5     //F1为全屏和普通屏的切换键
 6     case Qt::Key_F1:
 7         fullscreen = !fullscreen;
 8         if (fullscreen)
 9         {
10             showFullScreen();
11         }
12         else
13         {
14             showNormal();
15         }
16         updateGL();
17         break;
18     //ESC为退出键
19     case Qt::Key_Escape:
20         close();
21     }
22 }

最后再向项目中添加main.cpp文件,更改内容如下:

 1 #include <QApplication>
 2 #include "myglwidget.h"
 3  
 4 int main(int argc, char *argv[])
 5 {
 6     QApplication app(argc, argv);
 7     
 8     MyGLWidget w;
 9     w.resize(400, 300);
10     w.show();
11     
12     return app.exec();
13 }

现在就可以运行程序查看效果了!

posted on 2020-11-27 14:29  一杯清酒邀明月  阅读(1073)  评论(0编辑  收藏  举报