2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2011-01-11 16:28:15 +01:00
|
|
|
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2011-04-13 08:42:33 +02:00
|
|
|
** Contact: Nokia Corporation (info@qt.nokia.com)
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
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
|
2011-05-06 15:05:37 +02:00
|
|
|
** Nokia at info@qt.nokia.com.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +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>
|
2010-04-26 14:02:09 +02:00
|
|
|
#include <texteditor/basetextdocumentlayout.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
#include <QtGui/QTextDocument>
|
|
|
|
|
#include <QtCore/QDebug>
|
|
|
|
|
|
|
|
|
|
using namespace CppEditor::Internal;
|
|
|
|
|
using namespace TextEditor;
|
|
|
|
|
using namespace CPlusPlus;
|
|
|
|
|
|
2008-12-16 13:19:11 +01:00
|
|
|
CppHighlighter::CppHighlighter(QTextDocument *document) :
|
2010-07-09 14:47:18 +02:00
|
|
|
TextEditor::SyntaxHighlighter(document)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 13:19:11 +01:00
|
|
|
void CppHighlighter::highlightBlock(const QString &text)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
const int previousState = previousBlockState();
|
2009-09-25 17:22:10 +02:00
|
|
|
int state = 0, initialBraceDepth = 0;
|
2008-12-02 12:01:29 +01:00
|
|
|
if (previousState != -1) {
|
|
|
|
|
state = previousState & 0xff;
|
2009-09-25 17:22:10 +02:00
|
|
|
initialBraceDepth = previousState >> 8;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2009-09-25 17:22:10 +02:00
|
|
|
int braceDepth = initialBraceDepth;
|
2009-05-06 16:48:51 +02:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
SimpleLexer tokenize;
|
|
|
|
|
tokenize.setQtMocRunEnabled(false);
|
2009-11-11 16:46:33 +01:00
|
|
|
tokenize.setObjCEnabled(false);
|
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
|
|
|
|
|
|
2010-05-20 15:10:26 +02:00
|
|
|
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);
|
2010-04-26 14:06:29 +02:00
|
|
|
BaseTextDocumentLayout::clearParentheses(currentBlock());
|
2009-09-09 14:58:47 +02:00
|
|
|
if (text.length()) // the empty line can still contain whitespace
|
2009-10-01 13:34:02 +02:00
|
|
|
setFormat(0, text.length(), m_formats[CppVisualWhitespace]);
|
2010-05-20 15:10:26 +02:00
|
|
|
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 ;-)
|
|
|
|
|
|
2011-03-17 11:03:08 +01:00
|
|
|
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()) {
|
|
|
|
|
setFormat(previousTokenEnd, tk.begin() - previousTokenEnd,
|
2009-10-01 13:34:02 +02:00
|
|
|
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()));
|
2010-05-20 15:10:26 +02:00
|
|
|
if (tk.is(T_LBRACE)) {
|
2008-12-02 12:01:29 +01:00
|
|
|
++braceDepth;
|
2010-05-20 15:10:26 +02:00
|
|
|
|
|
|
|
|
// 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) {
|
2010-05-20 15:10:26 +02:00
|
|
|
++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()));
|
2010-05-20 15:10:26 +02:00
|
|
|
if (tk.is(T_RBRACE)) {
|
2009-09-16 16:16:45 +02:00
|
|
|
--braceDepth;
|
2010-05-20 15:10:26 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
|
2011-03-17 11:03:08 +01:00
|
|
|
bool highlightCurrentWordAsPreprocessor = expectPreprocessorKeyword;
|
2009-02-20 11:52:27 +01:00
|
|
|
|
2011-03-17 11:03:08 +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]);
|
2011-03-17 11:03:08 +01:00
|
|
|
expectPreprocessorKeyword = true;
|
2008-12-02 12:01:29 +01:00
|
|
|
} else if (highlightCurrentWordAsPreprocessor &&
|
2011-03-17 11:03:08 +01:00
|
|
|
(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]);
|
2011-03-17 11:03:08 +01:00
|
|
|
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
|
|
|
|
2011-03-17 11:03:08 +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
|
|
|
|
2010-02-15 12:23:48 +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))
|
2010-06-29 17:57:15 +02:00
|
|
|
highlightLine(text, 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))
|
2010-06-29 17:57:15 +02:00
|
|
|
highlightLine(text, tk.begin(), tk.length(), m_formats[CppStringFormat]);
|
2009-02-20 11:52:27 +01:00
|
|
|
|
|
|
|
|
else if (tk.isComment()) {
|
|
|
|
|
|
2009-09-29 11:34:11 +02:00
|
|
|
if (tk.is(T_COMMENT) || tk.is(T_CPP_COMMENT))
|
2010-06-29 17:57:15 +02:00
|
|
|
highlightLine(text, 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;
|
2010-05-20 15:10:26 +02:00
|
|
|
// 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());
|
2010-05-20 15:10:26 +02:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// mark the trailing white spaces
|
2009-09-09 14:58:47 +02:00
|
|
|
{
|
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)
|
2010-04-16 17:42:29 +02:00
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-26 14:06:29 +02:00
|
|
|
BaseTextDocumentLayout::setParentheses(currentBlock(), parentheses);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2009-09-25 17:22:10 +02:00
|
|
|
// if the block is ifdefed out, we only store the parentheses, but
|
2010-05-20 15:10:26 +02:00
|
|
|
|
2009-09-25 17:22:10 +02:00
|
|
|
// do not adjust the brace depth.
|
2010-05-20 15:10:26 +02:00
|
|
|
if (BaseTextDocumentLayout::ifdefedOut(currentBlock())) {
|
2009-09-25 17:22:10 +02:00
|
|
|
braceDepth = initialBraceDepth;
|
2010-05-20 15:10:26 +02:00
|
|
|
foldingIndent = initialBraceDepth;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BaseTextDocumentLayout::setFoldingIndent(currentBlock(), foldingIndent);
|
2009-05-06 16:48:51 +02:00
|
|
|
|
|
|
|
|
// 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) {
|
|
|
|
|
int delta = braceDepth - oldBraceDepth;
|
|
|
|
|
QTextBlock block = currentBlock().next();
|
2009-11-23 15:15:56 +01:00
|
|
|
while (block.isValid() && block.userState() != -1) {
|
2010-04-26 14:06:29 +02:00
|
|
|
BaseTextDocumentLayout::changeBraceDepth(block, delta);
|
2010-05-20 15:10:26 +02:00
|
|
|
BaseTextDocumentLayout::changeFoldingIndent(block, delta);
|
2009-05-06 16:48:51 +02:00
|
|
|
block = block.next();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
setCurrentBlockState((braceDepth << 8) | tokenize.state());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-12-16 13:19:11 +01:00
|
|
|
bool CppHighlighter::isPPKeyword(const QStringRef &text) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
switch (text.length())
|
|
|
|
|
{
|
|
|
|
|
case 2:
|
|
|
|
|
if (text.at(0) == 'i' && text.at(1) == 'f')
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 4:
|
|
|
|
|
if (text.at(0) == 'e' && text == QLatin1String("elif"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'e' && text == QLatin1String("else"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 5:
|
|
|
|
|
if (text.at(0) == 'i' && text == QLatin1String("ifdef"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'u' && text == QLatin1String("undef"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'e' && text == QLatin1String("endif"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'e' && text == QLatin1String("error"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 6:
|
|
|
|
|
if (text.at(0) == 'i' && text == QLatin1String("ifndef"))
|
|
|
|
|
return true;
|
2009-02-05 13:01:35 +01:00
|
|
|
if (text.at(0) == 'i' && text == QLatin1String("import"))
|
|
|
|
|
return true;
|
2008-12-02 12:01:29 +01:00
|
|
|
else if (text.at(0) == 'd' && text == QLatin1String("define"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'p' && text == QLatin1String("pragma"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 7:
|
|
|
|
|
if (text.at(0) == 'i' && text == QLatin1String("include"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'w' && text == QLatin1String("warning"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 12:
|
|
|
|
|
if (text.at(0) == 'i' && text == QLatin1String("include_next"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 13:19:11 +01:00
|
|
|
bool CppHighlighter::isQtKeyword(const QStringRef &text) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
switch (text.length()) {
|
|
|
|
|
case 4:
|
|
|
|
|
if (text.at(0) == 'e' && text == QLatin1String("emit"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'S' && text == QLatin1String("SLOT"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 5:
|
|
|
|
|
if (text.at(0) == 's' && text == QLatin1String("slots"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 6:
|
|
|
|
|
if (text.at(0) == 'S' && text == QLatin1String("SIGNAL"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 7:
|
|
|
|
|
if (text.at(0) == 's' && text == QLatin1String("signals"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'f' && text == QLatin1String("foreach"))
|
|
|
|
|
return true;
|
|
|
|
|
else if (text.at(0) == 'f' && text == QLatin1String("forever"))
|
|
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-16 17:42:29 +02:00
|
|
|
void CppHighlighter::highlightLine(const QString &text, int position, int length,
|
|
|
|
|
const QTextCharFormat &format)
|
2009-10-08 12:56:56 +02:00
|
|
|
{
|
|
|
|
|
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;
|
2010-04-16 17:39:55 +02:00
|
|
|
if (isSpace)
|
|
|
|
|
setFormat(start, tokenLength, visualSpaceFormat);
|
|
|
|
|
else if (format.isValid())
|
|
|
|
|
setFormat(start, tokenLength, format);
|
2009-10-08 12:56:56 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-16 13:19:11 +01:00
|
|
|
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
|
|
|
|
2010-06-01 10:08:00 +02:00
|
|
|
if (word.length() > 2 && word.at(0) == QLatin1Char('Q')) {
|
|
|
|
|
if (word.at(1) == QLatin1Char('_') // Q_
|
2010-06-25 14:12:01 +10:00
|
|
|
|| (word.at(1) == QLatin1Char('T') && word.at(2) == QLatin1Char('_'))) { // QT_
|
2010-06-01 10:08:00 +02:00
|
|
|
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
|
|
|
|
2010-06-01 10:08:00 +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) {
|
2010-04-16 17:42:29 +02:00
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-16 17:42:29 +02:00
|
|
|
highlightLine(text, initial, it - uc - initial, format);
|
2009-02-20 11:52:27 +01:00
|
|
|
}
|
|
|
|
|
|