我有一个QAbstractListModel
和一个QJsonArray
显示一个drag & drop implementation:
class NoteListModel : public QAbstractListModel
{
Q_OBJECT
public:
explicit NoteListModel(QObject *parent = nullptr);
~NoteListModel() override;
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
Qt::ItemFlags flags(const QModelIndex& index) const override;
// Drag & Drop:
bool canDropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) const override;
bool dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) override;
QStringList mimeTypes() const override {
QStringList mimes = QAbstractListModel::mimeTypes();
mimes.prepend("application/flynote_json");
return mimes;
}
QMimeData *mimeData(const QModelIndexList &indexes) const override;
Qt::DropActions supportedDropActions() const override;
private:
QJsonArray noteArray;
};
当我在mimeData()
方法中调用基本实现时,出现此错误:
QVariant :: save:无法保存“ QJsonValue”类型(类型ID:45)
QVariant :: save中的ASSERT失败:“要保存的类型无效”
QMimeData *NoteListModel::mimeData(const QModelIndexList &indexes) const
{
QMimeData *ret = nullptr;
if (indexes.size() == 1){
QModelIndex mi = indexes.first();
if (mi.isValid()){
ret = QAbstractListModel::mimeData(indexes);
QJsonDocument json_mime(noteArray.at(mi.row()).toObject());
ret->setData("application/flynote_json", json_mime.toJson());
}
}
return ret;
}
所以没问题,就像doc中描述的那样,我添加了以下几行:
// In the header (outside the class)
Q_DECLARE_METATYPE(QJsonValue)
QDataStream &operator<<(QDataStream &out, const QJsonValue &myObj){ /*...*/ }
QDataStream &operator>>(QDataStream &in, QJsonValue &myObj){ /*...*/ }
// In the constructor
qRegisterMetaType<QJsonValue>("QJsonValue");
qRegisterMetaTypeStreamOperators<QJsonValue>("QJsonValue");
但是我有同样的问题,我做错了什么?
答案 0 :(得分:1)
我试图重现您的示例,因为我真的无法通过查看您的代码来解释此行为。我使用Qt 5.5编译了项目,并且能够重现错误消息:
QVariant::save: unable to save type 'QJsonValue' (type id: 45).
有趣的是,该错误在Qt 5.13中消失了,因为QVariant
在此版本中已经包含QJson-Types。
看来,我找到了一种解决这种defect
的方法,如下所示:
#include <QApplication>
#include <QDebug>
#include <QDataStream>
#include <QJsonDocument>
QDataStream& operator<<(QDataStream& out, const QJsonDocument& myObj) {
return out;
}
QDataStream& operator >> (QDataStream & in, QJsonDocument& myObj) {
return in;
}
Q_DECLARE_METATYPE(QJsonDocument)
int main(int argc, char** args) {
qDebug() << qRegisterMetaTypeStreamOperators<QJsonDocument>();
QVariant var = QVariant::fromValue(QJsonDocument());
qDebug() << var.type();
qDebug() << var.typeName();
QDataStream stream;
var.load(stream); // Workaround: Just call load at least once before any save
var.save(stream);
var.save(stream);
}
如果在所有load
的调用之前至少调用一次save
,该错误将消失。至少对于Qt 5.5。
不幸的是,我没有手头的Qt 5.5来真正找到这种奇怪行为的原因。我也不知道是什么使QJsonDocument
在其他类中如此特别,它们不需要调用load
。