| 
									
										
										
										
											2011-01-24 12:30:21 +01:00
										 |  |  | #include <QtDeclarative/QtDeclarative>
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | #include <QtDeclarative/private/qdeclarativemetatype_p.h>
 | 
					
						
							| 
									
										
										
										
											2010-06-09 14:24:23 +02:00
										 |  |  | #include <QtDeclarative/private/qdeclarativeopenmetaobject_p.h>
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | #include <QtDeclarative/QDeclarativeView>
 | 
					
						
							| 
									
										
										
										
											2011-01-24 12:30:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <QtGui/QApplication>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QtCore/QSet>
 | 
					
						
							|  |  |  | #include <QtCore/QMetaObject>
 | 
					
						
							|  |  |  | #include <QtCore/QMetaProperty>
 | 
					
						
							|  |  |  | #include <QtCore/QDebug>
 | 
					
						
							|  |  |  | #include <QtCore/private/qobject_p.h>
 | 
					
						
							|  |  |  | #include <QtCore/private/qmetaobject_p.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | #include "qmlstreamwriter.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-26 10:32:27 +01:00
										 |  |  | #ifdef QT_SIMULATOR
 | 
					
						
							|  |  |  | #include <QtGui/private/qsimulatorconnection_p.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:15:09 +01:00
										 |  |  | #ifdef Q_OS_UNIX
 | 
					
						
							|  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | void collectReachableMetaObjects(const QMetaObject *meta, QSet<const QMetaObject *> *metas) | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! meta || metas->contains(meta)) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // dynamic meta objects break things badly, so just ignore them
 | 
					
						
							| 
									
										
										
										
											2010-06-09 14:24:23 +02:00
										 |  |  |     const QMetaObjectPrivate *mop = reinterpret_cast<const QMetaObjectPrivate *>(meta->d.data); | 
					
						
							|  |  |  |     if (!(mop->flags & DynamicMetaObject)) | 
					
						
							|  |  |  |         metas->insert(meta); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     collectReachableMetaObjects(meta->superClass(), metas); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | QString currentProperty; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void collectReachableMetaObjects(QObject *object, QSet<const QMetaObject *> *metas) | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! object) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QMetaObject *meta = object->metaObject(); | 
					
						
							| 
									
										
										
										
											2010-09-15 14:13:33 +02:00
										 |  |  |     qDebug() << "Processing object" << meta->className(); | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     collectReachableMetaObjects(meta, metas); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (int index = 0; index < meta->propertyCount(); ++index) { | 
					
						
							|  |  |  |         QMetaProperty prop = meta->property(index); | 
					
						
							|  |  |  |         if (QDeclarativeMetaType::isQObject(prop.userType())) { | 
					
						
							| 
									
										
										
										
											2010-09-15 14:13:33 +02:00
										 |  |  |             qDebug() << "  Processing property" << prop.name(); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:15:09 +01:00
										 |  |  |             currentProperty = QString("%1::%2").arg(meta->className(), prop.name()); | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // if the property was not initialized during construction,
 | 
					
						
							|  |  |  |             // accessing a member of oo is going to cause a segmentation fault
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |             QObject *oo = QDeclarativeMetaType::toQObject(prop.read(object)); | 
					
						
							| 
									
										
										
										
											2010-03-31 14:09:49 +02:00
										 |  |  |             if (oo && !metas->contains(oo->metaObject())) | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |                 collectReachableMetaObjects(oo, metas); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:15:09 +01:00
										 |  |  |             currentProperty.clear(); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | void collectReachableMetaObjects(const QDeclarativeType *ty, QSet<const QMetaObject *> *metas) | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     collectReachableMetaObjects(ty->metaObject(), metas); | 
					
						
							| 
									
										
										
										
											2011-02-10 14:46:03 +01:00
										 |  |  |     if (ty->attachedPropertiesType()) | 
					
						
							|  |  |  |         collectReachableMetaObjects(ty->attachedPropertiesType(), metas); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | /* We want to add the MetaObject for 'Qt' to the list, this is a
 | 
					
						
							|  |  |  |    simple way to access it. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | class FriendlyQObject: public QObject | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | public: | 
					
						
							|  |  |  |     static const QMetaObject *qtMeta() { return &staticQtMetaObject; } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | /* When we dump a QMetaObject, we want to list all the types it is exported as.
 | 
					
						
							|  |  |  |    To do this, we need to find the QDeclarativeTypes associated with this | 
					
						
							|  |  |  |    QMetaObject. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | static QHash<QByteArray, QSet<const QDeclarativeType *> > qmlTypesByCppName; | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | static QHash<QByteArray, QByteArray> cppToId; | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | /* Takes a C++ type name, such as Qt::LayoutDirection or QString and
 | 
					
						
							|  |  |  |    maps it to how it should appear in the description file. | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |    These names need to be unique globally, so we don't change the C++ symbol's | 
					
						
							|  |  |  |    name much. It is mostly used to for explicit translations such as | 
					
						
							|  |  |  |    QString->string and translations for extended QML objects. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | QByteArray convertToId(const QByteArray &cppName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return cppToId.value(cppName, cppName); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | QSet<const QMetaObject *> collectReachableMetaObjects(const QString &importCode, QDeclarativeEngine *engine) | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     QSet<const QMetaObject *> metas; | 
					
						
							|  |  |  |     metas.insert(FriendlyQObject::qtMeta()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QHash<QByteArray, QSet<QByteArray> > extensions; | 
					
						
							|  |  |  |     foreach (const QDeclarativeType *ty, QDeclarativeMetaType::qmlTypes()) { | 
					
						
							|  |  |  |         qmlTypesByCppName[ty->metaObject()->className()].insert(ty); | 
					
						
							|  |  |  |         if (ty->isExtendedType()) { | 
					
						
							|  |  |  |             extensions[ty->typeName()].insert(ty->metaObject()->className()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         collectReachableMetaObjects(ty, &metas); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // Adjust ids of extended objects.
 | 
					
						
							|  |  |  |     // The chain ends up being:
 | 
					
						
							|  |  |  |     // __extended__.originalname - the base object
 | 
					
						
							|  |  |  |     // __extension_0_.originalname - first extension
 | 
					
						
							|  |  |  |     // ..
 | 
					
						
							|  |  |  |     // __extension_n-2_.originalname - second to last extension
 | 
					
						
							|  |  |  |     // originalname - last extension
 | 
					
						
							|  |  |  |     // ### does this actually work for multiple extensions? it seems like the prototypes might be wrong
 | 
					
						
							|  |  |  |     foreach (const QByteArray &extendedCpp, extensions.keys()) { | 
					
						
							|  |  |  |         cppToId.remove(extendedCpp); | 
					
						
							|  |  |  |         const QByteArray extendedId = convertToId(extendedCpp); | 
					
						
							|  |  |  |         cppToId.insert(extendedCpp, "__extended__." + extendedId); | 
					
						
							|  |  |  |         QSet<QByteArray> extensionCppNames = extensions.value(extendedCpp); | 
					
						
							|  |  |  |         int c = 0; | 
					
						
							|  |  |  |         foreach (const QByteArray &extensionCppName, extensionCppNames) { | 
					
						
							|  |  |  |             if (c != extensionCppNames.size() - 1) { | 
					
						
							|  |  |  |                 QByteArray adjustedName = QString("__extension__%1.%2").arg(QString::number(c), QString(extendedId)).toAscii(); | 
					
						
							|  |  |  |                 cppToId.insert(extensionCppName, adjustedName); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 cppToId.insert(extensionCppName, extendedId); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             ++c; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // find even more QMetaObjects by instantiating QML types and running
 | 
					
						
							|  |  |  |     // over the instances
 | 
					
						
							|  |  |  |     foreach (const QDeclarativeType *ty, QDeclarativeMetaType::qmlTypes()) { | 
					
						
							|  |  |  |         if (ty->isExtendedType()) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         QByteArray tyName = ty->qmlTypeName(); | 
					
						
							|  |  |  |         tyName = tyName.mid(tyName.lastIndexOf('/') + 1); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         QByteArray code = importCode.toUtf8(); | 
					
						
							|  |  |  |         code += tyName; | 
					
						
							|  |  |  |         code += " {}\n"; | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         QDeclarativeComponent c(engine); | 
					
						
							|  |  |  |         c.setData(code, QUrl("typeinstance")); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         QObject *object = c.create(); | 
					
						
							|  |  |  |         if (object) | 
					
						
							|  |  |  |             collectReachableMetaObjects(object, &metas); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             qDebug() << "Could not create" << tyName << ":" << c.errorString(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     return metas; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | class Dumper | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QmlStreamWriter *qml; | 
					
						
							|  |  |  |     QString relocatableModuleUri; | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | public: | 
					
						
							|  |  |  |     Dumper(QmlStreamWriter *qml) : qml(qml) {} | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     void setRelocatableModuleUri(const QString &uri) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         relocatableModuleUri = uri; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     void dump(const QMetaObject *meta) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         qml->writeStartObject("Component"); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         QByteArray id = convertToId(meta->className()); | 
					
						
							|  |  |  |         qml->writeScriptBinding(QLatin1String("name"), enquote(id)); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         for (int index = meta->classInfoCount() - 1 ; index >= 0 ; --index) { | 
					
						
							|  |  |  |             QMetaClassInfo classInfo = meta->classInfo(index); | 
					
						
							|  |  |  |             if (QLatin1String(classInfo.name()) == QLatin1String("DefaultProperty")) { | 
					
						
							|  |  |  |                 qml->writeScriptBinding(QLatin1String("defaultProperty"), enquote(QLatin1String(classInfo.value()))); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         if (meta->superClass()) | 
					
						
							|  |  |  |             qml->writeScriptBinding(QLatin1String("prototype"), enquote(convertToId(meta->superClass()->className()))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         QSet<const QDeclarativeType *> qmlTypes = qmlTypesByCppName.value(meta->className()); | 
					
						
							|  |  |  |         if (!qmlTypes.isEmpty()) { | 
					
						
							|  |  |  |             QStringList exports; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             foreach (const QDeclarativeType *qmlTy, qmlTypes) { | 
					
						
							|  |  |  |                 QString qmlTyName = qmlTy->qmlTypeName(); | 
					
						
							| 
									
										
										
										
											2011-02-09 13:38:59 +01:00
										 |  |  |                 // some qmltype names are missing the actual names, ignore that import
 | 
					
						
							|  |  |  |                 if (qmlTyName.endsWith('/')) | 
					
						
							|  |  |  |                     continue; | 
					
						
							| 
									
										
										
										
											2011-02-08 13:22:34 +01:00
										 |  |  |                 if (!relocatableModuleUri.isNull() | 
					
						
							|  |  |  |                         && qmlTyName.startsWith(relocatableModuleUri + QLatin1Char('/'))) { | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |                     qmlTyName.remove(0, relocatableModuleUri.size() + 1); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 exports += enquote(QString("%1 %2.%3").arg( | 
					
						
							|  |  |  |                                        qmlTyName, | 
					
						
							|  |  |  |                                        QString::number(qmlTy->majorVersion()), | 
					
						
							|  |  |  |                                        QString::number(qmlTy->minorVersion()))); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |             // ensure exports are sorted and don't change order when the plugin is dumped again
 | 
					
						
							|  |  |  |             exports.removeDuplicates(); | 
					
						
							|  |  |  |             qSort(exports); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |             qml->writeArrayBinding(QLatin1String("exports"), exports); | 
					
						
							| 
									
										
										
										
											2011-02-10 14:46:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (const QMetaObject *attachedType = (*qmlTypes.begin())->attachedPropertiesType()) { | 
					
						
							|  |  |  |                 qml->writeScriptBinding(QLatin1String("attachedType"), enquote( | 
					
						
							|  |  |  |                                             convertToId(attachedType->className()))); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         for (int index = meta->enumeratorOffset(); index < meta->enumeratorCount(); ++index) | 
					
						
							|  |  |  |             dump(meta->enumerator(index)); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         for (int index = meta->propertyOffset(); index < meta->propertyCount(); ++index) | 
					
						
							|  |  |  |             dump(meta->property(index)); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         for (int index = meta->methodOffset(); index < meta->methodCount(); ++index) | 
					
						
							|  |  |  |             dump(meta->method(index)); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         qml->writeEndObject(); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     void writeEasingCurve() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         qml->writeStartObject("Component"); | 
					
						
							|  |  |  |         qml->writeScriptBinding(QLatin1String("name"), enquote(QLatin1String("QEasingCurve"))); | 
					
						
							|  |  |  |         qml->writeScriptBinding(QLatin1String("prototype"), enquote(QLatin1String("QDeclarativeEasingValueType"))); | 
					
						
							|  |  |  |         qml->writeEndObject(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     static QString enquote(const QString &string) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return QString("\"%1\"").arg(string); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     /* Removes pointer and list annotations from a type name, returning
 | 
					
						
							|  |  |  |        what was removed in isList and isPointer | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     static void removePointerAndList(QByteArray *typeName, bool *isList, bool *isPointer) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         static QByteArray declListPrefix = "QDeclarativeListProperty<"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (typeName->endsWith('*')) { | 
					
						
							|  |  |  |             *isPointer = true; | 
					
						
							|  |  |  |             typeName->truncate(typeName->length() - 1); | 
					
						
							|  |  |  |             removePointerAndList(typeName, isList, isPointer); | 
					
						
							|  |  |  |         } else if (typeName->startsWith(declListPrefix)) { | 
					
						
							|  |  |  |             *isList = true; | 
					
						
							|  |  |  |             typeName->truncate(typeName->length() - 1); // get rid of the suffix '>'
 | 
					
						
							|  |  |  |             *typeName = typeName->mid(declListPrefix.size()); | 
					
						
							|  |  |  |             removePointerAndList(typeName, isList, isPointer); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-20 12:02:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         *typeName = convertToId(*typeName); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     void writeTypeProperties(QByteArray typeName, bool isWritable) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         bool isList = false, isPointer = false; | 
					
						
							|  |  |  |         removePointerAndList(&typeName, &isList, &isPointer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qml->writeScriptBinding(QLatin1String("type"), enquote(typeName)); | 
					
						
							|  |  |  |         if (isList) | 
					
						
							|  |  |  |             qml->writeScriptBinding(QLatin1String("isList"), QLatin1String("true")); | 
					
						
							|  |  |  |         if (!isWritable) | 
					
						
							|  |  |  |             qml->writeScriptBinding(QLatin1String("isReadonly"), QLatin1String("true")); | 
					
						
							|  |  |  |         if (isPointer) | 
					
						
							|  |  |  |             qml->writeScriptBinding(QLatin1String("isPointer"), QLatin1String("true")); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     void dump(const QMetaProperty &prop) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         qml->writeStartObject("Property"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qml->writeScriptBinding(QLatin1String("name"), enquote(QString::fromUtf8(prop.name()))); | 
					
						
							|  |  |  |         writeTypeProperties(prop.typeName(), prop.isWritable()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qml->writeEndObject(); | 
					
						
							| 
									
										
										
										
											2010-03-31 14:09:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     void dump(const QMetaMethod &meth) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (meth.methodType() == QMetaMethod::Signal) { | 
					
						
							|  |  |  |             if (meth.access() != QMetaMethod::Protected) | 
					
						
							|  |  |  |                 return; // nothing to do.
 | 
					
						
							|  |  |  |         } else if (meth.access() != QMetaMethod::Public) { | 
					
						
							|  |  |  |             return; // nothing to do.
 | 
					
						
							| 
									
										
										
										
											2010-09-30 11:21:23 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         QByteArray name = meth.signature(); | 
					
						
							|  |  |  |         int lparenIndex = name.indexOf('('); | 
					
						
							|  |  |  |         if (lparenIndex == -1) { | 
					
						
							|  |  |  |             return; // invalid signature
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         name = name.left(lparenIndex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (meth.methodType() == QMetaMethod::Signal) | 
					
						
							|  |  |  |             qml->writeStartObject(QLatin1String("Signal")); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             qml->writeStartObject(QLatin1String("Method")); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         qml->writeScriptBinding(QLatin1String("name"), enquote(name)); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         const QString typeName = convertToId(meth.typeName()); | 
					
						
							|  |  |  |         if (! typeName.isEmpty()) | 
					
						
							|  |  |  |             qml->writeScriptBinding(QLatin1String("type"), enquote(typeName)); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         for (int i = 0; i < meth.parameterTypes().size(); ++i) { | 
					
						
							|  |  |  |             QByteArray argName = meth.parameterNames().at(i); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |             qml->writeStartObject(QLatin1String("Parameter")); | 
					
						
							|  |  |  |             if (! argName.isEmpty()) | 
					
						
							|  |  |  |                 qml->writeScriptBinding(QLatin1String("name"), enquote(argName)); | 
					
						
							|  |  |  |             writeTypeProperties(meth.parameterTypes().at(i), true); | 
					
						
							|  |  |  |             qml->writeEndObject(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qml->writeEndObject(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void dump(const QMetaEnum &e) | 
					
						
							| 
									
										
										
										
											2010-03-03 11:36:42 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         qml->writeStartObject(QLatin1String("Enum")); | 
					
						
							|  |  |  |         qml->writeScriptBinding(QLatin1String("name"), enquote(QString::fromUtf8(e.name()))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         QList<QPair<QString, QString> > namesValues; | 
					
						
							|  |  |  |         for (int index = 0; index < e.keyCount(); ++index) { | 
					
						
							|  |  |  |             namesValues.append(qMakePair(enquote(QString::fromUtf8(e.key(index))), QString::number(e.value(index)))); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qml->writeScriptObjectLiteralBinding(QLatin1String("values"), namesValues); | 
					
						
							|  |  |  |         qml->writeEndObject(); | 
					
						
							| 
									
										
										
										
											2010-03-03 11:36:42 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-03-03 11:36:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-20 16:02:28 +01:00
										 |  |  | enum ExitCode { | 
					
						
							|  |  |  |     EXIT_INVALIDARGUMENTS = 1, | 
					
						
							|  |  |  |     EXIT_SEGV = 2, | 
					
						
							|  |  |  |     EXIT_IMPORTERROR = 3 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:15:09 +01:00
										 |  |  | #ifdef Q_OS_UNIX
 | 
					
						
							|  |  |  | void sigSegvHandler(int) { | 
					
						
							|  |  |  |     fprintf(stderr, "Error: qmldump SEGV\n"); | 
					
						
							|  |  |  |     if (!currentProperty.isEmpty()) | 
					
						
							|  |  |  |         fprintf(stderr, "While processing the property '%s', which probably has uninitialized data.\n", currentProperty.toLatin1().constData()); | 
					
						
							| 
									
										
										
										
											2011-01-20 16:02:28 +01:00
										 |  |  |     exit(EXIT_SEGV); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:15:09 +01:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 13:22:34 +01:00
										 |  |  | void printUsage(const QString &appName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     qWarning() << qPrintable(QString( | 
					
						
							|  |  |  |                                  "Usage: %1 [--notrelocatable] module.uri version [module/import/path]\n" | 
					
						
							|  |  |  |                                  "       %1 --builtins\n" | 
					
						
							|  |  |  |                                  "Example: %1 Qt.labs.particles 4.7 /home/user/dev/qt-install/imports").arg( | 
					
						
							|  |  |  |                                  appName)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | int main(int argc, char *argv[]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-12 13:15:09 +01:00
										 |  |  | #ifdef Q_OS_UNIX
 | 
					
						
							|  |  |  |     // qmldump may crash, but we don't want any crash handlers to pop up
 | 
					
						
							|  |  |  |     // therefore we intercept the segfault and just exit() ourselves
 | 
					
						
							|  |  |  |     struct sigaction action; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sigemptyset(&action.sa_mask); | 
					
						
							|  |  |  |     action.sa_handler = &sigSegvHandler; | 
					
						
							|  |  |  |     action.sa_flags   = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sigaction(SIGSEGV, &action, 0); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-26 10:32:27 +01:00
										 |  |  | #ifdef QT_SIMULATOR
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // Running this application would bring up the Qt Simulator (since it links QtGui), avoid that!
 | 
					
						
							| 
									
										
										
										
											2010-11-26 10:32:27 +01:00
										 |  |  |     QtSimulatorPrivate::SimulatorConnection::createStubInstance(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     QApplication app(argc, argv); | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     const QStringList args = app.arguments(); | 
					
						
							|  |  |  |     const QString appName = QFileInfo(app.applicationFilePath()).baseName(); | 
					
						
							| 
									
										
										
										
											2011-02-08 13:22:34 +01:00
										 |  |  |     if (!(args.size() >= 3 || (args.size() == 2 && args.at(1) == QLatin1String("--builtins")))) { | 
					
						
							|  |  |  |         printUsage(appName); | 
					
						
							| 
									
										
										
										
											2011-01-20 16:02:28 +01:00
										 |  |  |         return EXIT_INVALIDARGUMENTS; | 
					
						
							| 
									
										
										
										
											2010-05-20 12:02:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     QString pluginImportUri; | 
					
						
							|  |  |  |     QString pluginImportVersion; | 
					
						
							| 
									
										
										
										
											2010-05-20 12:02:46 +02:00
										 |  |  |     QString pluginImportPath; | 
					
						
							| 
									
										
										
										
											2011-02-08 13:22:34 +01:00
										 |  |  |     bool relocatable = true; | 
					
						
							|  |  |  |     if (args.size() >= 3) { | 
					
						
							|  |  |  |         QStringList positionalArgs; | 
					
						
							|  |  |  |         foreach (const QString &arg, args) { | 
					
						
							|  |  |  |             if (!arg.startsWith("--")) { | 
					
						
							|  |  |  |                 positionalArgs.append(arg); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (arg == QLatin1String("--notrelocatable")) { | 
					
						
							|  |  |  |                 relocatable = false; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 qWarning() << "Invalid argument: " << arg; | 
					
						
							|  |  |  |                 return EXIT_INVALIDARGUMENTS; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (positionalArgs.size() != 3 && positionalArgs.size() != 4) { | 
					
						
							|  |  |  |             qWarning() << "Incorrect number of positional arguments"; | 
					
						
							|  |  |  |             return EXIT_INVALIDARGUMENTS; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pluginImportUri = positionalArgs[1]; | 
					
						
							|  |  |  |         pluginImportVersion = positionalArgs[2]; | 
					
						
							|  |  |  |         if (positionalArgs.size() >= 4) | 
					
						
							|  |  |  |             pluginImportPath = positionalArgs[3]; | 
					
						
							| 
									
										
										
										
											2010-05-20 12:02:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     QDeclarativeView view; | 
					
						
							|  |  |  |     QDeclarativeEngine *engine = view.engine(); | 
					
						
							| 
									
										
										
										
											2010-05-20 12:02:46 +02:00
										 |  |  |     if (!pluginImportPath.isEmpty()) | 
					
						
							|  |  |  |         engine->addImportPath(pluginImportPath); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // find all QMetaObjects reachable from the builtin module
 | 
					
						
							|  |  |  |     QByteArray importCode("import QtQuick 1.0\n"); | 
					
						
							|  |  |  |     QSet<const QMetaObject *> defaultReachable = collectReachableMetaObjects(importCode, engine); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // this will hold the meta objects we want to dump information of
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     QSet<const QMetaObject *> metas; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     if (pluginImportUri.isEmpty()) { | 
					
						
							|  |  |  |         metas = defaultReachable; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // find all QMetaObjects reachable when the specified module is imported
 | 
					
						
							|  |  |  |         importCode += QString("import %0 %1\n").arg(pluginImportUri, pluginImportVersion).toAscii(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // create a component with these imports to make sure the imports are valid
 | 
					
						
							|  |  |  |         // and to populate the declarative meta type system
 | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             QByteArray code = importCode; | 
					
						
							|  |  |  |             code += "QtObject {}"; | 
					
						
							|  |  |  |             QDeclarativeComponent c(engine); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             c.setData(code, QUrl("typelist")); | 
					
						
							|  |  |  |             c.create(); | 
					
						
							|  |  |  |             if (!c.errors().isEmpty()) { | 
					
						
							|  |  |  |                 foreach (const QDeclarativeError &error, c.errors()) | 
					
						
							|  |  |  |                     qWarning() << error.toString(); | 
					
						
							|  |  |  |                 return EXIT_IMPORTERROR; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-06-01 12:03:17 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         QSet<const QMetaObject *> candidates = collectReachableMetaObjects(importCode, engine); | 
					
						
							|  |  |  |         candidates.subtract(defaultReachable); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Also eliminate meta objects with the same classname.
 | 
					
						
							|  |  |  |         // This is required because extended objects seem not to share
 | 
					
						
							|  |  |  |         // a single meta object instance.
 | 
					
						
							|  |  |  |         QSet<QByteArray> defaultReachableNames; | 
					
						
							|  |  |  |         foreach (const QMetaObject *mo, defaultReachable) | 
					
						
							|  |  |  |             defaultReachableNames.insert(QByteArray(mo->className())); | 
					
						
							|  |  |  |         foreach (const QMetaObject *mo, candidates) { | 
					
						
							|  |  |  |             if (!defaultReachableNames.contains(mo->className())) | 
					
						
							|  |  |  |                 metas.insert(mo); | 
					
						
							| 
									
										
										
										
											2010-06-01 12:03:17 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // setup static rewrites of type names
 | 
					
						
							|  |  |  |     cppToId.insert("QString", "string"); | 
					
						
							|  |  |  |     cppToId.insert("QDeclarativeEasingValueType::Type", "Type"); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // start dumping data
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     QByteArray bytes; | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     QmlStreamWriter qml(&bytes); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     qml.writeStartDocument(); | 
					
						
							|  |  |  |     qml.writeLibraryImport(QLatin1String("QtQuick.tooling"), 1, 0); | 
					
						
							|  |  |  |     qml.write("\n" | 
					
						
							|  |  |  |               "// This file describes the plugin-supplied types contained in the library.\n" | 
					
						
							|  |  |  |               "// It is used for QML tooling purposes only.\n" | 
					
						
							|  |  |  |               "\n"); | 
					
						
							|  |  |  |     qml.writeStartObject("Module"); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // put the metaobjects into a map so they are always dumped in the same order
 | 
					
						
							| 
									
										
										
										
											2010-03-05 10:17:18 +01:00
										 |  |  |     QMap<QString, const QMetaObject *> nameToMeta; | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     foreach (const QMetaObject *meta, metas) | 
					
						
							| 
									
										
										
										
											2010-09-30 11:21:23 +02:00
										 |  |  |         nameToMeta.insert(convertToId(meta->className()), meta); | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Dumper dumper(&qml); | 
					
						
							| 
									
										
										
										
											2011-02-08 13:22:34 +01:00
										 |  |  |     if (relocatable) | 
					
						
							|  |  |  |         dumper.setRelocatableModuleUri(pluginImportUri); | 
					
						
							| 
									
										
										
										
											2010-03-05 10:17:18 +01:00
										 |  |  |     foreach (const QMetaObject *meta, nameToMeta) { | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |         dumper.dump(meta); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-03-15 17:41:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // define QEasingCurve as an extension of QDeclarativeEasingValueType, this way
 | 
					
						
							|  |  |  |     // properties using the QEasingCurve type get useful type information.
 | 
					
						
							|  |  |  |     if (pluginImportUri.isEmpty()) | 
					
						
							|  |  |  |         dumper.writeEasingCurve(); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     qml.writeEndObject(); | 
					
						
							|  |  |  |     qml.writeEndDocument(); | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::cout << bytes.constData(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-07 11:34:49 +01:00
										 |  |  |     // workaround to avoid crashes on exit
 | 
					
						
							| 
									
										
										
										
											2010-02-25 13:53:58 +01:00
										 |  |  |     QTimer timer; | 
					
						
							|  |  |  |     timer.setSingleShot(true); | 
					
						
							|  |  |  |     timer.setInterval(0); | 
					
						
							|  |  |  |     QObject::connect(&timer, SIGNAL(timeout()), &app, SLOT(quit())); | 
					
						
							|  |  |  |     timer.start(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return app.exec(); | 
					
						
							|  |  |  | } |