OpenGL实现3D空间中移动图像

 更新时间:2020年08月18日 16:38:43   作者:yangchuankai  
这篇文章主要为大家详细介绍了OpenGL实现3D空间中移动图像,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

Qt_OpenGL:3D空间中移动图像,供大家参考,具体内容如下

//.h

#ifndef GLWIDGET_H
#define GLWIDGET_H
 
#include <QGLWidget>
#include <QtOpenGL>
 
class QGLWidget;
class QTimer;
 
typedef struct Stars{
public:
 int r, g, b;
 GLfloat dist, angle;
}Stars;
 
class GLWidget : public QGLWidget
{
 Q_OBJECT
 
public:
 GLWidget(QWidget *parent = 0);
 ~GLWidget();
 
protected:
 void initializeGL();
 void paintGL();
 void resizeGL(int w, int h);
 void keyPressEvent(QKeyEvent*);
 void timerEvent(QTimerEvent*);
 
private:
 bool fullscreen;
 GLfloat rotate_angle;
 GLfloat zoom;
 GLfloat title;
 GLfloat spin;
 GLuint loop;
 bool twinkle;
 GLfloat blend;
 
private:
 void loadTextures();
 
 GLuint texture[1];
 
};
 
#endif // GLWIDGET_H

//.cpp

#include "glwidget.h"
#include <glut.h>
#include <QtGui>
#include <QtCore>
 
//好吧我承认全部变量不好
GLfloat light_ambient[4] = {0.5,0.5,0.5,1.0};
GLfloat light_diffiuse[4] = {1.0,1.0,1.0,1.0};
GLfloat light_position[4] = {0.0,0.0,2.0,0.0};
 
static const int num = 50;
static Stars stars[num];
 
GLWidget::GLWidget(QWidget *parent)
 : QGLWidget(parent)
{
 fullscreen = false;
 rotate_angle = 0.0;
 zoom = -15.0;
 title = 90.0;
 spin = 0.1;
 loop = 0;
 twinkle = false;
 blend = false;
 
 startTimer(5);
}
 
void GLWidget::initializeGL(){
 
 setGeometry(300,150,500,500);
 loadTextures();
 glEnable(GL_TEXTURE_2D);
 glShadeModel(GL_SMOOTH);
 glClearColor(0.0, 0.0, 0.0, 0.5);
 glClearDepth(1.0);
 glEnable(GL_DEPTH_TEST);
 glDepthFunc(GL_LEQUAL);
 glHint(GL_PERSPECTIVE_CORRECTION_HINT,GL_NICEST);
 
 glBlendFunc(GL_SRC_ALPHA,GL_ONE);
 glEnable(GL_BLEND);
 
 //为num个星星对象赋初值
 for(loop = 0; loop < num; ++loop){
 stars[loop].angle = 0.0;
 stars[loop].dist = (float(loop)/num) * 5.0;
 stars[loop].r = rand() % 256;
 stars[loop].g = rand() % 256;
 stars[loop].b = rand() % 256;
 }
}
 
void GLWidget::paintGL(){
 
 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
 glBindTexture(GL_TEXTURE_2D, texture[0]);
 for(loop = 0; loop < num; ++loop){
 glLoadIdentity();
 glTranslatef(0.0, 0.0, zoom); //移向屏幕里面
 glRotatef(title, 1.0, 0.0, 0.0); //沿x轴旋转title
 glRotatef(stars[loop].angle, 0.0, 1.0, 0.0); //每个星星沿y轴旋转自己的角度
 glTranslatef(stars[loop].dist, 0.0, 0.0); //平移
 glRotatef(-stars[loop].angle, 0.0, 1.0, 0.0); //沿y轴反转
 glRotatef(-title, 1.0, 0.0, 0.0); //沿x轴反转
 if(twinkle){ //如果星星闪烁
  glColor4ub(stars[num-loop-1].r, stars[num-loop-1].g,
   stars[num-loop-1].b, 255);
  glBegin(GL_QUADS);
  glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f,-1.0f, 0.0f);
  glTexCoord2f(1.0f, 0.0f); glVertex3f( 1.0f,-1.0f, 0.0f);
  glTexCoord2f(1.0f, 1.0f); glVertex3f( 1.0f, 1.0f, 0.0f);
  glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, 0.0f);
  glEnd();
 }
 //如果星星不闪烁
 glRotatef(spin, 0.0f, 0.0f, 1.0f); //沿z轴自转spin
 glColor4ub(stars[loop].r, stars[loop].g, stars[loop].b, 255);
 glBegin(GL_QUADS);
  glTexCoord2f(0.0f, 0.0f); glVertex3f(-1.0f,-1.0f, 0.0f);
  glTexCoord2f(1.0f, 0.0f); glVertex3f( 1.0f,-1.0f, 0.0f);
  glTexCoord2f(1.0f, 1.0f); glVertex3f( 1.0f, 1.0f, 0.0f);
  glTexCoord2f(0.0f, 1.0f); glVertex3f(-1.0f, 1.0f, 0.0f);
 glEnd();
 
 spin += 0.01f;
 stars[loop].angle += float(loop)/num;
 stars[loop].dist -= 0.01f; //距离逐渐减小,即越来越靠近屏幕
 if(stars[loop].dist < 0){
  stars[loop].dist += 5.0f;
  stars[loop].r = rand() % 256;
  stars[loop].g = rand() % 256;
  stars[loop].b = rand() % 256;
 }
 }
}
 
void GLWidget::resizeGL(int w, int h){
 
 if( 0 == h){
 h = 1;
 }
 glViewport(0, 0, (GLsizei)w, (GLsizei)h);
 glMatrixMode(GL_PROJECTION);
 glLoadIdentity();
 gluPerspective(45.0, (GLdouble)w/(GLdouble)h, 0.1, 100.0);
 glColor4f(1.0f, 1.0f, 1.0f, 0.5f);
 glBlendFunc(GL_SRC_ALPHA, GL_ONE);
 glMatrixMode(GL_MODELVIEW);
 glLoadIdentity();
}
 
void GLWidget::keyPressEvent(QKeyEvent *event){
 
 switch(event->key()){
 
 case Qt::Key_T:{
  twinkle = !twinkle;
  updateGL();
  break;
 }
 case Qt::Key_B:{
  blend = !blend;
  if(blend){
  glEnable(GL_BLEND);
  glDisable(GL_DEPTH_TEST);
  }else{
  glDisable(GL_BLEND);
  glEnable(GL_DEPTH_TEST); //色彩混合和深度缓存不能同时开启
  }
  updateGL();
  break;
 }
 case Qt::Key_PageUp:{ //移向屏幕
  zoom -= 0.2;
  updateGL();
  break;
 }
 case Qt::Key_PageDown:{ //移向屏幕外
  zoom += 0.2;
  updateGL();
  break;
 }
 case Qt::Key_Up:{  //加快旋转速度
  title += 0.5;
  updateGL();
  break;
 }
 case Qt::Key_Down:{
  title -= 0.5;
  updateGL();
  break;
 }
 case Qt::Key_F1:{
  fullscreen = !fullscreen;
  if(fullscreen){
  showFullScreen();
  }else{
  setGeometry(300,150,500,500);
  showNormal();
  }
  updateGL();
  break;
 }
 case Qt::Key_Escape:{
  close();
 }
 }
}
 
void GLWidget::loadTextures(){
 
 QImage tex, buf;
 if(!buf.load(":Star.bmp")){
 qWarning("Cannot open the image...");
 QImage dummy(128, 128, QImage::Format_RGB32);
 dummy.fill(Qt::green);
 buf = dummy;
 }
 tex = convertToGLFormat(buf);
 glGenTextures(1, &texture[0]);
 glBindTexture(GL_TEXTURE_2D, texture[0]);
 glTexImage2D(GL_TEXTURE_2D, 0, 3, tex.width(), tex.height(), 0, GL_RGBA,
   GL_UNSIGNED_BYTE, tex.bits());
 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
}
 
void GLWidget::timerEvent(QTimerEvent *){
 updateGL();
}
 
GLWidget::~GLWidget()
{
}

//main.cpp

#include "glwidget.h"
#include <QApplication>
 
int main(int argc, char *argv[])
{
 QApplication a(argc, argv);
 GLWidget w;
 w.show();
 
 return a.exec();
}

运行结果截图:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

您可能感兴趣的文章:

相关文章

  • visual studio 2019编译c++17的方法

    visual studio 2019编译c++17的方法

    这篇文章主要介绍了visual studio 2019编译c++17的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-03-03
  • 手把手教你用C语言实现三子棋

    手把手教你用C语言实现三子棋

    三子棋是黑白棋的一种。三子棋是一种民间传统游戏,又叫九宫棋、圈圈叉叉、一条龙、井字棋等。这篇文章就教你如何用C语言实现三子棋的功能
    2021-08-08
  • 利用Matlab复刻举牌加油小人生成器

    利用Matlab复刻举牌加油小人生成器

    upuptoyou是一款非常有创意的小工具,可以在线生成举牌小人,看起来很可爱,也比较有趣,并能用于表白,或节日送祝福等场景。本文将用Matlab复刻这一小工具,需要的可以参考一下
    2022-03-03
  • 带你理解C语言中的汉诺塔公式

    带你理解C语言中的汉诺塔公式

    大家好,本篇文章主要讲的是带你理解C语言中的汉诺塔公式,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下
    2022-01-01
  • C语言 解压华为固件的实例代码

    C语言 解压华为固件的实例代码

    这是解压华为固件(update.app)的C语言。。其实这也是我翻115翻出来的。。。
    2013-08-08
  • C++名称空间介绍

    C++名称空间介绍

    这篇文章主要介绍了C++名称空间,这是一个很重要的概念——名称空间,有些书本里翻译成了命名空间,对应的英文是namespace。下面来看文章对此的详细介绍吧,需要的小伙伴可以参考一下

    2021-12-12
  • Qt QML使用虚拟键盘的示例代码

    Qt QML使用虚拟键盘的示例代码

    这篇文章主要为大家详细介绍了Qt QML使用虚拟键盘的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-01-01
  • C++实现LeetCode(63.不同的路径之二)

    C++实现LeetCode(63.不同的路径之二)

    这篇文章主要介绍了C++实现LeetCode(63.不同的路径之二),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • c语言实现向上取整计算方法

    c语言实现向上取整计算方法

    这篇文章主要介绍了c语言实现向上取整计算方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-07-07
  • C与C++之间相互调用实例方法讲解

    C与C++之间相互调用实例方法讲解

    这篇文章主要介绍了C与C++之间相互调用的实例方法,大家参考使用吧
    2013-12-12

最新评论