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

所属分类: 软件编程 / C 语言 阅读数: 137
收藏 0 赞 0 分享

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();
}

运行结果截图:

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

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多