QML QTQuick ChartView 将指针传递给 C++
QML QTQuick ChartView pass pointer to C++
我正在尝试基于 qtcharts-qmloscilloscope-example here
制作类似 Qt Quick 应用程序的示波器
在此示例中,轨迹(QTQuick ChartView)预先分配在 QML 中并通过计时器更新。
我希望能够在运行时添加和删除跟踪。
现有应用程序将对基础数据数组的引用作为 QPointF 对象的 QAbstractSeries 传递。此操作在计时器上触发,因此:
Timer {
id: refreshTimer
interval: 1 / 1 * 1000 // 1 Hz
running: dataSource.isRunning
repeat: true
onTriggered: {
dataSource.updateTime();
//dataSource.update(chartView.series(0));
//dataSource.update(chartView.series(1));
//dataSource.update(chartView.series(2));
dataSource.update(chartView);
}
}
现有的更新方法如下所示:
void DataSource::update(QAbstractSeries * series)
{
...
}
如果您只需要固定数量的跟踪并且它们都是单独更新的,那么这没问题。但我希望能够在打开和关闭时添加跟踪。
我尝试将 chartView ID 传递给 update(QChartView *) 函数,但这总是因空指针而中断。
Q_INVOKABLE void DataSource::update(QChartView * view)
{
...
}
我也试过在顶层使用 window->findChildren 并将其传递给 DataSource 的实例。这得到一个有效的指针,但类型为 QQuickItem。如果我将其转换为 QChartView,我也会得到一个空指针。
如何正确地将指向 QChartView 对象的指针传递给 C++?
ChartView 不是 QChartView,而是 QQuickItem,因此转换不起作用。
因此您不能直接访问这些方法,而是使用 QMetaObject,如下所示:
helper.h
#ifndef HELPER_H
#define HELPER_H
#include <QObject>
class QQuickItem;
class Helper : public QObject
{
Q_OBJECT
public:
using QObject::QObject;
Q_INVOKABLE void createSerie(QQuickItem *chartview);
Q_INVOKABLE void removeAllSeries(QQuickItem *chartview);
};
#endif // HELPER_H
helper.cpp
#include "helper.h"
#include <QAbstractAxis>
#include <QAbstractSeries>
#include <QLineSeries>
#include <QMetaObject>
#include <QQuickItem>
#include <random>
#include <cstring>
QT_CHARTS_USE_NAMESPACE
void Helper::createSerie(QQuickItem *chartview){
if(!chartview)
return;
const QMetaObject *mo = chartview->metaObject();
if(std::strcmp(mo->className(), "QtCharts::DeclarativeChart") != 0)
return;
int ix = mo->indexOfEnumerator("SeriesType");
QMetaEnum me = mo->enumerator(ix);
int type = me.keyToValue("SeriesTypeLine");
QAbstractAxis *axis_x = nullptr;
QMetaObject::invokeMethod(chartview, "axisX", Qt::DirectConnection,
Q_RETURN_ARG(QAbstractAxis *, axis_x));
QAbstractAxis *axis_y = nullptr;
QMetaObject::invokeMethod(chartview, "axisY", Qt::DirectConnection,
Q_RETURN_ARG(QAbstractAxis *, axis_y));
QAbstractSeries *serie = nullptr;
QMetaObject::invokeMethod(chartview, "createSeries", Qt::DirectConnection,
Q_RETURN_ARG(QAbstractSeries *, serie),
Q_ARG(int, type),
Q_ARG(QString, "serie from c++"),
Q_ARG(QAbstractAxis *, axis_x),
Q_ARG(QAbstractAxis *, axis_y));
if(QLineSeries *line_serie = qobject_cast<QLineSeries *>(serie)){
static std::default_random_engine e;
static std::uniform_real_distribution<> dis(0, 3);
for(int i=0; i < 14; i++){
line_serie->append(i, dis(e));
}
}
}
void Helper::removeAllSeries(QQuickItem *chartview){
if(!chartview)
return;
const QMetaObject *mo = chartview->metaObject();
if(std::strcmp(mo->className(), "QtCharts::DeclarativeChart") != 0)
return;
QMetaObject::invokeMethod(chartview, "removeAllSeries", Qt::DirectConnection);
}
main.qml
import QtQuick 2.14
import QtQuick.Window 2.14
import QtQuick.Layouts 1.14
import QtQuick.Controls 2.4
import QtCharts 2.14
Window {
visible: true
width: 640
height: 480
title: qsTr("Hello World")
ColumnLayout{
anchors.fill: parent
RowLayout{
Button{
text: "Create serie"
Layout.fillWidth: true
onClicked: helper.createSerie(chartview)
}
Button{
text: "Clear series"
Layout.fillWidth: true
onClicked: helper.removeAllSeries(chartview);
}
}
ChartView {
id: chartview
title: "Line"
antialiasing: true
Layout.fillWidth: true
Layout.fillHeight: true
LineSeries {
name: "LineSeries"
XYPoint { x: 0; y: 0 }
XYPoint { x: 3; y: 2.1 }
XYPoint { x: 8; y: 3.3 }
XYPoint { x: 10; y: 2.1 }
XYPoint { x: 11; y: 4.9 }
XYPoint { x: 12; y: 3.0 }
XYPoint { x: 13; y: 3.3 }
}
axes: [
ValueAxis{
id: xAxis
min: 1.0
max: 15.0
},
ValueAxis{
id: yAxis
min: 0.0
max: 5.0
}
]
}
}
}
下面link是完整代码
存在不使用 QMetaObject::invokeMethod()
的更好变体:可以直接使用 QChart
。只需要扩展以下解决方案 一点:
*.h
public:
Q_INVOKABLE void setSeries(QAbstractSeries *series);
[...]
private:
QXYSeries *mSeries;
QChart *mChart;
[...]
*.cpp
void DataSource::setSeries(QAbstractSeries *series)
{
if (series) {
mSeries = static_cast<QXYSeries *>(series);
mChart = mSeries->chart();
}
}
我正在尝试基于 qtcharts-qmloscilloscope-example here
制作类似 Qt Quick 应用程序的示波器在此示例中,轨迹(QTQuick ChartView)预先分配在 QML 中并通过计时器更新。
我希望能够在运行时添加和删除跟踪。
现有应用程序将对基础数据数组的引用作为 QPointF 对象的 QAbstractSeries 传递。此操作在计时器上触发,因此:
Timer {
id: refreshTimer
interval: 1 / 1 * 1000 // 1 Hz
running: dataSource.isRunning
repeat: true
onTriggered: {
dataSource.updateTime();
//dataSource.update(chartView.series(0));
//dataSource.update(chartView.series(1));
//dataSource.update(chartView.series(2));
dataSource.update(chartView);
}
}
现有的更新方法如下所示:
void DataSource::update(QAbstractSeries * series)
{
...
}
如果您只需要固定数量的跟踪并且它们都是单独更新的,那么这没问题。但我希望能够在打开和关闭时添加跟踪。
我尝试将 chartView ID 传递给 update(QChartView *) 函数,但这总是因空指针而中断。
Q_INVOKABLE void DataSource::update(QChartView * view)
{
...
}
我也试过在顶层使用 window->findChildren 并将其传递给 DataSource 的实例。这得到一个有效的指针,但类型为 QQuickItem。如果我将其转换为 QChartView,我也会得到一个空指针。
如何正确地将指向 QChartView 对象的指针传递给 C++?
ChartView 不是 QChartView,而是 QQuickItem,因此转换不起作用。
因此您不能直接访问这些方法,而是使用 QMetaObject,如下所示:
helper.h
#ifndef HELPER_H
#define HELPER_H
#include <QObject>
class QQuickItem;
class Helper : public QObject
{
Q_OBJECT
public:
using QObject::QObject;
Q_INVOKABLE void createSerie(QQuickItem *chartview);
Q_INVOKABLE void removeAllSeries(QQuickItem *chartview);
};
#endif // HELPER_H
helper.cpp
#include "helper.h"
#include <QAbstractAxis>
#include <QAbstractSeries>
#include <QLineSeries>
#include <QMetaObject>
#include <QQuickItem>
#include <random>
#include <cstring>
QT_CHARTS_USE_NAMESPACE
void Helper::createSerie(QQuickItem *chartview){
if(!chartview)
return;
const QMetaObject *mo = chartview->metaObject();
if(std::strcmp(mo->className(), "QtCharts::DeclarativeChart") != 0)
return;
int ix = mo->indexOfEnumerator("SeriesType");
QMetaEnum me = mo->enumerator(ix);
int type = me.keyToValue("SeriesTypeLine");
QAbstractAxis *axis_x = nullptr;
QMetaObject::invokeMethod(chartview, "axisX", Qt::DirectConnection,
Q_RETURN_ARG(QAbstractAxis *, axis_x));
QAbstractAxis *axis_y = nullptr;
QMetaObject::invokeMethod(chartview, "axisY", Qt::DirectConnection,
Q_RETURN_ARG(QAbstractAxis *, axis_y));
QAbstractSeries *serie = nullptr;
QMetaObject::invokeMethod(chartview, "createSeries", Qt::DirectConnection,
Q_RETURN_ARG(QAbstractSeries *, serie),
Q_ARG(int, type),
Q_ARG(QString, "serie from c++"),
Q_ARG(QAbstractAxis *, axis_x),
Q_ARG(QAbstractAxis *, axis_y));
if(QLineSeries *line_serie = qobject_cast<QLineSeries *>(serie)){
static std::default_random_engine e;
static std::uniform_real_distribution<> dis(0, 3);
for(int i=0; i < 14; i++){
line_serie->append(i, dis(e));
}
}
}
void Helper::removeAllSeries(QQuickItem *chartview){
if(!chartview)
return;
const QMetaObject *mo = chartview->metaObject();
if(std::strcmp(mo->className(), "QtCharts::DeclarativeChart") != 0)
return;
QMetaObject::invokeMethod(chartview, "removeAllSeries", Qt::DirectConnection);
}
main.qml
import QtQuick 2.14
import QtQuick.Window 2.14
import QtQuick.Layouts 1.14
import QtQuick.Controls 2.4
import QtCharts 2.14
Window {
visible: true
width: 640
height: 480
title: qsTr("Hello World")
ColumnLayout{
anchors.fill: parent
RowLayout{
Button{
text: "Create serie"
Layout.fillWidth: true
onClicked: helper.createSerie(chartview)
}
Button{
text: "Clear series"
Layout.fillWidth: true
onClicked: helper.removeAllSeries(chartview);
}
}
ChartView {
id: chartview
title: "Line"
antialiasing: true
Layout.fillWidth: true
Layout.fillHeight: true
LineSeries {
name: "LineSeries"
XYPoint { x: 0; y: 0 }
XYPoint { x: 3; y: 2.1 }
XYPoint { x: 8; y: 3.3 }
XYPoint { x: 10; y: 2.1 }
XYPoint { x: 11; y: 4.9 }
XYPoint { x: 12; y: 3.0 }
XYPoint { x: 13; y: 3.3 }
}
axes: [
ValueAxis{
id: xAxis
min: 1.0
max: 15.0
},
ValueAxis{
id: yAxis
min: 0.0
max: 5.0
}
]
}
}
}
下面link是完整代码
存在不使用 QMetaObject::invokeMethod()
的更好变体:可以直接使用 QChart
。只需要扩展以下解决方案
*.h
public:
Q_INVOKABLE void setSeries(QAbstractSeries *series);
[...]
private:
QXYSeries *mSeries;
QChart *mChart;
[...]
*.cpp
void DataSource::setSeries(QAbstractSeries *series)
{
if (series) {
mSeries = static_cast<QXYSeries *>(series);
mChart = mSeries->chart();
}
}