当使用自己的类型作为属性类型时,我当前遇到了问题。
我的自定义类型是在命名空间中定义的。当我把它放入全局命名空间时,一切都没有任何问题。
是否可以在命名空间中使用自定义类型作为属性值?
这是一个最小的例子(我使用的是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 )
答案 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。