From 546d52e605b6a74d9a5be26fb374b831a46f2550 Mon Sep 17 00:00:00 2001 From: Stephen Kelly Date: Sun, 8 Sep 2013 08:49:29 +0200 Subject: Add a convenient way to get a type-erased smart pointer from a QVariant. Any known smart pointer in a QVariant can be handled in this way. The metatype system can be informed of new smart pointer types using an existing macro which is now documented. This is very similar to the existing infrastructure for containers. Change-Id: Iac4f9fabbc5a0626c04e1185d51d720b8b54603d Reviewed-by: Olivier Goffart --- .../snippets/code/src_corelib_kernel_qmetatype.cpp | 28 ++++++++++++++ src/corelib/kernel/qmetatype.cpp | 16 ++++++++ src/corelib/kernel/qmetatype.h | 45 ++++++++++++++++++++++ 3 files changed, 89 insertions(+) (limited to 'src/corelib') diff --git a/src/corelib/doc/snippets/code/src_corelib_kernel_qmetatype.cpp b/src/corelib/doc/snippets/code/src_corelib_kernel_qmetatype.cpp index cb1346f74c..0af10af36d 100644 --- a/src/corelib/doc/snippets/code/src_corelib_kernel_qmetatype.cpp +++ b/src/corelib/doc/snippets/code/src_corelib_kernel_qmetatype.cpp @@ -143,3 +143,31 @@ void someFunc() //! [11] +//! [12] +QPointer fp(new QFile); +QVariant var = QVariant::fromValue(fp); +// ... +if (var.canConvert()) { + QObject *sp = var.value(); + qDebug() << sp->metaObject()->className(); // Prints 'QFile'. +} +//! [12] + +//! [13] + +#include + +Q_DECLARE_SMART_POINTER_METATYPE(std::shared_ptr) + +void someFunc() +{ + auto smart_ptr = std::make_shared(); + QVariant var = QVariant::fromValue(smart_ptr); + // ... + if (var.canConvert()) { + QObject *sp = var.value(); + qDebug() << sp->metaObject()->className(); // Prints 'QFile'. + } +} + +//! [13] diff --git a/src/corelib/kernel/qmetatype.cpp b/src/corelib/kernel/qmetatype.cpp index 2ab6681bb9..70c2e236dd 100644 --- a/src/corelib/kernel/qmetatype.cpp +++ b/src/corelib/kernel/qmetatype.cpp @@ -174,6 +174,22 @@ struct DefinedTypesFilter { \snippet code/src_corelib_kernel_qmetatype.cpp 11 */ +/*! + \macro Q_DECLARE_SMART_POINTER_METATYPE(SmartPointer) + \relates QMetaType + + This macro makes the smart pointer \a SmartPointer known to QMetaType as a + smart pointer. This makes it possible to put an instance of SmartPointer into + a QVariant, if T is a type which inherits QObject. + + Note that the QWeakPointer, QSharedPointer and QPointer already have built-in + support, and it is not necessary to use this macro with them. + + This example shows a typical use of Q_DECLARE_SMART_POINTER_METATYPE(): + + \snippet code/src_corelib_kernel_qmetatype.cpp 13 +*/ + /*! \enum QMetaType::Type diff --git a/src/corelib/kernel/qmetatype.h b/src/corelib/kernel/qmetatype.h index bd4963e4f1..2d9133dbae 100644 --- a/src/corelib/kernel/qmetatype.h +++ b/src/corelib/kernel/qmetatype.h @@ -373,6 +373,8 @@ struct ConverterFunctor : public AbstractConverterFunction struct AssociativeValueTypeIsMetaType; template struct IsMetaTypePair; + template + struct MetaTypeSmartPointerHelper; } class Q_CORE_EXPORT QMetaType { @@ -656,6 +658,7 @@ private: friend bool qRegisterAssociativeConverter(); template friend struct QtPrivate::AssociativeValueTypeIsMetaType; template friend struct QtPrivate::IsMetaTypePair; + template friend struct QtPrivate::MetaTypeSmartPointerHelper; #endif #else public: @@ -1463,6 +1466,12 @@ namespace QtPrivate template struct MetaTypePairHelper : IsPair {}; + template + struct MetaTypeSmartPointerHelper + { + static bool registerConverter(int) { return false; } + }; + Q_CORE_EXPORT bool isBuiltinType(const QByteArray &type); } // namespace QtPrivate @@ -1530,6 +1539,24 @@ namespace QtPrivate { { enum DefinedType { Defined = defined }; }; + + template + struct QSmartPointerConvertFunctor + { + QObject* operator()(const SmartPointer &p) const + { + return p.operator->(); + } + }; + + template + struct QSmartPointerConvertFunctor > + { + QObject* operator()(const QWeakPointer &p) const + { + return p.data(); + } + }; } template @@ -1565,6 +1592,7 @@ int qRegisterNormalizedMetaType(const QT_PREPEND_NAMESPACE(QByteArray) &normaliz QtPrivate::SequentialContainerConverterHelper::registerConverter(id); QtPrivate::AssociativeContainerConverterHelper::registerConverter(id); QtPrivate::MetaTypePairHelper::registerConverter(id); + QtPrivate::MetaTypeSmartPointerHelper::registerConverter(id); } return id; @@ -1828,6 +1856,23 @@ struct SharedPointerMetaTypeIdHelper, true> \ return newId; \ } \ }; \ +template \ +struct MetaTypeSmartPointerHelper , \ + typename QEnableIf::Value >::Type> \ +{ \ + static bool registerConverter(int id) \ + { \ + const int toId = QMetaType::QObjectStar; \ + if (!QMetaType::hasRegisteredConverterFunction(id, toId)) { \ + QtPrivate::QSmartPointerConvertFunctor > o; \ + static const QtPrivate::ConverterFunctor, \ + QObject*, \ + QSmartPointerConvertFunctor > > f(o); \ + return QMetaType::registerConverterFunction(&f, id, toId); \ + } \ + return true; \ + } \ +}; \ } \ template \ struct QMetaTypeId< SMART_POINTER > \ -- cgit v1.2.3