2014-01-23 14:00:21 +01:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2015-01-14 18:07:15 +01:00
|
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
|
|
|
** Contact: http://www.qt.io/licensing
|
2014-01-23 14:00:21 +01:00
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
|
**
|
|
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2015-01-14 18:07:15 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms and
|
|
|
|
|
** conditions see http://www.qt.io/terms-conditions. For further information
|
2014-10-01 13:21:18 +02:00
|
|
|
** use the contact form at http://www.qt.io/contact-us.
|
2014-01-23 14:00:21 +01:00
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
2014-10-01 13:21:18 +02:00
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2014-01-23 14:00:21 +01:00
|
|
|
**
|
2015-01-14 18:07:15 +01:00
|
|
|
** In addition, as a special exception, The Qt Company gives you certain additional
|
|
|
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
2014-01-23 14:00:21 +01:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "qmljseditordocument.h"
|
|
|
|
|
|
2014-03-05 15:58:12 +01:00
|
|
|
#include "qmljseditorconstants.h"
|
2014-01-23 15:28:06 +01:00
|
|
|
#include "qmljseditordocument_p.h"
|
2014-01-23 14:00:21 +01:00
|
|
|
#include "qmljshighlighter.h"
|
2014-02-06 12:59:00 +01:00
|
|
|
#include "qmljssemantichighlighter.h"
|
2014-01-24 16:53:16 +01:00
|
|
|
#include "qmljssemanticinfoupdater.h"
|
2014-01-30 17:18:35 +01:00
|
|
|
#include "qmloutlinemodel.h"
|
2014-01-23 14:00:21 +01:00
|
|
|
|
2014-01-23 16:21:05 +01:00
|
|
|
#include <qmljstools/qmljsindenter.h>
|
2014-01-23 15:28:06 +01:00
|
|
|
#include <qmljstools/qmljsmodelmanager.h>
|
2014-01-23 16:21:05 +01:00
|
|
|
#include <qmljstools/qmljsqtstylecodeformatter.h>
|
2014-01-23 14:00:21 +01:00
|
|
|
|
|
|
|
|
using namespace QmlJSEditor;
|
2014-01-24 16:53:16 +01:00
|
|
|
using namespace QmlJS;
|
|
|
|
|
using namespace QmlJS::AST;
|
|
|
|
|
using namespace QmlJSTools;
|
2014-01-23 14:00:21 +01:00
|
|
|
|
2014-01-23 15:28:06 +01:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
enum {
|
2014-01-30 17:18:35 +01:00
|
|
|
UPDATE_DOCUMENT_DEFAULT_INTERVAL = 100,
|
|
|
|
|
UPDATE_OUTLINE_INTERVAL = 500
|
2014-01-23 15:28:06 +01:00
|
|
|
};
|
|
|
|
|
|
2014-02-07 12:15:02 +01:00
|
|
|
struct Declaration
|
|
|
|
|
{
|
|
|
|
|
QString text;
|
|
|
|
|
int startLine;
|
|
|
|
|
int startColumn;
|
|
|
|
|
int endLine;
|
|
|
|
|
int endColumn;
|
|
|
|
|
|
|
|
|
|
Declaration()
|
|
|
|
|
: startLine(0),
|
|
|
|
|
startColumn(0),
|
|
|
|
|
endLine(0),
|
|
|
|
|
endColumn(0)
|
|
|
|
|
{ }
|
|
|
|
|
};
|
|
|
|
|
|
2014-01-24 16:53:16 +01:00
|
|
|
class FindIdDeclarations: protected Visitor
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
typedef QHash<QString, QList<AST::SourceLocation> > Result;
|
|
|
|
|
|
|
|
|
|
Result operator()(Document::Ptr doc)
|
|
|
|
|
{
|
|
|
|
|
_ids.clear();
|
|
|
|
|
_maybeIds.clear();
|
|
|
|
|
if (doc && doc->qmlProgram())
|
|
|
|
|
doc->qmlProgram()->accept(this);
|
|
|
|
|
return _ids;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
QString asString(AST::UiQualifiedId *id)
|
|
|
|
|
{
|
|
|
|
|
QString text;
|
|
|
|
|
for (; id; id = id->next) {
|
|
|
|
|
if (!id->name.isEmpty())
|
|
|
|
|
text += id->name;
|
|
|
|
|
else
|
|
|
|
|
text += QLatin1Char('?');
|
|
|
|
|
|
|
|
|
|
if (id->next)
|
|
|
|
|
text += QLatin1Char('.');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return text;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void accept(AST::Node *node)
|
|
|
|
|
{ AST::Node::acceptChild(node, this); }
|
|
|
|
|
|
|
|
|
|
using Visitor::visit;
|
|
|
|
|
using Visitor::endVisit;
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::UiScriptBinding *node)
|
|
|
|
|
{
|
|
|
|
|
if (asString(node->qualifiedId) == QLatin1String("id")) {
|
|
|
|
|
if (AST::ExpressionStatement *stmt = AST::cast<AST::ExpressionStatement*>(node->statement)) {
|
|
|
|
|
if (AST::IdentifierExpression *idExpr = AST::cast<AST::IdentifierExpression *>(stmt->expression)) {
|
|
|
|
|
if (!idExpr->name.isEmpty()) {
|
|
|
|
|
const QString &id = idExpr->name.toString();
|
|
|
|
|
QList<AST::SourceLocation> *locs = &_ids[id];
|
|
|
|
|
locs->append(idExpr->firstSourceLocation());
|
|
|
|
|
locs->append(_maybeIds.value(id));
|
|
|
|
|
_maybeIds.remove(id);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
accept(node->statement);
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::IdentifierExpression *node)
|
|
|
|
|
{
|
|
|
|
|
if (!node->name.isEmpty()) {
|
|
|
|
|
const QString &name = node->name.toString();
|
|
|
|
|
|
|
|
|
|
if (_ids.contains(name))
|
|
|
|
|
_ids[name].append(node->identifierToken);
|
|
|
|
|
else
|
|
|
|
|
_maybeIds[name].append(node->identifierToken);
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
Result _ids;
|
|
|
|
|
Result _maybeIds;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class FindDeclarations: protected Visitor
|
|
|
|
|
{
|
|
|
|
|
QList<Declaration> _declarations;
|
|
|
|
|
int _depth;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
QList<Declaration> operator()(AST::Node *node)
|
|
|
|
|
{
|
|
|
|
|
_depth = -1;
|
|
|
|
|
_declarations.clear();
|
|
|
|
|
accept(node);
|
|
|
|
|
return _declarations;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
using Visitor::visit;
|
|
|
|
|
using Visitor::endVisit;
|
|
|
|
|
|
|
|
|
|
QString asString(AST::UiQualifiedId *id)
|
|
|
|
|
{
|
|
|
|
|
QString text;
|
|
|
|
|
for (; id; id = id->next) {
|
|
|
|
|
if (!id->name.isEmpty())
|
|
|
|
|
text += id->name;
|
|
|
|
|
else
|
|
|
|
|
text += QLatin1Char('?');
|
|
|
|
|
|
|
|
|
|
if (id->next)
|
|
|
|
|
text += QLatin1Char('.');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return text;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void accept(AST::Node *node)
|
|
|
|
|
{ AST::Node::acceptChild(node, this); }
|
|
|
|
|
|
|
|
|
|
void init(Declaration *decl, AST::UiObjectMember *member)
|
|
|
|
|
{
|
|
|
|
|
const SourceLocation first = member->firstSourceLocation();
|
|
|
|
|
const SourceLocation last = member->lastSourceLocation();
|
|
|
|
|
decl->startLine = first.startLine;
|
|
|
|
|
decl->startColumn = first.startColumn;
|
|
|
|
|
decl->endLine = last.startLine;
|
|
|
|
|
decl->endColumn = last.startColumn + last.length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void init(Declaration *decl, AST::ExpressionNode *expressionNode)
|
|
|
|
|
{
|
|
|
|
|
const SourceLocation first = expressionNode->firstSourceLocation();
|
|
|
|
|
const SourceLocation last = expressionNode->lastSourceLocation();
|
|
|
|
|
decl->startLine = first.startLine;
|
|
|
|
|
decl->startColumn = first.startColumn;
|
|
|
|
|
decl->endLine = last.startLine;
|
|
|
|
|
decl->endColumn = last.startColumn + last.length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::UiObjectDefinition *node)
|
|
|
|
|
{
|
|
|
|
|
++_depth;
|
|
|
|
|
|
|
|
|
|
Declaration decl;
|
|
|
|
|
init(&decl, node);
|
|
|
|
|
|
|
|
|
|
decl.text.fill(QLatin1Char(' '), _depth);
|
|
|
|
|
if (node->qualifiedTypeNameId)
|
|
|
|
|
decl.text.append(asString(node->qualifiedTypeNameId));
|
|
|
|
|
else
|
|
|
|
|
decl.text.append(QLatin1Char('?'));
|
|
|
|
|
|
|
|
|
|
_declarations.append(decl);
|
|
|
|
|
|
|
|
|
|
return true; // search for more bindings
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual void endVisit(AST::UiObjectDefinition *)
|
|
|
|
|
{
|
|
|
|
|
--_depth;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::UiObjectBinding *node)
|
|
|
|
|
{
|
|
|
|
|
++_depth;
|
|
|
|
|
|
|
|
|
|
Declaration decl;
|
|
|
|
|
init(&decl, node);
|
|
|
|
|
|
|
|
|
|
decl.text.fill(QLatin1Char(' '), _depth);
|
|
|
|
|
|
|
|
|
|
decl.text.append(asString(node->qualifiedId));
|
|
|
|
|
decl.text.append(QLatin1String(": "));
|
|
|
|
|
|
|
|
|
|
if (node->qualifiedTypeNameId)
|
|
|
|
|
decl.text.append(asString(node->qualifiedTypeNameId));
|
|
|
|
|
else
|
|
|
|
|
decl.text.append(QLatin1Char('?'));
|
|
|
|
|
|
|
|
|
|
_declarations.append(decl);
|
|
|
|
|
|
|
|
|
|
return true; // search for more bindings
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual void endVisit(AST::UiObjectBinding *)
|
|
|
|
|
{
|
|
|
|
|
--_depth;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::UiScriptBinding *)
|
|
|
|
|
{
|
|
|
|
|
++_depth;
|
|
|
|
|
|
|
|
|
|
#if 0 // ### ignore script bindings for now.
|
|
|
|
|
Declaration decl;
|
|
|
|
|
init(&decl, node);
|
|
|
|
|
|
|
|
|
|
decl.text.fill(QLatin1Char(' '), _depth);
|
|
|
|
|
decl.text.append(asString(node->qualifiedId));
|
|
|
|
|
|
|
|
|
|
_declarations.append(decl);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return false; // more more bindings in this subtree.
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual void endVisit(AST::UiScriptBinding *)
|
|
|
|
|
{
|
|
|
|
|
--_depth;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::FunctionExpression *)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::FunctionDeclaration *ast)
|
|
|
|
|
{
|
|
|
|
|
if (ast->name.isEmpty())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
Declaration decl;
|
|
|
|
|
init(&decl, ast);
|
|
|
|
|
|
|
|
|
|
decl.text.fill(QLatin1Char(' '), _depth);
|
|
|
|
|
decl.text += ast->name;
|
|
|
|
|
|
|
|
|
|
decl.text += QLatin1Char('(');
|
|
|
|
|
for (FormalParameterList *it = ast->formals; it; it = it->next) {
|
|
|
|
|
if (!it->name.isEmpty())
|
|
|
|
|
decl.text += it->name;
|
|
|
|
|
|
|
|
|
|
if (it->next)
|
|
|
|
|
decl.text += QLatin1String(", ");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
decl.text += QLatin1Char(')');
|
|
|
|
|
|
|
|
|
|
_declarations.append(decl);
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::VariableDeclaration *ast)
|
|
|
|
|
{
|
|
|
|
|
if (ast->name.isEmpty())
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
Declaration decl;
|
|
|
|
|
decl.text.fill(QLatin1Char(' '), _depth);
|
|
|
|
|
decl.text += ast->name;
|
|
|
|
|
|
|
|
|
|
const SourceLocation first = ast->identifierToken;
|
|
|
|
|
decl.startLine = first.startLine;
|
|
|
|
|
decl.startColumn = first.startColumn;
|
|
|
|
|
decl.endLine = first.startLine;
|
|
|
|
|
decl.endColumn = first.startColumn + first.length;
|
|
|
|
|
|
|
|
|
|
_declarations.append(decl);
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class CreateRanges: protected AST::Visitor
|
|
|
|
|
{
|
|
|
|
|
QTextDocument *_textDocument;
|
|
|
|
|
QList<Range> _ranges;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
QList<Range> operator()(QTextDocument *textDocument, Document::Ptr doc)
|
|
|
|
|
{
|
|
|
|
|
_textDocument = textDocument;
|
|
|
|
|
_ranges.clear();
|
|
|
|
|
if (doc && doc->ast() != 0)
|
|
|
|
|
doc->ast()->accept(this);
|
|
|
|
|
return _ranges;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
using AST::Visitor::visit;
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::UiObjectBinding *ast)
|
|
|
|
|
{
|
|
|
|
|
if (ast->initializer && ast->initializer->lbraceToken.length)
|
|
|
|
|
_ranges.append(createRange(ast, ast->initializer));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::UiObjectDefinition *ast)
|
|
|
|
|
{
|
|
|
|
|
if (ast->initializer && ast->initializer->lbraceToken.length)
|
|
|
|
|
_ranges.append(createRange(ast, ast->initializer));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::FunctionExpression *ast)
|
|
|
|
|
{
|
|
|
|
|
_ranges.append(createRange(ast));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::FunctionDeclaration *ast)
|
|
|
|
|
{
|
|
|
|
|
_ranges.append(createRange(ast));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual bool visit(AST::UiScriptBinding *ast)
|
|
|
|
|
{
|
|
|
|
|
if (AST::Block *block = AST::cast<AST::Block *>(ast->statement))
|
|
|
|
|
_ranges.append(createRange(ast, block));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Range createRange(AST::UiObjectMember *member, AST::UiObjectInitializer *ast)
|
|
|
|
|
{
|
|
|
|
|
return createRange(member, member->firstSourceLocation(), ast->rbraceToken);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Range createRange(AST::FunctionExpression *ast)
|
|
|
|
|
{
|
|
|
|
|
return createRange(ast, ast->lbraceToken, ast->rbraceToken);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Range createRange(AST::UiScriptBinding *ast, AST::Block *block)
|
|
|
|
|
{
|
|
|
|
|
return createRange(ast, block->lbraceToken, block->rbraceToken);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Range createRange(AST::Node *ast, AST::SourceLocation start, AST::SourceLocation end)
|
|
|
|
|
{
|
|
|
|
|
Range range;
|
|
|
|
|
|
|
|
|
|
range.ast = ast;
|
|
|
|
|
|
|
|
|
|
range.begin = QTextCursor(_textDocument);
|
|
|
|
|
range.begin.setPosition(start.begin());
|
|
|
|
|
|
|
|
|
|
range.end = QTextCursor(_textDocument);
|
|
|
|
|
range.end.setPosition(end.end());
|
|
|
|
|
|
|
|
|
|
return range;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2014-01-23 15:28:06 +01:00
|
|
|
}
|
|
|
|
|
|
2014-01-24 16:53:16 +01:00
|
|
|
namespace QmlJSEditor {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2014-01-23 15:28:06 +01:00
|
|
|
QmlJSEditorDocumentPrivate::QmlJSEditorDocumentPrivate(QmlJSEditorDocument *parent)
|
2014-02-10 14:18:32 +01:00
|
|
|
: q(parent),
|
2014-02-06 12:59:00 +01:00
|
|
|
m_semanticInfoDocRevision(-1),
|
|
|
|
|
m_semanticHighlighter(new SemanticHighlighter(parent)),
|
2014-01-30 17:18:35 +01:00
|
|
|
m_semanticHighlightingNecessary(false),
|
|
|
|
|
m_outlineModelNeedsUpdate(false),
|
|
|
|
|
m_outlineModel(new QmlOutlineModel(parent))
|
2014-01-23 15:28:06 +01:00
|
|
|
{
|
2014-01-24 16:53:16 +01:00
|
|
|
ModelManagerInterface *modelManager = ModelManagerInterface::instance();
|
|
|
|
|
|
|
|
|
|
// code model
|
2014-02-10 14:18:32 +01:00
|
|
|
m_updateDocumentTimer.setInterval(UPDATE_DOCUMENT_DEFAULT_INTERVAL);
|
|
|
|
|
m_updateDocumentTimer.setSingleShot(true);
|
|
|
|
|
connect(q->document(), SIGNAL(contentsChanged()), &m_updateDocumentTimer, SLOT(start()));
|
|
|
|
|
connect(&m_updateDocumentTimer, SIGNAL(timeout()), this, SLOT(reparseDocument()));
|
2014-01-24 16:53:16 +01:00
|
|
|
connect(modelManager, SIGNAL(documentUpdated(QmlJS::Document::Ptr)),
|
|
|
|
|
this, SLOT(onDocumentUpdated(QmlJS::Document::Ptr)));
|
|
|
|
|
|
|
|
|
|
// semantic info
|
|
|
|
|
m_semanticInfoUpdater = new SemanticInfoUpdater(this);
|
|
|
|
|
connect(m_semanticInfoUpdater, SIGNAL(updated(QmlJSTools::SemanticInfo)),
|
|
|
|
|
this, SLOT(acceptNewSemanticInfo(QmlJSTools::SemanticInfo)));
|
|
|
|
|
m_semanticInfoUpdater->start();
|
|
|
|
|
|
|
|
|
|
// library info changes
|
2014-02-10 14:18:32 +01:00
|
|
|
m_reupdateSemanticInfoTimer.setInterval(UPDATE_DOCUMENT_DEFAULT_INTERVAL);
|
|
|
|
|
m_reupdateSemanticInfoTimer.setSingleShot(true);
|
|
|
|
|
connect(&m_reupdateSemanticInfoTimer, SIGNAL(timeout()), this, SLOT(reupdateSemanticInfo()));
|
2014-01-24 16:53:16 +01:00
|
|
|
connect(modelManager, SIGNAL(libraryInfoUpdated(QString,QmlJS::LibraryInfo)),
|
2014-02-10 14:18:32 +01:00
|
|
|
&m_reupdateSemanticInfoTimer, SLOT(start()));
|
2014-01-30 17:18:35 +01:00
|
|
|
|
|
|
|
|
// outline model
|
2014-02-10 14:18:32 +01:00
|
|
|
m_updateOutlineModelTimer.setInterval(UPDATE_OUTLINE_INTERVAL);
|
|
|
|
|
m_updateOutlineModelTimer.setSingleShot(true);
|
|
|
|
|
connect(&m_updateOutlineModelTimer, SIGNAL(timeout()), this, SLOT(updateOutlineModel()));
|
2014-04-29 16:39:16 +02:00
|
|
|
|
2014-12-21 21:54:30 +02:00
|
|
|
modelManager->updateSourceFiles(QStringList(parent->filePath().toString()), false);
|
2014-01-24 16:53:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QmlJSEditorDocumentPrivate::~QmlJSEditorDocumentPrivate()
|
|
|
|
|
{
|
|
|
|
|
m_semanticInfoUpdater->abort();
|
|
|
|
|
m_semanticInfoUpdater->wait();
|
2014-01-23 15:28:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QmlJSEditorDocumentPrivate::invalidateFormatterCache()
|
|
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
CreatorCodeFormatter formatter(q->tabSettings());
|
|
|
|
|
formatter.invalidateCache(q->document());
|
2014-01-23 15:28:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QmlJSEditorDocumentPrivate::reparseDocument()
|
|
|
|
|
{
|
2014-12-21 21:54:30 +02:00
|
|
|
ModelManagerInterface::instance()->updateSourceFiles(QStringList(q->filePath().toString()),
|
|
|
|
|
false);
|
2014-01-23 15:28:06 +01:00
|
|
|
}
|
|
|
|
|
|
2014-01-24 16:53:16 +01:00
|
|
|
void QmlJSEditorDocumentPrivate::onDocumentUpdated(Document::Ptr doc)
|
|
|
|
|
{
|
2014-12-21 21:54:30 +02:00
|
|
|
if (q->filePath().toString() != doc->fileName())
|
2014-01-24 16:53:16 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// text document has changed, simply wait for the next onDocumentUpdated
|
2014-02-10 14:18:32 +01:00
|
|
|
if (doc->editorRevision() != q->document()->revision())
|
2014-01-24 16:53:16 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (doc->ast()) {
|
|
|
|
|
// got a correctly parsed (or recovered) file.
|
|
|
|
|
m_semanticInfoDocRevision = doc->editorRevision();
|
|
|
|
|
m_semanticInfoUpdater->update(doc, ModelManagerInterface::instance()->snapshot());
|
|
|
|
|
}
|
2014-02-10 14:18:32 +01:00
|
|
|
emit q->updateCodeWarnings(doc);
|
2014-01-24 16:53:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QmlJSEditorDocumentPrivate::reupdateSemanticInfo()
|
|
|
|
|
{
|
|
|
|
|
// If the editor is newer than the semantic info (possibly with update in progress),
|
|
|
|
|
// new semantic infos won't be accepted anyway. We'll get a onDocumentUpdated anyhow.
|
2014-02-10 14:18:32 +01:00
|
|
|
if (q->document()->revision() != m_semanticInfoDocRevision)
|
2014-01-24 16:53:16 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
m_semanticInfoUpdater->reupdate(ModelManagerInterface::instance()->snapshot());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QmlJSEditorDocumentPrivate::acceptNewSemanticInfo(const SemanticInfo &semanticInfo)
|
|
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
if (semanticInfo.revision() != q->document()->revision()) {
|
2014-01-24 16:53:16 +01:00
|
|
|
// ignore outdated semantic infos
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_semanticInfo = semanticInfo;
|
|
|
|
|
Document::Ptr doc = semanticInfo.document;
|
|
|
|
|
|
|
|
|
|
// create the ranges
|
|
|
|
|
CreateRanges createRanges;
|
2014-02-10 14:18:32 +01:00
|
|
|
m_semanticInfo.ranges = createRanges(q->document(), doc);
|
2014-01-24 16:53:16 +01:00
|
|
|
|
|
|
|
|
// Refresh the ids
|
|
|
|
|
FindIdDeclarations updateIds;
|
|
|
|
|
m_semanticInfo.idLocations = updateIds(doc);
|
|
|
|
|
|
2014-01-30 17:18:35 +01:00
|
|
|
m_outlineModelNeedsUpdate = true;
|
2014-02-06 12:59:00 +01:00
|
|
|
m_semanticHighlightingNecessary = true;
|
2014-01-30 17:18:35 +01:00
|
|
|
|
2014-02-10 14:18:32 +01:00
|
|
|
emit q->semanticInfoUpdated(m_semanticInfo); // calls triggerPendingUpdates as necessary
|
2014-01-30 17:18:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QmlJSEditorDocumentPrivate::updateOutlineModel()
|
|
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
if (q->isSemanticInfoOutdated())
|
2014-01-30 17:18:35 +01:00
|
|
|
return; // outline update will be retriggered when semantic info is updated
|
|
|
|
|
|
|
|
|
|
m_outlineModel->update(m_semanticInfo);
|
2014-01-24 16:53:16 +01:00
|
|
|
}
|
|
|
|
|
|
2014-02-07 13:45:51 +01:00
|
|
|
} // Internal
|
|
|
|
|
|
2014-01-23 14:00:21 +01:00
|
|
|
QmlJSEditorDocument::QmlJSEditorDocument()
|
2014-02-10 14:18:32 +01:00
|
|
|
: d(new Internal::QmlJSEditorDocumentPrivate(this))
|
2014-01-23 14:00:21 +01:00
|
|
|
{
|
2014-03-05 15:58:12 +01:00
|
|
|
setId(Constants::C_QMLJSEDITOR_ID);
|
2014-01-23 14:00:21 +01:00
|
|
|
connect(this, SIGNAL(tabSettingsChanged()),
|
2014-02-10 14:18:32 +01:00
|
|
|
d, SLOT(invalidateFormatterCache()));
|
2014-08-27 13:38:02 +02:00
|
|
|
setSyntaxHighlighter(new QmlJSHighlighter(document()));
|
2014-02-07 13:45:51 +01:00
|
|
|
setIndenter(new Internal::Indenter);
|
2014-01-23 14:00:21 +01:00
|
|
|
}
|
|
|
|
|
|
2014-01-23 15:28:06 +01:00
|
|
|
QmlJSEditorDocument::~QmlJSEditorDocument()
|
2014-01-23 14:00:21 +01:00
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
delete d;
|
2014-01-23 14:00:21 +01:00
|
|
|
}
|
2014-01-24 16:53:16 +01:00
|
|
|
|
|
|
|
|
const SemanticInfo &QmlJSEditorDocument::semanticInfo() const
|
|
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
return d->m_semanticInfo;
|
2014-01-24 16:53:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool QmlJSEditorDocument::isSemanticInfoOutdated() const
|
|
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
return d->m_semanticInfo.revision() != document()->revision();
|
2014-01-24 16:53:16 +01:00
|
|
|
}
|
|
|
|
|
|
2014-01-29 14:10:43 +01:00
|
|
|
QVector<QTextLayout::FormatRange> QmlJSEditorDocument::diagnosticRanges() const
|
|
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
return d->m_diagnosticRanges;
|
2014-01-29 14:10:43 +01:00
|
|
|
}
|
|
|
|
|
|
2014-02-07 13:45:51 +01:00
|
|
|
Internal::QmlOutlineModel *QmlJSEditorDocument::outlineModel() const
|
2014-01-30 17:18:35 +01:00
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
return d->m_outlineModel;
|
2014-01-30 17:18:35 +01:00
|
|
|
}
|
|
|
|
|
|
2014-01-29 14:10:43 +01:00
|
|
|
void QmlJSEditorDocument::setDiagnosticRanges(const QVector<QTextLayout::FormatRange> &ranges)
|
|
|
|
|
{
|
2014-02-10 14:18:32 +01:00
|
|
|
d->m_diagnosticRanges = ranges;
|
2014-01-29 14:10:43 +01:00
|
|
|
}
|
|
|
|
|
|
2014-02-06 12:59:00 +01:00
|
|
|
void QmlJSEditorDocument::applyFontSettings()
|
|
|
|
|
{
|
2014-09-22 18:43:31 +02:00
|
|
|
TextDocument::applyFontSettings();
|
2014-02-10 14:18:32 +01:00
|
|
|
d->m_semanticHighlighter->updateFontSettings(fontSettings());
|
2014-02-06 12:59:00 +01:00
|
|
|
if (!isSemanticInfoOutdated()) {
|
2014-02-10 14:18:32 +01:00
|
|
|
d->m_semanticHighlightingNecessary = false;
|
|
|
|
|
d->m_semanticHighlighter->rerun(d->m_semanticInfo);
|
2014-02-06 12:59:00 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void QmlJSEditorDocument::triggerPendingUpdates()
|
|
|
|
|
{
|
2014-09-22 18:43:31 +02:00
|
|
|
TextDocument::triggerPendingUpdates(); // calls applyFontSettings if necessary
|
2014-02-06 12:59:00 +01:00
|
|
|
// might still need to rehighlight if font settings did not change
|
2014-02-10 14:18:32 +01:00
|
|
|
if (d->m_semanticHighlightingNecessary && !isSemanticInfoOutdated()) {
|
|
|
|
|
d->m_semanticHighlightingNecessary = false;
|
|
|
|
|
d->m_semanticHighlighter->rerun(d->m_semanticInfo);
|
2014-02-06 12:59:00 +01:00
|
|
|
}
|
2014-02-10 14:18:32 +01:00
|
|
|
if (d->m_outlineModelNeedsUpdate && !isSemanticInfoOutdated()) {
|
|
|
|
|
d->m_outlineModelNeedsUpdate = false;
|
|
|
|
|
d->m_updateOutlineModelTimer.start();
|
2014-01-30 17:18:35 +01:00
|
|
|
}
|
2014-02-06 12:59:00 +01:00
|
|
|
}
|
|
|
|
|
|
2014-01-24 16:53:16 +01:00
|
|
|
} // QmlJSEditor
|