Qt 3D教程(三)实现对模型材质參数的控制
Qt 3D教程(三)实现对模型材质參数的控制
蒋彩阳原创文章,首发地址:http://blog.csdn.net/gamesdev/article/details/47131841。欢迎同行前来探讨。
上一篇教程介绍的是显示一个三维模型的基本步骤,接下来我们须要实现的是加入材质,而且希望我们通过button来控制材质的參数。
这种效果看起来非常像一个3D模型材质编辑器的样子。
那我们来尝试一下吧。
首先我们对Settings这个类进行改动,给它增添一些属性,比方说环境光、漫反射、镜面反射以及反射系数。
通过Q_PROPERTY宏以及一系列的setter和getter函数,我们就能够做到这一点。
class Settings: public QObject
{
Q_OBJECT
Q_PROPERTY( bool showModel READ showModel WRITE setShowModel NOTIFY showModelChanged )
Q_PROPERTY( QColor ambient READ ambient WRITE setAmbient NOTIFY ambientChanged )
Q_PROPERTY( QColor diffuse READ diffuse WRITE setDiffuse NOTIFY diffuseChanged )
Q_PROPERTY( QColor specular READ specular WRITE setSpecular NOTIFY specularChanged )
Q_PROPERTY( float shininess READ shininess WRITE setShininess NOTIFY shininessChanged )
public:
explicit Settings( QObject* parent = Q_NULLPTR ); bool showModel( void ) { return m_showModel; }
void setShowModel( bool showModel ); QColor ambient( void ) { return m_ambient; }
void setAmbient( const QColor& ambient ); QColor diffuse( void ) { return m_diffuse; }
void setDiffuse( const QColor& diffuse ); QColor specular( void ) { return m_specular; }
void setSpecular( const QColor& specular ); float shininess( void ) { return m_shininess; }
void setShininess( float shininess );
signals:
void showModelChanged( void );
void ambientChanged( void );
void diffuseChanged( void );
void specularChanged( void );
void shininessChanged( void );
protected:
bool m_showModel;
QColor m_ambient, m_diffuse, m_specular;
float m_shininess;
};
以下是Settings一些函数的实现:
Settings::Settings( QObject* parent ): QObject( parent )
{
m_showModel = true;
m_ambient = QColor( 153, 51, 26 );
m_diffuse = QColor( 51, 153, 26 );
m_specular = QColor( 153, 230, 26 );
m_shininess = 0.6;
} void Settings::setShowModel( bool showModel )
{
if ( m_showModel == showModel ) return;
m_showModel = showModel;
emit showModelChanged( );
} void Settings::setAmbient( const QColor& ambient )
{
if ( m_ambient == ambient ) return;
m_ambient = ambient;
emit ambientChanged( );
} void Settings::setDiffuse( const QColor& diffuse )
{
if ( m_diffuse == diffuse ) return;
m_diffuse = diffuse;
emit diffuseChanged( );
} void Settings::setSpecular( const QColor& specular )
{
if ( m_specular == specular ) return;
m_specular = specular;
emit specularChanged( );
} void Settings::setShininess( float shininess )
{
if ( m_shininess == shininess ) return;
m_shininess = shininess;
emit shininessChanged( );
}
随后我们声明槽函数,点击环境光、漫反射和镜面反射的时候。它都会设置button的背景色,然后设置m_settings的相关成员。
void MainWindow::decorateButton( QPushButton* button, const QColor& color )
{
QString styleSheetTemplate( "background: rgb( %1, %2, %3 )" );
QString styleSheet = styleSheetTemplate.
arg( color.red( ) ).arg( color.green( ) ).
arg( color.blue( ) );
button->setStyleSheet( styleSheet );
} void MainWindow::on_ambientButton_clicked()
{
QPushButton* button = qobject_cast<QPushButton*>( sender( ) );
QColor color, prevColor;
prevColor = m_settings.ambient( );
color = QColorDialog::getColor( prevColor, this, "请选择一个颜色" );
decorateButton( button, color );
m_settings.setAmbient( color );
} void MainWindow::on_diffuseButton_clicked()
{
QPushButton* button = qobject_cast<QPushButton*>( sender( ) );
QColor color, prevColor;
prevColor = m_settings.diffuse( );
color = QColorDialog::getColor( prevColor, this, "请选择一个颜色" );
decorateButton( button, color );
m_settings.setDiffuse( color );
} void MainWindow::on_specularButton_clicked()
{
QPushButton* button = qobject_cast<QPushButton*>( sender( ) );
QColor color, prevColor;
prevColor = m_settings.specular( );
color = QColorDialog::getColor( prevColor, this, "请选择一个颜色" );
decorateButton( button, color );
m_settings.setSpecular( color );
} void MainWindow::on_shininessEdit_returnPressed( void )
{
m_settings.setShininess( ui->shininessEdit->text( ).toFloat( ) );
}
最后我们在QML中加入PhongMaterial这个类,这个类在C++中是Qt3D::Render::QPhongMaterial,它提供了基于Phong光照模型的这种材质,提供了一种很真实的显示效果。我们使用_settings这个上下文属性将上述的材质属性绑定到PhongMaterial中。加入了PhongMaterial的QML代码例如以下:
import Qt3D 2.0
import Qt3D.Renderer 2.0 Entity
{
id: root Camera
{
id: camera
position: Qt.vector3d( 0.0, 20.0, 100.0 )
projectionType: CameraLens.PerspectiveProjection
fieldOfView: 45
aspectRatio: 16.0 / 9.0
nearPlane : 0.1
farPlane : 1000.0
upVector: Qt.vector3d( 0.0, 1.0, 0.0 )
viewCenter: Qt.vector3d( 0.0, 20.0, 0.0 )
} components: FrameGraph
{
ForwardRenderer
{
clearColor: Qt.rgba( 0.2, 0, 0, 1 )
camera: camera
}
} Entity
{
Mesh
{
id: chestMesh
source: "qrc:/assets/Chest.obj"
enabled: _settings.showModel
} // 新加入的内容
PhongMaterial
{
id: phongMaterial
ambient: _settings.ambient
diffuse: _settings.diffuse
specular: _settings.specular
shininess: _settings.shininess
} components: [ chestMesh, phongMaterial ]
} Configuration
{
controlledCamera: camera
}
}
程序执行截图例如以下:
本次教程的代码均在我的github中,感兴趣的同行们能够通过git clone或者是直接下载我的git项目来获取到本套教程的全部源码。
Qt 3D教程(三)实现对模型材质參数的控制的更多相关文章
- Hadoop作业性能指标及參数调优实例 (三)Hadoop作业性能參数调优方法
作者: Shu, Alison Hadoop作业性能调优的两种场景: 一.用户观察到作业性能差,主动寻求帮助. (一)eBayEagle作业性能分析器 1. Hadoop作业性能异常指标 2. Had ...
- Qt 3D教程(二)初步显示3D的内容
Qt3D教程(二)初步显示3D的内容 前一篇很easy,全然就没有牵涉到3D的内容,它仅仅是我们搭建3D应用的基本框架而已,而这一篇.我们将要利用它来初步地显示3D的内容了! 本次目的是将程序中间的内 ...
- Chromium与CEF的多进程模型及相关參数
CEF基于Chromium,也是多进程模型.关于进程模型.參考这里:https://www.chromium.org/developers/design-documents/process-model ...
- [译]Vulkan教程(31)加载模型
[译]Vulkan教程(31)加载模型 Loading models 加载模型 Introduction 入门 Your program is now ready to render textured ...
- Chem 3D中怎么创建立体模型
ChemDraw作为一款很受大家欢迎的化学绘图软件,其在绘制平面化学方面的功能已经非常的强大了,其实它也可以绘制3D图形.Chem 3D就是绘制3D图形的重要组件.而且为了满足不同的用户绘图的需求,可 ...
- 使用Qt 3D Studio 2.4显着提升性能(渲染速度提高了565%)
发布于2019年6月18日星期二11评论Qt 3D Studio 2.4显着改善性能 发表于Biz Circuit&Dev Loop,设计,图形,性能,Qt 3D Studio 除了有效使用系 ...
- 『MXNet』第三弹_Gluon模型参数
MXNet中含有init包,它包含了多种模型初始化方法. from mxnet import init, nd from mxnet.gluon import nn net = nn.Sequenti ...
- 移植QT5.6到嵌入式开发板(史上最详细的QT移植教程)
目前网上的大多数 QT 移植教程还都停留在 qt4.8 版本,或者还有更老的 Qtopia ,但是目前 Qt 已经发展到最新的 5.7 版本了,我个人也已经使用了很长一段时间的 qt5.6 for w ...
- Smart3D系列教程8之 《模型合并——相邻地区多次建模结果合并》
迄今为止,Wish3D已经出品推出了7篇系列教程,从倾斜摄影的原理方法.采集照片的技巧.Smart3D各模块的功能应用.小物件的照片重建.大区域的地形重建到DSM及正射影像的处理生产,立足于建模软件的 ...
随机推荐
- 上机题目(中级)- 用小数形式输出指定符号出现的频率 (Java)
题目例如以下:
- POJ 题目3461 Oulipo(KMP)
Oulipo Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 26479 Accepted: 10550 Descript ...
- [JavaEE]Spring配置文件总结
首先来看一个标准的Spring配置文件 applicationContext.xml <?xml version="1.0" encoding="UTF-8&quo ...
- 爬虫概念与编程学习之如何爬取视频网站页面(用HttpClient)(二)
先看,前一期博客,理清好思路. 爬虫概念与编程学习之如何爬取网页源代码(一) 不多说,直接上代码. 编写代码 运行 <!DOCTYPE html><html><head& ...
- C语言文件操作函数之ferror & feof & clearerr
这些函数都是和文件读写时发生错误有关,下面一一分析: 1:ferror 原型:int ferror(FILE * fp) 作用:测试一个文件流是否被设置了错误标识符,如果是返回非 0 整数,否则返回 ...
- (转载)Android项目实战(二十八):Zxing二维码实现及优化
Android项目实战(二十八):Zxing二维码实现及优化 前言: 多年之前接触过zxing实现二维码,没想到今日项目中再此使用竟然使用的还是zxing,百度之,竟是如此牛的玩意. 当然,项目中 ...
- MFC框架下Opengl窗口闪屏问题解决方案
转自https://blog.csdn.net/niusiqiang/article/details/43116153 虽然启用了双缓冲,但是仍然会出闪屏的情况,这是由于OpenGL自己有刷新背景的函 ...
- CorelDRAW 2018新增功能介绍
2018年4月11日,CorelDRAW 2018正式对外发布,最新版设计软件包已经过全面更新,是近几年来发行的最强大版本,可协助绘图专业人士将创意转换为令人惊艳的专业视觉设计.CorelDRAW 2 ...
- js浏览器问题
前段时间做了个项目,里面关于手机移动端下载的问题 开始是判断微信.ios和android系统的下载 因为微信屏蔽点击事件和链接的缘故,需要通过打开新页面来进行下载 ios和android的下载分别为不 ...
- 关于layui.laypage.render 刷新首页没有分页问题
前言: 最近写项目遇到一个问题,就是使用vue里的layui.laypage.render 分页时,刷新首页会只有一页,但后台传来的数据是有50多页的,所有的数据也都一一对应,调了好久debug,终于 ...