forked from qt-creator/qt-creator
Introduced QmlJS::LookupContext.
This commit is contained in:
@@ -20,6 +20,7 @@ HEADERS += \
|
|||||||
$$PWD/qmljslink.h \
|
$$PWD/qmljslink.h \
|
||||||
$$PWD/qmljscheck.h \
|
$$PWD/qmljscheck.h \
|
||||||
$$PWD/qmljsscopebuilder.h \
|
$$PWD/qmljsscopebuilder.h \
|
||||||
|
$$PWD/qmljslookupcontext.h \
|
||||||
$$PWD/qmljslineinfo.h \
|
$$PWD/qmljslineinfo.h \
|
||||||
$$PWD/qmljscompletioncontextfinder.h \
|
$$PWD/qmljscompletioncontextfinder.h \
|
||||||
$$PWD/qmljscomponentversion.h \
|
$$PWD/qmljscomponentversion.h \
|
||||||
@@ -38,6 +39,7 @@ SOURCES += \
|
|||||||
$$PWD/qmljslink.cpp \
|
$$PWD/qmljslink.cpp \
|
||||||
$$PWD/qmljscheck.cpp \
|
$$PWD/qmljscheck.cpp \
|
||||||
$$PWD/qmljsscopebuilder.cpp \
|
$$PWD/qmljsscopebuilder.cpp \
|
||||||
|
$$PWD/qmljslookupcontext.cpp \
|
||||||
$$PWD/qmljslineinfo.cpp \
|
$$PWD/qmljslineinfo.cpp \
|
||||||
$$PWD/qmljscompletioncontextfinder.cpp \
|
$$PWD/qmljscompletioncontextfinder.cpp \
|
||||||
$$PWD/qmljscomponentversion.cpp \
|
$$PWD/qmljscomponentversion.cpp \
|
||||||
|
|||||||
88
src/libs/qmljs/qmljslookupcontext.cpp
Normal file
88
src/libs/qmljs/qmljslookupcontext.cpp
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
**
|
||||||
|
** This file is part of Qt Creator
|
||||||
|
**
|
||||||
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
||||||
|
**
|
||||||
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
||||||
|
**
|
||||||
|
** Commercial Usage
|
||||||
|
**
|
||||||
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
||||||
|
** accordance with the Qt Commercial License Agreement provided with the
|
||||||
|
** Software or, alternatively, in accordance with the terms contained in
|
||||||
|
** a written agreement between you and Nokia.
|
||||||
|
**
|
||||||
|
** 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.
|
||||||
|
**
|
||||||
|
** If you are unsure which license is appropriate for your use, please
|
||||||
|
** contact the sales department at http://qt.nokia.com/contact.
|
||||||
|
**
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#include "qmljslookupcontext.h"
|
||||||
|
#include "qmljsinterpreter.h"
|
||||||
|
#include "qmljslink.h"
|
||||||
|
#include "qmljsscopebuilder.h"
|
||||||
|
#include "qmljsmodelmanagerinterface.h"
|
||||||
|
#include "qmljsevaluate.h"
|
||||||
|
|
||||||
|
using namespace QmlJS;
|
||||||
|
|
||||||
|
class QmlJS::LookupContextData
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LookupContextData(Document::Ptr doc, const Snapshot &snapshot, const QList<AST::Node *> &path)
|
||||||
|
: context(&interp),
|
||||||
|
doc(doc),
|
||||||
|
snapshot(snapshot),
|
||||||
|
link(&context, doc, snapshot, ModelManagerInterface::instance()->importPaths())
|
||||||
|
{
|
||||||
|
ScopeBuilder scopeBuilder(doc, &context);
|
||||||
|
scopeBuilder.push(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
Interpreter::Engine interp;
|
||||||
|
Interpreter::Context context;
|
||||||
|
Document::Ptr doc;
|
||||||
|
Snapshot snapshot;
|
||||||
|
Link link;
|
||||||
|
};
|
||||||
|
|
||||||
|
LookupContext::LookupContext(Document::Ptr doc, const Snapshot &snapshot, const QList<AST::Node *> &path)
|
||||||
|
: d(new LookupContextData(doc, snapshot, path))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
LookupContext::~LookupContext()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
LookupContext::Ptr LookupContext::create(Document::Ptr doc, const Snapshot &snapshot, const QList<AST::Node *> &path)
|
||||||
|
{
|
||||||
|
Ptr ptr(new LookupContext(doc, snapshot, path));
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const Interpreter::Value *LookupContext::evaluate(AST::Node *node) const
|
||||||
|
{
|
||||||
|
Evaluate check(&d->context);
|
||||||
|
return check(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
Interpreter::Engine *LookupContext::engine() const
|
||||||
|
{
|
||||||
|
return &d->interp;
|
||||||
|
}
|
||||||
|
|
||||||
|
Interpreter::Context *LookupContext::context() const
|
||||||
|
{
|
||||||
|
return &d->context;
|
||||||
|
}
|
||||||
72
src/libs/qmljs/qmljslookupcontext.h
Normal file
72
src/libs/qmljs/qmljslookupcontext.h
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
/**************************************************************************
|
||||||
|
**
|
||||||
|
** This file is part of Qt Creator
|
||||||
|
**
|
||||||
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
||||||
|
**
|
||||||
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
||||||
|
**
|
||||||
|
** Commercial Usage
|
||||||
|
**
|
||||||
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
||||||
|
** accordance with the Qt Commercial License Agreement provided with the
|
||||||
|
** Software or, alternatively, in accordance with the terms contained in
|
||||||
|
** a written agreement between you and Nokia.
|
||||||
|
**
|
||||||
|
** 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.
|
||||||
|
**
|
||||||
|
** If you are unsure which license is appropriate for your use, please
|
||||||
|
** contact the sales department at http://qt.nokia.com/contact.
|
||||||
|
**
|
||||||
|
**************************************************************************/
|
||||||
|
|
||||||
|
#ifndef QMLJSLOOKUPCONTEXT_H
|
||||||
|
#define QMLJSLOOKUPCONTEXT_H
|
||||||
|
|
||||||
|
#include "qmljsdocument.h"
|
||||||
|
#include "parser/qmljsastfwd_p.h"
|
||||||
|
|
||||||
|
#include <QtCore/QSharedPointer>
|
||||||
|
#include <QtCore/QScopedPointer>
|
||||||
|
|
||||||
|
namespace QmlJS {
|
||||||
|
|
||||||
|
class LookupContextData;
|
||||||
|
|
||||||
|
namespace Interpreter {
|
||||||
|
class Value;
|
||||||
|
class Engine;
|
||||||
|
class Context;
|
||||||
|
}
|
||||||
|
|
||||||
|
class QMLJS_EXPORT LookupContext
|
||||||
|
{
|
||||||
|
LookupContext(const Document::Ptr doc, const Snapshot &snapshot, const QList<AST::Node *> &path);
|
||||||
|
|
||||||
|
public:
|
||||||
|
~LookupContext();
|
||||||
|
|
||||||
|
typedef QSharedPointer<LookupContext> Ptr;
|
||||||
|
|
||||||
|
static Ptr create(const Document::Ptr doc, const Snapshot &snapshot,
|
||||||
|
const QList<AST::Node *> &path);
|
||||||
|
|
||||||
|
const Interpreter::Value *evaluate(AST::Node *node) const;
|
||||||
|
Interpreter::Engine *engine() const;
|
||||||
|
Interpreter::Context *context() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
QScopedPointer<LookupContextData> d;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // end of namespace QmlJS
|
||||||
|
|
||||||
|
#endif // QMLJSLOOKUPCONTEXT_H
|
||||||
|
|
||||||
@@ -31,12 +31,20 @@
|
|||||||
|
|
||||||
using namespace QmlJS;
|
using namespace QmlJS;
|
||||||
|
|
||||||
|
static ModelManagerInterface *g_instance = 0;
|
||||||
|
|
||||||
ModelManagerInterface::ModelManagerInterface(QObject *parent)
|
ModelManagerInterface::ModelManagerInterface(QObject *parent)
|
||||||
: QObject(parent)
|
: QObject(parent)
|
||||||
{
|
{
|
||||||
|
Q_ASSERT(! g_instance);
|
||||||
|
g_instance = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
ModelManagerInterface::~ModelManagerInterface()
|
ModelManagerInterface::~ModelManagerInterface()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ModelManagerInterface *ModelManagerInterface::instance()
|
||||||
|
{
|
||||||
|
return g_instance;
|
||||||
|
}
|
||||||
|
|||||||
@@ -80,6 +80,8 @@ public:
|
|||||||
ModelManagerInterface(QObject *parent = 0);
|
ModelManagerInterface(QObject *parent = 0);
|
||||||
virtual ~ModelManagerInterface();
|
virtual ~ModelManagerInterface();
|
||||||
|
|
||||||
|
static ModelManagerInterface *instance();
|
||||||
|
|
||||||
virtual QmlJS::Snapshot snapshot() const = 0;
|
virtual QmlJS::Snapshot snapshot() const = 0;
|
||||||
virtual void updateSourceFiles(const QStringList &files,
|
virtual void updateSourceFiles(const QStringList &files,
|
||||||
bool emitDocumentOnDiskChanged) = 0;
|
bool emitDocumentOnDiskChanged) = 0;
|
||||||
|
|||||||
@@ -33,12 +33,10 @@
|
|||||||
|
|
||||||
#include <qmljs/qmljsmodelmanagerinterface.h>
|
#include <qmljs/qmljsmodelmanagerinterface.h>
|
||||||
#include <qmljs/parser/qmljsast_p.h>
|
#include <qmljs/parser/qmljsast_p.h>
|
||||||
#include <qmljs/qmljsbind.h>
|
|
||||||
#include <qmljs/qmljsinterpreter.h>
|
#include <qmljs/qmljsinterpreter.h>
|
||||||
|
#include <qmljs/qmljslookupcontext.h>
|
||||||
#include <qmljs/qmljsscanner.h>
|
#include <qmljs/qmljsscanner.h>
|
||||||
#include <qmljs/qmljsevaluate.h>
|
|
||||||
#include <qmljs/qmljscompletioncontextfinder.h>
|
#include <qmljs/qmljscompletioncontextfinder.h>
|
||||||
#include <qmljs/qmljslink.h>
|
|
||||||
#include <qmljs/qmljsscopebuilder.h>
|
#include <qmljs/qmljsscopebuilder.h>
|
||||||
|
|
||||||
#include <texteditor/basetexteditor.h>
|
#include <texteditor/basetexteditor.h>
|
||||||
@@ -689,13 +687,9 @@ int CodeCompletion::startCompletion(TextEditor::ITextEditable *editor)
|
|||||||
const QIcon symbolIcon = iconForColor(Qt::darkCyan);
|
const QIcon symbolIcon = iconForColor(Qt::darkCyan);
|
||||||
const QIcon keywordIcon = iconForColor(Qt::darkYellow);
|
const QIcon keywordIcon = iconForColor(Qt::darkYellow);
|
||||||
|
|
||||||
Interpreter::Engine interp;
|
const QList<AST::Node *> path = semanticInfo.astPath(editor->position());
|
||||||
Interpreter::Context context(&interp);
|
LookupContext::Ptr lookupContext = LookupContext::create(document, snapshot, path);
|
||||||
Link link(&context, document, snapshot, m_modelManager->importPaths());
|
Interpreter::Context *context = lookupContext->context();
|
||||||
|
|
||||||
// Set up the current scope chain.
|
|
||||||
ScopeBuilder scopeBuilder(document, &context);
|
|
||||||
scopeBuilder.push(semanticInfo.astPath(editor->position()));
|
|
||||||
|
|
||||||
// Search for the operator that triggered the completion.
|
// Search for the operator that triggered the completion.
|
||||||
QChar completionOperator;
|
QChar completionOperator;
|
||||||
@@ -708,7 +702,7 @@ int CodeCompletion::startCompletion(TextEditor::ITextEditable *editor)
|
|||||||
|
|
||||||
const Interpreter::ObjectValue *qmlScopeType = 0;
|
const Interpreter::ObjectValue *qmlScopeType = 0;
|
||||||
if (contextFinder.isInQmlContext())
|
if (contextFinder.isInQmlContext())
|
||||||
qmlScopeType = context.lookupType(document.data(), contextFinder.qmlObjectTypeName());
|
qmlScopeType = context->lookupType(document.data(), contextFinder.qmlObjectTypeName());
|
||||||
|
|
||||||
if (completionOperator.isSpace() || completionOperator.isNull() || isDelimiter(completionOperator) ||
|
if (completionOperator.isSpace() || completionOperator.isNull() || isDelimiter(completionOperator) ||
|
||||||
(completionOperator == QLatin1Char('(') && m_startPosition != editor->position())) {
|
(completionOperator == QLatin1Char('(') && m_startPosition != editor->position())) {
|
||||||
@@ -721,7 +715,7 @@ int CodeCompletion::startCompletion(TextEditor::ITextEditable *editor)
|
|||||||
doGlobalCompletion = false;
|
doGlobalCompletion = false;
|
||||||
doJsKeywordCompletion = false;
|
doJsKeywordCompletion = false;
|
||||||
|
|
||||||
EnumerateProperties enumerateProperties(&context);
|
EnumerateProperties enumerateProperties(context);
|
||||||
enumerateProperties.setGlobalCompletion(true);
|
enumerateProperties.setGlobalCompletion(true);
|
||||||
enumerateProperties.setEnumerateGeneratedSlots(true);
|
enumerateProperties.setEnumerateGeneratedSlots(true);
|
||||||
|
|
||||||
@@ -733,11 +727,11 @@ int CodeCompletion::startCompletion(TextEditor::ITextEditable *editor)
|
|||||||
m_completions.append(idPropertyCompletion);
|
m_completions.append(idPropertyCompletion);
|
||||||
|
|
||||||
addCompletionsPropertyLhs(enumerateProperties(qmlScopeType), symbolIcon, PropertyOrder);
|
addCompletionsPropertyLhs(enumerateProperties(qmlScopeType), symbolIcon, PropertyOrder);
|
||||||
addCompletions(enumerateProperties(context.scopeChain().qmlTypes), symbolIcon, TypeOrder);
|
addCompletions(enumerateProperties(context->scopeChain().qmlTypes), symbolIcon, TypeOrder);
|
||||||
|
|
||||||
if (ScopeBuilder::isPropertyChangesObject(&context, qmlScopeType)
|
if (ScopeBuilder::isPropertyChangesObject(context, qmlScopeType)
|
||||||
&& context.scopeChain().qmlScopeObjects.size() == 2) {
|
&& context->scopeChain().qmlScopeObjects.size() == 2) {
|
||||||
addCompletions(enumerateProperties(context.scopeChain().qmlScopeObjects.first()), symbolIcon, SymbolOrder);
|
addCompletions(enumerateProperties(context->scopeChain().qmlScopeObjects.first()), symbolIcon, SymbolOrder);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -748,7 +742,7 @@ int CodeCompletion::startCompletion(TextEditor::ITextEditable *editor)
|
|||||||
const Interpreter::Value *value = qmlScopeType;
|
const Interpreter::Value *value = qmlScopeType;
|
||||||
foreach (const QString &name, contextFinder.bindingPropertyName()) {
|
foreach (const QString &name, contextFinder.bindingPropertyName()) {
|
||||||
if (const Interpreter::ObjectValue *objectValue = value->asObjectValue()) {
|
if (const Interpreter::ObjectValue *objectValue = value->asObjectValue()) {
|
||||||
value = objectValue->property(name, &context);
|
value = objectValue->property(name, context);
|
||||||
if (!value)
|
if (!value)
|
||||||
break;
|
break;
|
||||||
} else {
|
} else {
|
||||||
@@ -772,7 +766,7 @@ int CodeCompletion::startCompletion(TextEditor::ITextEditable *editor)
|
|||||||
|
|
||||||
if (doGlobalCompletion) {
|
if (doGlobalCompletion) {
|
||||||
// It's a global completion.
|
// It's a global completion.
|
||||||
EnumerateProperties enumerateProperties(&context);
|
EnumerateProperties enumerateProperties(context);
|
||||||
enumerateProperties.setGlobalCompletion(true);
|
enumerateProperties.setGlobalCompletion(true);
|
||||||
addCompletions(enumerateProperties(), symbolIcon, SymbolOrder);
|
addCompletions(enumerateProperties(), symbolIcon, SymbolOrder);
|
||||||
}
|
}
|
||||||
@@ -813,14 +807,13 @@ int CodeCompletion::startCompletion(TextEditor::ITextEditable *editor)
|
|||||||
QmlJS::AST::ExpressionNode *expression = expressionUnderCursor(tc);
|
QmlJS::AST::ExpressionNode *expression = expressionUnderCursor(tc);
|
||||||
|
|
||||||
if (expression != 0 && ! isLiteral(expression)) {
|
if (expression != 0 && ! isLiteral(expression)) {
|
||||||
Evaluate evaluate(&context);
|
|
||||||
|
|
||||||
// Evaluate the expression under cursor.
|
// Evaluate the expression under cursor.
|
||||||
const Interpreter::Value *value = interp.convertToObject(evaluate(expression));
|
Interpreter::Engine *interp = lookupContext->engine();
|
||||||
|
const Interpreter::Value *value = interp->convertToObject(lookupContext->evaluate(expression));
|
||||||
//qDebug() << "type:" << interp.typeId(value);
|
//qDebug() << "type:" << interp.typeId(value);
|
||||||
|
|
||||||
if (value && completionOperator == QLatin1Char('.')) { // member completion
|
if (value && completionOperator == QLatin1Char('.')) { // member completion
|
||||||
EnumerateProperties enumerateProperties(&context);
|
EnumerateProperties enumerateProperties(context);
|
||||||
if (contextFinder.isInLhsOfBinding() && qmlScopeType && expressionUnderCursor.text().at(0).isLower())
|
if (contextFinder.isInLhsOfBinding() && qmlScopeType && expressionUnderCursor.text().at(0).isLower())
|
||||||
addCompletionsPropertyLhs(enumerateProperties(value), symbolIcon, PropertyOrder);
|
addCompletionsPropertyLhs(enumerateProperties(value), symbolIcon, PropertyOrder);
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -37,11 +37,8 @@
|
|||||||
#include <coreplugin/editormanager/editormanager.h>
|
#include <coreplugin/editormanager/editormanager.h>
|
||||||
#include <debugger/debuggerconstants.h>
|
#include <debugger/debuggerconstants.h>
|
||||||
#include <extensionsystem/pluginmanager.h>
|
#include <extensionsystem/pluginmanager.h>
|
||||||
#include <qmljs/qmljsbind.h>
|
#include <qmljs/qmljslookupcontext.h>
|
||||||
#include <qmljs/qmljsevaluate.h>
|
|
||||||
#include <qmljs/qmljsinterpreter.h>
|
#include <qmljs/qmljsinterpreter.h>
|
||||||
#include <qmljs/qmljslink.h>
|
|
||||||
#include <qmljs/qmljsscopebuilder.h>
|
|
||||||
#include <qmljs/parser/qmljsast_p.h>
|
#include <qmljs/parser/qmljsast_p.h>
|
||||||
#include <texteditor/itexteditor.h>
|
#include <texteditor/itexteditor.h>
|
||||||
#include <texteditor/basetexteditor.h>
|
#include <texteditor/basetexteditor.h>
|
||||||
@@ -151,17 +148,11 @@ void HoverHandler::updateHelpIdAndTooltip(TextEditor::ITextEditor *editor, int p
|
|||||||
if (node && !(AST::cast<AST::StringLiteral *>(node) != 0 || AST::cast<AST::NumericLiteral *>(node) != 0)) {
|
if (node && !(AST::cast<AST::StringLiteral *>(node) != 0 || AST::cast<AST::NumericLiteral *>(node) != 0)) {
|
||||||
QList<AST::Node *> astPath = semanticInfo.astPath(pos);
|
QList<AST::Node *> astPath = semanticInfo.astPath(pos);
|
||||||
|
|
||||||
Interpreter::Engine interp;
|
LookupContext::Ptr lookupContext = LookupContext::create(qmlDocument, snapshot, astPath);
|
||||||
Interpreter::Context context(&interp);
|
const Interpreter::Value *value = lookupContext->evaluate(node);
|
||||||
Link link(&context, qmlDocument, snapshot, m_modelManager->importPaths());
|
|
||||||
ScopeBuilder scopeBuilder(qmlDocument, &context);
|
|
||||||
scopeBuilder.push(astPath);
|
|
||||||
|
|
||||||
Evaluate check(&context);
|
|
||||||
const Interpreter::Value *value = check(node);
|
|
||||||
|
|
||||||
QStringList baseClasses;
|
QStringList baseClasses;
|
||||||
m_toolTip = prettyPrint(value, &context, &baseClasses);
|
m_toolTip = prettyPrint(value, lookupContext->context(), &baseClasses);
|
||||||
|
|
||||||
foreach (const QString &baseClass, baseClasses) {
|
foreach (const QString &baseClass, baseClasses) {
|
||||||
QString helpId = QLatin1String("QML.");
|
QString helpId = QLatin1String("QML.");
|
||||||
@@ -193,7 +184,7 @@ void HoverHandler::updateHelpIdAndTooltip(TextEditor::ITextEditor *editor, int p
|
|||||||
}
|
}
|
||||||
|
|
||||||
QString HoverHandler::prettyPrint(const QmlJS::Interpreter::Value *value, QmlJS::Interpreter::Context *context,
|
QString HoverHandler::prettyPrint(const QmlJS::Interpreter::Value *value, QmlJS::Interpreter::Context *context,
|
||||||
QStringList *baseClasses) const
|
QStringList *baseClasses) const
|
||||||
{
|
{
|
||||||
if (! value)
|
if (! value)
|
||||||
return QString();
|
return QString();
|
||||||
|
|||||||
Reference in New Issue
Block a user