当前位置:首页 > 芯闻号 > 充电吧
[导读]这一课将把如下图片做成一个飘动的旗帜,其实主要还是用到了纹理映射。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();
}

运行效果



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

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭