2009-04-22 15:21:04 +02:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2010-03-05 11:25:49 +01:00
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
2009-04-22 15:21:04 +02:00
|
|
|
**
|
2009-06-17 00:01:27 +10:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2009-04-22 15:21:04 +02:00
|
|
|
**
|
|
|
|
|
** 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
|
2009-08-14 09:30:56 +02:00
|
|
|
** contact the sales department at http://qt.nokia.com/contact.
|
2009-04-22 15:21:04 +02:00
|
|
|
**
|
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
2010-01-15 17:20:03 +01:00
|
|
|
#ifndef QMLJSEDITOR_H
|
|
|
|
|
#define QMLJSEDITOR_H
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2010-01-18 16:15:23 +01:00
|
|
|
#include <qmljs/qmljsdocument.h>
|
|
|
|
|
#include <qmljs/qmljsscanner.h>
|
2009-04-22 15:21:04 +02:00
|
|
|
#include <texteditor/basetexteditor.h>
|
2009-05-14 14:09:00 +02:00
|
|
|
|
2010-02-16 10:36:09 +01:00
|
|
|
#include <QtCore/QWaitCondition>
|
2010-07-12 14:45:22 +02:00
|
|
|
#include <QtCore/QModelIndex>
|
2010-02-16 10:36:09 +01:00
|
|
|
#include <QtCore/QMutex>
|
|
|
|
|
#include <QtCore/QThread>
|
|
|
|
|
|
2009-04-22 15:21:04 +02:00
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
|
class QComboBox;
|
|
|
|
|
class QTimer;
|
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
|
|
|
|
|
namespace Core {
|
2009-09-22 17:25:18 +02:00
|
|
|
class ICore;
|
2009-04-22 15:21:04 +02:00
|
|
|
}
|
|
|
|
|
|
2010-06-09 15:56:03 +02:00
|
|
|
namespace QmlJS {
|
|
|
|
|
class ModelManagerInterface;
|
2010-07-07 13:09:30 +02:00
|
|
|
class IContextPane;
|
2010-06-09 15:56:03 +02:00
|
|
|
}
|
2009-09-04 16:51:11 +02:00
|
|
|
|
2010-06-09 15:56:03 +02:00
|
|
|
namespace QmlJSEditor {
|
2010-03-17 11:37:08 +01:00
|
|
|
class Highlighter;
|
2009-09-04 16:51:11 +02:00
|
|
|
|
2009-04-22 15:21:04 +02:00
|
|
|
namespace Internal {
|
|
|
|
|
|
2010-01-15 17:20:03 +01:00
|
|
|
class QmlJSTextEditor;
|
2010-07-12 14:45:22 +02:00
|
|
|
class QmlOutlineModel;
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2010-01-15 17:20:03 +01:00
|
|
|
class QmlJSEditorEditable : public TextEditor::BaseTextEditorEditable
|
2009-04-22 15:21:04 +02:00
|
|
|
{
|
2009-09-22 17:25:18 +02:00
|
|
|
Q_OBJECT
|
2009-05-06 15:45:19 +02:00
|
|
|
|
2009-04-22 15:21:04 +02:00
|
|
|
public:
|
2010-01-15 17:20:03 +01:00
|
|
|
QmlJSEditorEditable(QmlJSTextEditor *);
|
2010-06-25 12:56:16 +02:00
|
|
|
Core::Context context() const;
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2009-09-22 17:25:18 +02:00
|
|
|
bool duplicateSupported() const { return true; }
|
|
|
|
|
Core::IEditor *duplicate(QWidget *parent);
|
2010-01-07 18:17:24 +01:00
|
|
|
QString id() const;
|
2009-09-22 17:25:18 +02:00
|
|
|
bool isTemporary() const { return false; }
|
2010-01-15 17:41:12 +01:00
|
|
|
virtual bool open(const QString & fileName);
|
2010-03-16 16:51:45 +01:00
|
|
|
virtual QString preferredMode() const;
|
2009-04-22 15:21:04 +02:00
|
|
|
|
|
|
|
|
private:
|
2010-06-25 12:56:16 +02:00
|
|
|
Core::Context m_context;
|
2009-04-22 15:21:04 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct Declaration
|
|
|
|
|
{
|
2009-09-22 17:25:18 +02:00
|
|
|
QString text;
|
|
|
|
|
int startLine;
|
|
|
|
|
int startColumn;
|
|
|
|
|
int endLine;
|
|
|
|
|
int endColumn;
|
|
|
|
|
|
|
|
|
|
Declaration()
|
|
|
|
|
: startLine(0),
|
|
|
|
|
startColumn(0),
|
|
|
|
|
endLine(0),
|
|
|
|
|
endColumn(0)
|
|
|
|
|
{ }
|
2009-04-22 15:21:04 +02:00
|
|
|
};
|
|
|
|
|
|
2010-01-25 14:18:53 +01:00
|
|
|
class Range
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
Range(): ast(0) {}
|
|
|
|
|
|
|
|
|
|
public: // attributes
|
2010-02-02 16:36:14 +01:00
|
|
|
QmlJS::AST::Node *ast;
|
2010-01-25 14:18:53 +01:00
|
|
|
QTextCursor begin;
|
|
|
|
|
QTextCursor end;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class SemanticInfo
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
SemanticInfo() {}
|
|
|
|
|
|
2010-01-26 14:14:01 +01:00
|
|
|
int revision() const;
|
|
|
|
|
|
2010-02-01 16:12:19 +01:00
|
|
|
// Returns the declaring member
|
2010-02-02 16:36:14 +01:00
|
|
|
QmlJS::AST::Node *declaringMember(int cursorPosition) const;
|
2010-02-01 16:12:19 +01:00
|
|
|
|
2010-02-02 13:18:56 +01:00
|
|
|
// Returns the AST node under cursor
|
|
|
|
|
QmlJS::AST::Node *nodeUnderCursor(int cursorPosition) const;
|
|
|
|
|
|
2010-02-19 10:16:33 +01:00
|
|
|
// Returns the list of nodes that enclose the given position.
|
|
|
|
|
QList<QmlJS::AST::Node *> astPath(int cursorPosition) const;
|
|
|
|
|
|
2010-01-25 14:18:53 +01:00
|
|
|
public: // attributes
|
|
|
|
|
QmlJS::Document::Ptr document;
|
2010-02-01 16:12:19 +01:00
|
|
|
QmlJS::Snapshot snapshot;
|
2010-01-25 14:18:53 +01:00
|
|
|
QList<Range> ranges;
|
2010-01-26 14:14:01 +01:00
|
|
|
QHash<QString, QList<QmlJS::AST::SourceLocation> > idLocations;
|
2010-01-26 14:26:26 +01:00
|
|
|
QList<Declaration> declarations;
|
2010-02-16 10:36:09 +01:00
|
|
|
|
|
|
|
|
// these are in addition to the parser messages in the document
|
|
|
|
|
QList<QmlJS::DiagnosticMessage> semanticMessages;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class SemanticHighlighter: public QThread
|
|
|
|
|
{
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
SemanticHighlighter(QObject *parent = 0);
|
|
|
|
|
virtual ~SemanticHighlighter();
|
|
|
|
|
|
|
|
|
|
void abort();
|
|
|
|
|
|
|
|
|
|
struct Source
|
|
|
|
|
{
|
|
|
|
|
QmlJS::Snapshot snapshot;
|
|
|
|
|
QString fileName;
|
|
|
|
|
QString code;
|
|
|
|
|
int line;
|
|
|
|
|
int column;
|
|
|
|
|
int revision;
|
|
|
|
|
bool force;
|
|
|
|
|
|
|
|
|
|
Source()
|
|
|
|
|
: line(0), column(0), revision(0), force(false)
|
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
Source(const QmlJS::Snapshot &snapshot,
|
|
|
|
|
const QString &fileName,
|
|
|
|
|
const QString &code,
|
|
|
|
|
int line, int column,
|
|
|
|
|
int revision)
|
|
|
|
|
: snapshot(snapshot), fileName(fileName),
|
|
|
|
|
code(code), line(line), column(column),
|
|
|
|
|
revision(revision), force(false)
|
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
void clear()
|
|
|
|
|
{
|
|
|
|
|
snapshot = QmlJS::Snapshot();
|
|
|
|
|
fileName.clear();
|
|
|
|
|
code.clear();
|
|
|
|
|
line = 0;
|
|
|
|
|
column = 0;
|
|
|
|
|
revision = 0;
|
|
|
|
|
force = false;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void rehighlight(const Source &source);
|
2010-06-09 15:56:03 +02:00
|
|
|
void setModelManager(QmlJS::ModelManagerInterface *modelManager);
|
2010-02-16 10:36:09 +01:00
|
|
|
|
|
|
|
|
Q_SIGNALS:
|
2010-02-16 15:26:58 +01:00
|
|
|
void changed(const QmlJSEditor::Internal::SemanticInfo &semanticInfo);
|
2010-02-16 10:36:09 +01:00
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
virtual void run();
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
bool isOutdated();
|
|
|
|
|
SemanticInfo semanticInfo(const Source &source);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QMutex m_mutex;
|
|
|
|
|
QWaitCondition m_condition;
|
|
|
|
|
bool m_done;
|
|
|
|
|
Source m_source;
|
|
|
|
|
SemanticInfo m_lastSemanticInfo;
|
2010-06-09 15:56:03 +02:00
|
|
|
QmlJS::ModelManagerInterface *m_modelManager;
|
2010-01-25 14:18:53 +01:00
|
|
|
};
|
|
|
|
|
|
2010-01-15 17:20:03 +01:00
|
|
|
class QmlJSTextEditor : public TextEditor::BaseTextEditor
|
2009-04-22 15:21:04 +02:00
|
|
|
{
|
2009-09-22 17:25:18 +02:00
|
|
|
Q_OBJECT
|
2009-04-22 15:21:04 +02:00
|
|
|
|
|
|
|
|
public:
|
2010-01-15 17:20:03 +01:00
|
|
|
QmlJSTextEditor(QWidget *parent = 0);
|
|
|
|
|
~QmlJSTextEditor();
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2009-09-22 17:25:18 +02:00
|
|
|
virtual void unCommentSelection();
|
2009-05-14 16:37:17 +02:00
|
|
|
|
2010-02-02 15:01:42 +01:00
|
|
|
SemanticInfo semanticInfo() const;
|
|
|
|
|
int documentRevision() const;
|
2010-06-03 15:49:29 +02:00
|
|
|
bool isOutdated() const;
|
2009-07-06 16:29:44 +02:00
|
|
|
|
2010-07-12 14:45:22 +02:00
|
|
|
QmlOutlineModel *outlineModel() const;
|
2010-07-12 16:40:15 +02:00
|
|
|
QModelIndex outlineModelIndex();
|
2010-07-12 14:45:22 +02:00
|
|
|
|
2009-04-22 15:21:04 +02:00
|
|
|
public slots:
|
2010-02-08 12:50:10 +01:00
|
|
|
void followSymbolUnderCursor();
|
2009-09-22 17:25:18 +02:00
|
|
|
virtual void setFontSettings(const TextEditor::FontSettings &);
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2010-07-01 11:09:47 +02:00
|
|
|
signals:
|
2010-07-12 14:45:22 +02:00
|
|
|
void outlineModelIndexChanged(const QModelIndex &index);
|
2010-07-01 11:09:47 +02:00
|
|
|
|
2009-04-22 15:21:04 +02:00
|
|
|
private slots:
|
2010-01-18 16:15:23 +01:00
|
|
|
void onDocumentUpdated(QmlJS::Document::Ptr doc);
|
2010-04-16 12:42:12 +02:00
|
|
|
void modificationChanged(bool);
|
2009-09-04 16:51:11 +02:00
|
|
|
|
2009-09-22 17:25:18 +02:00
|
|
|
void updateDocument();
|
|
|
|
|
void updateDocumentNow();
|
|
|
|
|
void jumpToMethod(int index);
|
|
|
|
|
void updateMethodBoxIndex();
|
|
|
|
|
void updateMethodBoxToolTip();
|
|
|
|
|
void updateFileName();
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2010-01-14 16:30:48 +01:00
|
|
|
void updateUses();
|
|
|
|
|
void updateUsesNow();
|
|
|
|
|
|
2009-09-22 17:25:18 +02:00
|
|
|
// refactoring ops
|
|
|
|
|
void renameIdUnderCursor();
|
2009-05-05 15:33:39 +02:00
|
|
|
|
2010-02-16 10:36:09 +01:00
|
|
|
void semanticRehighlight();
|
2010-04-06 11:57:08 +02:00
|
|
|
void forceSemanticRehighlight();
|
2010-02-16 15:26:58 +01:00
|
|
|
void updateSemanticInfo(const QmlJSEditor::Internal::SemanticInfo &semanticInfo);
|
2010-07-07 13:09:30 +02:00
|
|
|
void onCursorPositionChanged();
|
2010-02-16 10:36:09 +01:00
|
|
|
|
2009-04-22 15:21:04 +02:00
|
|
|
protected:
|
2009-09-22 17:25:18 +02:00
|
|
|
void contextMenuEvent(QContextMenuEvent *e);
|
2010-07-07 13:09:30 +02:00
|
|
|
bool event(QEvent *e);
|
|
|
|
|
void wheelEvent(QWheelEvent *event);
|
2009-09-22 17:25:18 +02:00
|
|
|
TextEditor::BaseTextEditorEditable *createEditableInterface();
|
2010-01-15 17:20:03 +01:00
|
|
|
void createToolBar(QmlJSEditorEditable *editable);
|
2009-09-11 14:42:50 +02:00
|
|
|
TextEditor::BaseTextEditor::Link findLinkAt(const QTextCursor &cursor, bool resolveTarget = true);
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2010-01-12 14:13:43 +01:00
|
|
|
//// brace matching
|
|
|
|
|
virtual bool contextAllowsAutoParentheses(const QTextCursor &cursor, const QString &textToInsert = QString()) const;
|
2010-06-15 12:07:15 +02:00
|
|
|
virtual bool contextAllowsElectricCharacters(const QTextCursor &cursor) const;
|
2010-01-12 14:13:43 +01:00
|
|
|
virtual bool isInComment(const QTextCursor &cursor) const;
|
2010-06-14 16:06:42 +02:00
|
|
|
virtual QString insertMatchingBrace(const QTextCursor &tc, const QString &text, QChar la, int *skippedChars) const;
|
2010-01-12 14:13:43 +01:00
|
|
|
virtual QString insertParagraphSeparator(const QTextCursor &tc) const;
|
|
|
|
|
|
2009-04-22 15:21:04 +02:00
|
|
|
private:
|
2010-06-14 16:06:42 +02:00
|
|
|
virtual bool isElectricCharacter(QChar ch) const;
|
2009-09-22 17:25:18 +02:00
|
|
|
virtual void indentBlock(QTextDocument *doc, QTextBlock block, QChar typedChar);
|
2010-01-19 12:54:17 +01:00
|
|
|
bool isClosingBrace(const QList<QmlJS::Token> &tokens) const;
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2009-09-22 17:25:18 +02:00
|
|
|
QString wordUnderCursor() const;
|
2009-05-05 15:33:39 +02:00
|
|
|
|
2010-02-16 10:36:09 +01:00
|
|
|
SemanticHighlighter::Source currentSource(bool force = false);
|
2010-07-12 14:45:22 +02:00
|
|
|
QModelIndex indexForPosition(unsigned cursorPosition, const QModelIndex &rootIndex = QModelIndex()) const;
|
2010-02-16 10:36:09 +01:00
|
|
|
|
2010-06-25 12:56:16 +02:00
|
|
|
const Core::Context m_context;
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2009-09-22 17:25:18 +02:00
|
|
|
QTimer *m_updateDocumentTimer;
|
2010-01-14 16:30:48 +01:00
|
|
|
QTimer *m_updateUsesTimer;
|
2010-04-01 15:47:52 +02:00
|
|
|
QTimer *m_semanticRehighlightTimer;
|
2010-07-12 17:03:49 +02:00
|
|
|
QTimer *m_updateMethodBoxTimer;
|
2009-09-22 17:25:18 +02:00
|
|
|
QComboBox *m_methodCombo;
|
2010-07-12 14:45:22 +02:00
|
|
|
QmlOutlineModel *m_outlineModel;
|
|
|
|
|
QModelIndex m_outlineModelIndex;
|
2010-06-09 15:56:03 +02:00
|
|
|
QmlJS::ModelManagerInterface *m_modelManager;
|
2010-01-14 16:30:48 +01:00
|
|
|
QTextCharFormat m_occurrencesFormat;
|
2010-02-15 12:27:25 +01:00
|
|
|
QTextCharFormat m_occurrencesUnusedFormat;
|
|
|
|
|
QTextCharFormat m_occurrenceRenameFormat;
|
2010-01-25 14:18:53 +01:00
|
|
|
|
2010-02-16 10:36:09 +01:00
|
|
|
SemanticHighlighter *m_semanticHighlighter;
|
2010-01-25 14:18:53 +01:00
|
|
|
SemanticInfo m_semanticInfo;
|
2010-07-07 13:09:30 +02:00
|
|
|
|
|
|
|
|
QmlJS::IContextPane *m_contextPane;
|
|
|
|
|
int m_oldCurserPosition;
|
2009-04-22 15:21:04 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace Internal
|
2010-01-15 17:20:03 +01:00
|
|
|
} // namespace QmlJSEditor
|
2009-04-22 15:21:04 +02:00
|
|
|
|
2010-01-15 17:20:03 +01:00
|
|
|
#endif // QMLJSEDITOR_H
|