2015-10-10 200 views
0

我试图将QVariantMap转换为从QObject派生的自定义类,但是当涉及到设置我的枚举类型的属性时,我从得到返回值false。代码低于:setProperty()返回false

MessageHeader.h文件:

// deserialization class header 
class MessageHeader : public QObject 
{ 
    Q_OBJECT 

public: 
    MessageHeader(QObject *parent = 0); 
    ~MessageHeader(); 

    enum class MessageType 
    { 
     none = 0, 
     foo = 1, 
     baa = 2 
    }; 

    Q_ENUM(MessageType) 
    Q_PROPERTY(MessageType type READ getType WRITE setType) 
    Q_PROPERTY(int ContentLength READ getContentLength WRITE setContentLength) 

    void setType(MessageType type); 
    void setContentLength(int ContentLength); 
    MessageType getType(); 
    int getContentLength(); 
    QString toString(); 

    MessageType type = MessageType::none; 
    int ContentLength = 0; 
}; 

MessageHeader.cpp文件:

MessageHeader::MessageHeader(QObject *parent) 
    : QObject(parent) 
{ 
} 

MessageHeader::~MessageHeader() 
{ 
} 

MessageType MessageHeader::getType() 
{ 
    return type; 
} 

int MessageHeader::getContentLength() 
{ 
    return ContentLength; 
} 

void MessageHeader::setType(MessageType type) 
{ 
    this->type = type; 
} 

void MessageHeader::setContentLength(int ContentLength) 
{ 
    this->ContentLength = ContentLength; 
} 

QString MessageHeader::toString() 
{ 
    return QString("NOT IMPLEMENTED YET"); 
} 

而反序列化函数模板帮手:

template<typename T> 
    T* Deserialize(const QString &json) 
    { 
     bool status = false; 

     QJson::Parser parser; 
     QVariantMap map = parser.parse(json.toUtf8(), &status).toMap(); 

     if(!status) 
      return NULL; 

     T *obj = new T(); //don't worry about this, I'll rather take this from paramters once this is working 
     QObject *p = (QObject *) obj; // cast done so that I see setProperty() method 
     for(QVariantMap::const_iterator iter = map.begin(); iter != map.end(); ++iter) 
     { 
      const char *name = iter.key().toLatin1(); 
      const QVariant value = iter.value(); 
      qDebug() << "setting " << name << "=" << value; 
      // the issue goes below. Here setProperty() return false. 
      // At this point, name = 'type' and value = 2 
      assert(p->setProperty(name, value)); 
     } 
    //QJson::QObjectHelper::qvariant2qobject(map, obj); 
    return obj; 
} 

的JSON输入字符串上面的函数是这样的:

"{\"ContentLength\": 100, \"type\": 2}" 

枚举类型别的之前注册在main funcction:

qRegisterMetaType<MessageType>("MessageType"); 

下面是本例中使用的QJson库。我建立它在Windows上使用此.pro文件

编辑:

我刚刚发现type属性不能被indexOfProperty()

qDebug() << "id = " << meta->indexOfProperty(name); // print -1, name = 'type' 
+0

False意味着它无法设置您要求的内容。 –

+0

是啊,我知道... – Jack

回答

1

枚举属性被发现只能如果变种设置类型是QStringQIntQUInt,可以看到here。因此,要成功设置枚举属性,变体需要是这些类型之一,而不是其他类型。 QJson解析任何无符号整数为QULongLong,可以看到here, line 84。因此,一种方法是分叉QJson并修改代码,以便将整数值转换为QIntQUInt或将枚举值作为字符串读取/写入。

此外,将语句放入assert不是一个好主意,但我认为您只是编写了试图找出问题的代码。

正如一个方面说明,根据Qt文档,

[qRegisterMetaType]仅对每隔一个用例Q_DECLARE_METATYPE和qMetaTypeId登记的别名(类型定义)是有用的()应被代替使用。

因此在您的标题中用Q_DECLARE_METATYPE(MessageHeader::MessageType)代替qRegisterMetaType<MessageHeader::MessageType>("MessageType")将是一个合理的举措。

+0

不幸的是,从枚举类改为枚举并没有解决。是的,我只是写了那个代码来试图找出问题所在。 – Jack

+0

@Jack啊确实,无用的枚举可以隐式转换为'int',而不是'int'。我完全取代了我的答案。 – Rostislav

+0

我加了重载并且移动到'Q_DECLARE_METATYPE',但仍然没有工作 – Jack