| 
									
										
										
										
											2010-07-09 15:47:07 +02:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-01-11 16:28:15 +01:00
										 |  |  | ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2010-07-09 15:47:07 +02:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** Contact: Nokia Corporation (info@qt.nokia.com) | 
					
						
							| 
									
										
										
										
											2010-07-09 15:47:07 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** 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.
 | 
					
						
							| 
									
										
										
										
											2010-07-09 15:47:07 +02:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-12-17 16:01:08 +01:00
										 |  |  | ** In addition, as a special exception, Nokia gives you certain additional | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** rights. These rights are described in the Nokia Qt LGPL Exception | 
					
						
							| 
									
										
										
										
											2010-12-17 16:01:08 +01:00
										 |  |  | ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** Other Usage | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Alternatively, this file may be used in accordance with the terms and | 
					
						
							|  |  |  | ** conditions contained in a signed written agreement between you and Nokia. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-12-17 16:01:08 +01:00
										 |  |  | ** If you have questions regarding the use of this file, please contact | 
					
						
							| 
									
										
										
										
											2011-05-06 15:05:37 +02:00
										 |  |  | ** Nokia at info@qt.nokia.com. | 
					
						
							| 
									
										
										
										
											2010-07-09 15:47:07 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  | #include "qmljsscopebuilder.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qmljsbind.h"
 | 
					
						
							| 
									
										
										
										
											2011-07-01 13:51:53 +02:00
										 |  |  | #include "qmljscontext.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  | #include "qmljsevaluate.h"
 | 
					
						
							|  |  |  | #include "parser/qmljsast_p.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using namespace QmlJS; | 
					
						
							|  |  |  | using namespace QmlJS::Interpreter; | 
					
						
							|  |  |  | using namespace QmlJS::AST; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  | ScopeBuilder::ScopeBuilder(Context *context, Document::Ptr doc) | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |     : _doc(doc) | 
					
						
							|  |  |  |     , _context(context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ScopeBuilder::~ScopeBuilder() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ScopeBuilder::push(AST::Node *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _nodes += node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // QML scope object
 | 
					
						
							|  |  |  |     Node *qmlObject = cast<UiObjectDefinition *>(node); | 
					
						
							|  |  |  |     if (! qmlObject) | 
					
						
							|  |  |  |         qmlObject = cast<UiObjectBinding *>(node); | 
					
						
							|  |  |  |     if (qmlObject) | 
					
						
							|  |  |  |         setQmlScopeObject(qmlObject); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-06 14:13:50 +02:00
										 |  |  |     // JS scopes
 | 
					
						
							|  |  |  |     switch (node->kind) { | 
					
						
							|  |  |  |     case Node::Kind_UiScriptBinding: | 
					
						
							|  |  |  |     case Node::Kind_FunctionDeclaration: | 
					
						
							|  |  |  |     case Node::Kind_FunctionExpression: | 
					
						
							|  |  |  |     case Node::Kind_UiPublicMember: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ObjectValue *scope = _doc->bind()->findAttachedJSScope(node); | 
					
						
							|  |  |  |         if (scope) | 
					
						
							|  |  |  |             _context->scopeChain().jsScopes += scope; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _context->scopeChain().update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-22 15:59:21 +02:00
										 |  |  | void ScopeBuilder::push(const QList<AST::Node *> &nodes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     foreach (Node *node, nodes) | 
					
						
							|  |  |  |         push(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  | void ScopeBuilder::pop() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Node *toRemove = _nodes.last(); | 
					
						
							|  |  |  |     _nodes.removeLast(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // JS scopes
 | 
					
						
							| 
									
										
										
										
											2011-06-06 14:13:50 +02:00
										 |  |  |     switch (toRemove->kind) { | 
					
						
							|  |  |  |     case Node::Kind_UiScriptBinding: | 
					
						
							|  |  |  |     case Node::Kind_FunctionDeclaration: | 
					
						
							|  |  |  |     case Node::Kind_FunctionExpression: | 
					
						
							|  |  |  |     case Node::Kind_UiPublicMember: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ObjectValue *scope = _doc->bind()->findAttachedJSScope(toRemove); | 
					
						
							|  |  |  |         if (scope) | 
					
						
							| 
									
										
										
										
											2010-11-24 14:42:10 +01:00
										 |  |  |             _context->scopeChain().jsScopes.removeLast(); | 
					
						
							| 
									
										
										
										
											2011-06-06 14:13:50 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2010-11-24 14:42:10 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // QML scope object
 | 
					
						
							|  |  |  |     if (! _nodes.isEmpty() | 
					
						
							|  |  |  |         && (cast<UiObjectDefinition *>(toRemove) || cast<UiObjectBinding *>(toRemove))) | 
					
						
							|  |  |  |         setQmlScopeObject(_nodes.last()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _context->scopeChain().update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-06 13:31:30 +02:00
										 |  |  | void ScopeBuilder::initializeRootScope() | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     ScopeChain &scopeChain = _context->scopeChain(); | 
					
						
							| 
									
										
										
										
											2010-11-23 12:57:48 +01:00
										 |  |  |     if (scopeChain.qmlComponentScope | 
					
						
							|  |  |  |             && scopeChain.qmlComponentScope->document == _doc) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  |     scopeChain = ScopeChain(); // reset
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-01 12:11:02 +02:00
										 |  |  |     Interpreter::ValueOwner *valueOwner = _context->valueOwner(); | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // ### TODO: This object ought to contain the global namespace additions by QML.
 | 
					
						
							| 
									
										
										
										
											2011-07-01 12:11:02 +02:00
										 |  |  |     scopeChain.globalScope = valueOwner->globalObject(); | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (! _doc) { | 
					
						
							|  |  |  |         scopeChain.update(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Bind *bind = _doc->bind(); | 
					
						
							|  |  |  |     QHash<Document *, ScopeChain::QmlComponentChain *> componentScopes; | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  |     const Snapshot &snapshot = _context->snapshot(); | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ScopeChain::QmlComponentChain *chain = new ScopeChain::QmlComponentChain; | 
					
						
							|  |  |  |     scopeChain.qmlComponentScope = QSharedPointer<const ScopeChain::QmlComponentChain>(chain); | 
					
						
							|  |  |  |     if (_doc->qmlProgram()) { | 
					
						
							|  |  |  |         componentScopes.insert(_doc.data(), chain); | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  |         makeComponentChain(_doc, snapshot, chain, &componentScopes); | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-27 10:43:06 +02:00
										 |  |  |         if (const Imports *imports = _context->imports(_doc.data())) { | 
					
						
							|  |  |  |             scopeChain.qmlTypes = imports->typeScope(); | 
					
						
							|  |  |  |             scopeChain.jsImports = imports->jsImportScope(); | 
					
						
							| 
									
										
										
										
											2011-02-10 17:03:52 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // add scope chains for all components that import this file
 | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  |         foreach (Document::Ptr otherDoc, snapshot) { | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  |             foreach (const ImportInfo &import, otherDoc->bind()->imports()) { | 
					
						
							|  |  |  |                 if (import.type() == ImportInfo::FileImport && _doc->fileName() == import.name()) { | 
					
						
							|  |  |  |                     ScopeChain::QmlComponentChain *component = new ScopeChain::QmlComponentChain; | 
					
						
							|  |  |  |                     componentScopes.insert(otherDoc.data(), component); | 
					
						
							|  |  |  |                     chain->instantiatingComponents += component; | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  |                     makeComponentChain(otherDoc, snapshot, component, &componentScopes); | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // ### TODO: Which type environment do scripts see?
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (bind->rootObjectValue()) | 
					
						
							|  |  |  |             scopeChain.jsScopes += bind->rootObjectValue(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     scopeChain.update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ScopeBuilder::makeComponentChain( | 
					
						
							|  |  |  |         Document::Ptr doc, | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  |         const Snapshot &snapshot, | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  |         ScopeChain::QmlComponentChain *target, | 
					
						
							|  |  |  |         QHash<Document *, ScopeChain::QmlComponentChain *> *components) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!doc->qmlProgram()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Bind *bind = doc->bind(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // add scopes for all components instantiating this one
 | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  |     foreach (Document::Ptr otherDoc, snapshot) { | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  |         if (otherDoc == doc) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         if (otherDoc->bind()->usesQmlPrototype(bind->rootObjectValue(), _context)) { | 
					
						
							|  |  |  |             if (components->contains(otherDoc.data())) { | 
					
						
							|  |  |  | //                target->instantiatingComponents += components->value(otherDoc.data());
 | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 ScopeChain::QmlComponentChain *component = new ScopeChain::QmlComponentChain; | 
					
						
							|  |  |  |                 components->insert(otherDoc.data(), component); | 
					
						
							|  |  |  |                 target->instantiatingComponents += component; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-04 11:12:45 +02:00
										 |  |  |                 makeComponentChain(otherDoc, snapshot, component, components); | 
					
						
							| 
									
										
										
										
											2010-09-24 14:05:34 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // build this component scope
 | 
					
						
							|  |  |  |     target->document = doc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  | void ScopeBuilder::setQmlScopeObject(Node *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ScopeChain &scopeChain = _context->scopeChain(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-12 13:58:23 +02:00
										 |  |  |     if (_doc->bind()->isGroupedPropertyBinding(node)) { | 
					
						
							|  |  |  |         UiObjectDefinition *definition = cast<UiObjectDefinition *>(node); | 
					
						
							|  |  |  |         if (!definition) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         const Value *v = scopeObjectLookup(definition->qualifiedTypeNameId); | 
					
						
							|  |  |  |         if (!v) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         const ObjectValue *object = v->asObjectValue(); | 
					
						
							|  |  |  |         if (!object) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         scopeChain.qmlScopeObjects.clear(); | 
					
						
							|  |  |  |         scopeChain.qmlScopeObjects += object; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const ObjectValue *scopeObject = _doc->bind()->findQmlObject(node); | 
					
						
							|  |  |  |     if (scopeObject) { | 
					
						
							| 
									
										
										
										
											2010-05-12 13:58:23 +02:00
										 |  |  |         scopeChain.qmlScopeObjects.clear(); | 
					
						
							| 
									
										
										
										
											2010-02-23 12:34:52 +01:00
										 |  |  |         scopeChain.qmlScopeObjects += scopeObject; | 
					
						
							| 
									
										
										
										
											2010-02-25 12:56:59 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         return; // Probably syntax errors, where we're working with a "recovered" AST.
 | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-03 11:36:42 +01:00
										 |  |  |     // check if the object has a Qt.ListElement or Qt.Connections ancestor
 | 
					
						
							|  |  |  |     // ### allow only signal bindings for Connections
 | 
					
						
							| 
									
										
										
										
											2010-11-12 14:53:00 +01:00
										 |  |  |     PrototypeIterator iter(scopeObject, _context); | 
					
						
							|  |  |  |     iter.next(); | 
					
						
							|  |  |  |     while (iter.hasNext()) { | 
					
						
							|  |  |  |         const ObjectValue *prototype = iter.next(); | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |         if (const QmlObjectValue *qmlMetaObject = dynamic_cast<const QmlObjectValue *>(prototype)) { | 
					
						
							| 
									
										
										
										
											2010-03-03 11:36:42 +01:00
										 |  |  |             if ((qmlMetaObject->className() == QLatin1String("ListElement") | 
					
						
							|  |  |  |                     || qmlMetaObject->className() == QLatin1String("Connections") | 
					
						
							| 
									
										
										
										
											2010-10-20 11:05:21 +02:00
										 |  |  |                     ) && (qmlMetaObject->packageName() == QLatin1String("Qt") | 
					
						
							|  |  |  |                           || qmlMetaObject->packageName() == QLatin1String("QtQuick"))) { | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |                 scopeChain.qmlScopeObjects.clear(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // check if the object has a Qt.PropertyChanges ancestor
 | 
					
						
							| 
									
										
										
										
											2010-11-12 14:53:00 +01:00
										 |  |  |     const ObjectValue *prototype = scopeObject->prototype(_context); | 
					
						
							| 
									
										
										
										
											2010-05-19 11:15:57 +02:00
										 |  |  |     prototype = isPropertyChangesObject(_context, prototype); | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |     // find the target script binding
 | 
					
						
							|  |  |  |     if (prototype) { | 
					
						
							|  |  |  |         UiObjectInitializer *initializer = 0; | 
					
						
							|  |  |  |         if (UiObjectDefinition *definition = cast<UiObjectDefinition *>(node)) | 
					
						
							|  |  |  |             initializer = definition->initializer; | 
					
						
							|  |  |  |         if (UiObjectBinding *binding = cast<UiObjectBinding *>(node)) | 
					
						
							|  |  |  |             initializer = binding->initializer; | 
					
						
							|  |  |  |         if (initializer) { | 
					
						
							|  |  |  |             for (UiObjectMemberList *m = initializer->members; m; m = m->next) { | 
					
						
							|  |  |  |                 if (UiScriptBinding *scriptBinding = cast<UiScriptBinding *>(m->member)) { | 
					
						
							| 
									
										
										
										
											2010-10-14 15:14:35 +02:00
										 |  |  |                     if (scriptBinding->qualifiedId && scriptBinding->qualifiedId->name | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |                             && scriptBinding->qualifiedId->name->asString() == QLatin1String("target") | 
					
						
							|  |  |  |                             && ! scriptBinding->qualifiedId->next) { | 
					
						
							| 
									
										
										
										
											2010-11-23 14:15:50 +01:00
										 |  |  |                         Evaluate evaluator(_context); | 
					
						
							|  |  |  |                         const Value *targetValue = evaluator(scriptBinding->statement); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (const ObjectValue *target = value_cast<const ObjectValue *>(targetValue)) { | 
					
						
							|  |  |  |                             scopeChain.qmlScopeObjects.prepend(target); | 
					
						
							|  |  |  |                         } else { | 
					
						
							|  |  |  |                             scopeChain.qmlScopeObjects.clear(); | 
					
						
							| 
									
										
										
										
											2010-02-19 15:55:11 +01:00
										 |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-12 13:58:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | const Value *ScopeBuilder::scopeObjectLookup(AST::UiQualifiedId *id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // do a name lookup on the scope objects
 | 
					
						
							|  |  |  |     const Value *result = 0; | 
					
						
							|  |  |  |     foreach (const ObjectValue *scopeObject, _context->scopeChain().qmlScopeObjects) { | 
					
						
							|  |  |  |         const ObjectValue *object = scopeObject; | 
					
						
							|  |  |  |         for (UiQualifiedId *it = id; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2010-10-14 15:14:35 +02:00
										 |  |  |             if (!it->name) | 
					
						
							|  |  |  |                 return 0; | 
					
						
							| 
									
										
										
										
											2011-05-24 11:50:10 +02:00
										 |  |  |             result = object->lookupMember(it->name->asString(), _context); | 
					
						
							| 
									
										
										
										
											2010-05-12 13:58:23 +02:00
										 |  |  |             if (!result) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             if (it->next) { | 
					
						
							|  |  |  |                 object = result->asObjectValue(); | 
					
						
							|  |  |  |                 if (!object) { | 
					
						
							|  |  |  |                     result = 0; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (result) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-05-19 11:15:57 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-25 11:51:34 +02:00
										 |  |  | const ObjectValue *ScopeBuilder::isPropertyChangesObject(const Context *context, | 
					
						
							| 
									
										
										
										
											2010-05-19 11:15:57 +02:00
										 |  |  |                                                    const ObjectValue *object) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-12 14:53:00 +01:00
										 |  |  |     PrototypeIterator iter(object, context); | 
					
						
							|  |  |  |     while (iter.hasNext()) { | 
					
						
							|  |  |  |         const ObjectValue *prototype = iter.next(); | 
					
						
							| 
									
										
										
										
											2010-05-19 11:15:57 +02:00
										 |  |  |         if (const QmlObjectValue *qmlMetaObject = dynamic_cast<const QmlObjectValue *>(prototype)) { | 
					
						
							|  |  |  |             if (qmlMetaObject->className() == QLatin1String("PropertyChanges") | 
					
						
							| 
									
										
										
										
											2010-10-01 13:52:05 +02:00
										 |  |  |                     && (qmlMetaObject->packageName() == QLatin1String("Qt") | 
					
						
							|  |  |  |                         || qmlMetaObject->packageName() == QLatin1String("QtQuick"))) | 
					
						
							| 
									
										
										
										
											2010-05-19 11:15:57 +02:00
										 |  |  |                 return prototype; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } |