【QML与C++混合编程】用QVariantList传递数组类型成员
2017.5.8 更新:Record类要用指针,QObject 不能有拷贝函数。
我有一个C++中自定义的ReaderModel,继承自QAbstractListModel类,传递给了QML。
它的me成员是一个Reader指针,Reader有个成员是RecordModel。
通过reader获取的recordModel,在qml中类型是QVariant(RecordModel),我没法把它作为一个ListView的model。
要怎么让它绑定给view呢?
我尝试者把数据拷贝到一个直接传给qml的recordModel,但是当数据之后发生了变化时,视图就不会更新,除非再次拷贝,这样效率不可观。
通过艰难地google查找相关问题,我最后的解决方案是:
取消这个RecordModel成员,用QVariantList来储存所有record。
简单地说就是传递自定义类中的自定义结构体数组。
作为解决方案的代码(如果不需要,完全可以不用ReaderModel,但是要用setContextProperty把reader变量传给qml):
record.h
#ifndef RECORD_H
#define RECORD_H
#include <QObject>
class Record: public QObject
{
Q_OBJECT
Q_PROPERTY(QString bookId READ bookId WRITE setBookId NOTIFY bookIdChanged)
Q_PROPERTY(int state READ state WRITE setState NOTIFY stateChanged)
public:
Record(const QString &bookId="",int state=0):
bookId_(bookId),state_(state){}
Record(const Record &r){
bookId_ = r.bookId_;
state_ = r.state_;
}
QString bookId() const;
int state() const;
public slots:
void setBookId(const QString &);
void setState(int);
private:
QString bookId_;
int state_;
signals:
void bookIdChanged();
void stateChanged();
};
Q_DECLARE_METATYPE(Record*)//元类型注册
#endif // RECORD_H
reader.h
#ifndef READER_H
#define READER_H
#include <QObject>
#include <QVariantList>
class Reader: public QObject
{
Q_OBJECT
Q_PROPERTY(QString id READ id WRITE setId NOTIFY idChanged)
Q_PROPERTY(QString password READ password WRITE setPassword NOTIFY passwordChanged)
Q_PROPERTY(QVariantList record READ record WRITE setRecord NOTIFY recordChanged)
public:
Reader(const QString &id = "",
const QString &password = "")
: id_(id),password_(password){}
QString id() const; //id
QString password() const; //密码
QVariantList record() const; //记录
Q_INVOKABLE void doSomething()const;
public slots:
void setId(const QString &);
void setPassword(const QString &);
void setRecord(const QVariantList &);
private:
QString id_;
QString password_;
QVariantList record_;//借书记录
signals:
void idChanged();
void passwordChanged();
void recordChanged();
};
#endif // READER_H
readerModel.h
#ifndef READERMODEL_H
#define READERMODEL_H
#include <QAbstractListModel>
#include <QJsonValueRef>
#include <QVariant>
#include "reader.h"
#include "record.h"
class ReaderModel : public QAbstractListModel
{
Q_OBJECT
Q_PROPERTY(Reader* me READ me WRITE setMe NOTIFY meChanged)
public:
enum ReaderRole {
IdRole = Qt::DisplayRole, //0
PasswordRole = Qt::UserRole,
RecordRole
};
Q_ENUM(ReaderRole)
ReaderModel(QObject *parent = nullptr){Q_UNUSED(parent);}
int rowCount(const QModelIndex & = QModelIndex()) const;
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const;
QHash<int, QByteArray> roleNames() const;
Q_INVOKABLE QVariantMap get(int row) const;
Q_INVOKABLE void append(const QString &id,
const QString &password);
Q_INVOKABLE void set(int row, const QString &id,
const QString &password);
Q_INVOKABLE void remove(int row);
Q_INVOKABLE Reader *me() const;//当前登录的用户对象指针
Q_INVOKABLE void setMe(Reader *r);
private:
QList<Reader*> m_readers;
Reader *me_;
signals:
void meChanged();
};
#endif // READERMODEL_H
reader.cpp
#include "reader.h"
#include "record.h"
#include <QVariant>
QString Reader::id() const
{
return id_;
}
QString Reader::password() const
{
return password_;
}
QVariantList Reader::record() const
{
return record_;
}
void Reader::setId(const QString &value)
{
if (id_ == value)
return;
id_ = value;
emit idChanged();
}
void Reader::setPassword(const QString &value)
{
if (password_ == value)
return;
password_ = value;
emit passwordChanged();
}
void Reader::setRecord(const QVariantList &value)
{
if (record_ == value)
return;
record_ = value;
emit recordChanged();
}
record.cpp
#include "record.h"
QString Record::bookId() const
{
return bookId_;
}
int Record::state() const
{
return state_;
}
void Record::setBookId(const QString &value)
{
if(bookId_ == value)
return;
bookId_ = value;
emit bookIdChanged();
}
void Record::setState(int value)
{
if(state_ == value)
return;
state_ = value;
emit stateChanged();
}
readerModel.cpp
#include "readermodel.h"
#include "reader.h"
int ReaderModel::rowCount(const QModelIndex & /*parent*/) const
{
return m_readers.count();
}
QVariant ReaderModel::data(const QModelIndex &index, int role) const
{
if(index.row() < rowCount())
switch(role){
case IdRole: return m_readers.at(index.row())->id();
case PasswordRole: return m_readers.at(index.row())->password();
case RecordRole: return m_readers.at(index.row())->record();
default: return QVariant();
}
return QVariant();
}
QHash<int, QByteArray> ReaderModel::roleNames() const
{
static const QHash<int, QByteArray> roles{
{IdRole, "id"},
{PasswordRole, "password"},
{RecordRole, "record"}
};
return roles;
}
QVariantMap ReaderModel::get(int row) const
{
Reader *reader = m_readers.value(row);
return { {"id", reader->id()},
{"password", reader->password()},
{"record", reader->record()}};
}
void ReaderModel::append(const QString &id, const QString &password)
{
int row = m_readers.count();
beginInsertRows(QModelIndex(), row, row);
Reader *r = new Reader(id, password, name, power, school, credit, money, unback);
m_readers.append(r);
endInsertRows();
}
void ReaderModel::set(int row, const QString &id, const QString &password)
{
if (row < 0 || row >= m_readers.count())
return;
Reader *r = new Reader(id, password == ""? m_readers[row]->password() : password);
m_readers.replace(row, r);
dataChanged(index(row, 0), index(row, 0), { IdRole,PasswordRole});
}
void ReaderModel::remove(int row)
{
if (row < 0 || row >= m_readers.count())
return;
beginRemoveRows(QModelIndex(), row, row);
m_readers.removeAt(row);
endRemoveRows();
}
Reader *ReaderModel::me() const
{
return me_;
}
void ReaderModel::setMe(Reader *r)
{
me_ = r;
emit meChanged();
}
main.cpp
#include <QGuiApplication>
#include <QQmlApplicationEngine>
#include <QtQml>
#include <QQmlContext>
#include "readermodel.h"
int main(int argc, char *argv[])
{
QGuiApplication app(argc, argv);
//向qml注册类型
qmlRegisterType<ReaderModel>("Backend", 1, 0, "ReaderModel");
ReaderModel *readerModel = new ReaderModel();
...//写入数据
QQmlApplicationEngine *engine = new QQmlApplicationEngine();
//向qml传递变量
engine->rootContext()->setContextProperty("readerModel", readerModel);
engine->load(QUrl(QLatin1String("qrc:/main.qml")));
return app.exec();
}
mail.qml:
ListView {
visible: true
id: recordView
width: parent.width
height: parent.height
model: readerModel.me.record
delegate:Rectangle{
property var record: readerModel.me.record[index]
RowLayout{
spacing: 10
Label {
text: record.bookId
}
Label {
text: record.state
}
}
Component.onCompleted: {
console.log(readerModel.me.record)
console.log("\n",readerModel.me.record[index])
console.log("\n",readerModel.me.record[index].state)
}
}
}
参考:[SOLVED] Cascaded QVariantList exposed to QML causes Error: Cannot assign [undefined] to QString
【QML与C++混合编程】用QVariantList传递数组类型成员的更多相关文章
- QML与C++混合编程详解(转)
原文转自:http://blog.csdn.net/ieearth/article/details/42243553 原文转自:https://www.cnblogs.com/findumars/p/ ...
- QML与C++混合编程详解
1.QML与C++为什么要混合编程 QML与C++为什么要混合编程,简单来说,就是使用QML高效便捷地构建UI,而C++则用来实现业务逻辑和复杂算法,下面介绍了两者间交互的方法与技巧. 2.QML访问 ...
- 【Qt】Qt Quick 之 QML 与 C++ 混合编程详解
Qt Quick 之 QML 与 C++ 混合编程详解 - CSDN博客 专栏:Qt Quick简明教程 - CSDN博客 .
- qml与c++混合编程
QML 与 C++ 混合编程内容:1. QML 扩展2. C++ 与 QML 交互3. 开发时要尽量避免使用的 QML 元素4. demo 讲解5. QML 语法C++ 与 QML 的交互是通过注册 ...
- 由基于qml,c++的串口调试工具浅谈qml与c++混合编程
最近在做一个基于sim900 的串口通信工具,基于qml和c++来实现. 首先,对于串口,qt有自带的QSerialPort,可以实现同步,和异步通信,qt creator也有自带的例子,本例子是从其 ...
- Qt Quick 之 QML 与 C++ 混合编程具体解释
Qt Quick 技术的引入.使得你能够高速构建 UI ,具有动画.各种绚丽效果的 UI 都不在话下.但它不是万能的.也有非常多局限性,原来 Qt 的一些技术,比方低阶的网络编程如 QTcpSocke ...
- java matlab混合编程之返回值Struct类型
java matlab混合编程的时候当返回值是Struct类型(matlab中的返回类型)如何来取得(java中)其值? 上网找,看到这个网页:http://www.mathworks.cn/cn/h ...
- 在Qt(C++)中与Python混合编程
一.PythonQt库 在Qt(C++)中与Python混合编程,可以使用PythonQt库. 网站首页:http://pythonqt.sourceforge.net 下载页面:https://so ...
- C# 托管和非托管混合编程
在非托管模块中实现你比较重要的算法,然后通过 CLR 的平台互操作,来使托管代码调用它,这样程序仍然能够正常工作,但对非托管的本地代码进行反编译,就很困难. 最直接的实现托管与非托管编程的方法就是 ...
随机推荐
- 001-电脑操作规范-2019年03月.doc
001-电脑操作规范-2019年03月.doc 本文作者:徐晓亮 BoAi 作者腾讯QQ号码:595076941 /////////////////////////////////////// ...
- Winform MDI窗体切换不闪烁的解决办法(测试通过)
https://stackoverflow.com/questions/5817632/beginupdate-endupdate-for-datagridview-request SuspendLa ...
- 03-命令图片.doc
- 【学习总结】win7下安装Ubuntu双系统的日常
参考文献 1 - [双系统中删除linux(win7适用) ] 2 - [win7(32位)U盘安装.卸载ubuntu(64位)双系统] 3 - [Windows下安装Ubuntu 16.04双系统] ...
- react的项目坑
首先在构造页面时 应该将页面的结构分析好. 在处理数据异步时 将数据结构进行完全的简单结构化. 使用redux时 注意返回的数据是深拷贝还是浅拷贝 否则会产生 数组不为空但是没有值的问题 使用自制数据 ...
- [转帖]Linux分页机制之概述--Linux内存管理(六)
Linux分页机制之概述--Linux内存管理(六) 2016年09月01日 19:46:08 JeanCheng 阅读数:5491 标签: linuxkernel内存管理分页架构更多 个人分类: ┈ ...
- Linux上的一些基本常用命令
上传下载文件:// 首先安装lrzsz # yum -y install lrzsz // 上传文件,执行命令rz,会跳出文件选择窗口,选择好文件,点击确认即可.# rz // 下载文件,执行命令sz ...
- 集合之LinkedHashMap(含JDK1.8源码分析)
一.前言 大多数的情况下,只要不涉及线程安全问题,map都可以使用hashMap,不过hashMap有一个问题,hashMap的迭代顺序不是hashMap的存储顺序,即hashMap中的元素是无序的. ...
- JS检测是否是360浏览器
// JavaScript Document //application/vnd.chromium.remoting-viewer 可能为360特有 var is360 = _mime("t ...
- javax.validation.UnexpectedTypeException: HV000030: No validator could be found for constraint-实体报错
使用hibernate validator出现上面的错误, 需要 注意 @NotNull 和 @NotEmpty 和@NotBlank 区别 @NotEmpty 用在集合类上面@NotBlank 用 ...