qt提供了q3d进行三维开发,虽然这个框架没有得到大量运用也不是那么成功,性能上也有很大的欠缺,但是普通的点到为止的应用展示还是可以的。 其中就包括华丽绚烂的三维图表,数据量不大的时候是可以使用的。 前面介绍了基础的q3d散点图、柱状图,本篇介绍基础的三维曲面图。
依赖QtDataVisualization。在安装qt的时候要选择安装QtDataVisualization模块。
【资料图】
Q3D的散点图,性能大约支撑1000个点可以不卡顿,具体依赖pc,1000个点是什么 概念,可以理解为:10x10x10的区域,每个区域一个数据点。
Q3D的柱状图,性能跟散点图类似。
Q3D的柱状图,性能跟散点图类似。
Q3DSurface类提供了渲染3D曲面图的方法。该类使开发人员能够渲染3D表面图,并通过自由旋转场景来查看它们。可以通过QSurface3DSeries控制曲面的视觉财产,例如绘制模式和着色。 Q3DSurface通过在用户用鼠标左键点击的数据点上显示高亮显示的球(当使用默认输入处理程序时)或通过QSurface3DSeries进行选择来支持选择。选择指针附带一个标签,在默认情况下,该标签显示数据点的值和点的坐标。轴上显示的值范围和标签格式可以通过QValue3DAxis进行控制。 要旋转图形,请按住鼠标右键并移动鼠标。缩放是使用鼠标滚轮完成的。两者都假设默认的输入处理程序正在使用中。 如果没有将任何轴明确设置为Q3DSurface,则会创建不带标签的临时默认轴。这些默认轴可以通过轴访问器进行修改,但只要明确设置了方向的任何轴,该方向的默认轴就会被破坏。
首先,构造Q3D曲面。由于在本例中,我们将图形作为顶级窗口运行,因此需要清除Qt::FramelessWindowHint标志,该标志在默认情况下设置:
Q3DSurface surface; surface.setFlags(surface.flags() ^ Qt::FramelessWindowHint);
现在Q3DSurface已准备好接收要渲染的数据。创建数据元素以接收值:
QSurfaceDataArray *data = new QSurfaceDataArray;QSurfaceDataRow *dataRow1 = new QSurfaceDataRow;QSurfaceDataRow *dataRow2 = new QSurfaceDataRow;
首先将数据喂给行元素,然后将它们的指针添加到数据元素:
*dataRow1 << QVector3D(0.0f, 0.1f, 0.5f) << QVector3D(1.0f, 0.5f, 0.5f);*dataRow2 << QVector3D(0.0f, 1.8f, 1.0f) << QVector3D(1.0f, 1.2f, 1.0f);*data << dataRow1 << dataRow2;、
创建新系列并为其设置数据:
QSurface3DSeries *series = new QSurface3DSeries;series->dataProxy()->resetArray(data); surface.addSeries(series);
最后,设置为可见:
surface.show();
创建和显示此图所需的完整代码为:
#include <QtDataVisualization>using namespace QtDataVisualization;int main(int argc, char **argv){ QGuiApplication app(argc, argv); Q3DSurface surface; surface.setFlags(surface.flags() ^ Qt::FramelessWindowHint); QSurfaceDataArray *data = new QSurfaceDataArray; QSurfaceDataRow *dataRow1 = new QSurfaceDataRow; QSurfaceDataRow *dataRow2 = new QSurfaceDataRow; *dataRow1 << QVector3D(0.0f, 0.1f, 0.5f) << QVector3D(1.0f, 0.5f, 0.5f); *dataRow2 << QVector3D(0.0f, 1.8f, 1.0f) << QVector3D(1.0f, 1.2f, 1.0f); *data << dataRow1 << dataRow2; QSurface3DSeries *series = new QSurface3DSeries; series->dataProxy()->resetArray(data); surface.addSeries(series); surface.show(); return app.exec();}
运行效果:
场景可以被旋转、放大,并且可以选择一个项目来查看其位置,但在这个最小的代码示例中不包括其他交互。
如何确认,则是在帮助文件中查看是否有Q3dscatter类。一般是安装了模块才会有对应的帮助文件。没有则重新安装qt或者单独安装该模块。
Q3d是在数据可视化模块中,需要在pro或者pri配置文件中添加。
QT += datavisualization
使用到Q3DBar相关类中添加头文件,主要使用到Q3DBar、QBar3DSeries、QBarDataRow等等。
#include <Q3DBars>#include <Q3DTheme>#include <QBar3DSeries>#include <QVector3D>
这时候还是无法使用对应的类,需要添加命名空间才行:
using namespace QtDataVisualization;
下面是包含注释的Q3DSurface基础构建流程(注意轴的显示,查看末尾“入坑一”,注意数据的成面规则,查看“入坑二”
_pQ3DSurface = new Q3DSurface();_pContainer = QWidget::createWindowContainer(_pQ3DSurface, this);// 设置轴文本{ // 注意笛卡尔坐标 _pQ3DSurface->axisX()->setTitle("经度(°)"); _pQ3DSurface->axisX()->setTitleVisible(true); _pQ3DSurface->axisY()->setTitle("高度(m)"); _pQ3DSurface->axisY()->setTitleVisible(true); _pQ3DSurface->axisZ()->setTitle("纬度(°)"); _pQ3DSurface->axisZ()->setTitleVisible(true);}// 设置轴范围{ // 注意笛卡尔坐标 _pQ3DSurface->axisX()->setRange(0, 359); _pQ3DSurface->axisY()->setRange(0, 100); _pQ3DSurface->axisZ()->setRange(0, 359);}// 生成一个曲线_pSurface3DSeries = new QSurface3DSeries(_pQ3DSurface);// 设置渲染平滑_pSurface3DSeries->setMeshSmooth(true);// 设置渲染模式// DrawWireframe : 绘制栅格// DrawSurface : 绘制表面// DrawSurfaceAndWireframe : 绘制栅格和图表面_pSurface3DSeries->setDrawMode(QSurface3DSeries::DrawSurface);// 视图添加该曲线_pQ3DSurface->addSeries(_pSurface3DSeries);// 设置阴影质量_pQ3DSurface->setShadowQuality(QAbstract3DGraph::ShadowQualitySoftLow);// 设置视角_pQ3DSurface->scene()->activeCamera()->setCameraPreset(Q3DCamera::CameraPresetIsometricLeft);// 设置子网格_pQ3DSurface->activeTheme()->setGridEnabled(true);#if 1// 添加模拟数据QSurfaceDataArray *pSurfaceDataArray = new QSurfaceDataArray;#if 1#if 1// 这是 z 纬度for(int n = 0; n < 360; n++){ QSurfaceDataRow *pSurfaceDataRow = new QSurfaceDataRow; // 这是 x 经度 for(int m = 0; m < 360; m++) { // 注意与笛卡尔坐标进行映射 *pSurfaceDataRow << QVector3D(m, n / 7 + m / 7, n); } *pSurfaceDataArray << pSurfaceDataRow;}#elsefor(int n = 0; n < 360; n++){ QSurfaceDataRow *pSurfaceDataRow = new QSurfaceDataRow; // 这是 x 经度 for(int m = 0; m < 360; m++) { // 注意与笛卡尔坐标进行映射 *pSurfaceDataRow << QVector3D(m, qrand() % 100, n); LOG << n << m; } *pSurfaceDataArray << pSurfaceDataRow;}#endif#elseQSurfaceDataRow *pSurfaceDataRow1 = new QSurfaceDataRow;QSurfaceDataRow *pSurfaceDataRow2 = new QSurfaceDataRow;QSurfaceDataRow *pSurfaceDataRow3 = new QSurfaceDataRow;// 行与行之间,要形成一个四点成面*pSurfaceDataRow1 << QVector3D(0, 0, 0) << QVector3D(359, 20, 0);*pSurfaceDataRow2 << QVector3D(50, 20, 179) << QVector3D(359, 40, 179);*pSurfaceDataRow3 << QVector3D(100, 80, 359) << QVector3D(359, 100, 359);*pSurfaceDataArray << pSurfaceDataRow1 << pSurfaceDataRow2 << pSurfaceDataRow3;#endif// 添加数据(自动冲掉之前的数据)_pSurface3DSeries->dataProxy()->resetArray(pSurfaceDataArray);#endif_pQ3DSurface->addSeries(_pSurface3DSeries);_pQ3DSurface->show();
#ifndef Q3DSURFACEWIDGET_H#define Q3DSURFACEWIDGET_H#include <QWidget>#include <Q3DSurface>#include <Q3DTheme>#include <QSurface3DSeries>#include <QVector3D>using namespace QtDataVisualization;namespace Ui {class Q3dSurfaceWidget;}class Q3dSurfaceWidget : public QWidget{ Q_OBJECTpublic: explicit Q3dSurfaceWidget(QWidget *parent = 0); ~Q3dSurfaceWidget();protected: void initControl();protected: void resizeEvent(QResizeEvent *event);private: Ui::Q3dSurfaceWidget *ui;private: Q3DSurface *_pQ3DSurface; // q3d平面曲线图 QWidget *_pContainer; // q3d窗口容器 QSurface3DSeries *_pSurface3DSeries ; // q3d柱状图数据};#endif // Q3DSURFACEWIDGET_H
#include "Q3dSurfaceWidget.h"#include "ui_Q3dSurfaceWidget.h"#include <Q3DTheme>#include <QDebug>#include <QDateTime>//#define LOG qDebug()<<__FILE__<<__LINE__//#define LOG qDebug()<<__FILE__<<__LINE__<<__FUNCTION__//#define LOG qDebug()<<__FILE__<<__LINE__<<QThread()::currentThread()//#define LOG qDebug()<<__FILE__<<__LINE__<<QDateTime::currentDateTime().toString("yyyy-MM-dd")#define LOG qDebug()<<__FILE__<<__LINE__<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss:zzz")Q3dSurfaceWidget::Q3dSurfaceWidget(QWidget *parent) : QWidget(parent), ui(new Ui::Q3dSurfaceWidget), _pQ3DSurface(0), _pContainer(0), _pSurface3DSeries(0){ ui->setupUi(this); QString version = "v1.0.0"; initControl();}Q3dSurfaceWidget::~Q3dSurfaceWidget(){ delete ui;}void Q3dSurfaceWidget::initControl(){ _pQ3DSurface = new Q3DSurface(); _pContainer = QWidget::createWindowContainer(_pQ3DSurface, this); // 设置轴文本 { // 注意笛卡尔坐标 _pQ3DSurface->axisX()->setTitle("经度(°)"); _pQ3DSurface->axisX()->setTitleVisible(true); _pQ3DSurface->axisY()->setTitle("高度(m)"); _pQ3DSurface->axisY()->setTitleVisible(true); _pQ3DSurface->axisZ()->setTitle("纬度(°)"); _pQ3DSurface->axisZ()->setTitleVisible(true); } // 设置轴范围 { // 注意笛卡尔坐标 _pQ3DSurface->axisX()->setRange(0, 359); _pQ3DSurface->axisY()->setRange(0, 100); _pQ3DSurface->axisZ()->setRange(0, 359); } // 生成一个曲线 _pSurface3DSeries = new QSurface3DSeries(_pQ3DSurface); // 设置渲染平滑 _pSurface3DSeries->setMeshSmooth(true); // 设置渲染模式 // DrawWireframe : 绘制栅格 // DrawSurface : 绘制表面 // DrawSurfaceAndWireframe : 绘制栅格和图表面 _pSurface3DSeries->setDrawMode(QSurface3DSeries::DrawSurface); // 视图添加该曲线 _pQ3DSurface->addSeries(_pSurface3DSeries); // 设置阴影质量 _pQ3DSurface->setShadowQuality(QAbstract3DGraph::ShadowQualitySoftLow); // 设置视角 _pQ3DSurface->scene()->activeCamera()->setCameraPreset(Q3DCamera::CameraPresetIsometricLeft); // 设置子网格 _pQ3DSurface->activeTheme()->setGridEnabled(true);#if 1 // 添加模拟数据 QSurfaceDataArray *pSurfaceDataArray = new QSurfaceDataArray;#if 1#if 1 // 这是 z 纬度 for(int n = 0; n < 360; n++) { QSurfaceDataRow *pSurfaceDataRow = new QSurfaceDataRow; // 这是 x 经度 for(int m = 0; m < 360; m++) { // 注意与笛卡尔坐标进行映射 *pSurfaceDataRow << QVector3D(m, n / 7 + m / 7, n); } *pSurfaceDataArray << pSurfaceDataRow; }#else for(int n = 0; n < 360; n++) { QSurfaceDataRow *pSurfaceDataRow = new QSurfaceDataRow; // 这是 x 经度 for(int m = 0; m < 360; m++) { // 注意与笛卡尔坐标进行映射 *pSurfaceDataRow << QVector3D(m, qrand() % 100, n); LOG << n << m; } *pSurfaceDataArray << pSurfaceDataRow; }#endif#else QSurfaceDataRow *pSurfaceDataRow1 = new QSurfaceDataRow; QSurfaceDataRow *pSurfaceDataRow2 = new QSurfaceDataRow; QSurfaceDataRow *pSurfaceDataRow3 = new QSurfaceDataRow; // 行与行之间,要形成一个四点成面 *pSurfaceDataRow1 << QVector3D(0, 0, 0) << QVector3D(359, 20, 0); *pSurfaceDataRow2 << QVector3D(50, 20, 179) << QVector3D(359, 40, 179); *pSurfaceDataRow3 << QVector3D(100, 80, 359) << QVector3D(359, 100, 359); *pSurfaceDataArray << pSurfaceDataRow1 << pSurfaceDataRow2 << pSurfaceDataRow3;#endif // 添加数据(自动冲掉之前的数据) _pSurface3DSeries->dataProxy()->resetArray(pSurfaceDataArray);#endif _pQ3DSurface->addSeries(_pSurface3DSeries); _pQ3DSurface->show();}void Q3dSurfaceWidget::resizeEvent(QResizeEvent *event){ if(_pContainer) { _pContainer->setGeometry(rect()); }}
x精度,y维度,z高度(海拔高度)映射错误
x,y,z实际是遵循笛卡尔坐标集
先理解坐标,然后z轴方向,数据也要替换(按照x,y,z来排列,改为x,z,y) &emso;
数据显示映射错误
点成面,需要遵循4点成面的规则,和opengl相关3点成面和4点成面的原理类似。
相邻行与行之间,要形成面,修改后展示如下:
标签:
前言qt提供了q3d进行三维开发,虽然这个框架没有得到大量运用也不是那么成功,性能上也有很大的欠缺,但...
淡蓝的天,青青的草,画面简单而优雅,透露出一种高冷中含有一丝温暖的气质,这就是“小清新”。小清新...
数字基建扩容提速-建好数字底座数字经济的关键是算力,算力的关键是数据中心、网络硬件和云计算。
五一小长假离我们越来越近。目前,该档期已有17部新片扎堆上映,题材涵盖动作、悬疑、犯罪、喜剧、爱情...
1、生理性黄疸2、由于新生儿胆红素的代谢特点,约50%-60%的足月儿和80%的早产儿会出现生理性黄疸。足月...
原标题:米兰国米再迎欧冠德比时隔20年再次半决赛相遇央视网消息:国米淘汰本菲卡晋级欧冠半决赛,将对...
据人民日报报导,4月18日,中国国家统计局发布数据,初步核算,今年一季度中国国内生产总值284997亿元,...
25记三分创历史!无字鹿大胜热火扳1-1巴特勒25分,热火队,雄鹿队,霍勒迪,吉米·巴特勒
今天来聊聊关于直流符号键盘能打出来,直流符号一横三点怎么打出来的文章,现在就为大家来简单介绍下直...
近日第四届教学大师奖、杰出教学奖和创新创业英才奖颁奖典礼举行清华大学电机系教授于歆杰荣获杰出教学...
泡泡玛特披露一季度最新业务情况线下迎大规模复苏
2023赛季中超联赛已正式拉开帷幕山东泰山新赛季主场首场比赛将于4月21日19:35在济南奥体中心体育场举行...
排班从之前的“做7休1”变成“做5休1”甚至“做4休3”,部分一线工人最近“没班加”的烦恼,从微观视角...
解答:1、t恤长袖牛仔外套黑色紧身牛仔裤。2、20度的天气很适合穿牛仔外套,外套里面是简单的长袖t恤,...
4月19日浙版传媒发布2022年度分配预案,拟10派3 5元(含税),预计派现金额合计为7 78亿元。派现额占净...
(记者王恩博)中国国家发改委新闻发言人孟玮19日在北京表示,中国将研究合理缩减外资准入负面清单。孟玮...
4月19日,光弘科技(300735)融资买入2067 24万元,融资偿还1343 0万元,融资净买入724 25万元,融资余额3 47亿元。
23山西债19票面利率为2 9300%
近期合金低位偏弱运行:其中硅铁抵抗式下跌,指数并未跌破前低;锰硅下跌趋势流畅,价格重心一路向南,...
潮新闻讯编者按:4月4日下午,张家界天门山景区发生一起“多人跳崖”事件,四人来自不同的四个省份、互...
一、申请书工伤认定要多长时间才能知道结果申请书工伤认定要六十天才能知道结果。根据《工伤保险条例》...
很多小伙伴的vivoy76s手机系统都是oriainos系统,那么我们用不习惯这个风格我们应该如何将这个风格改成...
原标题:马秀兰当选武威市人民政府市长4月16日,在武威市第五届人民代表大会第三次会议第二次全体会议上...
宝宝足部稚嫩,步伐不稳,开始学走路时,家长除了日常要做好引导防护之外,为孩子选一双舒适护足的鞋子...
新华社北京4月19日电(记者吴文诩)北京市公安局刑事侦查总队总队长孙海涛19日在北京长峰医院火灾事故情况...
报道称,乌军使用迫击炮打击新卡霍夫卡市中心,城市部分地区电力中断,地方当局要求居民进入掩体
4月19日,第31届世界大学生夏季运动会进入倒计时100天。四川在线记者走进东安湖体育公园、成都现代五项...
小伙伴们,你们好,今天小夏来聊聊一篇关于高达00第2季,关于高达00第2季简述的文章,网友们对这件事情都...
柚子的含糖量通常为6 3%-12%。柚实大如瓠,香甜多汁,单果重一般有1 25-2千克,最大可达3 5千克;果...
前面给大家介绍了好几个逃生型的道具,很多玩家都说,在游戏中只有逃生型的道具吗?当然不是,本期就带...
Copyright @ 2015-2018 东方it网版权所有 备案号:沪ICP备2020036824号-8 联系邮箱:562 66 29@qq.com