/**************************************************************************** ** ** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). ** Contact: Qt Software Information (qt-info@nokia.com) ** ** This file is part of the QtDeclarative module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** No Commercial Usage ** This file contains pre-release code and may not be distributed. ** You may use this file in accordance with the terms and conditions ** contained in the either Technology Preview License Agreement or the ** Beta Release License Agreement. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Nokia gives you certain ** additional rights. These rights are described in the Nokia Qt LGPL ** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this ** package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** If you are unsure which license is appropriate for your use, please ** contact the sales department at qt-sales@nokia.com. ** $QT_END_LICENSE$ ** ****************************************************************************/ #undef QT3_SUPPORT // don't want it here - it just causes bugs (which is why we removed it) #include #include #include #include #include #ifdef QT_SCRIPTTOOLS_LIB #include #endif #include #include #include #include #include #include #include #include #include #include #include "qml.h" #include #include #include "private/qmlbasicscript_p.h" #include "qmlengine.h" #include "qmlcontext.h" #include "qmlexpression.h" #include #include #include #include #include #include #include #include "private/qmlcomponentjs_p.h" #include "private/qmlmetaproperty_p.h" #include #include #include #include Q_DECLARE_METATYPE(QmlMetaProperty) Q_DECLARE_METATYPE(QList); QT_BEGIN_NAMESPACE DEFINE_BOOL_CONFIG_OPTION(qmlDebugger, QML_DEBUGGER) DEFINE_BOOL_CONFIG_OPTION(qmlImportTrace, QML_IMPORT_TRACE) QML_DEFINE_TYPE(Qt,4,6,(QT_VERSION&0x00ff00)>>8,Object,QObject) struct StaticQtMetaObject : public QObject { static const QMetaObject *get() { return &static_cast (0)->staticQtMetaObject; } }; QScriptValue desktopOpenUrl(QScriptContext *ctxt, QScriptEngine *e) { if(!ctxt->argumentCount()) return e->newVariant(QVariant(false)); bool ret = QDesktopServices::openUrl(QUrl(ctxt->argument(0).toString())); return e->newVariant(QVariant(ret)); } QmlEnginePrivate::QmlEnginePrivate(QmlEngine *e) : rootContext(0), currentExpression(0), isDebugging(false), contextClass(0), objectClass(0), valueTypeClass(0), nodeListClass(0), namedNodeMapClass(0), scriptEngine(this), rootComponent(0), networkAccessManager(0), typeManager(e), uniqueId(1) { QScriptValue qtObject = scriptEngine.newQMetaObject(StaticQtMetaObject::get()); QScriptValue desktopObject = scriptEngine.newObject(); desktopObject.setProperty(QLatin1String("openUrl"),scriptEngine.newFunction(desktopOpenUrl, 1)); qtObject.setProperty(QLatin1String("DesktopServices"), desktopObject); scriptEngine.globalObject().setProperty(QLatin1String("Qt"), qtObject); qt_add_qmlxmlhttprequest(&scriptEngine); qtObject.setProperty(QLatin1String("rgba"), scriptEngine.newFunction(QmlEnginePrivate::rgba, 4)); qtObject.setProperty(QLatin1String("hsla"), scriptEngine.newFunction(QmlEnginePrivate::hsla, 4)); qtObject.setProperty(QLatin1String("rect"), scriptEngine.newFunction(QmlEnginePrivate::rect, 4)); qtObject.setProperty(QLatin1String("point"), scriptEngine.newFunction(QmlEnginePrivate::point, 2)); qtObject.setProperty(QLatin1String("size"), scriptEngine.newFunction(QmlEnginePrivate::size, 2)); qtObject.setProperty(QLatin1String("vector3d"), scriptEngine.newFunction(QmlEnginePrivate::vector, 3)); } QmlEnginePrivate::~QmlEnginePrivate() { delete rootContext; rootContext = 0; delete contextClass; contextClass = 0; delete objectClass; objectClass = 0; delete networkAccessManager; networkAccessManager = 0; delete nodeListClass; nodeListClass = 0; delete namedNodeMapClass; namedNodeMapClass = 0; for(int ii = 0; ii < bindValues.count(); ++ii) clear(bindValues[ii]); for(int ii = 0; ii < parserStatus.count(); ++ii) clear(parserStatus[ii]); } void QmlEnginePrivate::clear(SimpleList &bvs) { bvs.clear(); } void QmlEnginePrivate::clear(SimpleList &pss) { for (int ii = 0; ii < pss.count; ++ii) { QmlParserStatus *ps = pss.at(ii); if(ps) ps->d = 0; } pss.clear(); } Q_GLOBAL_STATIC(QmlEngineDebugServer, qmlEngineDebugServer); void QmlEnginePrivate::init() { Q_Q(QmlEngine); scriptEngine.installTranslatorFunctions(); contextClass = new QmlContextScriptClass(q); objectClass = new QmlObjectScriptClass(q); valueTypeClass = new QmlValueTypeScriptClass(q); rootContext = new QmlContext(q,true); #ifdef QT_SCRIPTTOOLS_LIB if (qmlDebugger()){ debugger = new QScriptEngineDebugger(q); debugger->attachTo(&scriptEngine); } #endif scriptEngine.globalObject().setProperty(QLatin1String("createQmlObject"), scriptEngine.newFunction(QmlEnginePrivate::createQmlObject, 1)); scriptEngine.globalObject().setProperty(QLatin1String("createComponent"), scriptEngine.newFunction(QmlEnginePrivate::createComponent, 1)); scriptEngine.globalObject().setProperty(QLatin1String("vector"), scriptEngine.newFunction(QmlEnginePrivate::vector, 3)); if (QCoreApplication::instance()->thread() == q->thread() && QmlEngineDebugServer::isDebuggingEnabled()) { qmlEngineDebugServer(); isDebugging = true; QmlEngineDebugServer::addEngine(q); qmlEngineDebugServer()->waitForClients(); } } QmlEnginePrivate::CapturedProperty::CapturedProperty(const QmlMetaProperty &p) : object(p.object()), coreIndex(p.coreIndex()), notifyIndex(p.property().notifySignalIndex()) { } //////////////////////////////////////////////////////////////////// typedef QHash, bool> FunctionCache; Q_GLOBAL_STATIC(FunctionCache, functionCache); QScriptClass::QueryFlags QmlEnginePrivate::queryObject(const QString &propName, uint *id, QObject *obj) { QScriptClass::QueryFlags rv = 0; QmlContext *ctxt = QmlEngine::contextForObject(obj); if (!ctxt) ctxt = rootContext; QmlMetaProperty prop(obj, propName, ctxt); if (prop.type() == QmlMetaProperty::Invalid) { QPair key = qMakePair(obj->metaObject(), propName); bool isFunction = false; if (functionCache()->contains(key)) { isFunction = functionCache()->value(key); } else { QScriptValue sobj = scriptEngine.newQObject(obj); QScriptValue func = sobj.property(propName); isFunction = func.isFunction(); functionCache()->insert(key, isFunction); } if (isFunction) { *id = QmlScriptClass::FunctionId; rv |= QScriptClass::HandlesReadAccess; } } else { *id = QmlScriptClass::PropertyId; *id |= prop.save(); rv |= QScriptClass::HandlesReadAccess; if (prop.isWritable()) rv |= QScriptClass::HandlesWriteAccess; } return rv; } struct QmlValueTypeReference { QmlValueType *type; QGuard object; int property; }; Q_DECLARE_METATYPE(QmlValueTypeReference); QScriptValue QmlEnginePrivate::propertyObject(const QScriptString &propName, QObject *obj, uint id) { if (id == QmlScriptClass::FunctionId) { QScriptValue sobj = scriptEngine.newQObject(obj); QScriptValue func = sobj.property(propName); return func; } else { QmlMetaProperty prop; prop.restore(id, obj); if (!prop.isValid()) return QScriptValue(); if (prop.needsChangedNotifier()) capturedProperties << CapturedProperty(prop); int propType = prop.propertyType(); if (propType < QVariant::UserType && valueTypes[propType]) { QmlValueTypeReference ref; ref.type = valueTypes[propType]; ref.object = obj; ref.property = prop.coreIndex(); return scriptEngine.newObject(valueTypeClass, scriptEngine.newVariant(QVariant::fromValue(ref))); } QVariant var = prop.read(); QObject *varobj = (propType < QVariant::UserType)?0:QmlMetaType::toQObject(var); if (!varobj) varobj = qvariant_cast(var); if (varobj) { return scriptEngine.newObject(objectClass, scriptEngine.newVariant(QVariant::fromValue(varobj))); } else { return qScriptValueFromValue(&scriptEngine, var); } } return QScriptValue(); } /*! \class QmlEngine \brief The QmlEngine class provides an environment for instantiating QML components. \mainclass Each QML component is instantiated in a QmlContext. QmlContext's are essential for passing data to QML components. In QML, contexts are arranged hierarchically and this hierarchy is managed by the QmlEngine. Prior to creating any QML components, an application must have created a QmlEngine to gain access to a QML context. The following example shows how to create a simple Text item. \code QmlEngine engine; QmlComponent component(&engine, "Text { text: \"Hello world!\" }"); QFxItem *item = qobject_cast(component.create()); //add item to view, etc ... \endcode In this case, the Text item will be created in the engine's \l {QmlEngine::rootContext()}{root context}. \sa QmlComponent QmlContext */ /*! Create a new QmlEngine with the given \a parent. */ QmlEngine::QmlEngine(QObject *parent) : QObject(*new QmlEnginePrivate(this), parent) { Q_D(QmlEngine); d->init(); qRegisterMetaType("QVariant"); } /*! Destroys the QmlEngine. Any QmlContext's created on this engine will be invalidated, but not destroyed (unless they are parented to the QmlEngine object). */ QmlEngine::~QmlEngine() { Q_D(QmlEngine); if (d->isDebugging) QmlEngineDebugServer::remEngine(this); } /*! Clears the engine's internal component cache. Normally the QmlEngine caches components loaded from qml files. This method clears this cache and forces the component to be reloaded. */ void QmlEngine::clearComponentCache() { Q_D(QmlEngine); d->typeManager.clearCache(); } /*! Returns the engine's root context. The root context is automatically created by the QmlEngine. Data that should be available to all QML component instances instantiated by the engine should be put in the root context. Additional data that should only be available to a subset of component instances should be added to sub-contexts parented to the root context. */ QmlContext *QmlEngine::rootContext() { Q_D(QmlEngine); return d->rootContext; } /*! Sets the common QNetworkAccessManager, \a network, used by all QML elements instantiated by this engine. Any previously set manager is deleted and \a network is owned by the QmlEngine. This method should only be called before any QmlComponents are instantiated. */ void QmlEngine::setNetworkAccessManager(QNetworkAccessManager *network) { Q_D(QmlEngine); delete d->networkAccessManager; d->networkAccessManager = network; } /*! Returns the common QNetworkAccessManager used by all QML elements instantiated by this engine. The default implements no caching, cookiejar, etc., just a default QNetworkAccessManager. */ QNetworkAccessManager *QmlEngine::networkAccessManager() const { Q_D(const QmlEngine); if (!d->networkAccessManager) d->networkAccessManager = new QNetworkAccessManager; return d->networkAccessManager; } /*! Return the base URL for this engine. The base URL is only used to resolve components when a relative URL is passed to the QmlComponent constructor. If a base URL has not been explicitly set, this method returns the application's current working directory. \sa setBaseUrl() */ QUrl QmlEngine::baseUrl() const { Q_D(const QmlEngine); if (d->baseUrl.isEmpty()) { return QUrl::fromLocalFile(QDir::currentPath() + QDir::separator()); } else { return d->baseUrl; } } /*! Set the base URL for this engine to \a url. \sa baseUrl() */ void QmlEngine::setBaseUrl(const QUrl &url) { Q_D(QmlEngine); d->baseUrl = url; } /*! Returns the QmlContext for the \a object, or 0 if no context has been set. When the QmlEngine instantiates a QObject, the context is set automatically. */ QmlContext *QmlEngine::contextForObject(const QObject *object) { if(!object) return 0; QObjectPrivate *priv = QObjectPrivate::get(const_cast(object)); QmlDeclarativeData *data = static_cast(priv->declarativeData); return data?data->context:0; } /*! Sets the QmlContext for the \a object to \a context. If the \a object already has a context, a warning is output, but the context is not changed. When the QmlEngine instantiates a QObject, the context is set automatically. */ void QmlEngine::setContextForObject(QObject *object, QmlContext *context) { QObjectPrivate *priv = QObjectPrivate::get(object); QmlDeclarativeData *data = static_cast(priv->declarativeData); if (data && data->context) { qWarning("QmlEngine::setContextForObject(): Object already has a QmlContext"); return; } if (!data) { priv->declarativeData = new QmlDeclarativeData(context); } else { data->context = context; } context->d_func()->contextObjects.append(object); } void qmlExecuteDeferred(QObject *object) { QmlDeclarativeData *data = QmlDeclarativeData::get(object); if (data && data->deferredComponent) { QmlVME vme; vme.runDeferred(object); data->deferredComponent->release(); data->deferredComponent = 0; } } QmlContext *qmlContext(const QObject *obj) { return QmlEngine::contextForObject(obj); } QmlEngine *qmlEngine(const QObject *obj) { QmlContext *context = QmlEngine::contextForObject(obj); return context?context->engine():0; } QObject *qmlAttachedPropertiesObjectById(int id, const QObject *object, bool create) { QmlDeclarativeData *data = QmlDeclarativeData::get(object); QObject *rv = data->attachedProperties?data->attachedProperties->value(id):0; if (rv || !create) return rv; QmlAttachedPropertiesFunc pf = QmlMetaType::attachedPropertiesFuncById(id); if (!pf) return 0; rv = pf(const_cast(object)); if (rv) { if (!data->attachedProperties) data->attachedProperties = new QHash(); data->attachedProperties->insert(id, rv); } return rv; } QmlDeclarativeData::QmlDeclarativeData(QmlContext *ctxt) : context(ctxt), bindings(0), deferredComponent(0), attachedProperties(0) { } void QmlDeclarativeData::destroyed(QObject *object) { if (deferredComponent) deferredComponent->release(); if (attachedProperties) delete attachedProperties; if (context) static_cast(QObjectPrivate::get(context))->contextObjects.removeAll(object); QmlAbstractBinding *binding = bindings; while (binding) { QmlAbstractBinding *next = binding->m_nextBinding; binding->m_prevBinding = 0; binding->m_nextBinding = 0; delete binding; binding = next; } delete this; } /*! Creates a QScriptValue allowing you to use \a object in QML script. \a engine is the QmlEngine it is to be created in. The QScriptValue returned is a QtScript Object, not a QtScript QObject, due to the special needs of QML requiring more functionality than a standard QtScript QObject. */ QScriptValue QmlEnginePrivate::qmlScriptObject(QObject* object, QmlEngine* engine) { QScriptEngine *scriptEngine = QmlEnginePrivate::getScriptEngine(engine); return scriptEngine->newObject(engine->d_func()->objectClass, scriptEngine->newQObject(object, QScriptEngine::AutoOwnership)); } /*! This function is intended for use inside QML only. In C++ just create a component object as usual. This function takes the URL of a QML file as its only argument. It returns a component object which can be used to create and load that QML file. Example QmlJS is below, remember that QML files that might be loaded over the network cannot be expected to be ready immediately. \code var component; var sprite; function finishCreation(){ if(component.isReady()){ sprite = component.createObject(); if(sprite == 0){ // Error Handling }else{ sprite.parent = page; sprite.x = 200; //... } }else if(component.isError()){ // Error Handling } } component = createComponent("Sprite.qml"); if(component.isReady()){ finishCreation(); }else{ component.statusChanged.connect(finishCreation); } \endcode If you are certain the files will be local, you could simplify to \code component = createComponent("Sprite.qml"); sprite = component.createObject(); if(sprite == 0){ // Error Handling print(component.errorsString()); }else{ sprite.parent = page; sprite.x = 200; //... } \endcode If you want to just create an arbitrary string of QML, instead of loading a qml file, consider the createQmlObject() function. */ QScriptValue QmlEnginePrivate::createComponent(QScriptContext *ctxt, QScriptEngine *engine) { QmlComponentJS* c; QmlEnginePrivate *activeEnginePriv = static_cast(engine)->p; QmlEngine* activeEngine = activeEnginePriv->q_func(); QmlContext* context = activeEnginePriv->currentExpression->context(); if(ctxt->argumentCount() != 1) { c = new QmlComponentJS(activeEngine); }else{ QUrl url = QUrl(context->resolvedUrl(ctxt->argument(0).toString())); if(!url.isValid()) url = QUrl(ctxt->argument(0).toString()); c = new QmlComponentJS(activeEngine, url, activeEngine); } c->setContext(context); return engine->newQObject(c); } /*! Creates a new object from the specified string of QML. It requires a second argument, which is the id of an existing QML object to use as the new object's parent. If a third argument is provided, this is used as the filepath that the qml came from. Example (where targetItem is the id of an existing QML item): \code newObject = createQmlObject('Rectangle {color: "red"; width: 20; height: 20}', targetItem, "dynamicSnippet1"); \endcode This function is intended for use inside QML only. It is intended to behave similarly to eval, but for creating QML elements. Returns the created object, or null if there is an error. In the case of an error, details of the error are output using qWarning(). Note that this function returns immediately, and therefore may not work if the QML loads new components. If you are trying to load a new component, for example from a QML file, consider the createComponent() function instead. 'New components' refers to external QML files that have not yet been loaded, and so it is safe to use createQmlObject to load built-in components. */ QScriptValue QmlEnginePrivate::createQmlObject(QScriptContext *ctxt, QScriptEngine *engine) { QmlEnginePrivate *activeEnginePriv = static_cast(engine)->p; QmlEngine* activeEngine = activeEnginePriv->q_func(); if(ctxt->argumentCount() < 2) return engine->nullValue(); QString qml = ctxt->argument(0).toString(); QUrl url; if(ctxt->argumentCount() > 2) url = QUrl(ctxt->argument(2).toString()); QObject *parentArg = ctxt->argument(1).data().toQObject(); QmlContext *qmlCtxt = qmlContext(parentArg); url = qmlCtxt->resolvedUrl(url); QmlComponent component(activeEngine, qml.toUtf8(), url); if(component.isError()) { QList errors = component.errors(); qWarning() <<"QmlEngine::createQmlObject():"; foreach (const QmlError &error, errors) qWarning() << " " << error; return engine->nullValue(); } QObject *obj = component.create(qmlCtxt); if(component.isError()) { QList errors = component.errors(); qWarning() <<"QmlEngine::createQmlObject():"; foreach (const QmlError &error, errors) qWarning() << " " << error; return engine->nullValue(); } if(obj) { obj->setParent(parentArg); obj->setProperty("parent", QVariant::fromValue(parentArg)); return qmlScriptObject(obj, activeEngine); } return engine->nullValue(); } /*! This function is intended for use inside QML only. In C++ just create a QVector3D as usual. This function takes three numeric components and combines them into a QVector3D value that can be used with any property that takes a QVector3D argument. The following QML code: \code transform: Rotation { id: Rotation origin.x: Container.width / 2; axis: vector(0, 1, 1) } \endcode is equivalent to: \code transform: Rotation { id: Rotation origin.x: Container.width / 2; axis.x: 0; axis.y: 1; axis.z: 0 } \endcode */ QScriptValue QmlEnginePrivate::vector(QScriptContext *ctxt, QScriptEngine *engine) { if(ctxt->argumentCount() < 3) return engine->nullValue(); qsreal x = ctxt->argument(0).toNumber(); qsreal y = ctxt->argument(1).toNumber(); qsreal z = ctxt->argument(2).toNumber(); return engine->newVariant(qVariantFromValue(QVector3D(x, y, z))); } QScriptValue QmlEnginePrivate::rgba(QScriptContext *ctxt, QScriptEngine *engine) { int argCount = ctxt->argumentCount(); if(argCount < 3) return engine->nullValue(); qsreal r = ctxt->argument(0).toNumber(); qsreal g = ctxt->argument(1).toNumber(); qsreal b = ctxt->argument(2).toNumber(); qsreal a = (argCount == 4) ? ctxt->argument(3).toNumber() : 1; return qScriptValueFromValue(engine, qVariantFromValue(QColor::fromRgbF(r, g, b, a))); } QScriptValue QmlEnginePrivate::hsla(QScriptContext *ctxt, QScriptEngine *engine) { int argCount = ctxt->argumentCount(); if(argCount < 3) return engine->nullValue(); qsreal h = ctxt->argument(0).toNumber(); qsreal s = ctxt->argument(1).toNumber(); qsreal l = ctxt->argument(2).toNumber(); qsreal a = (argCount == 4) ? ctxt->argument(3).toNumber() : 1; return qScriptValueFromValue(engine, qVariantFromValue(QColor::fromHslF(h, s, l, a))); } QScriptValue QmlEnginePrivate::rect(QScriptContext *ctxt, QScriptEngine *engine) { if(ctxt->argumentCount() < 4) return engine->nullValue(); qsreal x = ctxt->argument(0).toNumber(); qsreal y = ctxt->argument(1).toNumber(); qsreal w = ctxt->argument(2).toNumber(); qsreal h = ctxt->argument(3).toNumber(); return qScriptValueFromValue(engine, qVariantFromValue(QRectF(x, y, w, h))); } QScriptValue QmlEnginePrivate::point(QScriptContext *ctxt, QScriptEngine *engine) { if(ctxt->argumentCount() < 2) return engine->nullValue(); qsreal x = ctxt->argument(0).toNumber(); qsreal y = ctxt->argument(1).toNumber(); return qScriptValueFromValue(engine, qVariantFromValue(QPointF(x, y))); } QScriptValue QmlEnginePrivate::size(QScriptContext *ctxt, QScriptEngine *engine) { if(ctxt->argumentCount() < 2) return engine->nullValue(); qsreal w = ctxt->argument(0).toNumber(); qsreal h = ctxt->argument(1).toNumber(); return qScriptValueFromValue(engine, qVariantFromValue(QSizeF(w, h))); } QmlScriptClass::QmlScriptClass(QmlEngine *bindengine) : QScriptClass(QmlEnginePrivate::getScriptEngine(bindengine)), engine(bindengine) { } QVariant QmlScriptClass::toVariant(QmlEngine *engine, const QScriptValue &val) { QmlEnginePrivate *ep = static_cast(QObjectPrivate::get(engine)); QScriptClass *sc = val.scriptClass(); if (!sc) { return val.toVariant(); } else if (sc == ep->contextClass) { return QVariant(); } else if (sc == ep->objectClass) { return QVariant::fromValue(val.data().toQObject()); } else if (sc == ep->valueTypeClass) { QmlValueTypeReference ref = qvariant_cast(val.data().toVariant()); if (!ref.object) return QVariant(); QMetaProperty p = ref.object->metaObject()->property(ref.property); return p.read(ref.object); } return QVariant(); } ///////////////////////////////////////////////////////////// /* The QmlContextScriptClass handles property access for a QmlContext via QtScript. */ QmlContextScriptClass::QmlContextScriptClass(QmlEngine *bindEngine) : QmlScriptClass(bindEngine) { } QmlContextScriptClass::~QmlContextScriptClass() { } QScriptClass::QueryFlags QmlContextScriptClass::queryProperty(const QScriptValue &object, const QScriptString &name, QueryFlags flags, uint *id) { Q_UNUSED(flags); QmlContext *bindContext = static_cast(object.data().toQObject()); QueryFlags rv = 0; QString propName = name.toString(); *id = InvalidId; if (bindContext->d_func()->propertyNames.contains(propName)) { rv |= HandlesReadAccess; *id = VariantPropertyId; } for (int ii = 0; !rv && ii < bindContext->d_func()->defaultObjects.count(); ++ii) { rv = QmlEnginePrivate::get(engine)->queryObject(propName, id, bindContext->d_func()->defaultObjects.at(ii)); if (rv) *id |= (ii << 24); } return rv; } QScriptValue QmlContextScriptClass::property(const QScriptValue &object, const QScriptString &name, uint id) { QmlContext *bindContext = static_cast(object.data().toQObject()); uint basicId = id & QmlScriptClass::ClassIdMask; QScriptEngine *scriptEngine = QmlEnginePrivate::getScriptEngine(engine); QmlEnginePrivate *ep = QmlEnginePrivate::get(engine); switch (basicId) { case VariantPropertyId: { QmlContextPrivate *contextPrivate = bindContext->d_func(); QString propName = name.toString(); int index = contextPrivate->propertyNames.value(propName); QScriptValue rv; if (index < contextPrivate->idValueCount) { rv = scriptEngine->newObject(ep->objectClass, scriptEngine->newVariant(QVariant::fromValue(contextPrivate->idValues[index].data()))); } else { QVariant value = contextPrivate->propertyValues.at(index); if (QmlMetaType::isObject(value.userType())) { rv = scriptEngine->newObject(ep->objectClass, scriptEngine->newVariant(value)); } else { rv = scriptEngine->newVariant(value); } } ep->capturedProperties << QmlEnginePrivate::CapturedProperty(bindContext, -1, index + bindContext->d_func()->notifyIndex); return rv; } default: { int objId = (id & ClassIdSelectorMask) >> 24; QObject *obj = bindContext->d_func()->defaultObjects.at(objId); QScriptValue rv = ep->propertyObject(name, obj, id & ~QmlScriptClass::ClassIdSelectorMask); if (rv.isValid()) { return rv; } break; } } return QScriptValue(); } void QmlContextScriptClass::setProperty(QScriptValue &object, const QScriptString &name, uint id, const QScriptValue &value) { Q_UNUSED(name); QmlContext *bindContext = static_cast(object.data().toQObject()); int objIdx = (id & QmlScriptClass::ClassIdSelectorMask) >> 24; QObject *obj = bindContext->d_func()->defaultObjects.at(objIdx); QmlMetaProperty prop; prop.restore(id, obj); QVariant v = QmlScriptClass::toVariant(engine, value); prop.write(v); } ///////////////////////////////////////////////////////////// QmlValueTypeScriptClass::QmlValueTypeScriptClass(QmlEngine *bindEngine) : QmlScriptClass(bindEngine) { } QmlValueTypeScriptClass::~QmlValueTypeScriptClass() { } QmlValueTypeScriptClass::QueryFlags QmlValueTypeScriptClass::queryProperty(const QScriptValue &object, const QScriptString &name, QueryFlags flags, uint *id) { Q_UNUSED(flags); QmlValueTypeReference ref = qvariant_cast(object.data().toVariant()); if (!ref.object) return 0; QByteArray propName = name.toString().toUtf8(); int idx = ref.type->metaObject()->indexOfProperty(propName.constData()); if (idx == -1) return 0; *id = idx; QMetaProperty prop = ref.object->metaObject()->property(idx); QmlValueTypeScriptClass::QueryFlags rv = QmlValueTypeScriptClass::HandlesReadAccess; if (prop.isWritable()) rv |= QmlValueTypeScriptClass::HandlesWriteAccess; return rv; } QScriptValue QmlValueTypeScriptClass::property(const QScriptValue &object, const QScriptString &name, uint id) { Q_UNUSED(name); QmlValueTypeReference ref = qvariant_cast(object.data().toVariant()); if (!ref.object) return QScriptValue(); ref.type->read(ref.object, ref.property); QMetaProperty p = ref.type->metaObject()->property(id); QVariant rv = p.read(ref.type); return static_cast(QObjectPrivate::get(engine))->scriptEngine.newVariant(rv); } void QmlValueTypeScriptClass::setProperty(QScriptValue &object, const QScriptString &name, uint id, const QScriptValue &value) { Q_UNUSED(name); QmlValueTypeReference ref = qvariant_cast(object.data().toVariant()); if (!ref.object) return; QVariant v = QmlScriptClass::toVariant(engine, value); ref.type->read(ref.object, ref.property); QMetaProperty p = ref.type->metaObject()->property(id); p.write(ref.type, v); ref.type->write(ref.object, ref.property); } ///////////////////////////////////////////////////////////// /* The QmlObjectScriptClass handles property access for QObjects via QtScript. It is also used to provide a more useful API in QtScript for QML. */ QScriptValue QmlObjectToString(QScriptContext *context, QScriptEngine *engine) { QObject* obj = context->thisObject().data().toQObject(); QString ret = QLatin1String("Qml Object, "); if(obj){ //###Should this be designer or developer details? Dev for now. //TODO: Can we print the id too? ret += QLatin1String("\""); ret += obj->objectName(); ret += QLatin1String("\" "); ret += obj->metaObject()->className(); ret += QLatin1String("(0x"); ret += QString::number((quintptr)obj,16); ret += QLatin1String(")"); }else{ ret += QLatin1String("null"); } return engine->newVariant(ret); } QScriptValue QmlObjectDestroy(QScriptContext *context, QScriptEngine *engine) { QObject* obj = context->thisObject().data().toQObject(); if(obj){ int delay = 0; if(context->argumentCount() > 0) delay = context->argument(0).toInt32(); QTimer::singleShot(delay, obj, SLOT(deleteLater())); //### Should this be delayed as well? context->thisObject().setData(QScriptValue(engine, 0)); } return engine->nullValue(); } QmlObjectScriptClass::QmlObjectScriptClass(QmlEngine *bindEngine) : QmlScriptClass(bindEngine) { engine = bindEngine; QScriptEngine *scriptEngine = QmlEnginePrivate::getScriptEngine(bindEngine); prototypeObject = scriptEngine->newObject(); prototypeObject.setProperty(QLatin1String("toStr"),//TODO: Why won't toString work? scriptEngine->newFunction(QmlObjectToString)); prototypeObject.setProperty(QLatin1String("destroy"), scriptEngine->newFunction(QmlObjectDestroy)); } QmlObjectScriptClass::~QmlObjectScriptClass() { } QScriptValue QmlObjectScriptClass::prototype() const { return prototypeObject; } QScriptClass::QueryFlags QmlObjectScriptClass::queryProperty(const QScriptValue &object, const QScriptString &name, QueryFlags flags, uint *id) { Q_UNUSED(flags); QObject *obj = object.data().toQObject(); QueryFlags rv = 0; QString propName = name.toString(); if (obj) rv = QmlEnginePrivate::get(engine)->queryObject(propName, id, obj); return rv; } QScriptValue QmlObjectScriptClass::property(const QScriptValue &object, const QScriptString &name, uint id) { QObject *obj = object.data().toQObject(); QScriptValue rv = QmlEnginePrivate::get(engine)->propertyObject(name, obj, id); if (rv.isValid()) return rv; return QScriptValue(); } void QmlObjectScriptClass::setProperty(QScriptValue &object, const QScriptString &name, uint id, const QScriptValue &value) { Q_UNUSED(name); QObject *obj = object.data().toQObject(); QmlMetaProperty prop; prop.restore(id, obj); QVariant v = QmlScriptClass::toVariant(engine, value); prop.write(v); } struct QmlEnginePrivate::ImportedNamespace { QStringList urls; QList majversions; QList minversions; QList isLibrary; QList isBuiltin; bool find(const QByteArray& type, int *vmajor, int *vminor, QmlType** type_return, QUrl* url_return) const { for (int i=0; i=0 ? line.indexOf(QLatin1Char(' '),space1+1) : -1; QString mapversions = line.mid(space1+1,space2<0?line.length()-space1-2:space2-space1-1); int dot = mapversions.indexOf(QLatin1Char('.')); int dash = mapversions.indexOf(QLatin1Char('-')); int mapvmaj = mapversions.left(dot).toInt(); if (mapvmaj==vmaj) { int mapvmin_from = (dash <= 0 ? mapversions.mid(dot+1) : mapversions.mid(dot+1,dash-dot-1)).toInt(); int mapvmin_to = dash <= 0 ? mapvmin_from : mapversions.mid(dash+1).toInt(); if (vmin >= mapvmin_from && vmin <= mapvmin_to) { QStringRef mapfile = space2<0 ? QStringRef() : line.midRef(space2+1,line.length()-space2-2); if (url_return) *url_return = url.resolved(mapfile.toString()); return true; } } } } while (!qmldir.atEnd()); } } else { // XXX search non-files too! (eg. zip files, see QT-524) QFileInfo f(url.toLocalFile()); if (f.exists()) { if (url_return) *url_return = url; return true; } } } } return false; } }; class QmlImportsPrivate { public: QmlImportsPrivate() : ref(1) { } ~QmlImportsPrivate() { foreach (QmlEnginePrivate::ImportedNamespace* s, set.values()) delete s; } bool add(const QUrl& base, const QString& uri, const QString& prefix, int vmaj, int vmin, QmlScriptParser::Import::Type importType, const QStringList& importPath) { QmlEnginePrivate::ImportedNamespace *s; if (prefix.isEmpty()) { s = &unqualifiedset; } else { s = set.value(prefix); if (!s) set.insert(prefix,(s=new QmlEnginePrivate::ImportedNamespace)); } QString url = uri; bool isbuiltin = false; if (importType == QmlScriptParser::Import::Library) { url.replace(QLatin1Char('.'),QLatin1Char('/')); bool found = false; foreach (QString p, importPath) { QString dir = p+QLatin1Char('/')+url; if (QFile::exists(dir+QLatin1String("/qmldir"))) { url = QLatin1String("file://")+dir; found = true; break; } } if (!found) { // XXX assume it is a built-in type qualifier isbuiltin = true; } } else { url = base.resolved(QUrl(url)).toString(); } s->urls.prepend(url); s->majversions.prepend(vmaj); s->minversions.prepend(vmin); s->isLibrary.prepend(importType == QmlScriptParser::Import::Library); s->isBuiltin.prepend(isbuiltin); return true; } bool find(const QByteArray& type, int *vmajor, int *vminor, QmlType** type_return, QUrl* url_return) { QmlEnginePrivate::ImportedNamespace *s = 0; int slash = type.indexOf('/'); if (slash >= 0) { while (!s) { s = set.value(QString::fromLatin1(type.left(slash))); int nslash = type.indexOf('/',slash+1); if (nslash > 0) slash = nslash; else break; } } else { s = &unqualifiedset; } QByteArray unqualifiedtype = slash < 0 ? type : type.mid(slash+1); // common-case opt (QString::mid works fine, but slower) if (s) { if (s->find(unqualifiedtype,vmajor,vminor,type_return,url_return)) return true; } if (url_return) { *url_return = base.resolved(QUrl(QLatin1String(type + ".qml"))); return true; } else { return false; } } QmlEnginePrivate::ImportedNamespace *findNamespace(const QString& type) { return set.value(type); } QUrl base; int ref; private: QmlEnginePrivate::ImportedNamespace unqualifiedset; QHash set; }; QmlEnginePrivate::Imports::Imports(const Imports ©) : d(copy.d) { ++d->ref; } QmlEnginePrivate::Imports &QmlEnginePrivate::Imports::operator =(const Imports ©) { ++copy.d->ref; if (--d->ref == 0) delete d; d = copy.d; return *this; } QmlEnginePrivate::Imports::Imports() : d(new QmlImportsPrivate) { } QmlEnginePrivate::Imports::~Imports() { if (--d->ref == 0) delete d; } /*! Sets the base URL to be used for all relative file imports added. */ void QmlEnginePrivate::Imports::setBaseUrl(const QUrl& url) { d->base = url; } /*! Returns the base URL to be used for all relative file imports added. */ QUrl QmlEnginePrivate::Imports::baseUrl() const { return d->base; } /*! Adds \a path as a directory where installed QML components are defined in a URL-based directory structure. For example, if you add \c /opt/MyApp/lib/qml and then load QML that imports \c com.mycompany.Feature, then QmlEngine will look in \c /opt/MyApp/lib/qml/com/mycompany/Feature/ for the components provided by that module (and in the case of versioned imports, for the \c qmldir file definiting the type version mapping. */ void QmlEngine::addImportPath(const QString& path) { if (qmlImportTrace()) qDebug() << "QmlEngine::addImportPath" << path; Q_D(QmlEngine); d->fileImportPath.prepend(path); } /*! \internal Adds information to \a imports such that subsequent calls to resolveType() will resolve types qualified by \a prefix by considering types found at the given \a uri. The uri is either a directory (if importType is FileImport), or a URI resolved using paths added via addImportPath() (if importType is LibraryImport). The \a prefix may be empty, in which case the import location is considered for unqualified types. The base URL must already have been set with Import::setBaseUrl(). */ bool QmlEnginePrivate::addToImport(Imports* imports, const QString& uri, const QString& prefix, int vmaj, int vmin, QmlScriptParser::Import::Type importType) const { bool ok = imports->d->add(imports->d->base,uri,prefix,vmaj,vmin,importType,fileImportPath); if (qmlImportTrace()) qDebug() << "QmlEngine::addToImport(" << imports << uri << prefix << vmaj << "." << vmin << (importType==QmlScriptParser::Import::Library? "Library" : "File") << ": " << ok; return ok; } /*! \internal Using the given \a imports, the given (namespace qualified) \a type is resolved to either an ImportedNamespace stored at \a ns_return, a QmlType stored at \a type_return, or a component located at \a url_return. If any return pointer is 0, the corresponding search is not done. \sa addToImport() */ bool QmlEnginePrivate::resolveType(const Imports& imports, const QByteArray& type, QmlType** type_return, QUrl* url_return, int *vmaj, int *vmin, ImportedNamespace** ns_return) const { ImportedNamespace* ns = imports.d->findNamespace(QLatin1String(type)); if (ns) { if (qmlImportTrace()) qDebug() << "QmlEngine::resolveType" << type << "is namespace for" << ns->urls; if (ns_return) *ns_return = ns; return true; } if (type_return || url_return) { if (imports.d->find(type,vmaj,vmin,type_return,url_return)) { if (qmlImportTrace()) { if (type_return && *type_return) qDebug() << "QmlEngine::resolveType" << type << "=" << (*type_return)->typeName(); if (url_return) qDebug() << "QmlEngine::resolveType" << type << "=" << *url_return; } return true; } if (qmlImportTrace()) qDebug() << "QmlEngine::resolveType" << type << "not found"; } return false; } /*! \internal Searching \e only in the namespace \a ns (previously returned in a call to resolveType(), \a type is found and returned to either a QmlType stored at \a type_return, or a component located at \a url_return. If either return pointer is 0, the corresponding search is not done. */ void QmlEnginePrivate::resolveTypeInNamespace(ImportedNamespace* ns, const QByteArray& type, QmlType** type_return, QUrl* url_return, int *vmaj, int *vmin ) const { ns->find(type,vmaj,vmin,type_return,url_return); } QT_END_NAMESPACE