我正在尝试编写一些与QML对象交互的Qt C ++代码。目标是将TCP套接字上的字符串附加到GUI上的文本日志中。每次收到新字符串时,我都会运行appendText()函数。我有一个当前正在使用的实现,它使用QWidgets和.ui文件。我需要一个完全相同的QML实现。我的QWidget实现使用textBrowser和append函数,如下所示。程序运行时“theString”正在改变,并附加每个更改,填写文本日志。
//update the text log with data received on TCP socket
void MainWindow::appendText() {
ui->textBrowser->append(theString);
}
这为我提供了所需的结果,将每个字符串附加到文本框中。输出应如下所示。
Control connection successful.
Data connection successful.
Control Packet Receieved:
1
Control Packet Receieved:
2
Control Packet Receieved:
3
Control Packet Receieved:
4
Control Packet Receieved:
1
Control Packet Receieved:
2
Control Packet Receieved:
3
Control Packet Receieved:
4
但是,在使用以下代码执行我认为与QML对象相同的功能时......
//update the text log with data received on TCP socket
void MainWindow::appendText() {
QMetaObject::invokeMethod(textbox, "append", Qt::DirectConnection, Q_ARG(QVariant, theString));
//QQmlProperty(textbox, "text").write(theString);
}
它只附加前两个字符串,除此之外不再。输出看起来像这样。
Control connection successful.
Data connection successful.
我已经查看了在C ++中广泛调用QML方法的文档,但仍然没有运气。任何帮助表示赞赏。谢谢你的时间。
答案 0 :(得分:1)
我无法重现您的问题。
可能的解决方案
使用import QtQuick.Controls 2.0
可能是一种解决方案。
在这种情况下,我收到以下错误消息:
QMetaObject::invokeMethod: No such method QQuickTextArea::append(QVariant) Candidates are: append(QString)
根据错误消息的建议,您现在应使用QString
代替QVariant
作为参数类型:
QMetaObject::invokeMethod(textbox, "append", Qt::DirectConnection, Q_ARG(QString, theString));
更好的替代方案
如Qt所述,您应该avoid manipulating QML object from C++(深入对象树):
警告:虽然可以使用C ++访问和操作对象树深处的QML对象,但我们建议您不要 在应用程序测试和原型设计之外采取这种方法。一 QML和C ++集成的优势在于实现的能力 QML用户界面与C ++逻辑和数据集后端分开, 如果C ++方面深入到QML中,这种策略就会破裂 组件直接操纵它们。
因此,它可能是在C ++中实现信号的更好的替代方案,该信号发出新接收的消息并从QML端连接到它。这种方法明确区分了用户界面和编程逻辑。
工作示例代码
以下代码每秒"test"
附加TextArea
。
main.cpp中:
#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <QTimer>
#include <QQuickItem>
QObject *textbox;
void onTimeout()
{
QMetaObject::invokeMethod(textbox, "append", Qt::DirectConnection, Q_ARG(QVariant, "test"));
}
int main(int argc, char *argv[])
{
QGuiApplication app(argc, argv);
QQmlApplicationEngine engine;
engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
QTimer t;
QObject::connect(&t, &QTimer::timeout, &onTimeout);
textbox = engine.rootObjects().first()->children().first();
t.start(1000);
return app.exec();
}
main.qml:
import QtQuick 2.0
import QtQuick.Window 2.2
import QtQuick.Controls 1.0
Window
{
visible: true
width: 600
height: 600
TextArea
{
id: textbox
anchors.fill: parent
}
}