Qt开发技术:Q3D图表开发笔记(二):Q3DBar三维柱状图介绍、Demo以及代码详解

简介: qt提供了q3d进行三维开发,虽然这个框架没有得到大量运用也不是那么成功,性能上也有很大的欠缺,但是普通的点到为止的应用展示还是可以的。  其中就包括华丽绚烂的三维图表,数据量不大的时候是可以使用的。  上一篇介绍了基础的q3d散点图,本篇介绍基础的柱状图。

前言

  qt提供了q3d进行三维开发,虽然这个框架没有得到大量运用也不是那么成功,性能上也有很大的欠缺,但是普通的点到为止的应用展示还是可以的。

  其中就包括华丽绚烂的三维图表,数据量不大的时候是可以使用的。

  上一篇介绍了基础的q3d散点图,本篇介绍基础的柱状图。


Demo:Q3DScatter散点图演示效果

  

  

  


Q3D提供的三维图表

  依赖QtDataVisualization。在安装qt的时候要选择安装QtDataVisualization模块。

Q3DScatter散点图

  Q3D的散点图,性能大约支撑1000个点可以不卡顿,具体依赖pc,1000个点是什么 概念,可以理解为:10x10x10的区域,每个区域一个数据点。

  

Q3DBars柱状图

  Q3D的柱状图,性能跟散点图类似。

  

Q3DSurface平面凹凸图,平面纹理图

  Q3D的柱状图,性能跟散点图类似。

  

# Q3DBar柱状图 ## 简介   Q3DBars类提供了用于渲染三维条形图的方法。   该类使开发人员能够以3D方式渲染条形图,并通过自由旋转场景来查看它们。

  • 旋转:通过按住鼠标右键并移动鼠标来完成的。
  • 缩放:通过鼠标滚轮完成的。
  • 选择:如果启用了选择,则通过鼠标左键进行选择。
  • 重置视角:通过单击鼠标滚轮,可以将场景重置为默认的摄影机视图。
      在触摸设备中,旋转通过点击和移动来完成,选择通过点击和按住来完成,缩放通过捏来完成。
      如果没有将任何轴明确设置为Q3DBar,则会创建不带标签的临时默认轴。这些默认轴可以通过轴访问器进行修改,但只要明确设置了方向的任何轴,该方向的默认轴就会被破坏。
    Q3DBars支持同时显示多个系列。并非所有系列都必须具有相同数量的行和列。行和列标签取自第一个添加的系列,除非明确定义为行和列轴。

构造最小Q3D柱状图

  首先,构造一个Q3DBars实例。由于在本例中,我们将图形作为顶级窗口运行,因此需要清除Qt::FramelessWindowHint标志,该标志在默认情况下设置:

Q3DBars bars;
bars.setFlags(bars.flags() ^ Qt::FramelessWindowHint);

  构造Q3DBar后,可以通过更改行和列轴上的范围来设置数据窗口。这不是强制性的,因为数据窗口将默认显示系列中的所有数据。如果数据量很大,通常最好只显示其中的一部分。例如,让我们将数据窗口设置为显示前五行和前五列:

bars.rowAxis()->setRange(0, 4);
bars.columnAxis()->setRange(0, 4);

  现在Q3DBars已经准备好接收要渲染的数据了。创建一个包含一行5个值的序列:

QBar3DSeries *series = new QBar3DSeries;
QBarDataRow *data = new QBarDataRow;
*data << 1.0f << 3.0f << 7.5f << 5.0f << 2.2f;
series->dataProxy()->addRow(data);
bars.addSeries(series);

  注意:数据窗口设置为5 x 5,但只添加了一行数据。没添加的其余的行将是空白的。

  最后,设置为可见:

bars.show();

  创建和显示此图所需的完整代码为:

#include <QtDataVisualization>
using namespace QtDataVisualization;
int main(int argc, char **argv)
{
    QGuiApplication app(argc, argv);
    Q3DBars bars;
    bars.setFlags(bars.flags() ^ Qt::FramelessWindowHint);
    bars.rowAxis()->setRange(0, 4);
    bars.columnAxis()->setRange(0, 4);
    QBar3DSeries *series = new QBar3DSeries;
    QBarDataRow *data = new QBarDataRow;
    *data << 1.0f << 3.0f << 7.5f << 5.0f << 2.2f;
    series->dataProxy()->addRow(data);
    bars.addSeries(series);
    bars.show();
    return app.exec();
}

  运行效果:

  

  场景可以被旋转、放大,并且可以选择一个项目来查看其位置,但在这个最小的代码示例中不包括其他交互。


Q3Ddemo构建流程解析

步骤一:确认安装QtDataVisualization模块

  如何确认,则是在帮助文件中查看是否有Q3dscatter类。一般是安装了模块才会有对应的帮助文件。没有则重新安装qt或者单独安装该模块。

  

步骤二:工程配置文件中加入模块

  Q3d是在数据可视化模块中,需要在pro或者pri配置文件中添加。

QT += datavisualization

  

步骤三:添加使用到的头文件

使用到Q3DBar相关类中添加头文件,主要使用到Q3DBar、QBar3DSeries、QBarDataRow等等。

#include <Q3DBars>
#include <Q3DTheme>
#include <QBar3DSeries>
#include <QVector3D>

  

步骤四:添加命名空间

  这时候还是无法使用对应的类,需要添加命名空间才行:

using namespace QtDataVisualization;

  

步骤五:Q3D的图标基础构建框架

  下面是包含注释的Q3DBar基础构建流程,其与散点图在轴(轴名称和轴使用等)和数据(添加数据方式为每一个占位基本都需要,除非你能直接阔过去,从头开始添加的中间有0则不好处理,需要添加0来占位,QBarDataRow自动依次排序的)

_pQ3DBars = new Q3DBars();
_pContainer = QWidget::createWindowContainer(_pQ3DBars, this);
// 设置轴文本
{
    QStringList strList;
    _pQ3DBars->rowAxis()->setTitle("年");
    _pQ3DBars->rowAxis()->setTitleVisible(true);
    strList.clear();
    strList << "2010" << "2011" << "2012" << "2013" << "2014"
               << "2015" << "2016" << "2017" << "2018" << "2019"
               << "2020" << "2021" << "2022" << "2023" ;
    _pQ3DBars->rowAxis()->setLabels(strList);
    _pQ3DBars->columnAxis()->setTitle("月");
    strList.clear();
    strList << "1" << "2" << "3" << "4" << "5"
                << "6" << "7" << "8" << "9" << "10"
                << "11" << "12";
    _pQ3DBars->columnAxis()->setLabels(strList);
    _pQ3DBars->columnAxis()->setTitleVisible(true);
    _pQ3DBars->valueAxis()->setTitle("销售额(万元)");
    _pQ3DBars->valueAxis()->setTitleVisible(true);
}
// 设置轴范围
{
    _pQ3DBars->rowAxis()->setRange(2010 - 2010, 2023 - 2010);   // 从0开始
    _pQ3DBars->columnAxis()->setRange(1 - 1, 12 - 1);           // 从0开始
    _pQ3DBars->valueAxis()->setRange(0, 40);
}
// 生成一个曲线
_pBar3DSeries = new QBar3DSeries(_pQ3DBars);
// 设置渲染平滑
_pBar3DSeries->setMeshSmooth(true);
// 视图添加该曲线
_pQ3DBars->addSeries(_pBar3DSeries);
// 设置阴影质量
_pQ3DBars->setShadowQuality(QAbstract3DGraph::ShadowQualitySoftLow);
// 设置视角
_pQ3DBars->scene()->activeCamera()->setCameraPreset(Q3DCamera::CameraPresetIsometricLeft);
// 设置子网格
_pQ3DBars->activeTheme()->setGridEnabled(true);
#if 1
// 添加模拟数据
QBarDataArray data;
for(int year = 2010; year <= 2023; year++)
{
    QBarDataRow *pBarDataRow = new QBarDataRow;
    for(int month = 1; month <= 12; month++)
    {
        if(year == 2023 && month >= 4)
        {
            LOG << year << month;
            // 当前2023年4月,无数据
            *pBarDataRow << 0;
        }else{
            *pBarDataRow << ((year - 2010) + month);
        }
    }
    data << pBarDataRow;
}
// 添加数据(自动冲掉之前的数据)
_pBar3DSeries->dataProxy()->addRows(data);
#endif


Demo源码

Q3dBarWidget.h
#ifndef Q3DBARWIDGET_H
#define Q3DBARWIDGET_H
#include <QWidget>
#include <Q3DBars>
#include <Q3DTheme>
#include <QBar3DSeries>
#include <QVector3D>
using namespace QtDataVisualization;
namespace Ui {
class Q3dBarWidget;
}
class Q3dBarWidget : public QWidget
{
    Q_OBJECT
public:
    explicit Q3dBarWidget(QWidget *parent = 0);
    ~Q3dBarWidget();
protected:
    void initControl();
protected:
    void resizeEvent(QResizeEvent *event);
private:
    Ui::Q3dBarWidget *ui;
private:
    Q3DBars *_pQ3DBars;             // q3d柱状视图
    QWidget *_pContainer;           // q3d窗口容器
    QBar3DSeries *_pBar3DSeries;    // q3d柱状图数据
};
#endif // Q3DBARWIDGET_H

Q3dBarWidget.cpp

#include "Q3dBarWidget.h"
#include "ui_Q3dBarWidget.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")
Q3dBarWidget::Q3dBarWidget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Q3dBarWidget),
    _pQ3DBars(0),
    _pContainer(0),
    _pBar3DSeries(0)
{
    ui->setupUi(this);
    QString version = "v1.0.0";
    setWindowTitle(QString("q3d柱状图示例 %1(作者:长沙红胖子 QQ:21497936 WX:15173255813 www.hpzwl.com").arg(version));
    initControl();
}
Q3dBarWidget::~Q3dBarWidget()
{
    delete ui;
}
void Q3dBarWidget::initControl()
{
    _pQ3DBars = new Q3DBars();
    _pContainer = QWidget::createWindowContainer(_pQ3DBars, this);
    // 设置轴文本
    {
        QStringList strList;
        _pQ3DBars->rowAxis()->setTitle("年");
        _pQ3DBars->rowAxis()->setTitleVisible(true);
        strList.clear();
        strList << "2010" << "2011" << "2012" << "2013" << "2014"
                << "2015" << "2016" << "2017" << "2018" << "2019"
                << "2020" << "2021" << "2022" << "2023" ;
        _pQ3DBars->rowAxis()->setLabels(strList);
        _pQ3DBars->columnAxis()->setTitle("月");
        strList.clear();
        strList << "1" << "2" << "3" << "4" << "5"
                << "6" << "7" << "8" << "9" << "10"
                << "11" << "12";
        _pQ3DBars->columnAxis()->setLabels(strList);
        _pQ3DBars->columnAxis()->setTitleVisible(true);
        _pQ3DBars->valueAxis()->setTitle("销售额(万元)");
        _pQ3DBars->valueAxis()->setTitleVisible(true);
    }
    // 设置轴范围
    {
        _pQ3DBars->rowAxis()->setRange(2010 - 2010, 2023 - 2010);   // 从0开始
        _pQ3DBars->columnAxis()->setRange(1 - 1, 12 - 1);           // 从0开始
        _pQ3DBars->valueAxis()->setRange(0, 40);
    }
    // 生成一个曲线
    _pBar3DSeries = new QBar3DSeries(_pQ3DBars);
    // 设置渲染平滑
    _pBar3DSeries->setMeshSmooth(true);
    // 视图添加该曲线
    _pQ3DBars->addSeries(_pBar3DSeries);
    // 设置阴影质量
    _pQ3DBars->setShadowQuality(QAbstract3DGraph::ShadowQualitySoftLow);
    // 设置视角
    _pQ3DBars->scene()->activeCamera()->setCameraPreset(Q3DCamera::CameraPresetIsometricLeft);
    // 设置子网格
    _pQ3DBars->activeTheme()->setGridEnabled(true);
#if 1
    // 添加模拟数据
    QBarDataArray data;
    for(int year = 2010; year <= 2023; year++)
    {
        QBarDataRow *pBarDataRow = new QBarDataRow;
        for(int month = 1; month <= 12; month++)
        {
            if(year == 2023 && month >= 4)
            {
                LOG << year << month;
                // 当前2023年4月,无数据
                *pBarDataRow << 0;
            }else{
                *pBarDataRow << ((year - 2010) + month);
            }
        }
        data << pBarDataRow;
    }
    // 添加数据(自动冲掉之前的数据)
    _pBar3DSeries->dataProxy()->addRows(data);
#endif
}
void Q3dBarWidget::resizeEvent(QResizeEvent *event)
{
    if(_pContainer)
    {
        _pContainer->setGeometry(rect());
    }
}


工程模板v1.1.0

  

相关实践学习
DataV Board用户界面概览
本实验带领用户熟悉DataV Board这款可视化产品的用户界面
阿里云实时数仓实战 - 项目介绍及架构设计
课程简介 1)学习搭建一个数据仓库的过程,理解数据在整个数仓架构的从采集、存储、计算、输出、展示的整个业务流程。 2)整个数仓体系完全搭建在阿里云架构上,理解并学会运用各个服务组件,了解各个组件之间如何配合联动。 3&nbsp;)前置知识要求 &nbsp; 课程大纲 第一章&nbsp;了解数据仓库概念 初步了解数据仓库是干什么的 第二章&nbsp;按照企业开发的标准去搭建一个数据仓库 数据仓库的需求是什么 架构 怎么选型怎么购买服务器 第三章&nbsp;数据生成模块 用户形成数据的一个准备 按照企业的标准,准备了十一张用户行为表 方便使用 第四章&nbsp;采集模块的搭建 购买阿里云服务器 安装 JDK 安装 Flume 第五章&nbsp;用户行为数据仓库 严格按照企业的标准开发 第六章&nbsp;搭建业务数仓理论基础和对表的分类同步 第七章&nbsp;业务数仓的搭建&nbsp; 业务行为数仓效果图&nbsp;&nbsp;
相关文章
【Qt 学习笔记】使用QtCreator创建及运行项目 | 项目初始代码解释
【Qt 学习笔记】使用QtCreator创建及运行项目 | 项目初始代码解释
1603 1
|
Web App开发 编解码 安全
视频会议技术 入门探究:WebRTC、Qt与FFmpeg在视频编解码中的应用
视频会议技术 入门探究:WebRTC、Qt与FFmpeg在视频编解码中的应用
1064 4
|
Linux 数据处理 C++
Linux系统编程 C/C++ 以及Qt 中的零拷贝技术: 从底层原理到高级应用(一)
Linux系统编程 C/C++ 以及Qt 中的零拷贝技术: 从底层原理到高级应用
508 0
|
存储 测试技术 UED
Qt中实现界面回放的艺术:从理论到代码“ (“The Art of Implementing UI Playback in Qt: From Theory to Code
Qt中实现界面回放的艺术:从理论到代码“ (“The Art of Implementing UI Playback in Qt: From Theory to Code
243 1
|
存储 Linux API
Linux系统编程 C/C++ 以及Qt 中的零拷贝技术: 从底层原理到高级应用(三)
Linux系统编程 C/C++ 以及Qt 中的零拷贝技术: 从底层原理到高级应用
195 1
|
消息中间件 Linux 数据处理
Linux系统编程 C/C++ 以及Qt 中的零拷贝技术: 从底层原理到高级应用(二)
Linux系统编程 C/C++ 以及Qt 中的零拷贝技术: 从底层原理到高级应用
301 1
|
8月前
|
编解码 UED
Qt侧边栏的动态切换:隐藏与显示技术详解
在现代用户界面设计中,侧边栏(Sidebar)是一个常见的组件,它为用户提供了导航和工具面板的功能。在某些应用场景下,我们可能需要动态地隐藏或显示侧边栏,以优化界面布局或提供更灵活的用户体验。本文将分享如何在Qt框架下实现侧边栏的隐藏与呈现,包括技术细节和代码示例。
739 3
|
8月前
|
UED
Qt侧边栏的动态展示:隐藏与呈现技术详解
在现代用户界面设计中,侧边栏(Sidebar)是一个常见的组件,它为用户提供了便捷的导航和操作入口。在Qt框架中,实现侧边栏的隐藏与呈现不仅能够提升应用的美观度,还能增强用户体验。本文将详细介绍如何在Qt中实现侧边栏的动态隐藏与呈现,包括技术要点和代码实现。
715 0
|
10月前
自己动手写QT多线程demo
本文是作者关于如何编写Qt多线程demo的教程,介绍了如何实现多线程功能,包括可暂停和继续的功能。文章提供了部分示例代码,展示了如何创建线程类、启动和管理线程,以及线程间的通信。同时,还提供了相关参考资料和免费下载链接。
200 0

推荐镜像

更多
  • qt