forked from qt-creator/qt-creator
Conflicts: src/libs/utils/ipaddresslineedit.cpp src/libs/utils/logging.h src/plugins/analyzerbase/AnalyzerBase.pluginspec.in src/plugins/android/Android.pluginspec.in src/plugins/android/androiddeploystep.cpp src/plugins/android/androiddeploystep.h src/plugins/android/androiddeploystepfactory.cpp src/plugins/android/androiddeploystepwidget.cpp src/plugins/android/androidpackagecreationfactory.cpp src/plugins/android/androidpackagecreationstep.cpp src/plugins/android/androidpackagecreationstep.h src/plugins/android/androidpackagecreationwidget.cpp src/plugins/android/androidpackagecreationwidget.h src/plugins/android/javafilewizard.cpp src/plugins/autotoolsprojectmanager/AutotoolsProjectManager.pluginspec.in src/plugins/baremetal/BareMetal.pluginspec.in src/plugins/bazaar/Bazaar.pluginspec.in src/plugins/beautifier/Beautifier.pluginspec.in src/plugins/bineditor/BinEditor.pluginspec.in src/plugins/bookmarks/Bookmarks.pluginspec.in src/plugins/clangcodemodel/ClangCodeModel.pluginspec.in src/plugins/clangcodemodel/clanghighlightingsupport.cpp src/plugins/clangcodemodel/clangsymbolsearcher.cpp src/plugins/classview/ClassView.pluginspec.in src/plugins/clearcase/ClearCase.pluginspec.in src/plugins/cmakeprojectmanager/CMakeProjectManager.pluginspec.in src/plugins/cmakeprojectmanager/cmakeeditorfactory.cpp src/plugins/cmakeprojectmanager/cmakehighlighter.cpp src/plugins/coreplugin/Core.pluginspec.in src/plugins/cpaster/CodePaster.pluginspec.in src/plugins/cppeditor/CppEditor.pluginspec.in src/plugins/cppeditor/cppfilewizard.cpp src/plugins/cpptools/CppTools.pluginspec.in src/plugins/cpptools/cpphighlightingsupportinternal.cpp src/plugins/cpptools/cppmodelmanagerinterface.cpp src/plugins/cpptools/cppmodelmanagerinterface.h src/plugins/cvs/CVS.pluginspec.in src/plugins/debugger/Debugger.pluginspec.in src/plugins/designer/Designer.pluginspec.in src/plugins/diffeditor/DiffEditor.pluginspec.in src/plugins/emacskeys/EmacsKeys.pluginspec.in src/plugins/fakevim/FakeVim.pluginspec.in src/plugins/genericprojectmanager/GenericProjectManager.pluginspec.in src/plugins/git/Git.pluginspec.in src/plugins/git/gitorious/gitorious.cpp src/plugins/git/gitorious/gitorious.h src/plugins/git/gitorious/gitoriousclonewizard.cpp src/plugins/git/gitorious/gitorioushostwidget.cpp src/plugins/git/gitorious/gitorioushostwidget.h src/plugins/git/gitorious/gitorioushostwizardpage.cpp src/plugins/git/gitorious/gitoriousprojectwidget.cpp src/plugins/git/gitorious/gitoriousprojectwidget.h src/plugins/git/gitorious/gitoriousprojectwizardpage.cpp src/plugins/git/gitorious/gitoriousprojectwizardpage.h src/plugins/git/gitorious/gitoriousrepositorywizardpage.cpp src/plugins/git/gitorious/gitoriousrepositorywizardpage.h src/plugins/glsleditor/GLSLEditor.pluginspec.in src/plugins/glsleditor/glsleditorfactory.cpp src/plugins/glsleditor/glslfilewizard.cpp src/plugins/helloworld/HelloWorld.pluginspec.in src/plugins/help/Help.pluginspec.in src/plugins/imageviewer/ImageViewer.pluginspec.in src/plugins/ios/Ios.pluginspec.in src/plugins/macros/Macros.pluginspec.in src/plugins/mercurial/Mercurial.pluginspec.in src/plugins/perforce/Perforce.pluginspec.in src/plugins/projectexplorer/ProjectExplorer.pluginspec.in src/plugins/pythoneditor/PythonEditor.pluginspec.in src/plugins/pythoneditor/pythoneditorwidget.cpp src/plugins/pythoneditor/wizard/pythonfilewizard.cpp src/plugins/qbsprojectmanager/QbsProjectManager.pluginspec.in src/plugins/qbsprojectmanager/defaultpropertyprovider.cpp src/plugins/qmakeprojectmanager/QmakeProjectManager.pluginspec.in src/plugins/qmakeprojectmanager/profileeditorfactory.cpp src/plugins/qmldesigner/QmlDesigner.pluginspec.in src/plugins/qmljseditor/QmlJSEditor.pluginspec.in src/plugins/qmljseditor/qmljseditorfactory.cpp src/plugins/qmljstools/QmlJSTools.pluginspec.in src/plugins/qmlprofiler/QmlProfiler.pluginspec.in src/plugins/qmlprojectmanager/QmlProjectManager.pluginspec.in src/plugins/qnx/Qnx.pluginspec.in src/plugins/qtsupport/QtSupport.pluginspec.in src/plugins/remotelinux/RemoteLinux.pluginspec.in src/plugins/resourceeditor/ResourceEditor.pluginspec.in src/plugins/resourceeditor/resourcewizard.h src/plugins/subversion/Subversion.pluginspec.in src/plugins/tasklist/TaskList.pluginspec.in src/plugins/texteditor/TextEditor.pluginspec.in src/plugins/texteditor/basetexteditor_p.h src/plugins/texteditor/basetextmark.cpp src/plugins/texteditor/codeassist/basicproposalitemlistmodel.h src/plugins/texteditor/codeassist/defaultassistinterface.h src/plugins/texteditor/codeassist/iassistproposalitem.cpp src/plugins/texteditor/itexteditor.cpp src/plugins/texteditor/itexteditor.h src/plugins/texteditor/itextmark.cpp src/plugins/texteditor/plaintexteditor.cpp src/plugins/texteditor/plaintexteditor.h src/plugins/texteditor/texteditoractionhandler.cpp src/plugins/todo/Todo.pluginspec.in src/plugins/updateinfo/UpdateInfo.pluginspec.in src/plugins/valgrind/Valgrind.pluginspec.in src/plugins/vcsbase/VcsBase.pluginspec.in src/plugins/welcome/Welcome.pluginspec.in src/plugins/winrt/WinRt.pluginspec.in tests/auto/debugger/temporarydir.h Change-Id: I254af8be8119fe9855287909e17d4b8ca9d2fc2f
338 lines
9.9 KiB
C++
338 lines
9.9 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
|
** Contact: http://www.qt-project.org/legal
|
|
**
|
|
** 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
|
|
** a written agreement between you and Digia. For licensing terms and
|
|
** conditions see http://www.qt.io/licensing. For further information
|
|
** use the contact form at http://www.qt.io/contact-us.
|
|
**
|
|
** 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 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.
|
|
**
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#ifndef QMLJSCODEFORMATTER_H
|
|
#define QMLJSCODEFORMATTER_H
|
|
|
|
#include "qmljs_global.h"
|
|
|
|
#include "qmljsscanner.h"
|
|
|
|
#include <QStack>
|
|
#include <QList>
|
|
#include <QVector>
|
|
#include <QMetaObject>
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
class QTextDocument;
|
|
class QTextBlock;
|
|
QT_END_NAMESPACE
|
|
|
|
namespace QmlJS {
|
|
|
|
class QMLJS_EXPORT CodeFormatter
|
|
{
|
|
Q_GADGET
|
|
public:
|
|
CodeFormatter();
|
|
virtual ~CodeFormatter();
|
|
|
|
// updates all states up until block if necessary
|
|
// it is safe to call indentFor on block afterwards
|
|
void updateStateUntil(const QTextBlock &block);
|
|
|
|
// calculates the state change introduced by changing a single line
|
|
void updateLineStateChange(const QTextBlock &block);
|
|
|
|
int indentFor(const QTextBlock &block);
|
|
int indentForNewLineAfter(const QTextBlock &block);
|
|
|
|
void setTabSize(int tabSize);
|
|
|
|
void invalidateCache(QTextDocument *document);
|
|
|
|
protected:
|
|
virtual void onEnter(int newState, int *indentDepth, int *savedIndentDepth) const = 0;
|
|
virtual void adjustIndent(const QList<Token> &tokens, int startLexerState, int *indentDepth) const = 0;
|
|
|
|
struct State;
|
|
class QMLJS_EXPORT BlockData
|
|
{
|
|
public:
|
|
BlockData();
|
|
|
|
QStack<State> m_beginState;
|
|
QStack<State> m_endState;
|
|
int m_indentDepth;
|
|
int m_blockRevision;
|
|
};
|
|
|
|
virtual void saveBlockData(QTextBlock *block, const BlockData &data) const = 0;
|
|
virtual bool loadBlockData(const QTextBlock &block, BlockData *data) const = 0;
|
|
|
|
virtual void saveLexerState(QTextBlock *block, int state) const = 0;
|
|
virtual int loadLexerState(const QTextBlock &block) const = 0;
|
|
|
|
public: // must be public to make Q_GADGET introspection work
|
|
enum StateType {
|
|
invalid = 0,
|
|
|
|
topmost_intro, // The first line in a "topmost" definition.
|
|
|
|
top_qml, // root state for qml
|
|
top_js, // root for js
|
|
objectdefinition_or_js, // file starts with identifier
|
|
|
|
multiline_comment_start,
|
|
multiline_comment_cont,
|
|
|
|
import_start, // after 'import'
|
|
import_maybe_dot_or_version_or_as, // after string or identifier
|
|
import_dot, // after .
|
|
import_maybe_as, // after version
|
|
import_as,
|
|
|
|
property_start, // after 'property'
|
|
default_property_start, // after 'default'
|
|
property_list_open, // after 'list' as a type
|
|
property_name, // after the type
|
|
property_maybe_initializer, // after the identifier
|
|
|
|
signal_start, // after 'signal'
|
|
signal_maybe_arglist, // after identifier
|
|
signal_arglist_open, // after '('
|
|
|
|
function_start, // after 'function'
|
|
function_arglist_open, // after '(' starting function argument list
|
|
function_arglist_closed, // after ')' in argument list, expecting '{'
|
|
|
|
binding_or_objectdefinition, // after an identifier
|
|
|
|
binding_assignment, // after : in a binding
|
|
objectdefinition_open, // after {
|
|
|
|
expression,
|
|
expression_continuation, // at the end of the line, when the next line definitely is a continuation
|
|
expression_maybe_continuation, // at the end of the line, when the next line may be an expression
|
|
expression_or_objectdefinition, // after a binding starting with an identifier ("x: foo")
|
|
expression_or_label, // when expecting a statement and getting an identifier
|
|
|
|
paren_open, // opening ( in expression
|
|
bracket_open, // opening [ in expression
|
|
objectliteral_open, // opening { in expression
|
|
|
|
objectliteral_assignment, // after : in object literal
|
|
|
|
bracket_element_start, // after starting bracket_open or after ',' in bracket_open
|
|
bracket_element_maybe_objectdefinition, // after an identifier in bracket_element_start
|
|
|
|
ternary_op, // The ? : operator
|
|
ternary_op_after_colon, // after the : in a ternary
|
|
|
|
jsblock_open,
|
|
|
|
empty_statement, // for a ';', will be popped directly
|
|
breakcontinue_statement, // for continue/break, may be followed by identifier
|
|
|
|
if_statement, // After 'if'
|
|
maybe_else, // after the first substatement in an if
|
|
else_clause, // The else line of an if-else construct.
|
|
|
|
condition_open, // Start of a condition in 'if', 'while', entered after opening paren
|
|
|
|
substatement, // The first line after a conditional or loop construct.
|
|
substatement_open, // The brace that opens a substatement block.
|
|
|
|
labelled_statement, // after a label
|
|
|
|
return_statement, // After 'return'
|
|
throw_statement, // After 'throw'
|
|
|
|
statement_with_condition, // After the 'for', 'while', ... token
|
|
statement_with_condition_paren_open, // While inside the (...)
|
|
|
|
try_statement, // after 'try'
|
|
catch_statement, // after 'catch', nested in try_statement
|
|
finally_statement, // after 'finally', nested in try_statement
|
|
maybe_catch_or_finally, // after ther closing '}' of try_statement and catch_statement, nested in try_statement
|
|
|
|
do_statement, // after 'do'
|
|
do_statement_while_paren_open, // after '(' in while clause
|
|
|
|
switch_statement, // After 'switch' token
|
|
case_start, // after a 'case' or 'default' token
|
|
case_cont // after the colon in a case/default
|
|
};
|
|
Q_ENUMS(StateType)
|
|
|
|
protected:
|
|
// extends Token::Kind from qmljsscanner.h
|
|
// the entries until EndOfExistingTokenKinds must match
|
|
enum TokenKind {
|
|
EndOfFile,
|
|
Keyword,
|
|
Identifier,
|
|
String,
|
|
Comment,
|
|
Number,
|
|
LeftParenthesis,
|
|
RightParenthesis,
|
|
LeftBrace,
|
|
RightBrace,
|
|
LeftBracket,
|
|
RightBracket,
|
|
Semicolon,
|
|
Colon,
|
|
Comma,
|
|
Dot,
|
|
Delimiter,
|
|
|
|
EndOfExistingTokenKinds,
|
|
|
|
Break,
|
|
Case,
|
|
Catch,
|
|
Continue,
|
|
Debugger,
|
|
Default,
|
|
Delete,
|
|
Do,
|
|
Else,
|
|
Finally,
|
|
For,
|
|
Function,
|
|
If,
|
|
In,
|
|
Instanceof,
|
|
New,
|
|
Return,
|
|
Switch,
|
|
This,
|
|
Throw,
|
|
Try,
|
|
Typeof,
|
|
Var,
|
|
Void,
|
|
While,
|
|
With,
|
|
|
|
Import,
|
|
Signal,
|
|
On,
|
|
As,
|
|
List,
|
|
Property,
|
|
|
|
Question,
|
|
PlusPlus,
|
|
MinusMinus
|
|
};
|
|
|
|
TokenKind extendedTokenKind(const QmlJS::Token &token) const;
|
|
|
|
struct State {
|
|
State()
|
|
: savedIndentDepth(0)
|
|
, type(0)
|
|
{}
|
|
|
|
State(quint8 ty, quint16 savedDepth)
|
|
: savedIndentDepth(savedDepth)
|
|
, type(ty)
|
|
{}
|
|
|
|
quint16 savedIndentDepth;
|
|
quint8 type;
|
|
|
|
bool operator==(const State &other) const {
|
|
return type == other.type
|
|
&& savedIndentDepth == other.savedIndentDepth;
|
|
}
|
|
};
|
|
|
|
State state(int belowTop = 0) const;
|
|
const QVector<State> &newStatesThisLine() const;
|
|
int tokenIndex() const;
|
|
int tokenCount() const;
|
|
const Token ¤tToken() const;
|
|
const Token &tokenAt(int idx) const;
|
|
int column(int position) const;
|
|
|
|
bool isBracelessState(int type) const;
|
|
bool isExpressionEndState(int type) const;
|
|
|
|
void dump() const;
|
|
QString stateToString(int type) const;
|
|
|
|
private:
|
|
void recalculateStateAfter(const QTextBlock &block);
|
|
void saveCurrentState(const QTextBlock &block);
|
|
void restoreCurrentState(const QTextBlock &block);
|
|
|
|
QStringRef currentTokenText() const;
|
|
|
|
int tokenizeBlock(const QTextBlock &block);
|
|
|
|
void turnInto(int newState);
|
|
|
|
bool tryInsideExpression(bool alsoExpression = false);
|
|
bool tryStatement();
|
|
|
|
void enter(int newState);
|
|
void leave(bool statementDone = false);
|
|
void correctIndentation(const QTextBlock &block);
|
|
|
|
private:
|
|
static QStack<State> initialState();
|
|
|
|
QStack<State> m_beginState;
|
|
QStack<State> m_currentState;
|
|
QStack<State> m_newStates;
|
|
|
|
QList<Token> m_tokens;
|
|
QString m_currentLine;
|
|
Token m_currentToken;
|
|
int m_tokenIndex;
|
|
|
|
// should store indent level and padding instead
|
|
int m_indentDepth;
|
|
|
|
int m_tabSize;
|
|
};
|
|
|
|
class QMLJS_EXPORT QtStyleCodeFormatter : public CodeFormatter
|
|
{
|
|
public:
|
|
QtStyleCodeFormatter();
|
|
|
|
void setIndentSize(int size);
|
|
|
|
protected:
|
|
void onEnter(int newState, int *indentDepth, int *savedIndentDepth) const Q_DECL_OVERRIDE;
|
|
void adjustIndent(const QList<QmlJS::Token> &tokens, int lexerState, int *indentDepth) const Q_DECL_OVERRIDE;
|
|
|
|
private:
|
|
int m_indentSize;
|
|
};
|
|
|
|
} // namespace QmlJS
|
|
|
|
#endif // QMLJSCODEFORMATTER_H
|