]> git.lyx.org Git - lyx.git/blobdiff - src/frontends/qt/LaTeXHighlighter.cpp
No need (any longer?) to create a new view for lyxfiles-open
[lyx.git] / src / frontends / qt / LaTeXHighlighter.cpp
index 453f98b749dbea8160a71b28b9a112194927b925..dba92704a353fd406ef6b2792dc223edf03326cb 100644 (file)
@@ -20,8 +20,14 @@ namespace lyx {
 namespace frontend {
 
 
-LaTeXHighlighter::LaTeXHighlighter(QTextDocument * parent, bool at_letter)
-       : QSyntaxHighlighter(parent), at_letter_(at_letter)
+LaTeXHighlighter::LaTeXHighlighter(QTextDocument * parent, bool at_letter, bool keyval)
+       : QSyntaxHighlighter(parent), at_letter_(at_letter), keyval_(keyval)
+{
+       setupColors();
+}
+
+
+void LaTeXHighlighter::setupColors()
 {
        auto blend = [](QColor color1, QColor color2) {
                int r = 0.5 * (color1.red() + color2.red());
@@ -38,106 +44,60 @@ LaTeXHighlighter::LaTeXHighlighter(QTextDocument * parent, bool at_letter)
        mathFormat.setForeground(blend(Qt::red, text_color));
        warningFormat.setForeground(Qt::red);
        warningFormat.setFontWeight(QFont::Bold);
+       keyFormat.setForeground(blend(Qt::darkRed, text_color));
+       keyFormat.setFontWeight(QFont::Bold);
+       valFormat.setForeground(blend(Qt::darkGreen, text_color));
 }
 
 
 void LaTeXHighlighter::highlightBlock(QString const & text)
 {
-#if QT_VERSION < 0x060000
-       // $ $
-       static const QRegExp exprMath("\\$[^\\$]*\\$");
-       int index = exprMath.indexIn(text);
-       while (index >= 0) {
-               int length = exprMath.matchedLength();
-               setFormat(index, length, mathFormat);
-               index = exprMath.indexIn(text, index + length);
-       }
-       // [ ]
-       static const QRegExp exprStartDispMath("(\\\\\\[|"
-               "\\\\begin\\{equation\\**\\}|"
-               "\\\\begin\\{eqnarray\\**\\}|"
-               "\\\\begin\\{align(ed|at)*\\**\\}|"
-               "\\\\begin\\{flalign\\**\\}|"
-               "\\\\begin\\{gather\\**\\}|"
-               "\\\\begin\\{multline\\**\\}|"
-               "\\\\begin\\{array\\**\\}|"
-               "\\\\begin\\{cases\\**\\}"
-               ")");
-       static const QRegExp exprEndDispMath("(\\\\\\]|"
-               "\\\\end\\{equation\\**\\}|"
-               "\\\\end\\{eqnarray\\**\\}|"
-               "\\\\end\\{align(ed|at)*\\**\\}|"
-               "\\\\end\\{flalign\\**\\}|"
-               "\\\\end\\{gather\\**\\}|"
-               "\\\\end\\{multline\\**\\}|"
-               "\\\\end\\{array\\**\\}|"
-               "\\\\end\\{cases\\**\\}"
-               ")");
-       int startIndex = 0;
-       // if previous block was in 'disp math'
-       // start search from 0 (for end disp math)
-       // otherwise, start search from 'begin disp math'
-       if (previousBlockState() != 1)
-               startIndex = exprStartDispMath.indexIn(text);
-       while (startIndex >= 0) {
-               int endIndex = exprEndDispMath.indexIn(text, startIndex);
-               int length;
-               if (endIndex == -1) {
-                       setCurrentBlockState(1);
-                       length = text.length() - startIndex;
-               } else {
-                       length = endIndex - startIndex + exprEndDispMath.matchedLength();
+       // keyval
+       if (keyval_) {
+               // Highlight key-val options. Used in some option widgets.
+               // 1. The keys. Might or might not have values
+               static QRegularExpression exprKeyvalkey("[^=,}]+");
+               // 2. These are grouped values such as "key1={val,val},key2=val"
+               static QRegularExpression exprKeyvalgval("[^=,{]+{[^}]+}");
+               // 3. And normal values if we don't find grouped ones
+               static QRegularExpression exprKeyvalval("[^,]+");
+               QRegularExpressionMatch matchkey = exprKeyvalkey.match(text);
+               int kvindex = matchkey.capturedStart(0);
+               while (kvindex >= 0) {
+                       int length = matchkey.capturedLength(0);
+                       setFormat(kvindex, length, keyFormat);
+                       if (text.size() > kvindex + length && text.at(kvindex + length) == '=') {
+                               QRegularExpressionMatch matchgval =
+                                       exprKeyvalgval.match(text, kvindex + length);
+                               int kvvindex = matchgval.capturedStart(0);
+                               if (kvvindex > 0) {
+                                       int vlength = matchgval.capturedLength(0);
+                                       length += vlength;
+                                       setFormat(kvvindex, vlength, valFormat);
+                               } else {
+                                       QRegularExpressionMatch matchval =
+                                               exprKeyvalval.match(text, kvindex + length);
+                                       kvvindex = matchval.capturedStart(0);
+                                       if (kvvindex > 0) {
+                                               int vlength = matchval.capturedLength(0);
+                                               length += vlength;
+                                               setFormat(kvvindex, vlength, valFormat);
+                                       }
+                               }
+                       }
+                       matchkey = exprKeyvalkey.match(text, kvindex + length);
+                       kvindex = matchkey.capturedStart(0);
                }
-               setFormat(startIndex, length, mathFormat);
-               startIndex = exprStartDispMath.indexIn(text, startIndex + length);
-       }
-       // \whatever
-       static const QRegExp exprKeywordAtOther("\\\\[A-Za-z]+");
-       // \wh@tever
-       static const QRegExp exprKeywordAtLetter("\\\\[A-Za-z@]+");
-       QRegExp const & exprKeyword = at_letter_ ? exprKeywordAtLetter
-                                                : exprKeywordAtOther;
-       index = exprKeyword.indexIn(text);
-       while (index >= 0) {
-               int length = exprKeyword.matchedLength();
-               setFormat(index, length, keywordFormat);
-               index = exprKeyword.indexIn(text, index + length);
        }
-       // %comment
-       // Treat a line as a comment starting at a percent sign
-       // * that is the first character in a line
-       // * that is preceded by
-       // ** an even number of backslashes
-       // ** any character other than a backslash
-       QRegExp exprComment("(?:^|[^\\\\])(?:\\\\\\\\)*(%).*$");
-       exprComment.indexIn(text);
-       index = exprComment.pos(1);
-       while (index >= 0) {
-               int const length = exprComment.matchedLength()
-                                - (index - exprComment.pos(0));
-               setFormat(index, length, commentFormat);
-               exprComment.indexIn(text, index + length);
-               index = exprComment.pos(1);
-       }
-       // <LyX Warning: ...>
-       QString lyxwarn = qt_("LyX Warning: ");
-       QRegExp exprWarning("<" + lyxwarn + "[^<]*>");
-       index = exprWarning.indexIn(text);
-       while (index >= 0) {
-               int length = exprWarning.matchedLength();
-               setFormat(index, length, warningFormat);
-               index = exprWarning.indexIn(text, index + length);
-       }
-#else
        // $ $
        static const QRegularExpression exprMath("\\$[^\\$]*\\$");
        QRegularExpressionMatch match = exprMath.match(text);
-       int index = match.capturedStart(1);
+       int index = match.capturedStart(0);
        while (index >= 0) {
-               int length = match.capturedEnd(1) - index;
+               int length = match.capturedLength(0);
                setFormat(index, length, mathFormat);
                match = exprMath.match(text, index + length);
-               index = match.capturedStart(1);
+               index = match.capturedStart(0);
        }
        // [ ]
        static const QRegularExpression exprStartDispMath("(\\\\\\[|"
@@ -166,21 +126,21 @@ void LaTeXHighlighter::highlightBlock(QString const & text)
        // otherwise, start search from 'begin disp math'
        if (previousBlockState() != 1) {
                match = exprStartDispMath.match(text);
-               startIndex = match.capturedStart(1);
+               startIndex = match.capturedStart(0);
        }
        while (startIndex >= 0) {
                match = exprEndDispMath.match(text, startIndex);
-               int endIndex = match.capturedStart(1);
+               int endIndex = match.capturedStart(0);
                int length;
                if (endIndex == -1) {
                        setCurrentBlockState(1);
                        length = text.length() - startIndex;
                } else {
-                       length = match.capturedEnd(1) - startIndex;
+                       length = endIndex - startIndex + match.capturedLength(0);
                }
                setFormat(startIndex, length, mathFormat);
                match = exprStartDispMath.match(text, startIndex + length);
-               startIndex = match.capturedStart(1);
+               startIndex = match.capturedStart(0);
        }
        // \whatever
        static const QRegularExpression exprKeywordAtOther("\\\\[A-Za-z]+");
@@ -189,12 +149,12 @@ void LaTeXHighlighter::highlightBlock(QString const & text)
        QRegularExpression const & exprKeyword = at_letter_
                        ? exprKeywordAtLetter : exprKeywordAtOther;
        match = exprKeyword.match(text);
-       index = match.capturedStart(1);
+       index = match.capturedStart(0);
        while (index >= 0) {
-               int length = match.capturedEnd(1) - index;
+               int length = match.capturedLength(0);
                setFormat(index, length, keywordFormat);
                match = exprKeyword.match(text, index + length);
-               index = match.capturedStart(1);
+               index = match.capturedStart(0);
        }
        // %comment
        // Treat a line as a comment starting at a percent sign
@@ -206,7 +166,7 @@ void LaTeXHighlighter::highlightBlock(QString const & text)
        match = exprComment.match(text);
        index = match.capturedStart(1);
        while (index >= 0) {
-               int const length = match.capturedEnd(1) - index
+               int const length = match.capturedLength(0)
                                 - (index - match.capturedStart(0));
                setFormat(index, length, commentFormat);
                match = exprComment.match(text, index + length);
@@ -216,14 +176,13 @@ void LaTeXHighlighter::highlightBlock(QString const & text)
        QString lyxwarn = qt_("LyX Warning: ");
        QRegularExpression exprWarning("<" + lyxwarn + "[^<]*>");
        match = exprWarning.match(text);
-       index = match.capturedStart(1);
+       index = match.capturedStart(0);
        while (index >= 0) {
-               int length = match.capturedEnd(1) - index;
+               int length = match.capturedLength(0);
                setFormat(index, length, warningFormat);
                match = exprWarning.match(text, index + length);
-               index = match.capturedStart(1);
+               index = match.capturedStart(0);
        }
-#endif
 }
 
 } // namespace frontend