- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试将 qml
中的项目值获取到我的 cpp
代码中,从那里我想将这些值添加到 CAN 消息中并通过 CAN 总线发送它们.
到目前为止,我可以成功地将许多qml
项的值和状态获取到我的cpp
中。此外,我可以使用静态值将 CAN 消息传输到 CAN 总线。但是,其中一些值不应是静态的,而应使用 qml
中的项目值动态更新。
这是backend.h:
#ifndef BACKEND_H
#define BACKEND_H
#include <QObject>
#include <QCanBusDevice>
class BackEnd : public QObject
{
Q_OBJECT
Q_PROPERTY(int elemVal READ getElemVal WRITE setElemVal NOTIFY elemValChanged)
public:
explicit BackEnd(QObject *parent = nullptr);
//elemVal
int getElemVal();
void setElemVal(const int &elemVal);
int m_elemVal;
//can
void run();
void oneShotConnectCan();
QCanBusDevice *m_canDevice = nullptr;
signals:
void elemValChanged();
public slots:
void sendCanFrame();
};
#endif // BACKEND_H
这是backend.cpp:
BackEnd::BackEnd(QObject *parent) : QObject(parent)
{
}
//elemVal get set
int BackEnd::getElemVal()
{
return m_elemVal;
}
void BackEnd::setElemVal(const int &elemVal)
{
if(elemVal == m_elemVal)
return;
m_elemVal = elemVal;
emit elemValChanged();
qDebug() << "elemVal is: " << m_elemVal;
}
//end of elemVal get set
...
CAN Bus initialization
...
void BackEnd::sendCanFrame()
{
quint32 frameid = 131;
QByteArray payload;
payload[0] = 0x04;
payload[1] = 0x03;
payload[2] = m_elemVal;
QCanBusFrame testFrame(frameid, payload);
testFrame.setFrameType(QCanBusFrame::DataFrame);
m_canDevice->writeFrame(testFrame);
if (m_canDevice->writeFrame(testFrame)) {
qDebug() << "test frame: " << testFrame.toString();
}
else {
qFatal("Write failed");
}
}
这是main.cpp:
int main(int argc, char *argv[])
{
QGuiApplication app(argc, argv);
qmlRegisterType<BackEnd>("io.qt.examples.backend", 1, 0, "BackEnd");
QQmlApplicationEngine engine;
engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
BackEnd m_can;
m_can.oneShotConnectCan(); //creating CAN device
m_can.run(); //sending the CAN message
return app.exec();
}
这是main.qml 文件:
Window {
id: window
objectName: "window"
visible: true
visibility: Window.FullScreen
onWindowStateChanged: {
console.log( "onWindowStateChanged (Window), state: " + windowState );
}
BackEnd{
id: backend
}
Dial {
id: dial
x: 181
y: 38
stepSize: 1
to: 255
value: backend.elemVal
onValueChanged: backend.elemVal = value
}
}
qDebug的实际输出是这样的:
elemVal is: 0
test frame : " 083 [3] 04 03 05"
test frame : " 083 [3] 04 03 05"
elemVal is: 1
elemVal is: 2
elemVal is: 3
test frame : " 083 [3] 04 03 05"
elemVal is: 4
elemVal is: 5
elemVal is: 6
test frame : " 083 [3] 04 03 05"
我的期望是:
elemVal is: 0
test frame : " 083 [3] 04 03 00"
test frame : " 083 [3] 04 03 00"
elemVal is: 1
elemVal is: 2
elemVal is: 3
test frame : " 083 [3] 04 03 03"
elemVal is: 4
elemVal is: 5
elemVal is: 6
test frame : " 083 [3] 04 03 06"
消息的第三个字节,在实际输出中为 05,应该随 m_elemVal
变量动态变化,该变量连接到 qml
中。虽然我可以读取和写入 m_elemVal
的值,但我无法将其写入我的 CAN 报文。
抱歉,如果帖子太长,请尽可能具体。
感谢任何帮助...
最佳答案
您在 QML 中创建的后端对象
BackEnd {
id: backend
}
与C++创建的不同:
BackEnd m_can;
有几种可能的解决方案:
与其他建议解决方案相比的优势在于,现在可以从任何 QML 访问后端对象,例如 console.log()
。
#include <QQmlContext>
// ...
int main(int argc, char *argv[])
{
QGuiApplication app(argc, argv);
BackEnd m_can;
m_can.oneShotConnectCan(); //creating CAN device
m_can.run(); //sending the CAN message
QQmlApplicationEngine engine;
engine.rootContext()->setContextProperty("backend", &m_can);
engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
return app.exec();
}
*.qml
Window {
id: window
objectName: "window"
visible: true
visibility: Window.FullScreen
onWindowStateChanged: {
console.log( "onWindowStateChanged (Window), state: " + windowState );
}
Dial {
id: dial
x: 181
y: 38
stepSize: 1
to: 255
onValueChanged: backend.elemVal = value
}
}
与其他答案相比的一个优势是不使用会产生问题的旧连接样式,因为它的验证是在运行时而不是编译时完成的。另外,代码依赖于QML结构。
我还假设您想在 elemVal 更改时发送一个帧,因此理想的是在 elemValChanged 和 sendCanFrame() 之间的信号之间建立连接:
BackEnd::BackEnd(QObject *parent) : QObject(parent)
{
connect(this, &Backend::elemValChanged, this, &Backend::sendCanFrame);
}
有时需要在创建对象后启动某些资源,在这种情况下,您可以使用 QML 中的 Component.onCompleted 或使用 QQmlParserStatus,在这种情况下,我将使用第二种方法。
*.h
class BackEnd : public QObject, public QQmlParserStatus
{
Q_OBJECT
Q_INTERFACES(QQmlParserStatus)
Q_PROPERTY(int elemVal READ getElemVal WRITE setElemVal NOTIFY elemValChanged)
public:
explicit BackEnd(QObject *parent = nullptr);
void classBegin();
void componentComplete();
//elemVal
int getElemVal();
void setElemVal(const int &elemVal);
signals:
void elemValChanged();
public slots:
void sendCanFrame();
private:
//can
void run();
void oneShotConnectCan();
QCanBusDevice *m_canDevice = nullptr;
int m_elemVal;
};
*.cpp
BackEnd::BackEnd(QObject *parent) : QObject(parent)
{
connect(this, &Backend::elemValChanged, this, &Backend::sendCanFrame);
}
void BackEnd::classBegin(){}
void BackEnd::componentComplete()
{
oneShotConnectCan(); //creating CAN device
run()
}
//elemVal get set
int BackEnd::getElemVal()
{
return m_elemVal;
}
void BackEnd::setElemVal(const int &elemVal)
{
if(elemVal == m_elemVal)
return;
m_elemVal = elemVal;
emit elemValChanged();
qDebug() << "elemVal is: " << m_elemVal;
}
//end of elemVal get set
// ...
// CAN Bus initialization
// ...
void BackEnd::sendCanFrame()
{
quint32 frameid = 131;
QByteArray payload;
payload[0] = 0x04;
payload[1] = 0x03;
payload[2] = m_elemVal;
QCanBusFrame testFrame(frameid, payload);
testFrame.setFrameType(QCanBusFrame::DataFrame);
m_canDevice->writeFrame(testFrame);
if (m_canDevice->writeFrame(testFrame)) {
qDebug() << "test frame: " << testFrame.toString();
}
else {
qFatal("Write failed");
}
}
main.cpp
static void registerTypes()
{
qmlRegisterType<BackEnd>("io.qt.examples.backend", 1, 0, "BackEnd");
}
Q_COREAPP_STARTUP_FUNCTION(registerTypes)
int main(int argc, char *argv[])
{
QGuiApplication app(argc, argv);
QQmlApplicationEngine engine;
engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
return app.exec();
}
*.qml
Window {
id: window
objectName: "window"
visible: true
visibility: Window.FullScreen
onWindowStateChanged: {
console.log( "onWindowStateChanged (Window), state: " + windowState );
}
BackEnd{
id: backend
elemVal : dial.value
}
Dial {
id: dial
x: 181
y: 38
stepSize: 1
to: 255
Component.onCompleted: value = backend.elemVal
}
}
关于c++ - 将它们传递给 cpp 后如何与 qml 项目值进行交互?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55952105/
这个问题在这里已经有了答案: Why can templates only be implemented in the header file? (18 个答案) 关闭 7 年前。 我的 .hpp
我想使用 yaml-cpp 发出一个带引号的字符串,所以它看起来像 时间戳:“2011 年 8 月 10 日 01:37:52” 在输出yaml文件中。我该怎么做?谢谢。 最佳答案 YAML::Emi
我理解了模板的概念以及为什么我们需要在头文件中定义模板成员函数。另一种选择是在 cpp 文件中定义模板函数并显式实例化模板类,如下所示。 模板.h #include using namespace
是否可以发出和读取(解析)二进制数据(图像、文件等)?如下所示: http://yaml.org/type/binary.html我如何在 yaml-cpp 中执行此操作? 最佳答案 截至revisi
我尝试查找此内容并使用头文件等得到混合结果。 基本上我有多个 .cpp 文件,其中包含我为使用二叉树而制作的所有函数,BST , 链表等 我不想复制和粘贴我需要的函数,我只想能够做一个: #inclu
我正在发出一个 YAML 文档,如下所示: YAML::Node doc; // ...populate doc... YAML::Emitter out; out << doc; 在节点层次结构的某
这个问题在这里已经有了答案: Access extern variable in C++ from another file (1 个回答) 关闭 4 年前。 考虑以下场景: MyFile.cpp:
所以我在上基础编程课,我们正在学习如何将文件链接在一起。问题是我遇到了一个似乎没有人能够修复的错误。我已经去过我的教授、学生助理和校园里的编程辅助实验室,但运气不佳。 我还在这里搜索了至少 10 篇与
在下面的代码中,我在使用 parser.GetNextDocument(doc); 解析我的 .yaml 文件时遇到了一些问题。经过大量调试后,我发现这里的(主要)问题是我的 for 循环没有运行,因
我们有以下类(class)考试成绩:完成本类(class)的学生中有 75 人参加了考试。我们想知道学生在考试中的表现如何,并给出了 75 名学生的分数。我们想编写一个程序,按以下方式总结和分析结果:
主要.cpp #include #include #include #include "cootie.h" using namespace std; int main() { cout
试图制作电子鸡程序,但编译器抛出未定义的对“Tamagotchi::age()”错误的引用 理想情况下,这段代码会返回电子鸡的年龄,它应该在开始时由类的构造函数初始化为 0。 我显然在某个地方搞砸了,
我一直在开发一个使用 Microsoft Visual Studio 2010 命令提示符编译原始 .cpp 文件并分析其输出的应用程序。我遇到了很多麻烦,网上似乎没有太多关于这个的资料。这是麻烦的代
我试图从另一个 .cpp 文件调用 c++ 函数。我使用了 .h header 。看看下面我做了什么。 我有一个f.h文件: #ifndef PACKAGENAME_ADD_H #define PAC
我在 CPP 中有一个函数,其原型(prototype)如下: char* complexFunction(char* arg1, ...); 我使用 DLLImport 属性从 C# 导入它。问题是
也许这是一个幼稚的问题 - 但有没有办法构建/安装 yaml-cpp,以便在构建包含 yaml.h 的项目时不需要使用 Boost 库 header ? IE:我正在开发一个使用 yaml-cpp 结
我有一个在 .cpp 函数中声明的静态函数,我不能在 header 中声明它,因为它不应该是可见的。我想在同一项目的另一个 .cpp 中重新使用它。 这有可能吗? 最佳答案 这里有两个问题: 这可能吗
我正在使用 php-cpp 为我的 php 代码创建扩展,当我尝试编译 main.cpp 文件的简单结构时,我得到这个错误。这是编译错误: main.cpp:15:5: error: ‘PHPCPP_
我决定将必要的代码减少到显示此错误所需的最低限度。我有一个存在于 hc_list.h 文件中的 STL 列表包装器模板类。完整代码如下: // hc_list.h file #ifndef HC_LI
您好,我目前正在尝试通过 AMQPCPP 将 RabbitMQ 集成到我的 VisualStudio 项目中。我只能使用 Windows PC,这对安装来说是一件很痛苦的事情。我想我能够使用 CMAK
我是一名优秀的程序员,十分优秀!