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

//运行结果截图:

时间: 2024-12-29 11:20:27

Qt_OpenGL:3D空间中移动图像的相关文章

3D空间中射线与轴向包围盒AABB的交叉检测算法

引言 在上一节中,我讲述了如何实现射线与三角形的交叉检测算法.但是,我们应该知道,在游戏开发中,一个模型有很多的三角形构成,如果要对所有的物体,所有的三角形进行这种检测,就算现在的计算机运算能力,也是无法高效的完成.所以,我们需要通过其他的手段来提早剔除一些不可能发生交叉的物体,这种早退的思想,大量的运用在3D游戏技术中.在本篇文章中,我将像大家讲述如何实现射线与轴向包围盒AABB的交叉检测.如果读者不明白什么是轴向包围盒,请看这篇文章. Ray-AABB交叉检测算法 现如今,有很多的Ray-A

在3D空间中绘制四边形

在3D空间中绘制四边形 四边形 GL_QUADS OpenGL的GL_QUADS图元用于绘制四边形,它根据每四个顶点绘制一个四边形. 注意,在使用四边形时,必需记住一个重要规则:一个四边形的四个角必须位于同一个平面中(不存在弯曲的四边形).如图所示 四边形带 GL_QUAD_STRIP 该图元指定一个连接的四边形带.它们都保持相同方向的环绕.如图所示 通用多边形 GL_POLYGON 我们可以用它绘制任意数量的多边形.与四边形一样,多边形的所有顶点也必须位于同一平面中.如果想越过这个规则,可以采

使3D空间中物体朝向和其速度方向一致的旋转矩阵计算方案

在3D空间中的物体以某一速度运动,有时候需要这个物体的朝向和速度的方向一致, 为了实现这个目标我们一般借助旋转矩阵 M 来将物体旋转到对应的朝向. 例如速度方向矢量 spdV: Vector3D(1,2,3), X轴基向量为 axis_x: Vector3D(1,0,0), 这个矢量的方向和3D物体不做任何旋转时候的默认朝向一致 3D矢量 cross_x 记录了 axis_x 叉乘 spdV 的结果. 算出矩阵 M 的方法一: 先计算出 spdV 和 axis_x 两矢量之间的弧度值 rad(可

3D空间中射线与三角形的交叉检测算法

引言 射线Ray,在3D图形学中有很多重要的应用.比如,pick操作就是使用射线Ray来实现的,还有诸如子弹射线的碰撞检测等等都可以使用射线Ray来完成.所以,在本次博客中,将会简单的像大家介绍下,如何进行Ray-Triangle的交叉检测. Ray-Triangle交叉检测算法 在Tomas Moller的MT97论文中,提出了一种新的算法.这种算法能够减少以前进行Ray-Triangle交叉检测所需要的内存消耗.在以前,进行Ray-Triangle交叉检测,主要是计算射线与三角形所构成的平面

2D和3D空间中计算两点之间的距离

自己在做游戏的忘记了Unity帮我们提供计算两点之间的距离,在百度搜索了下. 原来有一个公式自己就写了一个方法O(∩_∩)O~,到僵尸到达某一个点之后就向另一个奔跑过去 /// <summary> /// 3维中如何计算两点之间的距离 /// </summary> /// <param name="p1"></param> /// <param name="p2"></param> /// &l

3D空间中射线与轴向包围盒AABB的交叉检测算法【转】

引言 在上一节中,我讲述了如何实现射线与三角形的交叉检测算法.但是,我们应该知道,在游戏开发中,一个模型有很多的三角形构成,如果要对所有的物体,所有的三角形进行这种检测,就算现在的计算机运算能力,也是无法高效的完成.所以,我们需要通过其他的手段来提早剔除一些不可能发生交叉的物体,这种早退的思想,大量的运用在3D游戏技术中.在本篇文章中,我将像大家讲述如何实现射线与轴向包围盒AABB的交叉检测.如果读者不明白什么是轴向包围盒,请看这篇文章. Ray-AABB交叉检测算法 现如今,有很多的Ray-A

3D空间中射线与轴向包围盒AABB的交叉检测算法 【转】

http://blog.csdn.net/i_dovelemon/article/details/38342739 引言 在上一节中,我讲述了如何实现射线与三角形的交叉检测算法. 但是,我们应该知道,在游戏开发中,一个模型有很多的三角形构成,如果要对所有的物体,所有的三角形进行这种检测,就算现在的计算机运算能力,也是无法高 效的完成.所以,我们需要通过其他的手段来提早剔除一些不可能发生交叉的物体,这种早退的思想,大量的运用在3D游戏技术中.在本篇文章中,我将像大家讲 述如何实现射线与轴向包围盒A

OpenGl学习进程(9)在3D空间的绘制实例

    本节将演示在3D空间中绘制图形的几个简单实例:     (1)在3D空间内绘制圆锥体: #include <GL/glut.h> #include <math.h> #pragma comment(linker,"/subsystem:\"windows\" /entry:\"mainCRTStartup\"") #define PI 3.1416 GLfloat xRot = 0; GLfloat yRot =

Direct3D 11 Tutorial 4: 3D Spaces_Direct3D 11 教程4:3D空间

概述 在上一个教程中,我们在应用程序窗口的中心成功渲染了一个三角形. 我们没有太注意我们在顶点缓冲区中拾取的顶点位置. 在本教程中,我们将深入研究3D位置和转换的细节. 本教程的结果将是渲染到屏幕的3D对象. 虽然之前的教程侧重于将2D对象渲染到3D世界,但在这里我们展示了一个3D对象. 资源目录 (SDK root)\Samples\C++\Direct3D11\Tutorials\Tutorial04 Github仓库 3D空间 在上一个教程中,三角形的顶点被有策略地放置,以在屏幕上完美地对