在最近接到的需求是这样的,画一个折线图,关键点使用空心的圆点标识出来,鼠标移动到关键点上,显示出当前数值;鼠标移走数值消失。

我们遇到这个需求的时候,第一时间就会想到使用 QLineSeries 画折线图。首先初始化

  1.  
    QChart *chart = new QChart();
  2.  
    chart->legend()->setVisible(false);
  3.  
    ui->chartView->setChart(chart);
  4.  
    ui->chartView->setRenderHint(QPainter::Antialiasing);
  5.  
     
  6.  
    chart->setBackgroundBrush(QBrush(QColor(248, 251, 255)));

将每个点添加到QLineSeries序列中。然后就会形成折线图。如下:

  1.  
    QChart *chart = ui->chartView->chart();
  2.  
    chart->removeAllSeries();
  3.  
    chart->removeAxis(chart->axisX());
  4.  
    chart->removeAxis(chart->axisY());
  5.  
     
  6.  
    //折线图
  7.  
    QLineSeries *series0 = new QLineSeries();
  8.  
     
  9.  
    QPen pen;
  10.  
    pen.setStyle(Qt::SolidLine);
  11.  
    pen.setWidth(4);
  12.  
    pen.setColor(QColor(21, 100, 255));
  13.  
    series0->setPen(pen);//折现序列的线条设置
  1.  
    QLineSeries *series0 = (QLineSeries *)ui->chartView->chart()->series().at(0);
  2.  
     
  3.  
    series0->clear();
  4.  
     
  5.  
    qsrand(QTime::currentTime().second());
  6.  
     
  7.  
    qreal t=0, y1, intv=1;
  8.  
    qreal rd;
  9.  
    int cnt=16;
  10.  
    for (int i=0; i<cnt; i++)
  11.  
    {
  12.  
    rd = (qrand() % 100);
  13.  
    y1=rd;
  14.  
    series0->append(t, y1);
  15.  
    t+=intv;
  16.  
    }

这是完成了第一步,画出来了折线图。但是对于那些圆点要显示出来的话我们可以考虑使用QScatterSeries来画一些离散的点。

  1.  
    QScatterSeries *series1 = new QScatterSeries();
  2.  
    series1->setMarkerShape(QScatterSeries::MarkerShapeCircle);//圆形的点
  3.  
    series1->setBorderColor(QColor(21, 100, 255)); //离散点边框颜色
  4.  
    series1->setBrush(QBrush(QColor(21, 100, 255)));//离散点背景色
  5.  
    series1->setMarkerSize(12); //离散点大小
  1.  
    QLineSeries *series0 = (QLineSeries *)ui->chartView->chart()->series().at(0);
  2.  
    QScatterSeries *series1 = (QScatterSeries *)ui->chartView->chart()->series().at(1);
  3.  
     
  4.  
    series0->clear();
  5.  
    series1->clear();
  6.  
     
  7.  
    qsrand(QTime::currentTime().second());
  8.  
     
  9.  
    qreal t=0, y1, intv=1;
  10.  
    qreal rd;
  11.  
    int cnt=16;
  12.  
    for (int i=0; i<cnt; i++)
  13.  
    {
  14.  
    rd = (qrand() % 100);
  15.  
    y1=rd;
  16.  
    series0->append(t, y1);
  17.  
    series1->append(t, y1);
  18.  
     
  19.  
    t+=intv;
  20.  
    }

然后我们添加了一些离散的点,效果如下图:

很显然,虽然添加了离散的圆形的点,但是并没有满足我们的需求,因为需求是空心的圆点。而且控件也没提供相关函数可以设置成空心。但是这里面有3个函数值得注意

  1.  
    series1->setBorderColor(QColor(21, 100, 255)); //离散点边框颜色
  2.  
    series1->setBrush(QBrush(QColor(21, 100, 255)));//离散点背景色
  3.  
    series1->setMarkerSize(12); //离散点大小

因为可以设置一个点的大小,边框和颜色。那我们如果想实现一个空心的离散点就可以这样做:

以同一个位置为圆心,画两个半径不同的实心圆。下面的圆半径大,颜色就是边框的颜色蓝色;上面的圆形半径小,颜色设置为白色。这样两个圆形叠加起来的效果,视觉上就是一个空心的圆形。按照这个思路,我们需要使用2个QScatterSeries序列

series0 : 半径较大,背景为蓝色,充当边框。

series1:半径较小,北京为白色,充电圆心。

  1.  
    //散点图(用于边框)
  2.  
    QScatterSeries *series1 = new QScatterSeries();
  3.  
    series1->setMarkerShape(QScatterSeries::MarkerShapeCircle);//圆形的点
  4.  
    series1->setBorderColor(QColor(21, 100, 255)); //边框颜色
  5.  
    series1->setBrush(QBrush(QColor(21, 100, 255)));//背景颜色
  6.  
    series1->setMarkerSize(12); //点大小
  7.  
     
  8.  
    //散点图(用于中心)
  9.  
    QScatterSeries *series2 = new QScatterSeries();
  10.  
    series2->setMarkerShape(QScatterSeries::MarkerShapeCircle);//圆形的点
  11.  
    series2->setBorderColor(Qt::white);//边框颜色
  12.  
    series2->setBrush(QBrush(Qt::white));//背景颜色
  13.  
    series2->setMarkerSize(6);//点大小
  14.  
     
  15.  
    chart->addSeries(series1);
  16.  
    chart->addSeries(series2);
  1.  
    QLineSeries *series0 = (QLineSeries *)ui->chartView->chart()->series().at(0);
  2.  
    QScatterSeries *series1 = (QScatterSeries *)ui->chartView->chart()->series().at(1);
  3.  
    QScatterSeries *series2 = (QScatterSeries *)ui->chartView->chart()->series().at(2);
  4.  
     
  5.  
    series0->clear();
  6.  
    series1->clear();
  7.  
    series2->clear();
  8.  
     
  9.  
    qsrand(QTime::currentTime().second());
  10.  
     
  11.  
    qreal t=0, y1, intv=1;
  12.  
    qreal rd;
  13.  
    int cnt=16;
  14.  
    for (int i=0; i<cnt; i++)
  15.  
    {
  16.  
    rd = (qrand() % 100);
  17.  
    y1=rd;
  18.  
    series0->append(t, y1);
  19.  
    series1->append(t, y1);
  20.  
    series2->append(t, y1);
  21.  
     
  22.  
    t+=intv;
  23.  
    }

效果如下:

做完这些,我们还有最后一个需求就是鼠标移动到这些离散的点上,要显示出当前点的数值。由于框架并没有提供相关的api,所以我们要自己完成这项工作。我们可以想象,显示的数值需要使用QLabel承载,当鼠标移动到这些点上,QLabel就show,移开就hide。那么怎么确定鼠标是否移动到这些离散点上呢?查阅文档,我们发现QCatterSeries有这样一个信号

他的意思就是,这是一个信号,当鼠标移动到上面,或者从上面移开就会发射这个信号,其中point是移动到哪个点上,当移动到上面,state=true;否则state就为false。

我们可以连接这个信号到我们自己的槽函数

connect(series2, &QScatterSeries::hovered, this, &TDMTrendChartForm::slotPointHoverd);//用于鼠标移动到点上显示数值
 
  1.  
    void TDMTrendChartForm::slotPointHoverd(const QPointF &point, bool state)
  2.  
    {
  3.  
    if (state) {
  4.  
    m_valueLabel->setText(QString::asprintf("%1.0f%", point.y()));
  5.  
     
  6.  
    QPoint curPos = mapFromGlobal(QCursor::pos());
  7.  
    m_valueLabel->move(curPos.x() - m_valueLabel->width() / 2, curPos.y() - m_valueLabel->height() * 1.5);//移动数值
  8.  
     
  9.  
    m_valueLabel->show();//显示出来
  10.  
    }
  11.  
    else
  12.  
    m_valueLabel->hide();//进行隐藏
  13.  
     
  14.  
    }

======================================================================================

补充:完善了代码。增加鼠标悬浮在离散点上,应该显示渐变的边框。

这样我们就完成了相关功能。完整代码可以在这里下载:https://download.csdn.net/download/xiezhongyuan07/10675931

======================================================================================

--------------------- 本文来自 漫步繁华街 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/xiezhongyuan07/article/details/82760103?utm_source=copy

QT QCharts QScatterSeries 空心点阵图,鼠标移动到上面显示数值,鼠标移开数值消失的更多相关文章

  1. jq实现鼠标移动到 图片上放大,移开图片缩小效果(打算封装成插件)

    先看代码 <script> $(function() { $('div').mouseover(function() { $('img').animate({ opacity: '0.9' ...

  2. WPF 鼠标移动到图片变大,移开还原,单击触发事件效果

    <Grid>         <Canvas x:Name="LayoutRoot">             <Image Cursor=" ...

  3. 鼠标划过用户名时在鼠标右下角显示div展示用户资料

    最近做一个网站论坛,为了方便会员之间相互了解,又不想再做一个页面展示用户资料,就想到了鼠标划过用户名时在鼠标右下角显示div展示用户资料这个效果, 这里要注意的该方法不是给每个用户名的旁边都绑定一个d ...

  4. 一个用 js 实现点阵图的编辑器演示

    这是个客户的需求,具体大概是可以在一个 24*8 的点阵图上自由绘制图形,然后数据的存储是按列依次记录,用0和1分别表示是否选中,最终串成一个字符串. 整体需求难度并不复杂,所以在写demo的时候就尽 ...

  5. Qt实战之开发软件数据获取助手(eventFilter处理鼠标按下,event处理鼠标松开)

    前段时间,受朋友委托,需要做一个能够获取别人软件文本框中内容的助手.当然这需要调用win api来解决问题.一开始,我想都没想,就用getWindowText()....居然没用,好郁闷.于是查msd ...

  6. 自己定义View之Chart图标系列(1)——点阵图

    近期要做一些图表类的需求,一開始就去github上看了看,发现开源的图表框架还是蛮多的.可是非常少有全然符合我的需求的.另外就是使用起来比較麻烦.所以就决定自己来造轮子了~~~ 今天要介绍的就是And ...

  7. Android实现二值点阵图识别

    好好学习,天天向上 本文已收录至我的Github仓库DayDayUP:github.com/RobodLee/DayDayUP,欢迎Star 前言 我这几天在做一个东西,就是一张像二维码这样的 n*n ...

  8. 小问题总结:鼠标点击到输入框(input)里的时候,输入框的提示消失,鼠标再移开,输入框提示出现

    问题如标题: 鼠标点击到输入框(input)里的时候,输入框的提示消失,鼠标再移开,输入框提示出现.如图所示:   做法如下: <input type="text" name ...

  9. CSS3鼠标悬停图片上浮显示描述代码

    效果:http://hovertree.com/texiao/css3/20/ 效果图: 代码如下: <!doctype html> <html lang="zh" ...

随机推荐

  1. EcOS安装

    从ubuntu 拷贝到 centos cd /media ls cd ./sf_EcOS 这个目录就是共享目录,名字可能不一样 cp -r studio.zip /home/ 1. 查看版本 cent ...

  2. JavaScript对象参考手册

    1.array 属性: constructor 返回原型函数: length 数组个数: prototype 向对象添加属性和方法 方法: concat() 连接两个或多个数组,并返回结果: fill ...

  3. 虚拟机Ubuntu16.04 The system is running in low-graphics mode解决方法!!

    虚拟机Ubuntu16.04无法进入图形界面 The system is running in low-graphics mode 安装的虚拟机Ubuntu16.04 64位本可以正常使用,在安装了许 ...

  4. Ext.example.msg()应用

    ①需要在开发包中包含文件夹example/shared中的example.js和example.css两个文件即可. ②在html文件中引入: <script src="../extj ...

  5. CSUOJ 1018 Avatar

    Description In the planet Pandora, Jake found an old encryption algorithm. The plaintext, key and ci ...

  6. 洛谷——P2756 飞行员配对方案问题

    P2756 飞行员配对方案问题 题目背景 第二次世界大战时期.. 题目描述 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞行员,其 ...

  7. 安装部署VMware vSphere 5.5文档 (6-6) 集群和vMotion

    部署VMware vSphere 5.5 实施文档 ########################################################################## ...

  8. 一列道出yield和生成器的真谛

    均匀大小的块 def chunks(l, n): """Yield successive n-sized chunks from l.""" ...

  9. java的反射机制(第三篇)

    本文转载自:http://c.biancheng.net/cpp/html/1782.html Person p=new Person();这是什么?当然是实例化一个对象了.可是这种实例化对象的方法存 ...

  10. VC6配置sqlite数据库

    SQLite官方下载只提供给我们一个sqlite3.dll跟一个sqlite3.def文件,并没有提供用于VC++6.0的lib文件,可以利用sqlite3.def文件生成,步骤如下: 1.下载DLL ...