Files
qt-creator/src/plugins/cppeditor/cpphighlighter.cpp

459 lines
16 KiB
C++
Raw Normal View History

/**************************************************************************
2008-12-02 12:01:29 +01:00
**
** This file is part of Qt Creator
**
** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies).
2008-12-02 12:01:29 +01:00
**
** Contact: Nokia Corporation (qt-info@nokia.com)
2008-12-02 12:01:29 +01: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-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
** Nokia at qt-info@nokia.com.
2008-12-02 12:01:29 +01:00
**
**************************************************************************/
2008-12-02 15:08:31 +01:00
2008-12-02 12:01:29 +01:00
#include "cpphighlighter.h"
2009-02-20 12:55:01 +01:00
#include <cpptools/cppdoxygen.h>
2008-12-02 12:01:29 +01:00
#include <Token.h>
#include <cplusplus/SimpleLexer.h>
#include <texteditor/basetextdocumentlayout.h>
2008-12-02 12:01:29 +01:00
#include <QTextDocument>
#include <QDebug>
2008-12-02 12:01:29 +01:00
using namespace CppEditor::Internal;
using namespace TextEditor;
using namespace CPlusPlus;
CppHighlighter::CppHighlighter(QTextDocument *document) :
TextEditor::SyntaxHighlighter(document)
2008-12-02 12:01:29 +01:00
{
}
void CppHighlighter::highlightBlock(const QString &text)
2008-12-02 12:01:29 +01:00
{
const int previousState = previousBlockState();
int state = 0, initialBraceDepth = 0;
2008-12-02 12:01:29 +01:00
if (previousState != -1) {
state = previousState & 0xff;
initialBraceDepth = previousState >> 8;
2008-12-02 12:01:29 +01:00
}
int braceDepth = initialBraceDepth;
2008-12-02 12:01:29 +01:00
SimpleLexer tokenize;
tokenize.setQtMocRunEnabled(false);
tokenize.setObjCEnabled(false);
tokenize.setCxx0xEnabled(true);
2008-12-02 12:01:29 +01:00
int initialState = state;
2010-06-29 17:57:15 +02:00
const QList<Token> tokens = tokenize(text, initialState);
2008-12-02 12:01:29 +01:00
state = tokenize.state(); // refresh the state
int foldingIndent = initialBraceDepth;
if (TextBlockUserData *userData = BaseTextDocumentLayout::testUserData(currentBlock())) {
userData->setFoldingIndent(0);
userData->setFoldingStartIncluded(false);
userData->setFoldingEndIncluded(false);
}
2008-12-02 12:01:29 +01:00
if (tokens.isEmpty()) {
setCurrentBlockState(previousState);
BaseTextDocumentLayout::clearParentheses(currentBlock());
if (text.length()) {// the empty line can still contain whitespace
if (!initialState)
setFormat(0, text.length(), m_formats[CppVisualWhitespace]);
else
setFormat(0, text.length(), m_formats[CppCommentFormat]);
}
BaseTextDocumentLayout::setFoldingIndent(currentBlock(), foldingIndent);
2008-12-02 12:01:29 +01:00
return;
}
2010-06-29 17:57:15 +02:00
const unsigned firstNonSpace = tokens.first().begin();
2008-12-02 12:01:29 +01:00
Parentheses parentheses;
parentheses.reserve(20); // assume wizard level ;-)
bool expectPreprocessorKeyword = false;
bool onlyHighlightComments = false;
2008-12-02 12:01:29 +01:00
for (int i = 0; i < tokens.size(); ++i) {
2010-06-29 17:57:15 +02:00
const Token &tk = tokens.at(i);
2008-12-02 12:01:29 +01:00
2010-06-29 17:57:15 +02:00
unsigned previousTokenEnd = 0;
2008-12-02 12:01:29 +01:00
if (i != 0) {
// mark the whitespaces
2010-06-29 17:57:15 +02:00
previousTokenEnd = tokens.at(i - 1).begin() +
2008-12-02 12:01:29 +01:00
tokens.at(i - 1).length();
}
2010-06-29 17:57:15 +02:00
if (previousTokenEnd != tk.begin()) {
if (initialState && tk.isComment())
setFormat(previousTokenEnd, tk.begin() - previousTokenEnd, m_formats[CppCommentFormat]);
else
setFormat(previousTokenEnd, tk.begin() - previousTokenEnd, m_formats[CppVisualWhitespace]);
2008-12-02 12:01:29 +01:00
}
if (tk.is(T_LPAREN) || tk.is(T_LBRACE) || tk.is(T_LBRACKET)) {
2010-06-29 17:57:15 +02:00
const QChar c = text.at(tk.begin());
parentheses.append(Parenthesis(Parenthesis::Opened, c, tk.begin()));
if (tk.is(T_LBRACE)) {
2008-12-02 12:01:29 +01:00
++braceDepth;
// if a folding block opens at the beginning of a line, treat the entire line
// as if it were inside the folding block
2010-06-29 17:57:15 +02:00
if (tk.begin() == firstNonSpace) {
++foldingIndent;
BaseTextDocumentLayout::userData(currentBlock())->setFoldingStartIncluded(true);
}
}
2008-12-02 12:01:29 +01:00
} else if (tk.is(T_RPAREN) || tk.is(T_RBRACE) || tk.is(T_RBRACKET)) {
2010-06-29 17:57:15 +02:00
const QChar c = text.at(tk.begin());
parentheses.append(Parenthesis(Parenthesis::Closed, c, tk.begin()));
if (tk.is(T_RBRACE)) {
--braceDepth;
if (braceDepth < foldingIndent) {
// unless we are at the end of the block, we reduce the folding indent
if (i == tokens.size()-1 || tokens.at(i+1).is(T_SEMICOLON))
BaseTextDocumentLayout::userData(currentBlock())->setFoldingEndIncluded(true);
else
foldingIndent = qMin(braceDepth, foldingIndent);
}
}
2008-12-02 12:01:29 +01:00
}
bool highlightCurrentWordAsPreprocessor = expectPreprocessorKeyword;
2009-02-20 11:52:27 +01:00
if (expectPreprocessorKeyword)
expectPreprocessorKeyword = false;
if (onlyHighlightComments && !tk.isComment())
continue;
2008-12-02 12:01:29 +01:00
if (i == 0 && tk.is(T_POUND)) {
2010-06-29 17:57:15 +02:00
highlightLine(text, tk.begin(), tk.length(), m_formats[CppPreprocessorFormat]);
expectPreprocessorKeyword = true;
2008-12-02 12:01:29 +01:00
} else if (highlightCurrentWordAsPreprocessor &&
(tk.isKeyword() || tk.is(T_IDENTIFIER)) && isPPKeyword(text.midRef(tk.begin(), tk.length()))) {
2010-06-29 17:57:15 +02:00
setFormat(tk.begin(), tk.length(), m_formats[CppPreprocessorFormat]);
const QStringRef ppKeyword = text.midRef(tk.begin(), tk.length());
if (ppKeyword == QLatin1String("error")
|| ppKeyword == QLatin1String("warning")
|| ppKeyword == QLatin1String("pragma")) {
onlyHighlightComments = true;
}
2009-02-20 11:52:27 +01:00
} else if (tk.is(T_NUMERIC_LITERAL))
2010-06-29 17:57:15 +02:00
setFormat(tk.begin(), tk.length(), m_formats[CppNumberFormat]);
2009-02-20 11:52:27 +01:00
else if (tk.is(T_STRING_LITERAL) || tk.is(T_CHAR_LITERAL) || tk.is(T_ANGLE_STRING_LITERAL) ||
tk.is(T_AT_STRING_LITERAL))
setFormat(tk.begin(), tk.length(), m_formats[CppStringFormat]);
2009-02-20 11:52:27 +01:00
2008-12-02 12:01:29 +01:00
else if (tk.is(T_WIDE_STRING_LITERAL) || tk.is(T_WIDE_CHAR_LITERAL))
setFormat(tk.begin(), tk.length(), m_formats[CppStringFormat]);
2009-02-20 11:52:27 +01:00
else if (tk.isComment()) {
if (tk.is(T_COMMENT) || tk.is(T_CPP_COMMENT))
setFormat(tk.begin(), tk.length(), m_formats[CppCommentFormat]);
2009-02-20 11:52:27 +01:00
else // a doxygen comment
2010-06-29 17:57:15 +02:00
highlightDoxygenComment(text, tk.begin(), tk.length());
2009-02-20 11:52:27 +01:00
2008-12-02 12:01:29 +01:00
// we need to insert a close comment parenthesis, if
// - the line starts in a C Comment (initalState != 0)
// - the first token of the line is a T_COMMENT (i == 0 && tk.is(T_COMMENT))
// - is not a continuation line (tokens.size() > 1 || ! state)
if (initialState && i == 0 && (tokens.size() > 1 || ! state)) {
--braceDepth;
// unless we are at the end of the block, we reduce the folding indent
if (i == tokens.size()-1)
BaseTextDocumentLayout::userData(currentBlock())->setFoldingEndIncluded(true);
else
foldingIndent = qMin(braceDepth, foldingIndent);
2010-06-29 17:57:15 +02:00
const int tokenEnd = tk.begin() + tk.length() - 1;
2008-12-02 12:01:29 +01:00
parentheses.append(Parenthesis(Parenthesis::Closed, QLatin1Char('-'), tokenEnd));
// clear the initial state.
initialState = 0;
}
2009-02-20 11:52:27 +01:00
2010-06-29 17:57:15 +02:00
} else if (tk.isKeyword() || isQtKeyword(text.midRef(tk.begin(), tk.length())) || tk.isObjCAtKeyword())
setFormat(tk.begin(), tk.length(), m_formats[CppKeywordFormat]);
2009-02-20 11:52:27 +01:00
2008-12-02 12:01:29 +01:00
else if (tk.isOperator())
2010-06-29 17:57:15 +02:00
setFormat(tk.begin(), tk.length(), m_formats[CppOperatorFormat]);
2009-02-20 11:52:27 +01:00
2008-12-02 12:01:29 +01:00
else if (i == 0 && tokens.size() > 1 && tk.is(T_IDENTIFIER) && tokens.at(1).is(T_COLON))
2010-06-29 17:57:15 +02:00
setFormat(tk.begin(), tk.length(), m_formats[CppLabelFormat]);
2009-02-20 11:52:27 +01:00
2008-12-02 12:01:29 +01:00
else if (tk.is(T_IDENTIFIER))
2010-06-29 17:57:15 +02:00
highlightWord(text.midRef(tk.begin(), tk.length()), tk.begin(), tk.length());
2008-12-02 12:01:29 +01:00
}
// mark the trailing white spaces
{
2010-06-29 17:57:15 +02:00
const Token tk = tokens.last();
const int lastTokenEnd = tk.begin() + tk.length();
2008-12-02 12:01:29 +01:00
if (text.length() > lastTokenEnd)
highlightLine(text, lastTokenEnd, text.length() - lastTokenEnd, QTextCharFormat());
2008-12-02 12:01:29 +01:00
}
if (! initialState && state && ! tokens.isEmpty()) {
parentheses.append(Parenthesis(Parenthesis::Opened, QLatin1Char('+'),
2010-06-29 17:57:15 +02:00
tokens.last().begin()));
2008-12-02 12:01:29 +01:00
++braceDepth;
}
BaseTextDocumentLayout::setParentheses(currentBlock(), parentheses);
2008-12-02 12:01:29 +01:00
// if the block is ifdefed out, we only store the parentheses, but
// do not adjust the brace depth.
if (BaseTextDocumentLayout::ifdefedOut(currentBlock())) {
braceDepth = initialBraceDepth;
foldingIndent = initialBraceDepth;
}
BaseTextDocumentLayout::setFoldingIndent(currentBlock(), foldingIndent);
// optimization: if only the brace depth changes, we adjust subsequent blocks
// to have QSyntaxHighlighter stop the rehighlighting
int currentState = currentBlockState();
if (currentState != -1) {
int oldState = currentState & 0xff;
int oldBraceDepth = currentState >> 8;
if (oldState == tokenize.state() && oldBraceDepth != braceDepth) {
BaseTextDocumentLayout::FoldValidator foldValidor;
foldValidor.setup(qobject_cast<BaseTextDocumentLayout *>(document()->documentLayout()));
int delta = braceDepth - oldBraceDepth;
QTextBlock block = currentBlock().next();
while (block.isValid() && block.userState() != -1) {
BaseTextDocumentLayout::changeBraceDepth(block, delta);
BaseTextDocumentLayout::changeFoldingIndent(block, delta);
foldValidor.process(block);
block = block.next();
}
foldValidor.finalize();
}
}
2008-12-02 12:01:29 +01:00
setCurrentBlockState((braceDepth << 8) | tokenize.state());
}
bool CppHighlighter::isPPKeyword(const QStringRef &text) const
2008-12-02 12:01:29 +01:00
{
switch (text.length())
{
case 2:
if (text.at(0) == QLatin1Char('i') && text.at(1) == QLatin1Char('f'))
2008-12-02 12:01:29 +01:00
return true;
break;
case 4:
if (text.at(0) == QLatin1Char('e')
&& (text == QLatin1String("elif") || text == QLatin1String("else")))
2008-12-02 12:01:29 +01:00
return true;
break;
case 5:
switch (text.at(0).toLatin1()) {
case 'i':
if (text == QLatin1String("ifdef"))
return true;
break;
case 'u':
if (text == QLatin1String("undef"))
return true;
break;
case 'e':
if (text == QLatin1String("endif") || text == QLatin1String("error"))
return true;
break;
}
2008-12-02 12:01:29 +01:00
break;
case 6:
switch (text.at(0).toLatin1()) {
case 'i':
if (text == QLatin1String("ifndef") || text == QLatin1String("import"))
return true;
break;
case 'd':
if (text == QLatin1String("define"))
return true;
break;
case 'p':
if (text == QLatin1String("pragma"))
return true;
break;
}
2008-12-02 12:01:29 +01:00
break;
case 7:
switch (text.at(0).toLatin1()) {
case 'i':
if (text == QLatin1String("include"))
return true;
break;
case 'w':
if (text == QLatin1String("warning"))
return true;
break;
}
2008-12-02 12:01:29 +01:00
break;
case 12:
if (text.at(0) == QLatin1Char('i') && text == QLatin1String("include_next"))
2008-12-02 12:01:29 +01:00
return true;
break;
default:
break;
}
return false;
}
bool CppHighlighter::isQtKeyword(const QStringRef &text) const
2008-12-02 12:01:29 +01:00
{
switch (text.length()) {
case 4:
switch (text.at(0).toLatin1()) {
case 'e':
if (text == QLatin1String("emit"))
return true;
break;
case 'S':
if (text == QLatin1String("SLOT"))
return true;
break;
}
2008-12-02 12:01:29 +01:00
break;
case 5:
if (text.at(0) == QLatin1Char('s') && text == QLatin1String("slots"))
2008-12-02 12:01:29 +01:00
return true;
break;
case 6:
if (text.at(0) == QLatin1Char('S') && text == QLatin1String("SIGNAL"))
2008-12-02 12:01:29 +01:00
return true;
break;
case 7:
switch (text.at(0).toLatin1()) {
case 's':
if (text == QLatin1String("signals"))
return true;
break;
case 'f':
if (text == QLatin1String("foreach") || text == QLatin1String("forever"))
return true;
break;
}
2008-12-02 12:01:29 +01:00
break;
default:
break;
}
return false;
}
void CppHighlighter::highlightLine(const QString &text, int position, int length,
const QTextCharFormat &format)
{
const QTextCharFormat visualSpaceFormat = m_formats[CppVisualWhitespace];
const int end = position + length;
int index = position;
while (index != end) {
const bool isSpace = text.at(index).isSpace();
const int start = index;
do { ++index; }
while (index != end && text.at(index).isSpace() == isSpace);
const int tokenLength = index - start;
if (isSpace)
setFormat(start, tokenLength, visualSpaceFormat);
else if (format.isValid())
setFormat(start, tokenLength, format);
}
}
void CppHighlighter::highlightWord(QStringRef word, int position, int length)
2008-12-02 12:01:29 +01:00
{
// try to highlight Qt 'identifiers' like QObject and Q_PROPERTY
2010-05-25 14:53:21 +02:00
if (word.length() > 2 && word.at(0) == QLatin1Char('Q')) {
if (word.at(1) == QLatin1Char('_') // Q_
|| (word.at(1) == QLatin1Char('T') && word.at(2) == QLatin1Char('_'))) { // QT_
for (int i = 1; i < word.length(); ++i) {
const QChar &ch = word.at(i);
if (! (ch.isUpper() || ch == QLatin1Char('_')))
return;
}
2010-05-25 14:53:21 +02:00
setFormat(position, length, m_formats[CppTypeFormat]);
}
2008-12-02 12:01:29 +01:00
}
}
2009-02-20 11:52:27 +01:00
2009-02-20 11:53:32 +01:00
void CppHighlighter::highlightDoxygenComment(const QString &text, int position, int)
2009-02-20 11:52:27 +01:00
{
int initial = position;
const QChar *uc = text.unicode();
const QChar *it = uc + position;
2009-02-20 12:08:34 +01:00
const QTextCharFormat &format = m_formats[CppDoxygenCommentFormat];
const QTextCharFormat &kwFormat = m_formats[CppDoxygenTagFormat];
2009-02-20 11:52:27 +01:00
while (! it->isNull()) {
if (it->unicode() == QLatin1Char('\\') ||
it->unicode() == QLatin1Char('@')) {
++it;
const QChar *start = it;
while (it->isLetterOrNumber() || it->unicode() == '_')
++it;
2009-02-20 12:55:01 +01:00
int k = CppTools::classifyDoxygenTag(start, it - start);
if (k != CppTools::T_DOXY_IDENTIFIER) {
highlightLine(text, initial, start - uc - initial, format);
2009-02-20 11:52:27 +01:00
setFormat(start - uc - 1, it - start + 1, kwFormat);
initial = it - uc;
}
} else
++it;
}
highlightLine(text, initial, it - uc - initial, format);
2009-02-20 11:52:27 +01:00
}