当前位置:首页 > > 充电吧
[导读]这一课将把如下图片做成一个飘动的旗帜,其实主要还是用到了纹理映射。lesson10.h#ifndef LESSON10_H #define LESSON10_H #include#include#i

这一课将把如下图片做成一个飘动的旗帜,其实主要还是用到了纹理映射。


lesson10.h

#ifndef LESSON10_H
#define LESSON10_H

#include#include#includeclass QPainter;
class QOpenGLContext;
class QOpenGLPaintDevice;

class Lesson10 : public QWindow, QOpenGLFunctions_1_1
{
    Q_OBJECT
public:
    explicit Lesson10(QWindow *parent = 0);
    ~Lesson10();

    virtual void render(QPainter *);
    virtual void render();
    virtual void initialize();

public slots:
    void renderNow();

protected:
    void exposeEvent(QExposeEvent *);
    void resizeEvent(QResizeEvent *);
    void keyPressEvent(QKeyEvent *); // 键盘事件
    void timerEvent(QTimerEvent *);  // 定时器

private:
    void loadGLTexture();

private:
    QOpenGLContext *m_context;

    GLfloat m_x_rotate;
    GLfloat m_y_rotate;
    GLfloat m_z_rotate;
    GLuint m_texture[1];

    //我们将使用points数组来存放网格各顶点独立的x,y,z坐标。这里网格由45×45点形成,
    //换句话说也就是由44格×44格的小方格子依次组成了。
    float m_points[45][45][3]; // Points网格顶点数组
};

#endif // LESSON10_H

lessson10.cpp

#include "lesson10.h"

#include#include#include#include#include#includeLesson10::Lesson10(QWindow *parent) :
    QWindow(parent)
  , m_context(0)
  , m_x_rotate(0.0f)
  , m_y_rotate(0.0f)
  , m_z_rotate(0.0f)
{
    setSurfaceType(QWindow::OpenGLSurface);
    startTimer(20);
}

Lesson10::~Lesson10()
{
    glDeleteTextures(1, &m_texture[0]);
}

void Lesson10::render(QPainter *painter)
{
    Q_UNUSED(painter);
}

void Lesson10::render()
{
    glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);

    glViewport(0,0,(GLint)width(),(GLint)height()); // 重置当前视口
    glMatrixMode(GL_PROJECTION);                    // 选择投影矩阵
    glLoadIdentity();                               // 重置投影矩阵为单位矩阵
    gluPerspective(45.0f,(GLdouble)width()/(GLdouble)height(),0.1f,100.0f);

    glMatrixMode(GL_MODELVIEW); // 选择模型视图矩阵
    glLoadIdentity();           // 重置模型视图矩阵为单位矩阵

    float float_x, float_y, float_xb, float_yb;		// 用来将旗形的波浪分割成很小的四边形
    glTranslatef(0.0f,0.0f,-12.0f);				    // 移入屏幕12个单位
    glRotatef(m_x_rotate,1.0f,0.0f,0.0f);			// 绕 X 轴旋转
    glRotatef(m_y_rotate,0.0f,1.0f,0.0f);			// 绕 Y 轴旋转
    glRotatef(m_z_rotate,0.0f,0.0f,1.0f);			// 绕 Z 轴旋转

    glBindTexture(GL_TEXTURE_2D, m_texture[0]);		// 选择纹理
    glBegin(GL_QUADS);					            // 四边形绘制开始
    for(int x = 0; x < 44; x++ )				    // 沿X平面0-44循环(45点)
    {
        for(int y = 0; y < 44; y++ )			    // 沿Y平面0-44循环(45点)
        {
            //接着开始使用循环进行多边形绘制。这里使用整型可以避免我以前所用的int()强制类型转换。
            float_x = float(x)/44.0f;		// 生成X浮点值
            float_y = float(y)/44.0f;		// 生成Y浮点值
            float_xb = float(x+1)/44.0f;	// X浮点值+0.0227f
            float_yb = float(y+1)/44.0f;	// Y浮点值+0.0227f
            //上面我们使用4个变量来存放纹理坐标。每个多边形(网格之间的四边形)分别映射了纹理的1/44×1/44部分。
            //循环首先确定左下顶点的值,然后我们据此得到其他三点的值。
            glTexCoord2f( float_x, float_y);	// 第一个纹理坐标 (左下角)
            glVertex3f( m_points[x][y][0], m_points[x][y][1], m_points[x][y][2] );
            glTexCoord2f( float_x, float_yb );	// 第二个纹理坐标 (左上角)
            glVertex3f( m_points[x][y+1][0], m_points[x][y+1][1], m_points[x][y+1][2] );
            glTexCoord2f( float_xb, float_yb );	// 第三个纹理坐标 (右上角)
            glVertex3f( m_points[x+1][y+1][0], m_points[x+1][y+1][1], m_points[x+1][y+1][2] );
            glTexCoord2f( float_xb, float_y );	// 第四个纹理坐标 (右下角)
            glVertex3f( m_points[x+1][y][0], m_points[x+1][y][1], m_points[x+1][y][2] );
        }
    }
    glEnd();						                 // 四边形绘制结束
}

void Lesson10::initialize()
{
    loadGLTexture();                      // 加载纹理
    glEnable(GL_TEXTURE_2D);              // 启用纹理映射
    glShadeModel(GL_SMOOTH);              // 启用平滑着色
    glClearColor(0.0f, 0.0f, 0.0f, 0.0f); // 黑色背景
    glClearDepth(1.0f);                   // 设置深度缓存
    glEnable(GL_DEPTH_TEST);              // 启用深度测试
    glDepthFunc(GL_LEQUAL);               // 深度测试类型
    // 接着告诉OpenGL我们希望进行最好的透视修正。这会十分轻微的影响性能。但使得透视图看起来好一点。
    glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST);

    glPolygonMode( GL_BACK, GL_FILL );			// 后表面完全填充
    glPolygonMode( GL_FRONT, GL_LINE );			// 前表面使用线条绘制
    // 上面的代码指定使用完全填充模式来填充多边形区域的背面(后面)。
    // 相反,多边形的正面(表面)则使用轮廓线填充了。这些方式完全取决于您的个人喜好。并且与多边形的方位或者顶点的方向有关。
    for(int x=0; x<45; x++)
    {
        for(int y=0; ysetFormat(requestedFormat());
        m_context->create();

        needsInitialize = true;
    }

    m_context->makeCurrent(this);

    if (needsInitialize) {
        initializeOpenGLFunctions();
        initialize();
    }

    render();

    m_context->swapBuffers(this);
}

void Lesson10::loadGLTexture()
{
    //现在载入图像,并将其转换为纹理。
    QImage image(":/image/Tim.bmp");
    image = image.convertToFormat(QImage::Format_RGB888);
    image = image.mirrored();
    glGenTextures(1, &m_texture[0]);//创建纹理
    //使用来自位图数据生成的典型纹理
    glBindTexture(GL_TEXTURE_2D, m_texture[0]);
    glTexImage2D(GL_TEXTURE_2D, 0, 3,image.width(), image.height(),
                 0, GL_RGB, GL_UNSIGNED_BYTE,image.bits());
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);	// 线形滤波
    glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);	// 线形滤波
}

void Lesson10::exposeEvent(QExposeEvent *event)
{
    Q_UNUSED(event);

    if (isExposed())
    {
        renderNow();
    }
}

void Lesson10::resizeEvent(QResizeEvent *event)
{
    Q_UNUSED(event);

    if (isExposed())
    {
        renderNow();
    }
}

void Lesson10::timerEvent(QTimerEvent *event)
{
    for(int y = 0; y < 45; y++ )			// Y平面循环
    {
        GLfloat hold = m_points[0][y][2];	// 存储当前左侧波浪值
        for(int x = 0; x < 44; x++)		    // 沿X平面循环
        {
            // 当前波浪值等于其右侧的波浪值
            m_points[x][y][2] = m_points[x+1][y][2];
        }
        m_points[44][y][2]=hold;			// 刚才的值成为最左侧的波浪值
    }
    //上面所作的事情是先存储每一行的第一个值,然后将波浪左移一下,使图象产生波浪。
    //存储的数值挪到末端以产生一个永无尽头的波浪纹理效果。
    //上面的代码由NeHe(2000年2月)修改过,以消除波浪间出现的细小锯齿。
    //现在增加 xrot , yrot 和 zrot 的值。
    m_x_rotate+=0.3f;								// X 轴旋转
    m_y_rotate+=0.2f;								// Y 轴旋转
    m_z_rotate+=0.4f;								// Z 轴旋转
    renderNow();
    QWindow::timerEvent(event);
}

void Lesson10::keyPressEvent(QKeyEvent *event)
{
    int key=event->key();
    switch(key)
    {
    }
}

main.cpp

#include#includeint main(int argc, char *argv[])
{
    QGuiApplication app(argc, argv);

    QSurfaceFormat format;
    format.setSamples(16);

    Lesson10 window;
    window.setFormat(format);
    window.resize(640, 480);
    window.show();

    return app.exec();
}

运行效果



本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭