2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2010-11-10 15:44:59 +01:00
|
|
|
**
|
2014-01-07 13:27:11 +01:00
|
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
2012-10-02 09:12:39 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
2010-11-10 15:44:59 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2010-11-10 15:44:59 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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://qt.digia.com/licensing. For further information
|
|
|
|
|
** use the contact form at http://qt.digia.com/contact-us.
|
2010-11-10 15:44:59 +01:00
|
|
|
**
|
|
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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.
|
|
|
|
|
**
|
|
|
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
2010-12-17 17:14:20 +01:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2010-11-10 15:44:59 +01:00
|
|
|
#include "glslhighlighter.h"
|
2010-12-01 11:49:36 +10:00
|
|
|
#include "glsleditor.h"
|
2010-11-10 15:44:59 +01:00
|
|
|
#include <glsl/glsllexer.h>
|
|
|
|
|
#include <glsl/glslparser.h>
|
2010-11-26 10:59:33 +01:00
|
|
|
#include <texteditor/basetextdocumentlayout.h>
|
2011-08-02 12:26:00 +02:00
|
|
|
#include <texteditor/basetextdocument.h>
|
2010-11-10 15:44:59 +01:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDebug>
|
2010-11-10 15:44:59 +01:00
|
|
|
|
2014-08-21 20:11:15 +02:00
|
|
|
using namespace GlslEditor;
|
|
|
|
|
using namespace GlslEditor::Internal;
|
2010-11-26 10:59:33 +01:00
|
|
|
using namespace TextEditor;
|
2010-11-10 15:44:59 +01:00
|
|
|
|
2013-08-14 13:52:13 +02:00
|
|
|
Highlighter::Highlighter(QTextDocument *parent)
|
|
|
|
|
: TextEditor::SyntaxHighlighter(parent)
|
|
|
|
|
{
|
|
|
|
|
init();
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-02 12:26:00 +02:00
|
|
|
Highlighter::Highlighter(BaseTextDocument *parent)
|
|
|
|
|
: TextEditor::SyntaxHighlighter(parent)
|
2013-08-14 13:52:13 +02:00
|
|
|
{
|
|
|
|
|
init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Highlighter::init()
|
2010-11-10 15:44:59 +01:00
|
|
|
{
|
2013-08-13 12:57:31 +02:00
|
|
|
static QVector<TextEditor::TextStyle> categories;
|
|
|
|
|
if (categories.isEmpty()) {
|
|
|
|
|
categories << TextEditor::C_NUMBER
|
|
|
|
|
<< TextEditor::C_STRING
|
|
|
|
|
<< TextEditor::C_TYPE
|
|
|
|
|
<< TextEditor::C_KEYWORD
|
|
|
|
|
<< TextEditor::C_OPERATOR
|
|
|
|
|
<< TextEditor::C_PREPROCESSOR
|
|
|
|
|
<< TextEditor::C_LABEL
|
|
|
|
|
<< TextEditor::C_COMMENT
|
|
|
|
|
<< TextEditor::C_DOXYGEN_COMMENT
|
|
|
|
|
<< TextEditor::C_DOXYGEN_TAG
|
|
|
|
|
<< TextEditor::C_VISUAL_WHITESPACE
|
|
|
|
|
<< TextEditor::C_REMOVED_LINE;
|
|
|
|
|
}
|
|
|
|
|
setTextFormatCategories(categories);
|
2010-11-10 15:44:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Highlighter::~Highlighter()
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Highlighter::highlightBlock(const QString &text)
|
|
|
|
|
{
|
2010-11-26 10:59:33 +01:00
|
|
|
const int previousState = previousBlockState();
|
|
|
|
|
int state = 0, initialBraceDepth = 0;
|
|
|
|
|
if (previousState != -1) {
|
|
|
|
|
state = previousState & 0xff;
|
|
|
|
|
initialBraceDepth = previousState >> 8;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int braceDepth = initialBraceDepth;
|
|
|
|
|
|
|
|
|
|
const QByteArray data = text.toLatin1();
|
|
|
|
|
GLSL::Lexer lex(/*engine=*/ 0, data.constData(), data.size());
|
|
|
|
|
lex.setState(state);
|
|
|
|
|
lex.setScanKeywords(false);
|
|
|
|
|
lex.setScanComments(true);
|
2014-08-22 17:16:02 +02:00
|
|
|
const int variant = languageVariant(parent()
|
2013-11-12 13:33:01 +01:00
|
|
|
? static_cast<BaseTextDocument*>(parent())->mimeType()
|
|
|
|
|
: QString());
|
2010-11-26 10:59:33 +01:00
|
|
|
lex.setVariant(variant);
|
|
|
|
|
|
|
|
|
|
int initialState = state;
|
|
|
|
|
|
|
|
|
|
QList<GLSL::Token> tokens;
|
|
|
|
|
GLSL::Token tk;
|
|
|
|
|
do {
|
|
|
|
|
lex.yylex(&tk);
|
|
|
|
|
tokens.append(tk);
|
|
|
|
|
} while (tk.isNot(GLSL::Parser::EOF_SYMBOL));
|
|
|
|
|
|
|
|
|
|
state = lex.state(); // refresh the state
|
|
|
|
|
|
|
|
|
|
int foldingIndent = initialBraceDepth;
|
|
|
|
|
if (TextBlockUserData *userData = BaseTextDocumentLayout::testUserData(currentBlock())) {
|
|
|
|
|
userData->setFoldingIndent(0);
|
|
|
|
|
userData->setFoldingStartIncluded(false);
|
|
|
|
|
userData->setFoldingEndIncluded(false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tokens.isEmpty()) {
|
|
|
|
|
setCurrentBlockState(previousState);
|
|
|
|
|
BaseTextDocumentLayout::clearParentheses(currentBlock());
|
|
|
|
|
if (text.length()) // the empty line can still contain whitespace
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(0, text.length(), formatForCategory(GLSLVisualWhitespace));
|
2010-11-26 10:59:33 +01:00
|
|
|
BaseTextDocumentLayout::setFoldingIndent(currentBlock(), foldingIndent);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const int firstNonSpace = tokens.first().begin();
|
|
|
|
|
|
|
|
|
|
Parentheses parentheses;
|
|
|
|
|
parentheses.reserve(20); // assume wizard level ;-)
|
|
|
|
|
|
|
|
|
|
bool highlightAsPreprocessor = false;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < tokens.size(); ++i) {
|
|
|
|
|
const GLSL::Token &tk = tokens.at(i);
|
|
|
|
|
|
|
|
|
|
int previousTokenEnd = 0;
|
|
|
|
|
if (i != 0) {
|
|
|
|
|
// mark the whitespaces
|
|
|
|
|
previousTokenEnd = tokens.at(i - 1).begin() +
|
|
|
|
|
tokens.at(i - 1).length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (previousTokenEnd != tk.begin()) {
|
|
|
|
|
setFormat(previousTokenEnd, tk.begin() - previousTokenEnd,
|
2013-08-13 12:57:31 +02:00
|
|
|
formatForCategory(GLSLVisualWhitespace));
|
2010-11-26 10:59:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (tk.is(GLSL::Parser::T_LEFT_PAREN) || tk.is(GLSL::Parser::T_LEFT_BRACE) || tk.is(GLSL::Parser::T_LEFT_BRACKET)) {
|
|
|
|
|
const QChar c = text.at(tk.begin());
|
|
|
|
|
parentheses.append(Parenthesis(Parenthesis::Opened, c, tk.begin()));
|
|
|
|
|
if (tk.is(GLSL::Parser::T_LEFT_BRACE)) {
|
|
|
|
|
++braceDepth;
|
|
|
|
|
|
|
|
|
|
// if a folding block opens at the beginning of a line, treat the entire line
|
|
|
|
|
// as if it were inside the folding block
|
|
|
|
|
if (tk.begin() == firstNonSpace) {
|
|
|
|
|
++foldingIndent;
|
|
|
|
|
BaseTextDocumentLayout::userData(currentBlock())->setFoldingStartIncluded(true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (tk.is(GLSL::Parser::T_RIGHT_PAREN) || tk.is(GLSL::Parser::T_RIGHT_BRACE) || tk.is(GLSL::Parser::T_RIGHT_BRACKET)) {
|
|
|
|
|
const QChar c = text.at(tk.begin());
|
|
|
|
|
parentheses.append(Parenthesis(Parenthesis::Closed, c, tk.begin()));
|
|
|
|
|
if (tk.is(GLSL::Parser::T_RIGHT_BRACE)) {
|
|
|
|
|
--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(GLSL::Parser::T_SEMICOLON))
|
|
|
|
|
BaseTextDocumentLayout::userData(currentBlock())->setFoldingEndIncluded(true);
|
|
|
|
|
else
|
|
|
|
|
foldingIndent = qMin(braceDepth, foldingIndent);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool highlightCurrentWordAsPreprocessor = highlightAsPreprocessor;
|
|
|
|
|
|
|
|
|
|
if (highlightAsPreprocessor)
|
|
|
|
|
highlightAsPreprocessor = false;
|
|
|
|
|
|
|
|
|
|
if (false /* && i == 0 && tk.is(GLSL::Parser::T_POUND)*/) {
|
2013-08-13 12:57:31 +02:00
|
|
|
highlightLine(text, tk.begin(), tk.length, formatForCategory(GLSLPreprocessorFormat));
|
2010-11-26 10:59:33 +01:00
|
|
|
highlightAsPreprocessor = true;
|
|
|
|
|
|
2013-07-17 00:01:45 +03:00
|
|
|
} else if (highlightCurrentWordAsPreprocessor && isPPKeyword(text.midRef(tk.begin(), tk.length))) {
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.begin(), tk.length, formatForCategory(GLSLPreprocessorFormat));
|
2010-11-26 10:59:33 +01:00
|
|
|
|
2013-07-17 00:01:45 +03:00
|
|
|
} else if (tk.is(GLSL::Parser::T_NUMBER)) {
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.begin(), tk.length, formatForCategory(GLSLNumberFormat));
|
2010-11-26 10:59:33 +01:00
|
|
|
|
2013-07-17 00:01:45 +03:00
|
|
|
} else if (tk.is(GLSL::Parser::T_COMMENT)) {
|
2013-08-13 12:57:31 +02:00
|
|
|
highlightLine(text, tk.begin(), tk.length, formatForCategory(GLSLCommentFormat));
|
2010-11-26 10:59:33 +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);
|
|
|
|
|
const int tokenEnd = tk.begin() + tk.length - 1;
|
|
|
|
|
parentheses.append(Parenthesis(Parenthesis::Closed, QLatin1Char('-'), tokenEnd));
|
|
|
|
|
|
|
|
|
|
// clear the initial state.
|
|
|
|
|
initialState = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else if (tk.is(GLSL::Parser::T_IDENTIFIER)) {
|
|
|
|
|
int kind = lex.findKeyword(data.constData() + tk.position, tk.length);
|
|
|
|
|
if (kind == GLSL::Parser::T_RESERVED)
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.position, tk.length, formatForCategory(GLSLReservedKeyword));
|
2010-11-26 10:59:33 +01:00
|
|
|
else if (kind != GLSL::Parser::T_IDENTIFIER)
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.position, tk.length, formatForCategory(GLSLKeywordFormat));
|
2010-11-26 10:59:33 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// mark the trailing white spaces
|
|
|
|
|
{
|
|
|
|
|
const GLSL::Token tk = tokens.last();
|
|
|
|
|
const int lastTokenEnd = tk.begin() + tk.length;
|
|
|
|
|
if (text.length() > lastTokenEnd)
|
|
|
|
|
highlightLine(text, lastTokenEnd, text.length() - lastTokenEnd, QTextCharFormat());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (! initialState && state && ! tokens.isEmpty()) {
|
|
|
|
|
parentheses.append(Parenthesis(Parenthesis::Opened, QLatin1Char('+'),
|
|
|
|
|
tokens.last().begin()));
|
|
|
|
|
++braceDepth;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BaseTextDocumentLayout::setParentheses(currentBlock(), parentheses);
|
|
|
|
|
|
|
|
|
|
// 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 == lex.state() && oldBraceDepth != braceDepth) {
|
|
|
|
|
int delta = braceDepth - oldBraceDepth;
|
|
|
|
|
QTextBlock block = currentBlock().next();
|
|
|
|
|
while (block.isValid() && block.userState() != -1) {
|
|
|
|
|
BaseTextDocumentLayout::changeBraceDepth(block, delta);
|
|
|
|
|
BaseTextDocumentLayout::changeFoldingIndent(block, delta);
|
|
|
|
|
block = block.next();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setCurrentBlockState((braceDepth << 8) | lex.state());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Highlighter::highlightLine(const QString &text, int position, int length,
|
|
|
|
|
const QTextCharFormat &format)
|
|
|
|
|
{
|
2013-08-13 12:57:31 +02:00
|
|
|
const QTextCharFormat visualSpaceFormat = formatForCategory(GLSLVisualWhitespace);
|
2010-11-26 10:59:33 +01:00
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Highlighter::isPPKeyword(const QStringRef &text) const
|
|
|
|
|
{
|
|
|
|
|
switch (text.length())
|
|
|
|
|
{
|
|
|
|
|
case 2:
|
2012-11-26 20:58:49 +02:00
|
|
|
if (text.at(0) == QLatin1Char('i') && text.at(1) == QLatin1Char('f'))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 4:
|
2012-11-26 20:58:49 +02:00
|
|
|
if (text.at(0) == QLatin1Char('e') && text == QLatin1String("elif"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
else if (text.at(0) == QLatin1Char('e') && text == QLatin1String("else"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 5:
|
2012-11-26 20:58:49 +02:00
|
|
|
if (text.at(0) == QLatin1Char('i') && text == QLatin1String("ifdef"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
else if (text.at(0) == QLatin1Char('u') && text == QLatin1String("undef"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
else if (text.at(0) == QLatin1Char('e') && text == QLatin1String("endif"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
else if (text.at(0) == QLatin1Char('e') && text == QLatin1String("error"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 6:
|
2012-11-26 20:58:49 +02:00
|
|
|
if (text.at(0) == QLatin1Char('i') && text == QLatin1String("ifndef"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
if (text.at(0) == QLatin1Char('i') && text == QLatin1String("import"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
else if (text.at(0) == QLatin1Char('d') && text == QLatin1String("define"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
else if (text.at(0) == QLatin1Char('p') && text == QLatin1String("pragma"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 7:
|
2012-11-26 20:58:49 +02:00
|
|
|
if (text.at(0) == QLatin1Char('i') && text == QLatin1String("include"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
2012-11-26 20:58:49 +02:00
|
|
|
else if (text.at(0) == QLatin1Char('w') && text == QLatin1String("warning"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 12:
|
2012-11-26 20:58:49 +02:00
|
|
|
if (text.at(0) == QLatin1Char('i') && text == QLatin1String("include_next"))
|
2010-11-26 10:59:33 +01:00
|
|
|
return true;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
void Highlighter::highlightBlock(const QString &text)
|
|
|
|
|
{
|
|
|
|
|
const int previousState = previousBlockState();
|
|
|
|
|
int state = 0, initialBraceDepth = 0;
|
|
|
|
|
if (previousState != -1) {
|
|
|
|
|
state = previousState & 0xff;
|
|
|
|
|
initialBraceDepth = previousState >> 8;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int braceDepth = initialBraceDepth;
|
|
|
|
|
|
|
|
|
|
Parentheses parentheses;
|
|
|
|
|
parentheses.reserve(20); // assume wizard level ;-)
|
|
|
|
|
|
2010-11-10 15:44:59 +01:00
|
|
|
const QByteArray data = text.toLatin1();
|
2010-11-11 12:01:37 +01:00
|
|
|
GLSL::Lexer lex(/*engine=*/ 0, data.constData(), data.size());
|
2010-11-26 10:59:33 +01:00
|
|
|
lex.setState(qMax(0, previousState));
|
2010-11-10 15:44:59 +01:00
|
|
|
lex.setScanKeywords(false);
|
|
|
|
|
lex.setScanComments(true);
|
2010-12-01 11:49:36 +10:00
|
|
|
const int variant = m_editor->languageVariant();
|
2010-11-23 17:05:55 +10:00
|
|
|
lex.setVariant(variant);
|
2010-11-26 10:59:33 +01:00
|
|
|
|
|
|
|
|
int foldingIndent = initialBraceDepth;
|
|
|
|
|
if (TextBlockUserData *userData = BaseTextDocumentLayout::testUserData(currentBlock())) {
|
|
|
|
|
userData->setFoldingIndent(0);
|
|
|
|
|
userData->setFoldingStartIncluded(false);
|
|
|
|
|
userData->setFoldingEndIncluded(false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<GLSL::Token> tokens;
|
2010-11-10 15:44:59 +01:00
|
|
|
GLSL::Token tk;
|
|
|
|
|
do {
|
|
|
|
|
lex.yylex(&tk);
|
2010-11-26 10:59:33 +01:00
|
|
|
tokens.append(tk);
|
|
|
|
|
} while (tk.isNot(GLSL::Parser::EOF_SYMBOL));
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < tokens.size(); ++i) {
|
|
|
|
|
const GLSL::Token &tk = tokens.at(i);
|
2010-11-10 15:44:59 +01:00
|
|
|
|
2013-07-17 00:01:45 +03:00
|
|
|
if (tk.is(GLSL::Parser::T_NUMBER)) {
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.position, tk.length, formatForCategory(GLSLNumberFormat);
|
2013-07-17 00:01:45 +03:00
|
|
|
} else if (tk.is(GLSL::Parser::T_COMMENT)) {
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.position, tk.length, Qt::darkGreen); // ### FIXME: formatForCategory(GLSLCommentFormat);
|
2013-07-17 00:01:45 +03:00
|
|
|
} else if (tk.is(GLSL::Parser::T_IDENTIFIER)) {
|
2010-11-23 17:05:55 +10:00
|
|
|
int kind = lex.findKeyword(data.constData() + tk.position, tk.length);
|
|
|
|
|
if (kind == GLSL::Parser::T_RESERVED)
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.position, tk.length, formatForCategory(GLSLReservedKeyword);
|
2010-11-23 17:05:55 +10:00
|
|
|
else if (kind != GLSL::Parser::T_IDENTIFIER)
|
2013-08-13 12:57:31 +02:00
|
|
|
setFormat(tk.position, tk.length, formatForCategory(GLSLKeywordFormat);
|
2010-11-26 10:59:33 +01:00
|
|
|
} else if (tk.is(GLSL::Parser::T_LEFT_PAREN) || tk.is(GLSL::Parser::T_LEFT_BRACE) || tk.is(GLSL::Parser::T_LEFT_BRACKET)) {
|
|
|
|
|
const QChar c = text.at(tk.begin());
|
|
|
|
|
parentheses.append(Parenthesis(Parenthesis::Opened, c, tk.begin()));
|
|
|
|
|
if (tk.is(GLSL::Parser::T_LEFT_BRACE)) {
|
|
|
|
|
++braceDepth;
|
|
|
|
|
|
|
|
|
|
// if a folding block opens at the beginning of a line, treat the entire line
|
|
|
|
|
// as if it were inside the folding block
|
|
|
|
|
// if (tk.begin() == firstNonSpace) {
|
|
|
|
|
// ++foldingIndent;
|
|
|
|
|
// BaseTextDocumentLayout::userData(currentBlock())->setFoldingStartIncluded(true);
|
|
|
|
|
// }
|
|
|
|
|
}
|
|
|
|
|
} else if (tk.is(GLSL::Parser::T_RIGHT_PAREN) || tk.is(GLSL::Parser::T_RIGHT_BRACE) || tk.is(GLSL::Parser::T_RIGHT_BRACKET)) {
|
|
|
|
|
const QChar c = text.at(tk.begin());
|
|
|
|
|
parentheses.append(Parenthesis(Parenthesis::Closed, c, tk.begin()));
|
|
|
|
|
if (tk.is(GLSL::Parser::T_RIGHT_BRACE)) {
|
|
|
|
|
--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(GLSL::Parser::T_SEMICOLON))
|
|
|
|
|
BaseTextDocumentLayout::userData(currentBlock())->setFoldingEndIncluded(true);
|
|
|
|
|
else
|
|
|
|
|
foldingIndent = qMin(braceDepth, foldingIndent);
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-11-10 15:44:59 +01:00
|
|
|
}
|
2010-11-26 10:59:33 +01:00
|
|
|
|
2010-11-10 15:44:59 +01:00
|
|
|
} while (tk.isNot(GLSL::Parser::EOF_SYMBOL));
|
2010-11-26 10:59:33 +01:00
|
|
|
setCurrentBlockState((braceDepth << 8) | lex.state());
|
2010-11-10 15:44:59 +01:00
|
|
|
}
|
2010-11-26 10:59:33 +01:00
|
|
|
#endif
|