自定义QChartView实现鼠标放在图表时,显示鼠标位置坐标值(x,y)-CSDN博客
阿里云国内75折 回扣 微信号:monov8 |
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6 |
前言因为需要一次性从文件中加载大量数据到图表中显示所以打算使用qchartview+qscrollarea当横坐标数据超出默认设定的显示范围之后重新设置chartview的宽度和scrollarea内容区域(scrollAreaWidgetContents)的宽度从而实现一次性显示所有数据的目的。因为这样显示之后如果滚动条向右边拖动时图表的纵坐标会看不到为了能够方便的显示纵坐标上的值于是想到在鼠标放在图表时通过一个标签显示鼠标所在位置的坐标。
一、自定义QChartView实现鼠标移动事件获取坐标值。
项目代码根目录添加c++头文件和源文件切记根目录不要在子目录。
showvaluechartview.h
#ifndef SHOWVALUECHARTVIEW_H
#define SHOWVALUECHARTVIEW_H
#include <QChartView>
class ShowValueChartView: public QChartView
{
Q_OBJECT
public:
explicit ShowValueChartView(QWidget *parent = Q_NULLPTR);
protected:
void mouseMoveEvent(QMouseEvent *e);
public: signals:
/** 鼠标在图表位置的数据改变时触发 **/
void onMousePositionValueChanged(QPointF valueGivenSeries);
};
#endif // SHOWVALUECHARTVIEW_H
showvaluechartview.cpp
#include "showvaluechartview.h"
ShowValueChartView::ShowValueChartView(QWidget *parent) : QChartView(parent)
{
setMouseTracking(true);
}
void ShowValueChartView::mouseMoveEvent(QMouseEvent *e)
{
auto const widgetPos = e->position();
auto const scenePos = this->chart()->mapToScene(QPoint(static_cast<int>(widgetPos.x()), static_cast<int>(widgetPos.y())));
auto const chartItemPos = chart()->mapFromScene(scenePos);
auto const valueGivenSeries = chart()->mapToValue(chartItemPos);
// qDebug() << "widgetPos:" << widgetPos;
// qDebug() << "scenePos:" << scenePos;
// qDebug() << "chartItemPos:" << chartItemPos;
// 鼠标所在位置的图表XY坐标值
// qDebug() << "valSeries:" << valueGivenSeries;
emit onMousePositionValueChanged(valueGivenSeries);
}
二、从文件中加载图表数据并从新设置图标宽度和滚动条内容控件宽度方便查看所有数据
1) 设计界面拖一个QWidget到显示图表的区域然后提升控件为ShowValueChartView文章可参考Qt创建自定义View和在布局中使用自定义View的方法_qt自定义布局_Zafir2023的博客-CSDN博客
2初始化chartview控件文章可参考
【精选】QChartView显示实时更新的温度曲线图即动态曲线图。_Zafir2023的博客-CSDN博客
中的initTempChartView函数。
// 初始化完chartview之后添加如下代码需要在windows定义槽函数并实现
// 关联鼠标在图表上移动时的显示坐标的槽函数
connect(ui->chartView, &ShowValueChartView::onMousePositionValueChanged,
this, &YourWindow::onMousePositionValueChanged);
3从文件中加载图表要显示的数据并添加到图表的坐标序列中。
// 此处以横纵坐标都是QValueAxis类型为例。若横坐标是日期型也类似
QList<QPointF> dataList;
for (int i = 0; i < 500; i++) {
QPointF tmp;
tmp.setX(i + 1);
tmp.setY(10);
dataList.append(tmp);
}
// 效率高加载速度快
splineSeries.replace(dataList);
// 重新设置chartview宽度
resizeChartView(dataList);
4根据数据量重新设置控件大小滚动显示图表内容
void YourWindow::resizeChartView(QList<QPointF> dataList) {
int valueCount = dataList.size();//从文件中加载的数据总个数/X轴的刻度个数
// 可根据默认图表宽度在能显示所有横坐标数字的情况下设置固定值
int defaultChartXTickCount = 默认图表的x轴刻度个数(默认图表宽度能显示的最大数据个数)。
// 这个可放在windows构造函数中获取一次即可此处为了简化defaultChartViewWidth 为成员变量
if (defaultChartViewWidth == 0) {
defaultChartViewWidth = ui->chartView->width();//图表默认宽度
}
if (valueCount > defaultChartXTickCount) {
// 设置横坐标数据范围
valueAxisX->setRange(1, valueCount);
//设置坐标轴的精度分成多少份(多少个刻度)最小间隔为Range/(TickCount - 1)
valueAxisX->setTickCount(valueCount);
// 图表视图横向放大倍数
int chartWidthMagnify = 1;
if (valueCount % defaultChartXTickCount == 0) {
chartWidthMagnify = valueCount / defaultChartXTickCount;
} else {
chartWidthMagnify = valueCount / defaultChartXTickCount + 1;
}
// 滚动条控件内部的内容控件宽度决定滚动显示内容的宽度
ui->scrollAreaWidgetContents->setMinimumWidth(defaultChartViewWidth * chartWidthMagnify);
// 图表宽度重新设置容纳所有数据
ui->chartView->setMinimumWidth(defaultChartViewWidth * chartWidthMagnify);
}
}