Qt::QOpenGLWidget渲染带光照的立方体

2023-11-27 03:40

本文主要是介绍Qt::QOpenGLWidget渲染带光照的立方体,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一 学前知识准备    

二 动手实践

三 参考文章


一 学前知识准备    

    这里为了方便不用之前的方法,之前的方法需要引入很多三方的数学库和纹理,模型等库,这里我们班用Qt工具来做更加简单方便。 在看过前面几篇文章之后,在正式开始之前还需要了解opengl渲染流程上的一些东西。

opengl坐标系统  了解opengl整个渲染流程坐标是如何转换的

OpenGL图形渲染管线、VBO、VAO、EBO概念及用例 opengl渲染cpu和gpu怎么传递大块数据

透视投影详解  理解摄像机的透视原理

OpenGL ES着色器语言之变量和数据类型  了解着色器程序中的基本变量类型

opengl如何渲染一个三角形   学会如何编写最简单着色器程序渲染最基本的渲染单位三角形

摄像机原理    理解3d场景中摄像机的实现原理

二 动手实践

立方体顶点着色器

#version 330 core
//顶点着色器uniform mat4 mvp_matrix;
uniform mat4 model_matrix;layout (location = 0) in vec3 a_position;   //空间坐标
layout (location = 1) in vec2 a_texcoord;   //纹理坐标
layout (location = 2) in vec3 a_normal;	    //法线向量out vec2 v_texcoord;
out vec3 FragPos;
out vec3 Normal;void main()
{FragPos = vec3(model_matrix * vec4(a_position, 1.0));Normal =  mat3(transpose(inverse(model_matrix))) * a_normal;  v_texcoord = a_texcoord;gl_Position = mvp_matrix * vec4(a_position,1);
}

立方体像素着色器

#version 330 core
//像素着色器in vec3 Normal;  
in vec3 FragPos;  uniform vec3 lightPos; 
uniform vec3 viewPos; 
uniform vec3 lightColor;
uniform sampler2D texture;in vec2 v_texcoord;void main()
{//环境光照float ambientStrength = 0.3;vec3 ambient = ambientStrength * lightColor;// 漫反射光照 vec3 norm = normalize(Normal);vec3 lightDir = normalize(lightPos - FragPos);float diff = max(dot(norm, lightDir), 0.0);vec3 diffuse = diff * lightColor;// 镜面光照float specularStrength = 1.0;vec3 viewDir = normalize(viewPos - FragPos);vec3 reflectDir = reflect(-lightDir, norm);  float spec = pow(max(dot(viewDir, reflectDir), 0.0), 2);vec3 specular = specularStrength * spec * lightColor;  //获取纹理中对应的颜色vec4 textureColor = texture2D(texture, v_texcoord);//把纹理中的颜色和光照参数相乘gl_FragColor = vec4(ambient + diffuse + specular, 1.0) * textureColor;
}

c++设置gpu数据

class MainWidget : public QOpenGLWidget, protected QOpenGLFunctions
{Q_OBJECT
public:explicit MainWidget(QWidget *parent = 0);~MainWidget();
protected:void mousePressEvent(QMouseEvent *e) override;void mouseReleaseEvent(QMouseEvent *e) override;void wheelEvent(QWheelEvent *event) override;void timerEvent(QTimerEvent *e) override;void initializeGL() override;void resizeGL(int w, int h) override;void paintGL() override;void initShaders();void initTextures();
private:QBasicTimer timer;QOpenGLShaderProgram cubeProgram;QOpenGLShaderProgram lightinhProgram;GeometryEngine *geometries;QOpenGLTexture *texture;QMatrix4x4 projection;QVector2D mousePressPosition;QVector3D rotationAxis;qreal angularSpeed;qreal viewChageDisOld;qreal viewChageDisNew;QQuaternion rotation;
};
#include "mainwidget.h"
#include <QMouseEvent>
#include <math.h>
#include <QTime>
#include <QDebug>//模型主动刷新帧率
#define ACTION_FPS  60
#define LIGHT_COLOR QVector3D(1.2f, 1.0f, 2.0f)
#define EYE_CENTER  QVector3D(0.0, 0.0, 8.0)
#define LIGHT_POS   QVector3D(2.5f, 2.0f, -1.0f)void MainWidget::initShaders()
{if (!lightinhProgram.addShaderFromSourceFile(QOpenGLShader::Vertex, ":/base_lighting.vs"))close();if (!lightinhProgram.addShaderFromSourceFile(QOpenGLShader::Fragment, ":/base_lighting.fs"))	close();if (!lightinhProgram.link())	close();if (!cubeProgram.addShaderFromSourceFile(QOpenGLShader::Vertex, ":/shader.vs"))close();if (!cubeProgram.addShaderFromSourceFile(QOpenGLShader::Fragment, ":/shader.fs"))close();if (!cubeProgram.link())close();geometries->setCubeGLData(&cubeProgram);geometries->setLightingGLData(&lightinhProgram);
}
void MainWidget::initTextures()
{texture = new QOpenGLTexture(QImage(":/cube.png").mirrored());texture->setMinificationFilter(QOpenGLTexture::Nearest);texture->setMagnificationFilter(QOpenGLTexture::Linear);texture->setWrapMode(QOpenGLTexture::Repeat);
}void MainWidget::resizeGL(int w, int h)
{qreal aspect = qreal(w) / qreal(h ? h : 1);const qreal zNear = 2.0, zFar = 20.0, fov = 45.0;projection.setToIdentity();projection.perspective(fov, aspect, zNear, zFar);
}//opengl渲染函数
void MainWidget::paintGL()
{glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);//开启深度测试glEnable(GL_DEPTH_TEST);//开启遮挡剔除glEnable(GL_CULL_FACE);texture->bind();qreal viewChangeRate = viewChageDisNew / 1000.f;viewChageDisOld = viewChageDisNew;QMatrix4x4 viewmatrix;//计算view矩阵viewmatrix.lookAt(EYE_CENTER, QVector3D(0, 0, -20), QVector3D(0, 1, 0));//模型坐标转换矩阵坐标QMatrix4x4 modelmatrix;//平移至左下角modelmatrix.translate(-1.6, -0.8, 0);//鼠标滚动旋转角度modelmatrix.rotate(rotation);//滚轮缩放modelmatrix.scale(1.0 * (1 + viewChangeRate));//设置cube着色器变量QMatrix4x4 mvpmatrix = projection * viewmatrix * modelmatrix;cubeProgram.bind();cubeProgram.setUniformValue("lightColor", LIGHT_COLOR);cubeProgram.setUniformValue("lightPos", LIGHT_POS);cubeProgram.setUniformValue("viewPos", EYE_CENTER);cubeProgram.setUniformValue("mvp_matrix", mvpmatrix);cubeProgram.setUniformValue("texture", 0);cubeProgram.setUniformValue("model_matrix", modelmatrix);geometries->drawCubeGeometry(&cubeProgram);//设置光源cube着色器变量QMatrix4x4 lampmodelmatrix;lampmodelmatrix.translate(LIGHT_POS);//缩放lampmodelmatrix.scale(0.4);QMatrix4x4 lampmvpmatrix = projection * viewmatrix * lampmodelmatrix;lightinhProgram.bind();lightinhProgram.setUniformValue("mvp_matrix", lampmvpmatrix);lightinhProgram.setUniformValue("lightColor", LIGHT_COLOR);geometries->drawLighting(&lightinhProgram);
}

这里只展示一些关键的代码,完整的代码地址:

gitee地址

github地址

最终效果展示:

三 参考文章

opengl教程之纹理

opengl教程之基础光照

OpenGL + Qt: 2 - 走向3D,画正四面体

这篇关于Qt::QOpenGLWidget渲染带光照的立方体的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/426989

相关文章

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【QT】基础入门学习

文章目录 浅析Qt应用程序的主函数使用qDebug()函数常用快捷键Qt 编码风格信号槽连接模型实现方案 信号和槽的工作机制Qt对象树机制 浅析Qt应用程序的主函数 #include "mywindow.h"#include <QApplication>// 程序的入口int main(int argc, char *argv[]){// argc是命令行参数个数,argv是

Python QT实现A-star寻路算法

目录 1、界面使用方法 2、注意事项 3、补充说明 用Qt5搭建一个图形化测试寻路算法的测试环境。 1、界面使用方法 设定起点: 鼠标左键双击,设定红色的起点。左键双击设定起点,用红色标记。 设定终点: 鼠标右键双击,设定蓝色的终点。右键双击设定终点,用蓝色标记。 设置障碍点: 鼠标左键或者右键按着不放,拖动可以设置黑色的障碍点。按住左键或右键并拖动,设置一系列黑色障碍点

使用Qt编程QtNetwork无法使用

使用 VS 构建 Qt 项目时 QtNetwork 无法使用的问题 - 摘叶飞镖 - 博客园 (cnblogs.com) 另外,强烈建议在使用QNetworkAccessManager之前看看这篇文章: Qt 之 QNetworkAccessManager踏坑记录-CSDN博客 C++ Qt开发:QNetworkAccessManager网络接口组件 阅读目录 1.1 通用API函数

Qt多语种开发教程

Qt作为跨平台的开发工具,早已应用到各行各业的软件开发中。 今天讲讲,Qt开发的正序怎么做多语言开发。就是说,你设置中文,就中文显示;设置英语就英文显示,设置繁体就繁体显示,设置发育就显示法语等。 开发环境(其实多语种这块根环境没太大关系):win10,Qt.5.12.10 一.先用QtCreator创建一个简单的桌面程序 1.工程就随便命名“LanguageTest”,其他默认。 2.在设计师

Qt中window frame的影响

window frame 在创建图形化界面的时候,会创建窗口主体,上面会多出一条,周围多次一圈细边,这就叫window frame窗口框架,这是操作系统自带的。 这个对geometry的一些属性有一定影响,主要体现在Qt坐标系体系: 窗口当中包含一个按钮,这个按钮的坐标系是以父元素为参考,那么这个参考是widget本体作为参考,还是window frame作为参考,这两种参考体系都存在

【Qt】定时器事件

定时器事件 在之前学习QTimer中实现了定时器的功能,而在QTimer背后是QTimerEvent定时器事件进行支撑的。在QObject中提供了一个timeEvent这个函数。 startTimer启动定时器killTimer关闭定时器 Qt 中在进⾏窗⼝程序的处理过程中,经常要周期性的执⾏某些操作,或者制作⼀些动画效果,使⽤定 时器就可以实现。所谓定时器就是在间隔⼀定时间后,去执⾏某⼀

QT 编译报错:C3861: ‘tr‘ identifier not found

问题: QT 编译报错:C3861: ‘tr’ identifier not found 原因 使用tr的地方所在的类没有继承自 QObject 类 或者在不在某一类中, 解决方案 就直接用类名引用 :QObject::tr( )

OpenGL ES 2.0渲染管线

http://codingnow.cn/opengles/1504.html Opengl es 2.0实现了可编程的图形管线,比起1.x的固定管线要复杂和灵活很多,由两部分规范组成:Opengl es 2.0 API规范和Opengl es着色语言规范。下图是Opengl es 2.0渲染管线,阴影部分是opengl es 2.0的可编程阶段。   1. 顶点着色器(Vert

在 Qt Creator 中,输入 /** 并按下Enter可以自动生成 Doxygen 风格的注释

在 Qt Creator 中,当你输入 /** 时,确实会自动补全标准的 Doxygen 风格注释。这是因为 Qt Creator 支持 Doxygen 以及类似的文档注释风格,并且提供了代码自动补全功能。 以下是如何在 Qt Creator 中使用和显示这些注释标记的步骤: 1. 自动补全 Doxygen 风格注释 在 Qt Creator 中,你可以这样操作: 在你的代码中,将光标放在