2022-01-04 23:07:50 +00:00
|
|
|
// Scintilla source code edit control
|
|
|
|
/** @file LexCPP.cxx
|
|
|
|
** Lexer for C++, C, Java, and JavaScript.
|
|
|
|
** Further folding features and configuration properties added by "Udo Lechner" <dlchnr(at)gmx(dot)net>
|
|
|
|
**/
|
|
|
|
// Copyright 1998-2005 by Neil Hodgson <neilh@scintilla.org>
|
|
|
|
// The License.txt file describes the conditions under which this software may be distributed.
|
|
|
|
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <cassert>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
#include <utility>
|
|
|
|
#include <string>
|
|
|
|
#include <string_view>
|
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <iterator>
|
|
|
|
#include <functional>
|
|
|
|
|
|
|
|
#include "ILexer.h"
|
|
|
|
#include "Scintilla.h"
|
|
|
|
#include "SciLexer.h"
|
|
|
|
|
|
|
|
#include "StringCopy.h"
|
|
|
|
#include "WordList.h"
|
|
|
|
#include "LexAccessor.h"
|
|
|
|
#include "Accessor.h"
|
|
|
|
#include "StyleContext.h"
|
|
|
|
#include "CharacterSet.h"
|
|
|
|
#include "LexerModule.h"
|
|
|
|
#include "OptionSet.h"
|
|
|
|
#include "SparseState.h"
|
|
|
|
#include "SubStyles.h"
|
|
|
|
|
|
|
|
using namespace Scintilla;
|
|
|
|
using namespace Lexilla;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
// Use an unnamed namespace to protect the functions and classes from name conflicts
|
|
|
|
|
|
|
|
constexpr bool IsSpaceEquiv(int state) noexcept {
|
|
|
|
return (state <= SCE_C_COMMENTDOC) ||
|
|
|
|
// including SCE_C_DEFAULT, SCE_C_COMMENT, SCE_C_COMMENTLINE
|
|
|
|
(state == SCE_C_COMMENTLINEDOC) || (state == SCE_C_COMMENTDOCKEYWORD) ||
|
|
|
|
(state == SCE_C_COMMENTDOCKEYWORDERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Preconditions: sc.currentPos points to a character after '+' or '-'.
|
|
|
|
// The test for pos reaching 0 should be redundant,
|
|
|
|
// and is in only for safety measures.
|
|
|
|
// Limitation: this code will give the incorrect answer for code like
|
|
|
|
// a = b+++/ptn/...
|
|
|
|
// Putting a space between the '++' post-inc operator and the '+' binary op
|
|
|
|
// fixes this, and is highly recommended for readability anyway.
|
|
|
|
bool FollowsPostfixOperator(const StyleContext &sc, LexAccessor &styler) {
|
|
|
|
Sci_Position pos = sc.currentPos;
|
|
|
|
while (--pos > 0) {
|
|
|
|
const char ch = styler[pos];
|
|
|
|
if (ch == '+' || ch == '-') {
|
|
|
|
return styler[pos - 1] == ch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool followsReturnKeyword(const StyleContext &sc, LexAccessor &styler) {
|
|
|
|
// Don't look at styles, so no need to flush.
|
|
|
|
Sci_Position pos = sc.currentPos;
|
|
|
|
const Sci_Position currentLine = styler.GetLine(pos);
|
|
|
|
const Sci_Position lineStartPos = styler.LineStart(currentLine);
|
|
|
|
while (--pos > lineStartPos) {
|
|
|
|
const char ch = styler.SafeGetCharAt(pos);
|
|
|
|
if (ch != ' ' && ch != '\t') {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const char *retBack = "nruter";
|
|
|
|
const char *s = retBack;
|
|
|
|
while (*s
|
|
|
|
&& pos >= lineStartPos
|
|
|
|
&& styler.SafeGetCharAt(pos) == *s) {
|
|
|
|
s++;
|
|
|
|
pos--;
|
|
|
|
}
|
|
|
|
return !*s;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr bool IsSpaceOrTab(int ch) noexcept {
|
|
|
|
return ch == ' ' || ch == '\t';
|
|
|
|
}
|
|
|
|
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
constexpr bool IsOperatorOrSpace(int ch) noexcept {
|
|
|
|
return isoperator(ch) || IsASpace(ch);
|
|
|
|
}
|
|
|
|
|
2022-01-04 23:07:50 +00:00
|
|
|
bool OnlySpaceOrTab(const std::string &s) noexcept {
|
|
|
|
for (const char ch : s) {
|
|
|
|
if (!IsSpaceOrTab(ch))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
using Tokens = std::vector<std::string>;
|
|
|
|
|
|
|
|
Tokens StringSplit(const std::string &text, int separator) {
|
|
|
|
Tokens vs(text.empty() ? 0 : 1);
|
|
|
|
for (const char ch : text) {
|
|
|
|
if (ch == separator) {
|
|
|
|
vs.emplace_back();
|
|
|
|
} else {
|
|
|
|
vs.back() += ch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return vs;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct BracketPair {
|
|
|
|
Tokens::iterator itBracket;
|
|
|
|
Tokens::iterator itEndBracket;
|
|
|
|
};
|
|
|
|
|
|
|
|
BracketPair FindBracketPair(Tokens &tokens) {
|
|
|
|
const Tokens::iterator itBracket = std::find(tokens.begin(), tokens.end(), "(");
|
|
|
|
if (itBracket != tokens.end()) {
|
|
|
|
size_t nest = 0;
|
|
|
|
for (Tokens::iterator itTok = itBracket; itTok != tokens.end(); ++itTok) {
|
|
|
|
if (*itTok == "(") {
|
|
|
|
nest++;
|
|
|
|
} else if (*itTok == ")") {
|
|
|
|
nest--;
|
|
|
|
if (nest == 0) {
|
|
|
|
return { itBracket, itTok };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return { tokens.end(), tokens.end() };
|
|
|
|
}
|
|
|
|
|
|
|
|
void highlightTaskMarker(StyleContext &sc, LexAccessor &styler,
|
|
|
|
int activity, const WordList &markerList, bool caseSensitive){
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
if (IsOperatorOrSpace(sc.chPrev) && !IsOperatorOrSpace(sc.ch) && markerList.Length()) {
|
|
|
|
std::string marker;
|
|
|
|
for (Sci_PositionU currPos = sc.currentPos; true; currPos++) {
|
|
|
|
const char ch = styler.SafeGetCharAt(currPos);
|
|
|
|
if (IsOperatorOrSpace(ch)) {
|
2022-01-04 23:07:50 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (caseSensitive)
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
marker.push_back(ch);
|
2022-01-04 23:07:50 +00:00
|
|
|
else
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
marker.push_back(MakeLowerCase(ch));
|
2022-01-04 23:07:50 +00:00
|
|
|
}
|
|
|
|
if (markerList.InList(marker)) {
|
|
|
|
sc.SetState(SCE_C_TASKMARKER|activity);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class EscapeSequence {
|
|
|
|
const CharacterSet setHexDigits = CharacterSet(CharacterSet::setDigits, "ABCDEFabcdef");
|
2023-05-31 23:11:12 +00:00
|
|
|
const CharacterSet setOctDigits = CharacterSet("01234567");
|
2022-01-04 23:07:50 +00:00
|
|
|
const CharacterSet setNoneNumeric;
|
|
|
|
const CharacterSet *escapeSetValid = nullptr;
|
|
|
|
int digitsLeft = 0;
|
|
|
|
public:
|
|
|
|
EscapeSequence() = default;
|
2022-10-12 18:45:40 +00:00
|
|
|
void resetEscapeState(int nextChar) noexcept {
|
2022-01-04 23:07:50 +00:00
|
|
|
digitsLeft = 0;
|
|
|
|
escapeSetValid = &setNoneNumeric;
|
|
|
|
if (nextChar == 'U') {
|
|
|
|
digitsLeft = 9;
|
|
|
|
escapeSetValid = &setHexDigits;
|
|
|
|
} else if (nextChar == 'u') {
|
|
|
|
digitsLeft = 5;
|
|
|
|
escapeSetValid = &setHexDigits;
|
|
|
|
} else if (nextChar == 'x') {
|
|
|
|
digitsLeft = 5;
|
|
|
|
escapeSetValid = &setHexDigits;
|
|
|
|
} else if (setOctDigits.Contains(nextChar)) {
|
|
|
|
digitsLeft = 3;
|
|
|
|
escapeSetValid = &setOctDigits;
|
|
|
|
}
|
|
|
|
}
|
2022-10-12 18:45:40 +00:00
|
|
|
bool atEscapeEnd(int currChar) const noexcept {
|
2022-01-04 23:07:50 +00:00
|
|
|
return (digitsLeft <= 0) || !escapeSetValid->Contains(currChar);
|
|
|
|
}
|
|
|
|
void consumeDigit() noexcept {
|
|
|
|
digitsLeft--;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::string GetRestOfLine(LexAccessor &styler, Sci_Position start, bool allowSpace) {
|
|
|
|
std::string restOfLine;
|
|
|
|
Sci_Position line = styler.GetLine(start);
|
|
|
|
Sci_Position pos = start;
|
|
|
|
Sci_Position endLine = styler.LineEnd(line);
|
|
|
|
char ch = styler.SafeGetCharAt(start, '\n');
|
|
|
|
while (pos < endLine) {
|
|
|
|
if (ch == '\\' && ((pos + 1) == endLine)) {
|
|
|
|
// Continuation line
|
|
|
|
line++;
|
|
|
|
pos = styler.LineStart(line);
|
|
|
|
endLine = styler.LineEnd(line);
|
|
|
|
ch = styler.SafeGetCharAt(pos, '\n');
|
|
|
|
} else {
|
|
|
|
const char chNext = styler.SafeGetCharAt(pos + 1, '\n');
|
|
|
|
if (ch == '/' && (chNext == '/' || chNext == '*'))
|
|
|
|
break;
|
|
|
|
if (allowSpace || (ch != ' ')) {
|
|
|
|
restOfLine += ch;
|
|
|
|
}
|
|
|
|
pos++;
|
|
|
|
ch = chNext;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return restOfLine;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr bool IsStreamCommentStyle(int style) noexcept {
|
|
|
|
return style == SCE_C_COMMENT ||
|
|
|
|
style == SCE_C_COMMENTDOC ||
|
|
|
|
style == SCE_C_COMMENTDOCKEYWORD ||
|
|
|
|
style == SCE_C_COMMENTDOCKEYWORDERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct PPDefinition {
|
|
|
|
Sci_Position line;
|
|
|
|
std::string key;
|
|
|
|
std::string value;
|
|
|
|
bool isUndef;
|
|
|
|
std::string arguments;
|
|
|
|
PPDefinition(Sci_Position line_, const std::string &key_, const std::string &value_, bool isUndef_ = false, const std::string &arguments_="") :
|
|
|
|
line(line_), key(key_), value(value_), isUndef(isUndef_), arguments(arguments_) {
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
constexpr int inactiveFlag = 0x40;
|
|
|
|
|
|
|
|
class LinePPState {
|
|
|
|
// Track the state of preprocessor conditionals to allow showing active and inactive
|
|
|
|
// code in different styles.
|
|
|
|
// Only works up to 31 levels of conditional nesting.
|
|
|
|
|
|
|
|
// state is a bit mask with 1 bit per level
|
|
|
|
// bit is 1 for level if section inactive, so any bits set = inactive style
|
|
|
|
int state = 0;
|
|
|
|
// ifTaken is a bit mask with 1 bit per level
|
|
|
|
// bit is 1 for level if some branch at this level has been taken
|
|
|
|
int ifTaken = 0;
|
|
|
|
// level is the nesting level of #if constructs
|
|
|
|
int level = -1;
|
|
|
|
static const int maximumNestingLevel = 31;
|
|
|
|
int maskLevel() const noexcept {
|
|
|
|
if (level >= 0) {
|
|
|
|
return 1 << level;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
public:
|
2022-08-27 07:35:52 +00:00
|
|
|
LinePPState() noexcept = default;
|
2022-01-04 23:07:50 +00:00
|
|
|
bool ValidLevel() const noexcept {
|
|
|
|
return level >= 0 && level < maximumNestingLevel;
|
|
|
|
}
|
|
|
|
bool IsActive() const noexcept {
|
|
|
|
return state == 0;
|
|
|
|
}
|
|
|
|
bool IsInactive() const noexcept {
|
|
|
|
return state != 0;
|
|
|
|
}
|
|
|
|
int ActiveState() const noexcept {
|
|
|
|
return state ? inactiveFlag : 0;
|
|
|
|
}
|
|
|
|
bool CurrentIfTaken() const noexcept {
|
|
|
|
return (ifTaken & maskLevel()) != 0;
|
|
|
|
}
|
|
|
|
void StartSection(bool on) noexcept {
|
|
|
|
level++;
|
|
|
|
if (ValidLevel()) {
|
|
|
|
if (on) {
|
|
|
|
state &= ~maskLevel();
|
|
|
|
ifTaken |= maskLevel();
|
|
|
|
} else {
|
|
|
|
state |= maskLevel();
|
|
|
|
ifTaken &= ~maskLevel();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void EndSection() noexcept {
|
|
|
|
if (ValidLevel()) {
|
|
|
|
state &= ~maskLevel();
|
|
|
|
ifTaken &= ~maskLevel();
|
|
|
|
}
|
|
|
|
level--;
|
|
|
|
}
|
|
|
|
void InvertCurrentLevel() noexcept {
|
|
|
|
if (ValidLevel()) {
|
|
|
|
state ^= maskLevel();
|
|
|
|
ifTaken |= maskLevel();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Hold the preprocessor state for each line seen.
|
|
|
|
// Currently one entry per line but could become sparse with just one entry per preprocessor line.
|
|
|
|
class PPStates {
|
|
|
|
std::vector<LinePPState> vlls;
|
|
|
|
public:
|
|
|
|
LinePPState ForLine(Sci_Position line) const noexcept {
|
|
|
|
if ((line > 0) && (vlls.size() > static_cast<size_t>(line))) {
|
|
|
|
return vlls[line];
|
|
|
|
}
|
2022-10-12 18:45:40 +00:00
|
|
|
return {};
|
2022-01-04 23:07:50 +00:00
|
|
|
}
|
|
|
|
void Add(Sci_Position line, LinePPState lls) {
|
|
|
|
vlls.resize(line+1);
|
|
|
|
vlls[line] = lls;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// An individual named option for use in an OptionSet
|
|
|
|
|
|
|
|
// Options used for LexerCPP
|
|
|
|
struct OptionsCPP {
|
2023-09-22 09:32:35 +00:00
|
|
|
bool stylingWithinPreprocessor = false;
|
|
|
|
bool identifiersAllowDollars = true;
|
|
|
|
bool trackPreprocessor = true;
|
|
|
|
bool updatePreprocessor = true;
|
|
|
|
bool verbatimStringsAllowEscapes = false;
|
|
|
|
bool triplequotedStrings = false;
|
|
|
|
bool hashquotedStrings = false;
|
|
|
|
bool backQuotedStrings = false;
|
|
|
|
bool escapeSequence = false;
|
|
|
|
bool fold = false;
|
|
|
|
bool foldSyntaxBased = true;
|
|
|
|
bool foldComment = false;
|
|
|
|
bool foldCommentMultiline = true;
|
|
|
|
bool foldCommentExplicit = true;
|
2022-01-04 23:07:50 +00:00
|
|
|
std::string foldExplicitStart;
|
|
|
|
std::string foldExplicitEnd;
|
2023-09-22 09:32:35 +00:00
|
|
|
bool foldExplicitAnywhere = false;
|
|
|
|
bool foldPreprocessor = false;
|
|
|
|
bool foldPreprocessorAtElse = false;
|
|
|
|
bool foldCompact = false;
|
|
|
|
bool foldAtElse = false;
|
2022-01-04 23:07:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const char *const cppWordLists[] = {
|
|
|
|
"Primary keywords and identifiers",
|
|
|
|
"Secondary keywords and identifiers",
|
|
|
|
"Documentation comment keywords",
|
|
|
|
"Global classes and typedefs",
|
|
|
|
"Preprocessor definitions",
|
|
|
|
"Task marker and error marker keywords",
|
|
|
|
nullptr,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct OptionSetCPP : public OptionSet<OptionsCPP> {
|
|
|
|
OptionSetCPP() {
|
|
|
|
DefineProperty("styling.within.preprocessor", &OptionsCPP::stylingWithinPreprocessor,
|
|
|
|
"For C++ code, determines whether all preprocessor code is styled in the "
|
|
|
|
"preprocessor style (0, the default) or only from the initial # to the end "
|
|
|
|
"of the command word(1).");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.allow.dollars", &OptionsCPP::identifiersAllowDollars,
|
|
|
|
"Set to 0 to disallow the '$' character in identifiers with the cpp lexer.");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.track.preprocessor", &OptionsCPP::trackPreprocessor,
|
|
|
|
"Set to 1 to interpret #if/#else/#endif to grey out code that is not active.");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.update.preprocessor", &OptionsCPP::updatePreprocessor,
|
|
|
|
"Set to 1 to update preprocessor definitions when #define found.");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.verbatim.strings.allow.escapes", &OptionsCPP::verbatimStringsAllowEscapes,
|
|
|
|
"Set to 1 to allow verbatim strings to contain escape sequences.");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.triplequoted.strings", &OptionsCPP::triplequotedStrings,
|
|
|
|
"Set to 1 to enable highlighting of triple-quoted strings.");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.hashquoted.strings", &OptionsCPP::hashquotedStrings,
|
|
|
|
"Set to 1 to enable highlighting of hash-quoted strings.");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.backquoted.strings", &OptionsCPP::backQuotedStrings,
|
|
|
|
"Set to 1 to enable highlighting of back-quoted raw strings .");
|
|
|
|
|
|
|
|
DefineProperty("lexer.cpp.escape.sequence", &OptionsCPP::escapeSequence,
|
|
|
|
"Set to 1 to enable highlighting of escape sequences in strings");
|
|
|
|
|
|
|
|
DefineProperty("fold", &OptionsCPP::fold);
|
|
|
|
|
|
|
|
DefineProperty("fold.cpp.syntax.based", &OptionsCPP::foldSyntaxBased,
|
|
|
|
"Set this property to 0 to disable syntax based folding.");
|
|
|
|
|
|
|
|
DefineProperty("fold.comment", &OptionsCPP::foldComment,
|
|
|
|
"This option enables folding multi-line comments and explicit fold points when using the C++ lexer. "
|
|
|
|
"Explicit fold points allows adding extra folding by placing a //{ comment at the start and a //} "
|
|
|
|
"at the end of a section that should fold.");
|
|
|
|
|
|
|
|
DefineProperty("fold.cpp.comment.multiline", &OptionsCPP::foldCommentMultiline,
|
|
|
|
"Set this property to 0 to disable folding multi-line comments when fold.comment=1.");
|
|
|
|
|
|
|
|
DefineProperty("fold.cpp.comment.explicit", &OptionsCPP::foldCommentExplicit,
|
|
|
|
"Set this property to 0 to disable folding explicit fold points when fold.comment=1.");
|
|
|
|
|
|
|
|
DefineProperty("fold.cpp.explicit.start", &OptionsCPP::foldExplicitStart,
|
|
|
|
"The string to use for explicit fold start points, replacing the standard //{.");
|
|
|
|
|
|
|
|
DefineProperty("fold.cpp.explicit.end", &OptionsCPP::foldExplicitEnd,
|
|
|
|
"The string to use for explicit fold end points, replacing the standard //}.");
|
|
|
|
|
|
|
|
DefineProperty("fold.cpp.explicit.anywhere", &OptionsCPP::foldExplicitAnywhere,
|
|
|
|
"Set this property to 1 to enable explicit fold points anywhere, not just in line comments.");
|
|
|
|
|
|
|
|
DefineProperty("fold.cpp.preprocessor.at.else", &OptionsCPP::foldPreprocessorAtElse,
|
|
|
|
"This option enables folding on a preprocessor #else or #endif line of an #if statement.");
|
|
|
|
|
|
|
|
DefineProperty("fold.preprocessor", &OptionsCPP::foldPreprocessor,
|
|
|
|
"This option enables folding preprocessor directives when using the C++ lexer. "
|
|
|
|
"Includes C#'s explicit #region and #endregion folding directives.");
|
|
|
|
|
|
|
|
DefineProperty("fold.compact", &OptionsCPP::foldCompact);
|
|
|
|
|
|
|
|
DefineProperty("fold.at.else", &OptionsCPP::foldAtElse,
|
|
|
|
"This option enables C++ folding on a \"} else {\" line of an if statement.");
|
|
|
|
|
|
|
|
DefineWordListSets(cppWordLists);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const char styleSubable[] = {SCE_C_IDENTIFIER, SCE_C_COMMENTDOCKEYWORD, 0};
|
|
|
|
|
|
|
|
LexicalClass lexicalClasses[] = {
|
|
|
|
// Lexer Cpp SCLEX_CPP SCE_C_:
|
|
|
|
0, "SCE_C_DEFAULT", "default", "White space",
|
|
|
|
1, "SCE_C_COMMENT", "comment", "Comment: /* */.",
|
|
|
|
2, "SCE_C_COMMENTLINE", "comment line", "Line Comment: //.",
|
|
|
|
3, "SCE_C_COMMENTDOC", "comment documentation", "Doc comment: block comments beginning with /** or /*!",
|
|
|
|
4, "SCE_C_NUMBER", "literal numeric", "Number",
|
|
|
|
5, "SCE_C_WORD", "keyword", "Keyword",
|
|
|
|
6, "SCE_C_STRING", "literal string", "Double quoted string",
|
|
|
|
7, "SCE_C_CHARACTER", "literal string character", "Single quoted string",
|
|
|
|
8, "SCE_C_UUID", "literal uuid", "UUIDs (only in IDL)",
|
|
|
|
9, "SCE_C_PREPROCESSOR", "preprocessor", "Preprocessor",
|
|
|
|
10, "SCE_C_OPERATOR", "operator", "Operators",
|
|
|
|
11, "SCE_C_IDENTIFIER", "identifier", "Identifiers",
|
|
|
|
12, "SCE_C_STRINGEOL", "error literal string", "End of line where string is not closed",
|
|
|
|
13, "SCE_C_VERBATIM", "literal string multiline raw", "Verbatim strings for C#",
|
|
|
|
14, "SCE_C_REGEX", "literal regex", "Regular expressions for JavaScript",
|
|
|
|
15, "SCE_C_COMMENTLINEDOC", "comment documentation line", "Doc Comment Line: line comments beginning with /// or //!.",
|
|
|
|
16, "SCE_C_WORD2", "identifier", "Keywords2",
|
|
|
|
17, "SCE_C_COMMENTDOCKEYWORD", "comment documentation keyword", "Comment keyword",
|
|
|
|
18, "SCE_C_COMMENTDOCKEYWORDERROR", "error comment documentation keyword", "Comment keyword error",
|
|
|
|
19, "SCE_C_GLOBALCLASS", "identifier", "Global class",
|
|
|
|
20, "SCE_C_STRINGRAW", "literal string multiline raw", "Raw strings for C++0x",
|
|
|
|
21, "SCE_C_TRIPLEVERBATIM", "literal string multiline raw", "Triple-quoted strings for Vala",
|
|
|
|
22, "SCE_C_HASHQUOTEDSTRING", "literal string", "Hash-quoted strings for Pike",
|
|
|
|
23, "SCE_C_PREPROCESSORCOMMENT", "comment preprocessor", "Preprocessor stream comment",
|
|
|
|
24, "SCE_C_PREPROCESSORCOMMENTDOC", "comment preprocessor documentation", "Preprocessor stream doc comment",
|
|
|
|
25, "SCE_C_USERLITERAL", "literal", "User defined literals",
|
|
|
|
26, "SCE_C_TASKMARKER", "comment taskmarker", "Task Marker",
|
|
|
|
27, "SCE_C_ESCAPESEQUENCE", "literal string escapesequence", "Escape sequence",
|
|
|
|
};
|
|
|
|
|
|
|
|
const int sizeLexicalClasses = static_cast<int>(std::size(lexicalClasses));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
class LexerCPP : public ILexer5 {
|
|
|
|
bool caseSensitive;
|
|
|
|
CharacterSet setWord;
|
|
|
|
CharacterSet setNegationOp;
|
|
|
|
CharacterSet setAddOp;
|
|
|
|
CharacterSet setMultOp;
|
|
|
|
CharacterSet setRelOp;
|
|
|
|
CharacterSet setLogicalOp;
|
|
|
|
CharacterSet setWordStart;
|
|
|
|
PPStates vlls;
|
|
|
|
std::vector<PPDefinition> ppDefineHistory;
|
|
|
|
WordList keywords;
|
|
|
|
WordList keywords2;
|
|
|
|
WordList keywords3;
|
|
|
|
WordList keywords4;
|
|
|
|
WordList ppDefinitions;
|
|
|
|
WordList markerList;
|
|
|
|
struct SymbolValue {
|
|
|
|
std::string value;
|
|
|
|
std::string arguments;
|
|
|
|
SymbolValue() noexcept = default;
|
|
|
|
SymbolValue(const std::string &value_, const std::string &arguments_) : value(value_), arguments(arguments_) {
|
|
|
|
}
|
|
|
|
SymbolValue &operator = (const std::string &value_) {
|
|
|
|
value = value_;
|
|
|
|
arguments.clear();
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
bool IsMacro() const noexcept {
|
|
|
|
return !arguments.empty();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
typedef std::map<std::string, SymbolValue> SymbolTable;
|
|
|
|
SymbolTable preprocessorDefinitionsStart;
|
|
|
|
OptionsCPP options;
|
|
|
|
OptionSetCPP osCPP;
|
|
|
|
EscapeSequence escapeSeq;
|
|
|
|
SparseState<std::string> rawStringTerminators;
|
|
|
|
enum { ssIdentifier, ssDocKeyword };
|
|
|
|
SubStyles subStyles;
|
|
|
|
std::string returnBuffer;
|
|
|
|
public:
|
|
|
|
explicit LexerCPP(bool caseSensitive_) :
|
|
|
|
caseSensitive(caseSensitive_),
|
|
|
|
setWord(CharacterSet::setAlphaNum, "._", true),
|
2023-05-31 23:11:12 +00:00
|
|
|
setNegationOp("!"),
|
|
|
|
setAddOp("+-"),
|
|
|
|
setMultOp("*/%"),
|
|
|
|
setRelOp("=!<>"),
|
|
|
|
setLogicalOp("|&"),
|
2022-01-04 23:07:50 +00:00
|
|
|
subStyles(styleSubable, 0x80, 0x40, inactiveFlag) {
|
|
|
|
}
|
|
|
|
// Deleted so LexerCPP objects can not be copied.
|
|
|
|
LexerCPP(const LexerCPP &) = delete;
|
|
|
|
LexerCPP(LexerCPP &&) = delete;
|
|
|
|
void operator=(const LexerCPP &) = delete;
|
|
|
|
void operator=(LexerCPP &&) = delete;
|
|
|
|
virtual ~LexerCPP() {
|
|
|
|
}
|
|
|
|
void SCI_METHOD Release() noexcept override {
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
int SCI_METHOD Version() const noexcept override {
|
|
|
|
return lvRelease5;
|
|
|
|
}
|
|
|
|
const char * SCI_METHOD PropertyNames() override {
|
|
|
|
return osCPP.PropertyNames();
|
|
|
|
}
|
|
|
|
int SCI_METHOD PropertyType(const char *name) override {
|
|
|
|
return osCPP.PropertyType(name);
|
|
|
|
}
|
|
|
|
const char * SCI_METHOD DescribeProperty(const char *name) override {
|
|
|
|
return osCPP.DescribeProperty(name);
|
|
|
|
}
|
|
|
|
Sci_Position SCI_METHOD PropertySet(const char *key, const char *val) override;
|
|
|
|
const char * SCI_METHOD DescribeWordListSets() override {
|
|
|
|
return osCPP.DescribeWordListSets();
|
|
|
|
}
|
|
|
|
Sci_Position SCI_METHOD WordListSet(int n, const char *wl) override;
|
|
|
|
void SCI_METHOD Lex(Sci_PositionU startPos, Sci_Position length, int initStyle, IDocument *pAccess) override;
|
|
|
|
void SCI_METHOD Fold(Sci_PositionU startPos, Sci_Position length, int initStyle, IDocument *pAccess) override;
|
|
|
|
|
|
|
|
void * SCI_METHOD PrivateCall(int, void *) noexcept override {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SCI_METHOD LineEndTypesSupported() noexcept override {
|
|
|
|
return SC_LINE_END_TYPE_UNICODE;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SCI_METHOD AllocateSubStyles(int styleBase, int numberStyles) override {
|
|
|
|
return subStyles.Allocate(styleBase, numberStyles);
|
|
|
|
}
|
|
|
|
int SCI_METHOD SubStylesStart(int styleBase) override {
|
|
|
|
return subStyles.Start(styleBase);
|
|
|
|
}
|
|
|
|
int SCI_METHOD SubStylesLength(int styleBase) override {
|
|
|
|
return subStyles.Length(styleBase);
|
|
|
|
}
|
|
|
|
int SCI_METHOD StyleFromSubStyle(int subStyle) override {
|
|
|
|
const int styleBase = subStyles.BaseStyle(MaskActive(subStyle));
|
|
|
|
const int inactive = subStyle & inactiveFlag;
|
|
|
|
return styleBase | inactive;
|
|
|
|
}
|
|
|
|
int SCI_METHOD PrimaryStyleFromStyle(int style) noexcept override {
|
|
|
|
return MaskActive(style);
|
|
|
|
}
|
|
|
|
void SCI_METHOD FreeSubStyles() override {
|
|
|
|
subStyles.Free();
|
|
|
|
}
|
|
|
|
void SCI_METHOD SetIdentifiers(int style, const char *identifiers) override {
|
|
|
|
subStyles.SetIdentifiers(style, identifiers);
|
|
|
|
}
|
|
|
|
int SCI_METHOD DistanceToSecondaryStyles() noexcept override {
|
|
|
|
return inactiveFlag;
|
|
|
|
}
|
|
|
|
const char * SCI_METHOD GetSubStyleBases() noexcept override {
|
|
|
|
return styleSubable;
|
|
|
|
}
|
|
|
|
int SCI_METHOD NamedStyles() override {
|
|
|
|
return std::max(subStyles.LastAllocated() + 1,
|
|
|
|
sizeLexicalClasses) +
|
|
|
|
inactiveFlag;
|
|
|
|
}
|
|
|
|
const char * SCI_METHOD NameOfStyle(int style) override {
|
|
|
|
if (style >= NamedStyles())
|
|
|
|
return "";
|
|
|
|
if (style < sizeLexicalClasses)
|
|
|
|
return lexicalClasses[style].name;
|
|
|
|
// TODO: inactive and substyles
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
const char * SCI_METHOD TagsOfStyle(int style) override {
|
|
|
|
if (style >= NamedStyles())
|
|
|
|
return "Excess";
|
|
|
|
returnBuffer.clear();
|
|
|
|
const int firstSubStyle = subStyles.FirstAllocated();
|
|
|
|
if (firstSubStyle >= 0) {
|
|
|
|
const int lastSubStyle = subStyles.LastAllocated();
|
|
|
|
if (((style >= firstSubStyle) && (style <= (lastSubStyle))) ||
|
|
|
|
((style >= firstSubStyle + inactiveFlag) && (style <= (lastSubStyle + inactiveFlag)))) {
|
|
|
|
int styleActive = style;
|
|
|
|
if (style > lastSubStyle) {
|
|
|
|
returnBuffer = "inactive ";
|
|
|
|
styleActive -= inactiveFlag;
|
|
|
|
}
|
|
|
|
const int styleMain = StyleFromSubStyle(styleActive);
|
|
|
|
returnBuffer += lexicalClasses[styleMain].tags;
|
|
|
|
return returnBuffer.c_str();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (style < sizeLexicalClasses)
|
|
|
|
return lexicalClasses[style].tags;
|
|
|
|
if (style >= inactiveFlag) {
|
|
|
|
returnBuffer = "inactive ";
|
|
|
|
const int styleActive = style - inactiveFlag;
|
|
|
|
if (styleActive < sizeLexicalClasses)
|
|
|
|
returnBuffer += lexicalClasses[styleActive].tags;
|
|
|
|
else
|
2022-08-27 07:35:52 +00:00
|
|
|
returnBuffer.clear();
|
2022-01-04 23:07:50 +00:00
|
|
|
return returnBuffer.c_str();
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
const char * SCI_METHOD DescriptionOfStyle(int style) override {
|
|
|
|
if (style >= NamedStyles())
|
|
|
|
return "";
|
|
|
|
if (style < sizeLexicalClasses)
|
|
|
|
return lexicalClasses[style].description;
|
|
|
|
// TODO: inactive and substyles
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
// ILexer5 methods
|
|
|
|
const char * SCI_METHOD GetName() override {
|
|
|
|
return caseSensitive ? "cpp" : "cppnocase";
|
|
|
|
}
|
|
|
|
int SCI_METHOD GetIdentifier() override {
|
|
|
|
return caseSensitive ? SCLEX_CPP : SCLEX_CPPNOCASE;
|
|
|
|
}
|
|
|
|
const char * SCI_METHOD PropertyGet(const char *key) override;
|
|
|
|
|
|
|
|
static ILexer5 *LexerFactoryCPP() {
|
|
|
|
return new LexerCPP(true);
|
|
|
|
}
|
|
|
|
static ILexer5 *LexerFactoryCPPInsensitive() {
|
|
|
|
return new LexerCPP(false);
|
|
|
|
}
|
|
|
|
constexpr static int MaskActive(int style) noexcept {
|
|
|
|
return style & ~inactiveFlag;
|
|
|
|
}
|
|
|
|
void EvaluateTokens(Tokens &tokens, const SymbolTable &preprocessorDefinitions);
|
|
|
|
Tokens Tokenize(const std::string &expr) const;
|
|
|
|
bool EvaluateExpression(const std::string &expr, const SymbolTable &preprocessorDefinitions);
|
|
|
|
};
|
|
|
|
|
|
|
|
Sci_Position SCI_METHOD LexerCPP::PropertySet(const char *key, const char *val) {
|
|
|
|
if (osCPP.PropertySet(&options, key, val)) {
|
|
|
|
if (strcmp(key, "lexer.cpp.allow.dollars") == 0) {
|
|
|
|
setWord = CharacterSet(CharacterSet::setAlphaNum, "._", true);
|
|
|
|
if (options.identifiersAllowDollars) {
|
|
|
|
setWord.Add('$');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char * SCI_METHOD LexerCPP::PropertyGet(const char *key) {
|
|
|
|
return osCPP.PropertyGet(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
Sci_Position SCI_METHOD LexerCPP::WordListSet(int n, const char *wl) {
|
|
|
|
WordList *wordListN = nullptr;
|
|
|
|
switch (n) {
|
|
|
|
case 0:
|
|
|
|
wordListN = &keywords;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
wordListN = &keywords2;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
wordListN = &keywords3;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
wordListN = &keywords4;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
wordListN = &ppDefinitions;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
wordListN = &markerList;
|
|
|
|
break;
|
2022-10-12 18:45:40 +00:00
|
|
|
default:
|
|
|
|
break;
|
2022-01-04 23:07:50 +00:00
|
|
|
}
|
|
|
|
Sci_Position firstModification = -1;
|
|
|
|
if (wordListN) {
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
if (wordListN->Set(wl)) {
|
2022-01-04 23:07:50 +00:00
|
|
|
firstModification = 0;
|
|
|
|
if (n == 4) {
|
|
|
|
// Rebuild preprocessorDefinitions
|
|
|
|
preprocessorDefinitionsStart.clear();
|
|
|
|
for (int nDefinition = 0; nDefinition < ppDefinitions.Length(); nDefinition++) {
|
|
|
|
const char *cpDefinition = ppDefinitions.WordAt(nDefinition);
|
|
|
|
const char *cpEquals = strchr(cpDefinition, '=');
|
|
|
|
if (cpEquals) {
|
|
|
|
std::string name(cpDefinition, cpEquals - cpDefinition);
|
|
|
|
std::string val(cpEquals+1);
|
|
|
|
const size_t bracket = name.find('(');
|
|
|
|
const size_t bracketEnd = name.find(')');
|
|
|
|
if ((bracket != std::string::npos) && (bracketEnd != std::string::npos)) {
|
|
|
|
// Macro
|
|
|
|
std::string args = name.substr(bracket + 1, bracketEnd - bracket - 1);
|
|
|
|
name = name.substr(0, bracket);
|
|
|
|
preprocessorDefinitionsStart[name] = SymbolValue(val, args);
|
|
|
|
} else {
|
|
|
|
preprocessorDefinitionsStart[name] = val;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
std::string name(cpDefinition);
|
|
|
|
std::string val("1");
|
|
|
|
preprocessorDefinitionsStart[name] = val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return firstModification;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SCI_METHOD LexerCPP::Lex(Sci_PositionU startPos, Sci_Position length, int initStyle, IDocument *pAccess) {
|
|
|
|
LexAccessor styler(pAccess);
|
|
|
|
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
const StyleContext::Transform transform = caseSensitive ?
|
|
|
|
StyleContext::Transform::none : StyleContext::Transform::lower;
|
|
|
|
|
2023-05-31 23:11:12 +00:00
|
|
|
const CharacterSet setOKBeforeRE("([{=,:;!%^&*|?~+-");
|
|
|
|
const CharacterSet setCouldBePostOp("+-");
|
2022-01-04 23:07:50 +00:00
|
|
|
|
|
|
|
const CharacterSet setDoxygen(CharacterSet::setAlpha, "$@\\&<>#{}[]");
|
|
|
|
|
|
|
|
setWordStart = CharacterSet(CharacterSet::setAlpha, "_", true);
|
|
|
|
|
2023-05-31 23:11:12 +00:00
|
|
|
const CharacterSet setInvalidRawFirst(" )\\\t\v\f\n");
|
2022-01-04 23:07:50 +00:00
|
|
|
|
|
|
|
if (options.identifiersAllowDollars) {
|
|
|
|
setWordStart.Add('$');
|
|
|
|
}
|
|
|
|
|
|
|
|
int chPrevNonWhite = ' ';
|
|
|
|
int visibleChars = 0;
|
|
|
|
bool lastWordWasUUID = false;
|
|
|
|
int styleBeforeDCKeyword = SCE_C_DEFAULT;
|
|
|
|
int styleBeforeTaskMarker = SCE_C_DEFAULT;
|
|
|
|
bool continuationLine = false;
|
|
|
|
bool isIncludePreprocessor = false;
|
|
|
|
bool isStringInPreprocessor = false;
|
|
|
|
bool inRERange = false;
|
|
|
|
bool seenDocKeyBrace = false;
|
|
|
|
|
|
|
|
Sci_Position lineCurrent = styler.GetLine(startPos);
|
|
|
|
if ((MaskActive(initStyle) == SCE_C_PREPROCESSOR) ||
|
|
|
|
(MaskActive(initStyle) == SCE_C_COMMENTLINE) ||
|
|
|
|
(MaskActive(initStyle) == SCE_C_COMMENTLINEDOC)) {
|
|
|
|
// Set continuationLine if last character of previous line is '\'
|
|
|
|
if (lineCurrent > 0) {
|
|
|
|
const Sci_Position endLinePrevious = styler.LineEnd(lineCurrent - 1);
|
|
|
|
if (endLinePrevious > 0) {
|
|
|
|
continuationLine = styler.SafeGetCharAt(endLinePrevious-1) == '\\';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// look back to set chPrevNonWhite properly for better regex colouring
|
|
|
|
if (startPos > 0) {
|
|
|
|
Sci_Position back = startPos;
|
|
|
|
while (--back && IsSpaceEquiv(MaskActive(styler.StyleAt(back))))
|
|
|
|
;
|
|
|
|
if (MaskActive(styler.StyleAt(back)) == SCE_C_OPERATOR) {
|
|
|
|
chPrevNonWhite = styler.SafeGetCharAt(back);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StyleContext sc(startPos, length, initStyle, styler);
|
|
|
|
LinePPState preproc = vlls.ForLine(lineCurrent);
|
|
|
|
|
|
|
|
bool definitionsChanged = false;
|
|
|
|
|
|
|
|
// Truncate ppDefineHistory before current line
|
|
|
|
|
|
|
|
if (!options.updatePreprocessor)
|
|
|
|
ppDefineHistory.clear();
|
|
|
|
|
|
|
|
const std::vector<PPDefinition>::iterator itInvalid = std::find_if(
|
|
|
|
ppDefineHistory.begin(), ppDefineHistory.end(),
|
|
|
|
[lineCurrent](const PPDefinition &p) noexcept { return p.line >= lineCurrent; });
|
|
|
|
if (itInvalid != ppDefineHistory.end()) {
|
|
|
|
ppDefineHistory.erase(itInvalid, ppDefineHistory.end());
|
|
|
|
definitionsChanged = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
SymbolTable preprocessorDefinitions = preprocessorDefinitionsStart;
|
|
|
|
for (const PPDefinition &ppDef : ppDefineHistory) {
|
|
|
|
if (ppDef.isUndef)
|
|
|
|
preprocessorDefinitions.erase(ppDef.key);
|
|
|
|
else
|
|
|
|
preprocessorDefinitions[ppDef.key] = SymbolValue(ppDef.value, ppDef.arguments);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string rawStringTerminator = rawStringTerminators.ValueAt(lineCurrent-1);
|
|
|
|
SparseState<std::string> rawSTNew(lineCurrent);
|
|
|
|
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
std::string currentText;
|
|
|
|
|
2022-01-04 23:07:50 +00:00
|
|
|
int activitySet = preproc.ActiveState();
|
|
|
|
|
|
|
|
const WordClassifier &classifierIdentifiers = subStyles.Classifier(SCE_C_IDENTIFIER);
|
|
|
|
const WordClassifier &classifierDocKeyWords = subStyles.Classifier(SCE_C_COMMENTDOCKEYWORD);
|
|
|
|
|
|
|
|
Sci_PositionU lineEndNext = styler.LineEnd(lineCurrent);
|
|
|
|
|
|
|
|
for (; sc.More();) {
|
|
|
|
|
|
|
|
if (sc.atLineStart) {
|
|
|
|
// Using MaskActive() is not needed in the following statement.
|
|
|
|
// Inside inactive preprocessor declaration, state will be reset anyway at the end of this block.
|
|
|
|
if ((sc.state == SCE_C_STRING) || (sc.state == SCE_C_CHARACTER)) {
|
|
|
|
// Prevent SCE_C_STRINGEOL from leaking back to previous line which
|
|
|
|
// ends with a line continuation by locking in the state up to this position.
|
|
|
|
sc.SetState(sc.state);
|
|
|
|
}
|
|
|
|
if ((MaskActive(sc.state) == SCE_C_PREPROCESSOR) && (!continuationLine)) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
// Reset states to beginning of colourise so no surprises
|
|
|
|
// if different sets of lines lexed.
|
|
|
|
visibleChars = 0;
|
|
|
|
lastWordWasUUID = false;
|
|
|
|
isIncludePreprocessor = false;
|
|
|
|
inRERange = false;
|
|
|
|
if (preproc.IsInactive()) {
|
|
|
|
activitySet = inactiveFlag;
|
|
|
|
sc.SetState(sc.state | activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sc.atLineEnd) {
|
|
|
|
lineCurrent++;
|
|
|
|
lineEndNext = styler.LineEnd(lineCurrent);
|
|
|
|
vlls.Add(lineCurrent, preproc);
|
2022-08-27 07:35:52 +00:00
|
|
|
if (!rawStringTerminator.empty()) {
|
2022-01-04 23:07:50 +00:00
|
|
|
rawSTNew.Set(lineCurrent-1, rawStringTerminator);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle line continuation generically.
|
|
|
|
if (sc.ch == '\\') {
|
|
|
|
if ((sc.currentPos+1) >= lineEndNext) {
|
|
|
|
lineCurrent++;
|
|
|
|
lineEndNext = styler.LineEnd(lineCurrent);
|
|
|
|
vlls.Add(lineCurrent, preproc);
|
2022-08-27 07:35:52 +00:00
|
|
|
if (!rawStringTerminator.empty()) {
|
2022-01-04 23:07:50 +00:00
|
|
|
rawSTNew.Set(lineCurrent-1, rawStringTerminator);
|
|
|
|
}
|
|
|
|
sc.Forward();
|
|
|
|
if (sc.ch == '\r' && sc.chNext == '\n') {
|
|
|
|
// Even in UTF-8, \r and \n are separate
|
|
|
|
sc.Forward();
|
|
|
|
}
|
|
|
|
continuationLine = true;
|
|
|
|
sc.Forward();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const bool atLineEndBeforeSwitch = sc.atLineEnd;
|
|
|
|
|
|
|
|
// Determine if the current state should terminate.
|
|
|
|
switch (MaskActive(sc.state)) {
|
|
|
|
case SCE_C_OPERATOR:
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
break;
|
|
|
|
case SCE_C_NUMBER:
|
|
|
|
// We accept almost anything because of hex. and number suffixes
|
|
|
|
if (sc.ch == '_') {
|
|
|
|
sc.ChangeState(SCE_C_USERLITERAL|activitySet);
|
|
|
|
} else if (!(setWord.Contains(sc.ch)
|
|
|
|
|| (sc.ch == '\'')
|
|
|
|
|| ((sc.ch == '+' || sc.ch == '-') && (sc.chPrev == 'e' || sc.chPrev == 'E' ||
|
|
|
|
sc.chPrev == 'p' || sc.chPrev == 'P')))) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_USERLITERAL:
|
|
|
|
if (!(setWord.Contains(sc.ch)))
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
break;
|
|
|
|
case SCE_C_IDENTIFIER:
|
|
|
|
if (sc.atLineStart || sc.atLineEnd || !setWord.Contains(sc.ch) || (sc.ch == '.')) {
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
sc.GetCurrentString(currentText, transform);
|
|
|
|
if (keywords.InList(currentText)) {
|
|
|
|
lastWordWasUUID = currentText == "uuid";
|
2022-01-04 23:07:50 +00:00
|
|
|
sc.ChangeState(SCE_C_WORD|activitySet);
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
} else if (keywords2.InList(currentText)) {
|
2022-01-04 23:07:50 +00:00
|
|
|
sc.ChangeState(SCE_C_WORD2|activitySet);
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
} else if (keywords4.InList(currentText)) {
|
2022-01-04 23:07:50 +00:00
|
|
|
sc.ChangeState(SCE_C_GLOBALCLASS|activitySet);
|
|
|
|
} else {
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
const int subStyle = classifierIdentifiers.ValueFor(currentText);
|
2022-01-04 23:07:50 +00:00
|
|
|
if (subStyle >= 0) {
|
|
|
|
sc.ChangeState(subStyle|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const bool literalString = sc.ch == '\"';
|
|
|
|
if (literalString || sc.ch == '\'') {
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
std::string_view s = currentText;
|
|
|
|
size_t lenS = s.length();
|
2022-01-04 23:07:50 +00:00
|
|
|
const bool raw = literalString && sc.chPrev == 'R' && !setInvalidRawFirst.Contains(sc.chNext);
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
if (raw) {
|
|
|
|
s.remove_suffix(1);
|
|
|
|
lenS--;
|
|
|
|
}
|
2022-01-04 23:07:50 +00:00
|
|
|
const bool valid =
|
|
|
|
(lenS == 0) ||
|
|
|
|
((lenS == 1) && ((s[0] == 'L') || (s[0] == 'u') || (s[0] == 'U'))) ||
|
|
|
|
((lenS == 2) && literalString && (s[0] == 'u') && (s[1] == '8'));
|
|
|
|
if (valid) {
|
|
|
|
if (literalString) {
|
|
|
|
if (raw) {
|
|
|
|
// Set the style of the string prefix to SCE_C_STRINGRAW but then change to
|
|
|
|
// SCE_C_DEFAULT as that allows the raw string start code to run.
|
|
|
|
sc.ChangeState(SCE_C_STRINGRAW|activitySet);
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else {
|
|
|
|
sc.ChangeState(SCE_C_STRING|activitySet);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
sc.ChangeState(SCE_C_CHARACTER|activitySet);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_DEFAULT | activitySet);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_PREPROCESSOR:
|
|
|
|
if (options.stylingWithinPreprocessor) {
|
|
|
|
if (IsASpace(sc.ch) || (sc.ch == '(')) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
} else if (isStringInPreprocessor && (sc.Match('>') || sc.Match('\"') || sc.atLineEnd)) {
|
|
|
|
isStringInPreprocessor = false;
|
|
|
|
} else if (!isStringInPreprocessor) {
|
|
|
|
if ((isIncludePreprocessor && sc.Match('<')) || sc.Match('\"')) {
|
|
|
|
isStringInPreprocessor = true;
|
|
|
|
} else if (sc.Match('/', '*')) {
|
|
|
|
if (sc.Match("/**") || sc.Match("/*!")) {
|
|
|
|
sc.SetState(SCE_C_PREPROCESSORCOMMENTDOC|activitySet);
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_PREPROCESSORCOMMENT|activitySet);
|
|
|
|
}
|
|
|
|
sc.Forward(); // Eat the *
|
|
|
|
} else if (sc.Match('/', '/')) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_PREPROCESSORCOMMENT:
|
|
|
|
case SCE_C_PREPROCESSORCOMMENTDOC:
|
|
|
|
if (sc.Match('*', '/')) {
|
|
|
|
sc.Forward();
|
|
|
|
sc.ForwardSetState(SCE_C_PREPROCESSOR|activitySet);
|
|
|
|
continue; // Without advancing in case of '\'.
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_COMMENT:
|
|
|
|
if (sc.Match('*', '/')) {
|
|
|
|
sc.Forward();
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else {
|
|
|
|
styleBeforeTaskMarker = SCE_C_COMMENT;
|
|
|
|
highlightTaskMarker(sc, styler, activitySet, markerList, caseSensitive);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_COMMENTDOC:
|
|
|
|
if (sc.Match('*', '/')) {
|
|
|
|
sc.Forward();
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else if (sc.ch == '@' || sc.ch == '\\') { // JavaDoc and Doxygen support
|
|
|
|
// Verify that we have the conditions to mark a comment-doc-keyword
|
|
|
|
if ((IsASpace(sc.chPrev) || sc.chPrev == '*') && (!IsASpace(sc.chNext))) {
|
|
|
|
styleBeforeDCKeyword = SCE_C_COMMENTDOC;
|
|
|
|
sc.SetState(SCE_C_COMMENTDOCKEYWORD|activitySet);
|
|
|
|
}
|
|
|
|
} else if ((sc.ch == '<' && sc.chNext != '/')
|
|
|
|
|| (sc.ch == '/' && sc.chPrev == '<')) { // XML comment style
|
|
|
|
styleBeforeDCKeyword = SCE_C_COMMENTDOC;
|
|
|
|
sc.ForwardSetState(SCE_C_COMMENTDOCKEYWORD | activitySet);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_COMMENTLINE:
|
|
|
|
if (sc.atLineStart && !continuationLine) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else {
|
|
|
|
styleBeforeTaskMarker = SCE_C_COMMENTLINE;
|
|
|
|
highlightTaskMarker(sc, styler, activitySet, markerList, caseSensitive);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_COMMENTLINEDOC:
|
|
|
|
if (sc.atLineStart && !continuationLine) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else if (sc.ch == '@' || sc.ch == '\\') { // JavaDoc and Doxygen support
|
|
|
|
// Verify that we have the conditions to mark a comment-doc-keyword
|
|
|
|
if ((IsASpace(sc.chPrev) || sc.chPrev == '/' || sc.chPrev == '!') && (!IsASpace(sc.chNext))) {
|
|
|
|
styleBeforeDCKeyword = SCE_C_COMMENTLINEDOC;
|
|
|
|
sc.SetState(SCE_C_COMMENTDOCKEYWORD|activitySet);
|
|
|
|
}
|
|
|
|
} else if ((sc.ch == '<' && sc.chNext != '/')
|
|
|
|
|| (sc.ch == '/' && sc.chPrev == '<')) { // XML comment style
|
|
|
|
styleBeforeDCKeyword = SCE_C_COMMENTLINEDOC;
|
|
|
|
sc.ForwardSetState(SCE_C_COMMENTDOCKEYWORD | activitySet);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_COMMENTDOCKEYWORD:
|
|
|
|
if ((styleBeforeDCKeyword == SCE_C_COMMENTDOC) && sc.Match('*', '/')) {
|
|
|
|
sc.ChangeState(SCE_C_COMMENTDOCKEYWORDERROR);
|
|
|
|
sc.Forward();
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
seenDocKeyBrace = false;
|
|
|
|
} else if (sc.ch == '[' || sc.ch == '{') {
|
|
|
|
seenDocKeyBrace = true;
|
|
|
|
} else if (!setDoxygen.Contains(sc.ch)
|
|
|
|
&& !(seenDocKeyBrace && (sc.ch == ',' || sc.ch == '.'))) {
|
|
|
|
if (!(IsASpace(sc.ch) || (sc.ch == 0))) {
|
|
|
|
sc.ChangeState(SCE_C_COMMENTDOCKEYWORDERROR|activitySet);
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
} else {
|
|
|
|
sc.GetCurrentString(currentText, transform);
|
|
|
|
assert(!currentText.empty());
|
|
|
|
std::string currentSuffix = currentText.substr(1);
|
|
|
|
if (!keywords3.InList(currentSuffix) && !keywords3.InList(currentText)) {
|
Update: Scintilla 5.3.6 and Lexilla 5.2.6
update to Scinitlla Release 5.3.6 (https://www.scintilla.org/scintilla536.zip)
Released 26 July 2023.
Redraw calltip after showing as didn't update when size of new text exactly same as previous. Feature #1486.
On Win32 fix reverse arrow cursor when scaled. Bug #2382.
On Win32 hide cursor when typing if that system preference has been chosen. Bug #2333.
On Win32 and Qt, stop aligning IME candidate window to target. It is now always aligned to start of composition string. This undoes part of feature #1300. Feature #1488, Bug #2391, Feature #1300.
On Qt, for IMEs, update micro focus when selection changes. This may move the location of IME popups to align with the caret.
On Qt, implement replacement for IMEs which may help with actions like reconversion. This is similar to delete-surrounding on GTK.
and Lexilla Release 5.2.6 (https://www.scintilla.org/lexilla526.zip)
Released 26 July 2023.
Include empty word list names in value returned by DescribeWordListSets and SCI_DESCRIBEKEYWORDSETS. Issue #175, Pull request #176.
Bash: style here-doc end delimiters as SCE_SH_HERE_DELIM instead of SCE_SH_HERE_Q. Issue #177.
Bash: allow '$' as last character in string. Issue #180, Pull request #181.
Bash: fix state after expansion. Highlight all numeric and file test operators. Don't highlight dash in long option as operator. Issue #182, Pull request #183.
Bash: strict checking of special parameters ($*, $@, $$, ...) with property lexer.bash.special.parameter to specify valid parameters. Issue #184, Pull request #186.
Bash: recognize keyword before redirection operators (< and >). Issue #188, Pull request #189.
Errorlist: recognize Bash diagnostic messages.
HTML: allow ASP block to terminate inside line comment. Issue #185.
HTML: fix folding with JSP/ASP.NET <%-- comment. Issue #191.
HTML: fix incremental styling of multi-line ASP.NET directive. Issue #191.
Matlab: improve arguments blocks. Add support for multiple arguments blocks. Prevent "arguments" from being keyword in function declaration line. Fix semicolon handling. Pull request #179.
Visual Prolog: add support for embedded syntax with SCE_VISUALPROLOG_EMBEDDED and SCE_VISUALPROLOG_PLACEHOLDER.
Styling of string literals changed with no differentiation between literals with quotes and those that are prefixed with "@". Quote characters are in a separate style (SCE_VISUALPROLOG_STRING_QUOTE) to contents (SCE_VISUALPROLOG_STRING).
SCE_VISUALPROLOG_CHARACTER, SCE_VISUALPROLOG_CHARACTER_TOO_MANY, SCE_VISUALPROLOG_CHARACTER_ESCAPE_ERROR, SCE_VISUALPROLOG_STRING_EOL_OPEN, and SCE_VISUALPROLOG_STRING_VERBATIM_SPECIAL were removed (replaced with SCE_VISUALPROLOG_UNUSED[1-5]). Pull request #178.
Fix #13901, fix #13911, fix #13943, close #13940
2023-07-27 17:57:12 +00:00
|
|
|
const int subStyleCDKW = classifierDocKeyWords.ValueFor(currentSuffix);
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
if (subStyleCDKW >= 0) {
|
|
|
|
sc.ChangeState(subStyleCDKW | activitySet);
|
|
|
|
} else {
|
|
|
|
sc.ChangeState(SCE_C_COMMENTDOCKEYWORDERROR | activitySet);
|
|
|
|
}
|
2022-01-04 23:07:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
sc.SetState(styleBeforeDCKeyword|activitySet);
|
|
|
|
seenDocKeyBrace = false;
|
|
|
|
} else if (sc.ch == '>') {
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
sc.GetCurrentString(currentText, transform);
|
|
|
|
if (!keywords3.InList(currentText)) {
|
|
|
|
const int subStyleCDKW = classifierDocKeyWords.ValueFor(currentText.substr(1));
|
2022-01-04 23:07:50 +00:00
|
|
|
if (subStyleCDKW >= 0) {
|
|
|
|
sc.ChangeState(subStyleCDKW | activitySet);
|
|
|
|
} else {
|
|
|
|
sc.ChangeState(SCE_C_COMMENTDOCKEYWORDERROR | activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sc.SetState(styleBeforeDCKeyword | activitySet);
|
|
|
|
seenDocKeyBrace = false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_STRING:
|
|
|
|
if (sc.atLineEnd) {
|
|
|
|
sc.ChangeState(SCE_C_STRINGEOL|activitySet);
|
|
|
|
} else if (isIncludePreprocessor) {
|
|
|
|
if (sc.ch == '>') {
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
isIncludePreprocessor = false;
|
|
|
|
}
|
|
|
|
} else if (sc.ch == '\\') {
|
|
|
|
if (options.escapeSequence) {
|
|
|
|
sc.SetState(SCE_C_ESCAPESEQUENCE|activitySet);
|
|
|
|
escapeSeq.resetEscapeState(sc.chNext);
|
|
|
|
}
|
|
|
|
sc.Forward(); // Skip all characters after the backslash
|
|
|
|
} else if (sc.ch == '\"') {
|
|
|
|
if (sc.chNext == '_') {
|
|
|
|
sc.ChangeState(SCE_C_USERLITERAL|activitySet);
|
|
|
|
} else {
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_ESCAPESEQUENCE:
|
|
|
|
escapeSeq.consumeDigit();
|
|
|
|
if (!escapeSeq.atEscapeEnd(sc.ch)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (sc.ch == '"') {
|
|
|
|
sc.SetState(SCE_C_STRING|activitySet);
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else if (sc.ch == '\\') {
|
|
|
|
escapeSeq.resetEscapeState(sc.chNext);
|
|
|
|
sc.Forward();
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_STRING|activitySet);
|
|
|
|
if (sc.atLineEnd) {
|
|
|
|
sc.ChangeState(SCE_C_STRINGEOL|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_HASHQUOTEDSTRING:
|
|
|
|
if (sc.ch == '\\') {
|
|
|
|
if (sc.chNext == '\"' || sc.chNext == '\'' || sc.chNext == '\\') {
|
|
|
|
sc.Forward();
|
|
|
|
}
|
|
|
|
} else if (sc.ch == '\"') {
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_STRINGRAW:
|
|
|
|
if (sc.Match(rawStringTerminator.c_str())) {
|
|
|
|
for (size_t termPos=rawStringTerminator.size(); termPos; termPos--)
|
|
|
|
sc.Forward();
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
2022-08-27 07:35:52 +00:00
|
|
|
rawStringTerminator.clear();
|
2022-01-04 23:07:50 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_CHARACTER:
|
|
|
|
if (sc.atLineEnd) {
|
|
|
|
sc.ChangeState(SCE_C_STRINGEOL|activitySet);
|
|
|
|
} else if (sc.ch == '\\') {
|
|
|
|
if (sc.chNext == '\"' || sc.chNext == '\'' || sc.chNext == '\\') {
|
|
|
|
sc.Forward();
|
|
|
|
}
|
|
|
|
} else if (sc.ch == '\'') {
|
|
|
|
if (sc.chNext == '_') {
|
|
|
|
sc.ChangeState(SCE_C_USERLITERAL|activitySet);
|
|
|
|
} else {
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_REGEX:
|
|
|
|
if (sc.atLineStart) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else if (!inRERange && sc.ch == '/') {
|
|
|
|
sc.Forward();
|
|
|
|
while (IsLowerCase(sc.ch))
|
|
|
|
sc.Forward(); // gobble regex flags
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
} else if (sc.ch == '\\' && ((sc.currentPos+1) < lineEndNext)) {
|
|
|
|
// Gobble up the escaped character
|
|
|
|
sc.Forward();
|
|
|
|
} else if (sc.ch == '[') {
|
|
|
|
inRERange = true;
|
|
|
|
} else if (sc.ch == ']') {
|
|
|
|
inRERange = false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_STRINGEOL:
|
|
|
|
if (sc.atLineStart) {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_VERBATIM:
|
|
|
|
if (options.verbatimStringsAllowEscapes && (sc.ch == '\\')) {
|
|
|
|
sc.Forward(); // Skip all characters after the backslash
|
|
|
|
} else if (sc.ch == '\"') {
|
|
|
|
if (sc.chNext == '\"') {
|
|
|
|
sc.Forward();
|
|
|
|
} else {
|
|
|
|
sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_TRIPLEVERBATIM:
|
|
|
|
if (sc.Match(R"(""")")) {
|
|
|
|
while (sc.Match('"')) {
|
|
|
|
sc.Forward();
|
|
|
|
}
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_UUID:
|
|
|
|
if (sc.atLineEnd || sc.ch == ')') {
|
|
|
|
sc.SetState(SCE_C_DEFAULT|activitySet);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SCE_C_TASKMARKER:
|
Update scintilla 5.3.4 and lexilla 5.2.4 with:
https://www.scintilla.org/scintilla534.zip
Released 8 March 2023.
Add multithreaded wrap to significantly improve performance of wrapping large files.
More typesafe bindings of *Full APIs in ScintillaCall. Feature #1477.
Fix overlapping of text with line end wrap marker. Bug #2378.
Fix clipping of line end wrap symbol for SC_WRAPVISUALFLAGLOC_END_BY_TEXT.
Where a multi-byte character contains multiple styles, display each byte as a representation. This makes it easier to see and fix lexers that change styles mid-character, commonly because they use fixed size buffers.
Fix a potential crash with autocompletion list fill-ups where a SCN_CHARADDED handler retriggered an autocompletion list, but with no items that match the typed character.
lexilla523
Released 8 March 2023.
Add scripts/PromoteNew.bat script to promote .new files after checking.
Makefile: Remove 1024-byte line length limit..
Ruby: Add new lexical classes for % literals SCE_RB_STRING_W (%w non-interpolable string array), SCE_RB_STRING_I (%i non-interpolable symbol array), SCE_RB_STRING_QI (%I interpolable symbol array), and SCE_RB_STRING_QS (%s symbol). Issue #124.
Ruby: Disambiguate %= which may be a quote or modulo assignment. Issue #124, Bug #1255, Bug #2182.
Ruby: Fix additional fold level for single character in SCE_RB_STRING_QW. Issue #132.
Ruby: Set SCE_RB_HERE_QQ for unquoted and double-quoted heredocs and SCE_RB_HERE_QX for backticks-quoted heredocs. Issue #134.
Ruby: Recognise #{} inside SCE_RB_HERE_QQ and SCE_RB_HERE_QX. Issue #134.
Ruby: Improve regex and heredoc recognition. Issue #136.
Ruby: Highlight #@, #@@ and #$ style interpolation. Issue #140.
Ruby: Fix folding for multiple heredocs started on one line. Fix folding when there is a space after heredoc opening delimiter. Issue #135.
YAML: Remove 1024-byte line length limit.
https://www.scintilla.org/lexilla524.zip
Released 13 March 2023.
C++: Fix failure to recognize keywords containing upper case. Issue #149.
GDScript: Support % and $ node paths. Issue #145, Pull request #146.
Close #13338
2023-03-10 02:37:21 +00:00
|
|
|
if (IsOperatorOrSpace(sc.ch)) {
|
2022-01-04 23:07:50 +00:00
|
|
|
sc.SetState(styleBeforeTaskMarker|activitySet);
|
|
|
|
styleBeforeTaskMarker = SCE_C_DEFAULT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sc.atLineEnd && !atLineEndBeforeSwitch) {
|
|
|
|
// State exit processing consumed characters up to end of line.
|
|
|
|
lineCurrent++;
|
|
|
|
lineEndNext = styler.LineEnd(lineCurrent);
|
|
|
|
vlls.Add(lineCurrent, preproc);
|
|
|
|
}
|
|
|
|
|
|
|
|
const bool atLineEndBeforeStateEntry = sc.atLineEnd;
|
|
|
|
|
|
|
|
// Determine if a new state should be entered.
|
|
|
|
if (MaskActive(sc.state) == SCE_C_DEFAULT) {
|
|
|
|
if (sc.Match('@', '\"')) {
|
|
|
|
sc.SetState(SCE_C_VERBATIM|activitySet);
|
|
|
|
sc.Forward();
|
|
|
|
} else if (options.triplequotedStrings && sc.Match(R"(""")")) {
|
|
|
|
sc.SetState(SCE_C_TRIPLEVERBATIM|activitySet);
|
|
|
|
sc.Forward(2);
|
|
|
|
} else if (options.hashquotedStrings && sc.Match('#', '\"')) {
|
|
|
|
sc.SetState(SCE_C_HASHQUOTEDSTRING|activitySet);
|
|
|
|
sc.Forward();
|
|
|
|
} else if (options.backQuotedStrings && sc.Match('`')) {
|
|
|
|
sc.SetState(SCE_C_STRINGRAW|activitySet);
|
|
|
|
rawStringTerminator = "`";
|
|
|
|
} else if (IsADigit(sc.ch) || (sc.ch == '.' && IsADigit(sc.chNext))) {
|
|
|
|
if (lastWordWasUUID) {
|
|
|
|
sc.SetState(SCE_C_UUID|activitySet);
|
|
|
|
lastWordWasUUID = false;
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_NUMBER|activitySet);
|
|
|
|
}
|
|
|
|
} else if (!sc.atLineEnd && (setWordStart.Contains(sc.ch) || (sc.ch == '@'))) {
|
|
|
|
if (lastWordWasUUID) {
|
|
|
|
sc.SetState(SCE_C_UUID|activitySet);
|
|
|
|
lastWordWasUUID = false;
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_IDENTIFIER|activitySet);
|
|
|
|
}
|
|
|
|
} else if (sc.Match('/', '*')) {
|
|
|
|
if (sc.Match("/**") || sc.Match("/*!")) { // Support of Qt/Doxygen doc. style
|
|
|
|
sc.SetState(SCE_C_COMMENTDOC|activitySet);
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_COMMENT|activitySet);
|
|
|
|
}
|
|
|
|
sc.Forward(); // Eat the * so it isn't used for the end of the comment
|
|
|
|
} else if (sc.Match('/', '/')) {
|
|
|
|
if ((sc.Match("///") && !sc.Match("////")) || sc.Match("//!"))
|
|
|
|
// Support of Qt/Doxygen doc. style
|
|
|
|
sc.SetState(SCE_C_COMMENTLINEDOC|activitySet);
|
|
|
|
else
|
|
|
|
sc.SetState(SCE_C_COMMENTLINE|activitySet);
|
|
|
|
} else if (sc.ch == '/'
|
|
|
|
&& (setOKBeforeRE.Contains(chPrevNonWhite)
|
|
|
|
|| followsReturnKeyword(sc, styler))
|
|
|
|
&& (!setCouldBePostOp.Contains(chPrevNonWhite)
|
|
|
|
|| !FollowsPostfixOperator(sc, styler))) {
|
|
|
|
sc.SetState(SCE_C_REGEX|activitySet); // JavaScript's RegEx
|
|
|
|
inRERange = false;
|
|
|
|
} else if (sc.ch == '\"') {
|
|
|
|
if (sc.chPrev == 'R') {
|
|
|
|
styler.Flush();
|
|
|
|
if (MaskActive(styler.StyleAt(sc.currentPos - 1)) == SCE_C_STRINGRAW) {
|
|
|
|
sc.SetState(SCE_C_STRINGRAW|activitySet);
|
|
|
|
rawStringTerminator = ")";
|
|
|
|
for (Sci_Position termPos = sc.currentPos + 1;; termPos++) {
|
|
|
|
const char chTerminator = styler.SafeGetCharAt(termPos, '(');
|
|
|
|
if (chTerminator == '(')
|
|
|
|
break;
|
|
|
|
rawStringTerminator += chTerminator;
|
|
|
|
}
|
|
|
|
rawStringTerminator += '\"';
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_STRING|activitySet);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
sc.SetState(SCE_C_STRING|activitySet);
|
|
|
|
}
|
|
|
|
isIncludePreprocessor = false; // ensure that '>' won't end the string
|
|
|
|
} else if (isIncludePreprocessor && sc.ch == '<') {
|
|
|
|
sc.SetState(SCE_C_STRING|activitySet);
|
|
|
|
} else if (sc.ch == '\'') {
|
|
|
|
sc.SetState(SCE_C_CHARACTER|activitySet);
|
|
|
|
} else if (sc.ch == '#' && visibleChars == 0) {
|
|
|
|
// Preprocessor commands are alone on their line
|
|
|
|
sc.SetState(SCE_C_PREPROCESSOR|activitySet);
|
|
|
|
// Skip whitespace between # and preprocessor word
|
|
|
|
do {
|
|
|
|
sc.Forward();
|
|
|
|
} while ((sc.ch == ' ' || sc.ch == '\t') && sc.More());
|
|
|
|
if (sc.Match("include")) {
|
|
|
|
isIncludePreprocessor = true;
|
|
|
|
} else {
|
|
|
|
if (options.trackPreprocessor && IsAlphaNumeric(sc.ch)) {
|
|
|
|
// If #if is nested too deeply (>31 levels) the active/inactive appearance
|
|
|
|
// will stop reflecting the code.
|
|
|
|
if (sc.Match("ifdef") || sc.Match("ifndef")) {
|
|
|
|
const bool isIfDef = sc.Match("ifdef");
|
|
|
|
const int startRest = isIfDef ? 5 : 6;
|
|
|
|
const std::string restOfLine = GetRestOfLine(styler, sc.currentPos + startRest + 1, false);
|
|
|
|
const bool foundDef = preprocessorDefinitions.find(restOfLine) != preprocessorDefinitions.end();
|
|
|
|
preproc.StartSection(isIfDef == foundDef);
|
|
|
|
} else if (sc.Match("if")) {
|
|
|
|
const std::string restOfLine = GetRestOfLine(styler, sc.currentPos + 2, true);
|
|
|
|
const bool ifGood = EvaluateExpression(restOfLine, preprocessorDefinitions);
|
|
|
|
preproc.StartSection(ifGood);
|
|
|
|
} else if (sc.Match("else")) {
|
|
|
|
// #else is shown as active if either preceding or following section is active
|
|
|
|
// as that means that it contributed to the result.
|
|
|
|
if (preproc.ValidLevel()) {
|
|
|
|
// If #else has no corresponding #if then take no action as invalid
|
|
|
|
if (!preproc.CurrentIfTaken()) {
|
|
|
|
// Inactive, may become active if parent scope active
|
|
|
|
assert(sc.state == (SCE_C_PREPROCESSOR | inactiveFlag));
|
|
|
|
preproc.InvertCurrentLevel();
|
|
|
|
activitySet = preproc.ActiveState();
|
|
|
|
// If following is active then show "else" as active
|
|
|
|
if (!activitySet)
|
|
|
|
sc.ChangeState(SCE_C_PREPROCESSOR);
|
|
|
|
} else if (preproc.IsActive()) {
|
|
|
|
// Active -> inactive
|
|
|
|
assert(sc.state == SCE_C_PREPROCESSOR);
|
|
|
|
preproc.InvertCurrentLevel();
|
|
|
|
activitySet = preproc.ActiveState();
|
|
|
|
// Continue to show "else" as active as it ends active section.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (sc.Match("elif")) {
|
|
|
|
// Ensure only one chosen out of #if .. #elif .. #elif .. #else .. #endif
|
|
|
|
// #elif is shown as active if either preceding or following section is active
|
|
|
|
// as that means that it contributed to the result.
|
|
|
|
if (preproc.ValidLevel()) {
|
|
|
|
if (!preproc.CurrentIfTaken()) {
|
|
|
|
// Inactive, if expression true then may become active if parent scope active
|
|
|
|
assert(sc.state == (SCE_C_PREPROCESSOR | inactiveFlag));
|
|
|
|
// Similar to #if
|
|
|
|
std::string restOfLine = GetRestOfLine(styler, sc.currentPos + 4, true);
|
|
|
|
const bool ifGood = EvaluateExpression(restOfLine, preprocessorDefinitions);
|
|
|
|
if (ifGood) {
|
|
|
|
preproc.InvertCurrentLevel();
|
|
|
|
activitySet = preproc.ActiveState();
|
|
|
|
if (!activitySet)
|
|
|
|
sc.ChangeState(SCE_C_PREPROCESSOR);
|
|
|
|
}
|
|
|
|
} else if (preproc.IsActive()) {
|
|
|
|
// Active -> inactive
|
|
|
|
assert(sc.state == SCE_C_PREPROCESSOR);
|
|
|
|
preproc.InvertCurrentLevel();
|
|
|
|
activitySet = preproc.ActiveState();
|
|
|
|
// Continue to show "elif" as active as it ends active section.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (sc.Match("endif")) {
|
|
|
|
preproc.EndSection();
|
|
|
|
activitySet = preproc.ActiveState();
|
|
|
|
sc.ChangeState(SCE_C_PREPROCESSOR|activitySet);
|
|
|
|
} else if (sc.Match("define")) {
|
|
|
|
if (options.updatePreprocessor && preproc.IsActive()) {
|
|
|
|
std::string restOfLine = GetRestOfLine(styler, sc.currentPos + 6, true);
|
|
|
|
size_t startName = 0;
|
|
|
|
while ((startName < restOfLine.length()) && IsSpaceOrTab(restOfLine[startName]))
|
|
|
|
startName++;
|
|
|
|
size_t endName = startName;
|
|
|
|
while ((endName < restOfLine.length()) && setWord.Contains(restOfLine[endName]))
|
|
|
|
endName++;
|
|
|
|
std::string key = restOfLine.substr(startName, endName-startName);
|
|
|
|
if ((endName < restOfLine.length()) && (restOfLine.at(endName) == '(')) {
|
|
|
|
// Macro
|
|
|
|
size_t endArgs = endName;
|
|
|
|
while ((endArgs < restOfLine.length()) && (restOfLine[endArgs] != ')'))
|
|
|
|
endArgs++;
|
|
|
|
std::string args = restOfLine.substr(endName + 1, endArgs - endName - 1);
|
|
|
|
size_t startValue = endArgs+1;
|
|
|
|
while ((startValue < restOfLine.length()) && IsSpaceOrTab(restOfLine[startValue]))
|
|
|
|
startValue++;
|
|
|
|
std::string value;
|
|
|
|
if (startValue < restOfLine.length())
|
|
|
|
value = restOfLine.substr(startValue);
|
|
|
|
preprocessorDefinitions[key] = SymbolValue(value, args);
|
|
|
|
ppDefineHistory.push_back(PPDefinition(lineCurrent, key, value, false, args));
|
|
|
|
definitionsChanged = true;
|
|
|
|
} else {
|
|
|
|
// Value
|
|
|
|
size_t startValue = endName;
|
|
|
|
while ((startValue < restOfLine.length()) && IsSpaceOrTab(restOfLine[startValue]))
|
|
|
|
startValue++;
|
|
|
|
std::string value = restOfLine.substr(startValue);
|
|
|
|
if (OnlySpaceOrTab(value))
|
|
|
|
value = "1"; // No value defaults to 1
|
|
|
|
preprocessorDefinitions[key] = value;
|
|
|
|
ppDefineHistory.push_back(PPDefinition(lineCurrent, key, value));
|
|
|
|
definitionsChanged = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (sc.Match("undef")) {
|
|
|
|
if (options.updatePreprocessor && preproc.IsActive()) {
|
|
|
|
const std::string restOfLine = GetRestOfLine(styler, sc.currentPos + 5, false);
|
|
|
|
Tokens tokens = Tokenize(restOfLine);
|
2022-10-12 18:45:40 +00:00
|
|
|
if (!tokens.empty()) {
|
2022-01-04 23:07:50 +00:00
|
|
|
const std::string key = tokens[0];
|
|
|
|
preprocessorDefinitions.erase(key);
|
|
|
|
ppDefineHistory.push_back(PPDefinition(lineCurrent, key, "", true));
|
|
|
|
definitionsChanged = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (isoperator(sc.ch)) {
|
|
|
|
sc.SetState(SCE_C_OPERATOR|activitySet);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sc.atLineEnd && !atLineEndBeforeStateEntry) {
|
|
|
|
// State entry processing consumed characters up to end of line.
|
|
|
|
lineCurrent++;
|
|
|
|
lineEndNext = styler.LineEnd(lineCurrent);
|
|
|
|
vlls.Add(lineCurrent, preproc);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsASpace(sc.ch) && !IsSpaceEquiv(MaskActive(sc.state))) {
|
|
|
|
chPrevNonWhite = sc.ch;
|
|
|
|
visibleChars++;
|
|
|
|
}
|
|
|
|
continuationLine = false;
|
|
|
|
sc.Forward();
|
|
|
|
}
|
|
|
|
const bool rawStringsChanged = rawStringTerminators.Merge(rawSTNew, lineCurrent);
|
|
|
|
if (definitionsChanged || rawStringsChanged)
|
|
|
|
styler.ChangeLexerState(startPos, startPos + length);
|
|
|
|
sc.Complete();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store both the current line's fold level and the next lines in the
|
|
|
|
// level store to make it easy to pick up with each increment
|
|
|
|
// and to make it possible to fiddle the current level for "} else {".
|
|
|
|
|
|
|
|
void SCI_METHOD LexerCPP::Fold(Sci_PositionU startPos, Sci_Position length, int initStyle, IDocument *pAccess) {
|
|
|
|
|
|
|
|
if (!options.fold)
|
|
|
|
return;
|
|
|
|
|
|
|
|
LexAccessor styler(pAccess);
|
|
|
|
|
|
|
|
const Sci_PositionU endPos = startPos + length;
|
|
|
|
int visibleChars = 0;
|
|
|
|
bool inLineComment = false;
|
|
|
|
Sci_Position lineCurrent = styler.GetLine(startPos);
|
|
|
|
int levelCurrent = SC_FOLDLEVELBASE;
|
|
|
|
if (lineCurrent > 0)
|
|
|
|
levelCurrent = styler.LevelAt(lineCurrent-1) >> 16;
|
|
|
|
Sci_PositionU lineStartNext = styler.LineStart(lineCurrent+1);
|
|
|
|
int levelMinCurrent = levelCurrent;
|
|
|
|
int levelNext = levelCurrent;
|
|
|
|
char chNext = styler[startPos];
|
|
|
|
int styleNext = MaskActive(styler.StyleAt(startPos));
|
|
|
|
int style = MaskActive(initStyle);
|
|
|
|
const bool userDefinedFoldMarkers = !options.foldExplicitStart.empty() && !options.foldExplicitEnd.empty();
|
|
|
|
for (Sci_PositionU i = startPos; i < endPos; i++) {
|
|
|
|
const char ch = chNext;
|
|
|
|
chNext = styler.SafeGetCharAt(i + 1);
|
|
|
|
const int stylePrev = style;
|
|
|
|
style = styleNext;
|
|
|
|
styleNext = MaskActive(styler.StyleAt(i + 1));
|
|
|
|
const bool atEOL = i == (lineStartNext-1);
|
|
|
|
if ((style == SCE_C_COMMENTLINE) || (style == SCE_C_COMMENTLINEDOC))
|
|
|
|
inLineComment = true;
|
|
|
|
if (options.foldComment && options.foldCommentMultiline && IsStreamCommentStyle(style) && !inLineComment) {
|
|
|
|
if (!IsStreamCommentStyle(stylePrev)) {
|
|
|
|
levelNext++;
|
|
|
|
} else if (!IsStreamCommentStyle(styleNext) && !atEOL) {
|
|
|
|
// Comments don't end at end of line and the next character may be unstyled.
|
|
|
|
levelNext--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (options.foldComment && options.foldCommentExplicit && ((style == SCE_C_COMMENTLINE) || options.foldExplicitAnywhere)) {
|
|
|
|
if (userDefinedFoldMarkers) {
|
|
|
|
if (styler.Match(i, options.foldExplicitStart.c_str())) {
|
|
|
|
levelNext++;
|
|
|
|
} else if (styler.Match(i, options.foldExplicitEnd.c_str())) {
|
|
|
|
levelNext--;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((ch == '/') && (chNext == '/')) {
|
|
|
|
const char chNext2 = styler.SafeGetCharAt(i + 2);
|
|
|
|
if (chNext2 == '{') {
|
|
|
|
levelNext++;
|
|
|
|
} else if (chNext2 == '}') {
|
|
|
|
levelNext--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (options.foldPreprocessor && (style == SCE_C_PREPROCESSOR)) {
|
|
|
|
if (ch == '#') {
|
|
|
|
Sci_PositionU j = i + 1;
|
|
|
|
while ((j < endPos) && IsASpaceOrTab(styler.SafeGetCharAt(j))) {
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
if (styler.Match(j, "region") || styler.Match(j, "if")) {
|
|
|
|
levelNext++;
|
|
|
|
} else if (styler.Match(j, "end")) {
|
|
|
|
levelNext--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.foldPreprocessorAtElse && (styler.Match(j, "else") || styler.Match(j, "elif"))) {
|
|
|
|
levelMinCurrent--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (options.foldSyntaxBased && (style == SCE_C_OPERATOR)) {
|
|
|
|
if (ch == '{' || ch == '[' || ch == '(') {
|
|
|
|
// Measure the minimum before a '{' to allow
|
|
|
|
// folding on "} else {"
|
|
|
|
if (options.foldAtElse && levelMinCurrent > levelNext) {
|
|
|
|
levelMinCurrent = levelNext;
|
|
|
|
}
|
|
|
|
levelNext++;
|
|
|
|
} else if (ch == '}' || ch == ']' || ch == ')') {
|
|
|
|
levelNext--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!IsASpace(ch))
|
|
|
|
visibleChars++;
|
|
|
|
if (atEOL || (i == endPos-1)) {
|
|
|
|
int levelUse = levelCurrent;
|
|
|
|
if ((options.foldSyntaxBased && options.foldAtElse) ||
|
|
|
|
(options.foldPreprocessor && options.foldPreprocessorAtElse)
|
|
|
|
) {
|
|
|
|
levelUse = levelMinCurrent;
|
|
|
|
}
|
|
|
|
int lev = levelUse | levelNext << 16;
|
|
|
|
if (visibleChars == 0 && options.foldCompact)
|
|
|
|
lev |= SC_FOLDLEVELWHITEFLAG;
|
|
|
|
if (levelUse < levelNext)
|
|
|
|
lev |= SC_FOLDLEVELHEADERFLAG;
|
|
|
|
if (lev != styler.LevelAt(lineCurrent)) {
|
|
|
|
styler.SetLevel(lineCurrent, lev);
|
|
|
|
}
|
|
|
|
lineCurrent++;
|
|
|
|
lineStartNext = styler.LineStart(lineCurrent+1);
|
|
|
|
levelCurrent = levelNext;
|
|
|
|
levelMinCurrent = levelCurrent;
|
|
|
|
if (atEOL && (i == static_cast<Sci_PositionU>(styler.Length()-1))) {
|
|
|
|
// There is an empty line at end of file so give it same level and empty
|
|
|
|
styler.SetLevel(lineCurrent, (levelCurrent | levelCurrent << 16) | SC_FOLDLEVELWHITEFLAG);
|
|
|
|
}
|
|
|
|
visibleChars = 0;
|
|
|
|
inLineComment = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LexerCPP::EvaluateTokens(Tokens &tokens, const SymbolTable &preprocessorDefinitions) {
|
|
|
|
|
|
|
|
// Remove whitespace tokens
|
|
|
|
tokens.erase(std::remove_if(tokens.begin(), tokens.end(), OnlySpaceOrTab), tokens.end());
|
|
|
|
|
|
|
|
// Evaluate defined statements to either 0 or 1
|
|
|
|
for (size_t i=0; (i+1)<tokens.size();) {
|
|
|
|
if (tokens[i] == "defined") {
|
|
|
|
const char *val = "0";
|
|
|
|
if (tokens[i+1] == "(") {
|
|
|
|
if (((i + 2)<tokens.size()) && (tokens[i + 2] == ")")) {
|
|
|
|
// defined()
|
|
|
|
tokens.erase(tokens.begin() + i + 1, tokens.begin() + i + 3);
|
|
|
|
} else if (((i+3)<tokens.size()) && (tokens[i+3] == ")")) {
|
|
|
|
// defined(<identifier>)
|
|
|
|
const SymbolTable::const_iterator it = preprocessorDefinitions.find(tokens[i+2]);
|
|
|
|
if (it != preprocessorDefinitions.end()) {
|
|
|
|
val = "1";
|
|
|
|
}
|
|
|
|
tokens.erase(tokens.begin() + i + 1, tokens.begin() + i + 4);
|
|
|
|
} else {
|
|
|
|
// Spurious '(' so erase as more likely to result in false
|
|
|
|
tokens.erase(tokens.begin() + i + 1, tokens.begin() + i + 2);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// defined <identifier>
|
|
|
|
const SymbolTable::const_iterator it = preprocessorDefinitions.find(tokens[i+1]);
|
|
|
|
if (it != preprocessorDefinitions.end()) {
|
|
|
|
val = "1";
|
|
|
|
}
|
|
|
|
tokens.erase(tokens.begin() + i + 1, tokens.begin() + i + 2);
|
|
|
|
}
|
|
|
|
tokens[i] = val;
|
|
|
|
} else {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Evaluate identifiers
|
|
|
|
constexpr size_t maxIterations = 100;
|
|
|
|
size_t iterations = 0; // Limit number of iterations in case there is a recursive macro.
|
|
|
|
for (size_t i = 0; (i<tokens.size()) && (iterations < maxIterations);) {
|
|
|
|
iterations++;
|
|
|
|
if (setWordStart.Contains(tokens[i][0])) {
|
|
|
|
const SymbolTable::const_iterator it = preprocessorDefinitions.find(tokens[i]);
|
|
|
|
if (it != preprocessorDefinitions.end()) {
|
|
|
|
// Tokenize value
|
|
|
|
Tokens macroTokens = Tokenize(it->second.value);
|
|
|
|
if (it->second.IsMacro()) {
|
|
|
|
if ((i + 1 < tokens.size()) && (tokens.at(i + 1) == "(")) {
|
|
|
|
// Create map of argument name to value
|
2022-08-27 07:35:52 +00:00
|
|
|
const Tokens argumentNames = StringSplit(it->second.arguments, ',');
|
2022-01-04 23:07:50 +00:00
|
|
|
std::map<std::string, std::string> arguments;
|
|
|
|
size_t arg = 0;
|
|
|
|
size_t tok = i+2;
|
|
|
|
while ((tok < tokens.size()) && (arg < argumentNames.size()) && (tokens.at(tok) != ")")) {
|
|
|
|
if (tokens.at(tok) != ",") {
|
|
|
|
arguments[argumentNames.at(arg)] = tokens.at(tok);
|
|
|
|
arg++;
|
|
|
|
}
|
|
|
|
tok++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove invocation
|
|
|
|
tokens.erase(tokens.begin() + i, tokens.begin() + tok + 1);
|
|
|
|
|
|
|
|
// Substitute values into macro
|
|
|
|
macroTokens.erase(std::remove_if(macroTokens.begin(), macroTokens.end(), OnlySpaceOrTab), macroTokens.end());
|
|
|
|
|
|
|
|
for (size_t iMacro = 0; iMacro < macroTokens.size();) {
|
|
|
|
if (setWordStart.Contains(macroTokens[iMacro][0])) {
|
|
|
|
const std::map<std::string, std::string>::const_iterator itFind =
|
|
|
|
arguments.find(macroTokens[iMacro]);
|
|
|
|
if (itFind != arguments.end()) {
|
|
|
|
// TODO: Possible that value will be expression so should insert tokenized form
|
|
|
|
macroTokens[iMacro] = itFind->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
iMacro++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert results back into tokens
|
|
|
|
tokens.insert(tokens.begin() + i, macroTokens.begin(), macroTokens.end());
|
|
|
|
|
|
|
|
} else {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Remove invocation
|
|
|
|
tokens.erase(tokens.begin() + i);
|
|
|
|
// Insert results back into tokens
|
|
|
|
tokens.insert(tokens.begin() + i, macroTokens.begin(), macroTokens.end());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Identifier not found and value defaults to zero
|
|
|
|
tokens[i] = "0";
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find bracketed subexpressions and recurse on them
|
|
|
|
BracketPair bracketPair = FindBracketPair(tokens);
|
|
|
|
while (bracketPair.itBracket != tokens.end()) {
|
|
|
|
Tokens inBracket(bracketPair.itBracket + 1, bracketPair.itEndBracket);
|
|
|
|
EvaluateTokens(inBracket, preprocessorDefinitions);
|
|
|
|
|
|
|
|
// The insertion is done before the removal because there were failures with the opposite approach
|
|
|
|
tokens.insert(bracketPair.itBracket, inBracket.begin(), inBracket.end());
|
|
|
|
|
2023-11-05 17:24:41 +00:00
|
|
|
// insert invalidated bracketPair. Use a new variable to avoid warning from Coverity.
|
|
|
|
const BracketPair pairToErase = FindBracketPair(tokens);
|
|
|
|
tokens.erase(pairToErase.itBracket, pairToErase.itEndBracket + 1);
|
2022-01-04 23:07:50 +00:00
|
|
|
|
|
|
|
bracketPair = FindBracketPair(tokens);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Evaluate logical negations
|
|
|
|
for (size_t j=0; (j+1)<tokens.size();) {
|
|
|
|
if (setNegationOp.Contains(tokens[j][0])) {
|
|
|
|
int isTrue = atoi(tokens[j+1].c_str());
|
|
|
|
if (tokens[j] == "!")
|
|
|
|
isTrue = !isTrue;
|
|
|
|
const Tokens::iterator itInsert =
|
|
|
|
tokens.erase(tokens.begin() + j, tokens.begin() + j + 2);
|
|
|
|
tokens.insert(itInsert, isTrue ? "1" : "0");
|
|
|
|
} else {
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Evaluate expressions in precedence order
|
|
|
|
enum precedence { precMult, precAdd, precRelative
|
|
|
|
, precLogical, /* end marker */ precLast };
|
|
|
|
for (int prec = precMult; prec < precLast; prec++) {
|
|
|
|
// Looking at 3 tokens at a time so end at 2 before end
|
|
|
|
for (size_t k=0; (k+2)<tokens.size();) {
|
|
|
|
const char chOp = tokens[k+1][0];
|
|
|
|
if (
|
|
|
|
((prec==precMult) && setMultOp.Contains(chOp)) ||
|
|
|
|
((prec==precAdd) && setAddOp.Contains(chOp)) ||
|
|
|
|
((prec==precRelative) && setRelOp.Contains(chOp)) ||
|
|
|
|
((prec==precLogical) && setLogicalOp.Contains(chOp))
|
|
|
|
) {
|
|
|
|
const int valA = atoi(tokens[k].c_str());
|
|
|
|
const int valB = atoi(tokens[k+2].c_str());
|
|
|
|
int result = 0;
|
|
|
|
if (tokens[k+1] == "+")
|
|
|
|
result = valA + valB;
|
|
|
|
else if (tokens[k+1] == "-")
|
|
|
|
result = valA - valB;
|
|
|
|
else if (tokens[k+1] == "*")
|
|
|
|
result = valA * valB;
|
|
|
|
else if (tokens[k+1] == "/")
|
|
|
|
result = valA / (valB ? valB : 1);
|
|
|
|
else if (tokens[k+1] == "%")
|
|
|
|
result = valA % (valB ? valB : 1);
|
|
|
|
else if (tokens[k+1] == "<")
|
|
|
|
result = valA < valB;
|
|
|
|
else if (tokens[k+1] == "<=")
|
|
|
|
result = valA <= valB;
|
|
|
|
else if (tokens[k+1] == ">")
|
|
|
|
result = valA > valB;
|
|
|
|
else if (tokens[k+1] == ">=")
|
|
|
|
result = valA >= valB;
|
|
|
|
else if (tokens[k+1] == "==")
|
|
|
|
result = valA == valB;
|
|
|
|
else if (tokens[k+1] == "!=")
|
|
|
|
result = valA != valB;
|
|
|
|
else if (tokens[k+1] == "||")
|
|
|
|
result = valA || valB;
|
|
|
|
else if (tokens[k+1] == "&&")
|
|
|
|
result = valA && valB;
|
|
|
|
const Tokens::iterator itInsert =
|
|
|
|
tokens.erase(tokens.begin() + k, tokens.begin() + k + 3);
|
|
|
|
tokens.insert(itInsert, std::to_string(result));
|
|
|
|
} else {
|
|
|
|
k++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Tokens LexerCPP::Tokenize(const std::string &expr) const {
|
|
|
|
// Break into tokens
|
|
|
|
Tokens tokens;
|
|
|
|
const char *cp = expr.c_str();
|
|
|
|
while (*cp) {
|
|
|
|
std::string word;
|
|
|
|
if (setWord.Contains(*cp)) {
|
|
|
|
// Identifiers and numbers
|
|
|
|
while (setWord.Contains(*cp)) {
|
|
|
|
word += *cp;
|
|
|
|
cp++;
|
|
|
|
}
|
|
|
|
} else if (IsSpaceOrTab(*cp)) {
|
|
|
|
while (IsSpaceOrTab(*cp)) {
|
|
|
|
word += *cp;
|
|
|
|
cp++;
|
|
|
|
}
|
|
|
|
} else if (setRelOp.Contains(*cp)) {
|
|
|
|
word += *cp;
|
|
|
|
cp++;
|
|
|
|
if (setRelOp.Contains(*cp)) {
|
|
|
|
word += *cp;
|
|
|
|
cp++;
|
|
|
|
}
|
|
|
|
} else if (setLogicalOp.Contains(*cp)) {
|
|
|
|
word += *cp;
|
|
|
|
cp++;
|
|
|
|
if (setLogicalOp.Contains(*cp)) {
|
|
|
|
word += *cp;
|
|
|
|
cp++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Should handle strings, characters, and comments here
|
|
|
|
word += *cp;
|
|
|
|
cp++;
|
|
|
|
}
|
|
|
|
tokens.push_back(word);
|
|
|
|
}
|
|
|
|
return tokens;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LexerCPP::EvaluateExpression(const std::string &expr, const SymbolTable &preprocessorDefinitions) {
|
|
|
|
Tokens tokens = Tokenize(expr);
|
|
|
|
|
|
|
|
EvaluateTokens(tokens, preprocessorDefinitions);
|
|
|
|
|
|
|
|
// "0" or "" -> false else true
|
|
|
|
const bool isFalse = tokens.empty() ||
|
2022-10-12 18:45:40 +00:00
|
|
|
((tokens.size() == 1) && (tokens[0].empty() || tokens[0] == "0"));
|
2022-01-04 23:07:50 +00:00
|
|
|
return !isFalse;
|
|
|
|
}
|
|
|
|
|
|
|
|
LexerModule lmCPP(SCLEX_CPP, LexerCPP::LexerFactoryCPP, "cpp", cppWordLists);
|
|
|
|
LexerModule lmCPPNoCase(SCLEX_CPPNOCASE, LexerCPP::LexerFactoryCPPInsensitive, "cppnocase", cppWordLists);
|