Qt模型拖放-无法保存类型QJsonValue

时间:2019-07-17 15:23:26

标签: c++ qt qdatastream qt5.9 qmetatype

我有一个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");

但是我有同样的问题,我做错了什么?

1 个答案:

答案 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