【问题标题】:QML QTQuick ChartView pass pointer to C++QML QTQuick ChartView 将指针传递给 C++
【发布时间】:2021-05-16 01:16:11
【问题描述】:

我正在尝试基于 qtcharts-qmloscilloscope-example here 制作类似 Qt Quick 应用程序的示波器

在此示例中,跟踪(QTQuick ChartView)在 QML 中预先分配并通过计时器更新。

我希望能够在运行时添加和删除跟踪。

现有应用程序将对底层数据数组的引用作为 QPointF 对象的 QAbstractSeries 传递。此操作在 Timer 上触发,因此:

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++?

【问题讨论】:

  • 您想在 C++ 中使用哪些 ChartView 方法?
  • 至少 removeAllSeries() 和 createSeries()

标签: c++ qt qml qt5 qchartview


【解决方案1】:

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);
}

ma​​in.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是完整代码。

【讨论】:

  • 你是对的,我删除了我的答案。通过我尝试实现的方法,我们需要将其转换为 QQuickItem 并使用 objectName。如果以后有时间,我会尝试恢复它。
  • @AlexanderV 主要问题是您仍然获得 QQuickItem(就像我一样)是您将无法直接使用它,但您必须使用 QMetaObject 来访问这些方法正如我在解决方案中提出的那样。
  • 对,我认为问题作者想要更多的 C++ 方式与之交互。单独的助手或 QML 单例更容易完成,可能使用一些智能模板。
  • @AlexanderV 我认为这不是 OP 想要的,也许我犯了一个错误。
  • 所以我的错误是假设 QML ChartView 是 QChartView 的一个实例?
【解决方案2】:

不使用QMetaObject::invokeMethod() 存在更好的变体:可以直接使用QChart。只需要扩展以下解决方案"Push QML ChartView updates from c++"

*.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();
    }
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2016-03-10
    • 2018-08-11
    • 2012-02-05
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-02-22
    相关资源
    最近更新 更多