zoukankan      html  css  js  c++  java
  • QT QCharts QScatterSeries 空心点阵图,鼠标移动到上面显示数值,鼠标移开数值消失

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

    我们遇到这个需求的时候,第一时间就会想到使用 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 

  • 相关阅读:
    编写第一个 .NET 微服务
    Docker 基础知识编排在开发机上设置和使用 Kubernetes 环境
    将微服务部署到 Azure Kubernetes 服务 (AKS) 实践
    .NET Task.Run vs Task.Factory.StartNew
    Python上下文管理器
    Python可迭代的对象与迭代器的对比
    开发你的第一个SpringBoot应用
    Flask的Blueprints和Views
    Flask项目发布流程
    长篇大论Python生成器
  • 原文地址:https://www.cnblogs.com/h2zZhou/p/9716832.html
Copyright © 2011-2022 走看看