将Qt属性与命名空间中定义的自定义类型一起使用

时间:2013-11-10 11:29:02

标签: c++ qt qt5

当使用自己的类型作为属性类型时,我当前遇到了问题。

我的自定义类型是在命名空间中定义的。当我把它放入全局命名空间时,一切都没有任何问题。

是否可以在命名空间中使用自定义类型作为属性值?

这是一个最小的例子(我使用的是Qt5.0和g ++ 4.7.3)

这是test.pro文件:

LANGUAGE  = C++
QT += core gui widgets
TARGET = test
QMAKE_CXXFLAGS += -std=c++11
HEADERS += test.h
SOURCES += test.cpp

这是test.h文件:

#include <QtCore>
#include <QtGui>
#include <QtWidgets>

namespace MyNamespace
{
  struct MyValue
  {
  private:
    QString a;
    int b;

  public:
    MyValue(const QString &a="", int b=0)
      : a(a), b(b)
    {
    }

    bool operator !=(const MyValue &other) const
    {
      return (this->a!=other.a) || (this->b!=other.b);
    }

    friend QDataStream &operator<<(QDataStream &stream, const MyNamespace::MyValue &value);
    friend QDataStream &operator>>(QDataStream &stream, MyNamespace::MyValue &value);
    friend QDebug operator<<(QDebug debug, const MyNamespace::MyValue &value);
  };

  inline QDataStream &operator<<(QDataStream &stream, const MyNamespace::MyValue &value)
  {
    stream << value.a;
    return stream << value.b;
  }

  inline QDataStream &operator>>(QDataStream &stream, MyNamespace::MyValue &value)
  {
    stream >> value.a;
    return stream >> value.b;
  }

  inline QDebug operator<<(QDebug debug, const MyNamespace::MyValue &value)
  {
    return debug << "MyValue("<<value.a<<", "<<value.b<<")";
  }
}

Q_DECLARE_METATYPE(MyNamespace::MyValue)


namespace AnotherNamespace
{
  typedef MyNamespace::MyValue MyValue;

  class MyClass : public QObject
  {
    Q_OBJECT

    Q_PROPERTY(MyValue value READ value WRITE setValue NOTIFY valueChanged)

  public:
    MyValue value()
    {
      return value_;
    }

  public slots:
    void setValue(const MyValue &value)
    {
      if(this->value() != value)
      {
        value_ = value;
        emit valueChanged(value);
      }
    }

  signals:
    void valueChanged(const MyValue &value);

  private:
    MyValue value_;
  };

}

这是test.cpp文件:

#include "test.h"

int main(int argc, char** argv)
{
  QApplication app(argc, argv);

  qRegisterMetaTypeStreamOperators<MyNamespace::MyValue>("MyNamespace::MyValue");


  AnotherNamespace::MyClass myObject;

  myObject.setValue(MyNamespace::MyValue("the answer", 42));

  QMetaObject metaObject = AnotherNamespace::MyClass::staticMetaObject;
  QMetaProperty metaProperty = metaObject.property(metaObject.indexOfProperty("value"));
  QVariant variant = metaProperty.read(&myObject);

  MyNamespace::MyValue value = variant.value<MyNamespace::MyValue>();

  qDebug() << value;


  qDebug() << "\nIt's interesting, that this is working without problems:";
  variant = QVariant::fromValue(MyNamespace::MyValue("2^4", 16));
  value = variant.value<MyNamespace::MyValue>();

  qDebug() << value;


  return 0;
}

此最小示例的输出为

QMetaProperty::read: Unable to handle unregistered datatype 'MyValue' for property 'AnotherNamespace::MyClass::value'
MyValue( "" ,  0 ) 

It's interesting, that this is working without problems: 
MyValue( "2^4" ,  16 ) 

正如alreaday所说,从最小的示例中删除任何名称空间使用后,一切都可以正常工作

MyValue( "the answer" ,  42 ) 

It's interesting, that this is working without problems: 
MyValue( "2^4" ,  16 ) 

1 个答案:

答案 0 :(得分:4)

在输入问题时我找到了答案。错误如下:

// ...
namespace AnotherNamespace
{
  typedef MyNamespace::MyValue MyValue;

  class MyClass : public QObject
  {
    Q_OBJECT

    Q_PROPERTY(MyValue value READ value WRITE setValue NOTIFY valueChanged)
// ...

使用Q_PROPERTY时,我必须提供带有所有命名空间的完整typename。所以不要使用

Q_PROPERTY(MyValue value READ value WRITE setValue NOTIFY valueChanged)

我必须使用

Q_PROPERTY(MyNamespace::MyValue value READ value WRITE setValue NOTIFY valueChanged)

Qt Property系统似乎无法识别typedef。