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教程(三)实现对模型材质參数的控制的更多相关文章

  1. Hadoop作业性能指标及參数调优实例 (三)Hadoop作业性能參数调优方法

    作者: Shu, Alison Hadoop作业性能调优的两种场景: 一.用户观察到作业性能差,主动寻求帮助. (一)eBayEagle作业性能分析器 1. Hadoop作业性能异常指标 2. Had ...

  2. Qt 3D教程(二)初步显示3D的内容

    Qt3D教程(二)初步显示3D的内容 前一篇很easy,全然就没有牵涉到3D的内容,它仅仅是我们搭建3D应用的基本框架而已,而这一篇.我们将要利用它来初步地显示3D的内容了! 本次目的是将程序中间的内 ...

  3. Chromium与CEF的多进程模型及相关參数

    CEF基于Chromium,也是多进程模型.关于进程模型.參考这里:https://www.chromium.org/developers/design-documents/process-model ...

  4. [译]Vulkan教程(31)加载模型

    [译]Vulkan教程(31)加载模型 Loading models 加载模型 Introduction 入门 Your program is now ready to render textured ...

  5. Chem 3D中怎么创建立体模型

    ChemDraw作为一款很受大家欢迎的化学绘图软件,其在绘制平面化学方面的功能已经非常的强大了,其实它也可以绘制3D图形.Chem 3D就是绘制3D图形的重要组件.而且为了满足不同的用户绘图的需求,可 ...

  6. 使用Qt 3D Studio 2.4显着提升性能(渲染速度提高了565%)

    发布于2019年6月18日星期二11评论Qt 3D Studio 2.4显着改善性能 发表于Biz Circuit&Dev Loop,设计,图形,性能,Qt 3D Studio 除了有效使用系 ...

  7. 『MXNet』第三弹_Gluon模型参数

    MXNet中含有init包,它包含了多种模型初始化方法. from mxnet import init, nd from mxnet.gluon import nn net = nn.Sequenti ...

  8. 移植QT5.6到嵌入式开发板(史上最详细的QT移植教程)

    目前网上的大多数 QT 移植教程还都停留在 qt4.8 版本,或者还有更老的 Qtopia ,但是目前 Qt 已经发展到最新的 5.7 版本了,我个人也已经使用了很长一段时间的 qt5.6 for w ...

  9. Smart3D系列教程8之 《模型合并——相邻地区多次建模结果合并》

    迄今为止,Wish3D已经出品推出了7篇系列教程,从倾斜摄影的原理方法.采集照片的技巧.Smart3D各模块的功能应用.小物件的照片重建.大区域的地形重建到DSM及正射影像的处理生产,立足于建模软件的 ...

随机推荐

  1. U盘安装 CentOS 64bit (dell c6100, CentOS6.3, 64bit)

    在淘宝买了一个server,dell c6100,64bit, 曾经系统是black apple.近期又买了一块企业级硬盘打算装CentOS. 综合各方面原因决定安装6.3版本号. 我參考了http: ...

  2. Uva 12012 Detection of Extraterrestrial 求循环节个数为1-n的最长子串长度 KMP

    题目链接:option=com_onlinejudge&Itemid=8&page=show_problem&problem=3163">点击打开链接 题意: ...

  3. Mina airQQ聊天 client篇(三)

    开发工具 (FlashBuilder4.7) 程序类型(Adobe Air) Flex Air做的桌面程序,效果还挺好看的.最主要是Socket这一块,它也是异步的,而且在Flex中的事件机制比較强大 ...

  4. 在Kali上安装打印机

    在Kali 2.0上安装打印机 最近在玩儿渗透测试,就把自己的办公电脑做成了Kali,可是发现办公室的网络打印机没办法正常使用,上网查了一下,把整个过程简单的记录一下,省的忘记了 1.安装cups a ...

  5. hdoj--2098--分拆素数和(枚举)

    分拆素数和 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Subm ...

  6. c++ std

    高中只是听说过stl,每次问老师老师都会说“有毒,千万别学”,于是stl有毒的言论深深的印在我脑海,看到就恐惧,于是一直没有学,但是大学后确实很多用到stl的地方必须去学习了. 现在想想老师当年的说法 ...

  7. [MySQL] 统计函数记录

    时间段统计========== 按年汇总,统计:select sum(mymoney) as totalmoney, count(*) as sheets from mytable group by ...

  8. MYSQL中 FIND_IN_SET 函数

    每天掌握一点,你的知识财富就多一点  今天在维护项目的时候发现了个MYSQL的FIND_IN_SET函数,之前接触太浅,今天又涨点知识了.下面是做个测试 1.创建一张test表,并添加数据 2.编写s ...

  9. Js radio

    <input type="radio" name="sex" value="1" />男 <input type=&quo ...

  10. json字符串与json对象的相互转换

    什么是 JSON ? JSON 指的是 JavaScript 对象表示法(JavaScript Object Notation) JSON 是轻量级的文本数据交换格式 JSON 独立于语言 * JSO ...