2009-04-24 23:35:41 +00:00
|
|
|
// Scintilla source code edit control
|
2013-08-28 00:44:27 +00:00
|
|
|
/** @file StyleContext.h
|
2009-04-24 23:35:41 +00:00
|
|
|
** Lexer infrastructure.
|
|
|
|
**/
|
|
|
|
// Copyright 1998-2004 by Neil Hodgson <neilh@scintilla.org>
|
|
|
|
// This file is in the public domain.
|
|
|
|
|
2010-08-21 23:59:56 +00:00
|
|
|
#ifndef STYLECONTEXT_H
|
|
|
|
#define STYLECONTEXT_H
|
|
|
|
|
2009-04-24 23:35:41 +00:00
|
|
|
namespace Scintilla {
|
2010-08-21 23:59:56 +00:00
|
|
|
|
2009-04-24 23:35:41 +00:00
|
|
|
// All languages handled so far can treat all characters >= 0x80 as one class
|
|
|
|
// which just continues the current token or starts an identifier if in default.
|
|
|
|
// DBCS treated specially as the second character can be < 0x80 and hence
|
|
|
|
// syntactically significant. UTF-8 avoids this as all trail bytes are >= 0x80
|
|
|
|
class StyleContext {
|
2010-08-21 23:59:56 +00:00
|
|
|
LexAccessor &styler;
|
2019-05-04 18:14:48 +00:00
|
|
|
IDocument *multiByteAccess;
|
|
|
|
Sci_PositionU endPos;
|
|
|
|
Sci_PositionU lengthDocument;
|
2015-06-07 21:19:26 +00:00
|
|
|
|
2013-08-28 00:44:27 +00:00
|
|
|
// Used for optimizing GetRelativeCharacter
|
2019-05-04 18:14:48 +00:00
|
|
|
Sci_PositionU posRelative;
|
|
|
|
Sci_PositionU currentPosLastRelative;
|
|
|
|
Sci_Position offsetRelative;
|
2013-08-28 00:44:27 +00:00
|
|
|
|
|
|
|
void GetNextChar() {
|
|
|
|
if (multiByteAccess) {
|
|
|
|
chNext = multiByteAccess->GetCharacterAndWidth(currentPos+width, &widthNext);
|
|
|
|
} else {
|
|
|
|
chNext = static_cast<unsigned char>(styler.SafeGetCharAt(currentPos+width, 0));
|
|
|
|
widthNext = 1;
|
2009-04-24 23:35:41 +00:00
|
|
|
}
|
2015-06-07 21:19:26 +00:00
|
|
|
// End of line determined from line end position, allowing CR, LF,
|
2013-08-28 00:44:27 +00:00
|
|
|
// CRLF and Unicode line ends as set by document.
|
|
|
|
if (currentLine < lineDocEnd)
|
2019-05-04 18:14:48 +00:00
|
|
|
atLineEnd = static_cast<Sci_Position>(currentPos) >= (lineStartNext-1);
|
2013-08-28 00:44:27 +00:00
|
|
|
else // Last line
|
2019-05-04 18:14:48 +00:00
|
|
|
atLineEnd = static_cast<Sci_Position>(currentPos) >= lineStartNext;
|
2009-04-24 23:35:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
2019-05-04 18:14:48 +00:00
|
|
|
Sci_PositionU currentPos;
|
|
|
|
Sci_Position currentLine;
|
|
|
|
Sci_Position lineDocEnd;
|
|
|
|
Sci_Position lineStartNext;
|
2009-04-24 23:35:41 +00:00
|
|
|
bool atLineStart;
|
|
|
|
bool atLineEnd;
|
|
|
|
int state;
|
|
|
|
int chPrev;
|
|
|
|
int ch;
|
2019-05-04 18:14:48 +00:00
|
|
|
Sci_Position width;
|
2009-04-24 23:35:41 +00:00
|
|
|
int chNext;
|
2019-05-04 18:14:48 +00:00
|
|
|
Sci_Position widthNext;
|
2009-04-24 23:35:41 +00:00
|
|
|
|
2019-05-04 18:14:48 +00:00
|
|
|
StyleContext(Sci_PositionU startPos, Sci_PositionU length,
|
2015-06-07 21:19:26 +00:00
|
|
|
int initStyle, LexAccessor &styler_, char chMask='\377') :
|
2009-04-24 23:35:41 +00:00
|
|
|
styler(styler_),
|
2019-05-04 18:14:48 +00:00
|
|
|
multiByteAccess(nullptr),
|
2009-04-24 23:35:41 +00:00
|
|
|
endPos(startPos + length),
|
2013-08-28 00:44:27 +00:00
|
|
|
posRelative(0),
|
|
|
|
currentPosLastRelative(0x7FFFFFFF),
|
|
|
|
offsetRelative(0),
|
2009-04-24 23:35:41 +00:00
|
|
|
currentPos(startPos),
|
2013-08-28 00:44:27 +00:00
|
|
|
currentLine(-1),
|
|
|
|
lineStartNext(-1),
|
2009-04-24 23:35:41 +00:00
|
|
|
atLineEnd(false),
|
|
|
|
state(initStyle & chMask), // Mask off all bits which aren't in the chMask.
|
|
|
|
chPrev(0),
|
|
|
|
ch(0),
|
2013-08-28 00:44:27 +00:00
|
|
|
width(0),
|
|
|
|
chNext(0),
|
|
|
|
widthNext(1) {
|
2021-02-21 04:53:09 +00:00
|
|
|
if (styler.Encoding() != EncodingType::eightBit) {
|
2013-08-28 00:44:27 +00:00
|
|
|
multiByteAccess = styler.MultiByteAccess();
|
|
|
|
}
|
2015-06-07 21:19:26 +00:00
|
|
|
styler.StartAt(startPos /*, chMask*/);
|
2009-04-24 23:35:41 +00:00
|
|
|
styler.StartSegment(startPos);
|
2013-08-28 00:44:27 +00:00
|
|
|
currentLine = styler.GetLine(startPos);
|
|
|
|
lineStartNext = styler.LineStart(currentLine+1);
|
2019-05-04 18:14:48 +00:00
|
|
|
lengthDocument = static_cast<Sci_PositionU>(styler.Length());
|
2013-08-28 00:44:27 +00:00
|
|
|
if (endPos == lengthDocument)
|
|
|
|
endPos++;
|
|
|
|
lineDocEnd = styler.GetLine(lengthDocument);
|
2019-05-04 18:14:48 +00:00
|
|
|
atLineStart = static_cast<Sci_PositionU>(styler.LineStart(currentLine)) == startPos;
|
2013-08-28 00:44:27 +00:00
|
|
|
|
|
|
|
// Variable width is now 0 so GetNextChar gets the char at currentPos into chNext/widthNext
|
|
|
|
width = 0;
|
|
|
|
GetNextChar();
|
|
|
|
ch = chNext;
|
|
|
|
width = widthNext;
|
|
|
|
|
|
|
|
GetNextChar();
|
2009-04-24 23:35:41 +00:00
|
|
|
}
|
2019-05-04 18:14:48 +00:00
|
|
|
// Deleted so StyleContext objects can not be copied.
|
|
|
|
StyleContext(const StyleContext &) = delete;
|
|
|
|
StyleContext &operator=(const StyleContext &) = delete;
|
2009-04-24 23:35:41 +00:00
|
|
|
void Complete() {
|
2013-08-28 00:44:27 +00:00
|
|
|
styler.ColourTo(currentPos - ((currentPos > lengthDocument) ? 2 : 1), state);
|
2010-09-05 22:56:27 +00:00
|
|
|
styler.Flush();
|
2009-04-24 23:35:41 +00:00
|
|
|
}
|
2021-02-21 04:53:09 +00:00
|
|
|
bool More() const noexcept {
|
2009-04-24 23:35:41 +00:00
|
|
|
return currentPos < endPos;
|
|
|
|
}
|
|
|
|
void Forward() {
|
|
|
|
if (currentPos < endPos) {
|
|
|
|
atLineStart = atLineEnd;
|
2013-08-28 00:44:27 +00:00
|
|
|
if (atLineStart) {
|
|
|
|
currentLine++;
|
|
|
|
lineStartNext = styler.LineStart(currentLine+1);
|
|
|
|
}
|
2009-04-24 23:35:41 +00:00
|
|
|
chPrev = ch;
|
2013-08-28 00:44:27 +00:00
|
|
|
currentPos += width;
|
2009-04-24 23:35:41 +00:00
|
|
|
ch = chNext;
|
2013-08-28 00:44:27 +00:00
|
|
|
width = widthNext;
|
|
|
|
GetNextChar();
|
2009-04-24 23:35:41 +00:00
|
|
|
} else {
|
|
|
|
atLineStart = false;
|
|
|
|
chPrev = ' ';
|
|
|
|
ch = ' ';
|
|
|
|
chNext = ' ';
|
|
|
|
atLineEnd = true;
|
|
|
|
}
|
|
|
|
}
|
2019-05-04 18:14:48 +00:00
|
|
|
void Forward(Sci_Position nb) {
|
|
|
|
for (Sci_Position i = 0; i < nb; i++) {
|
2009-04-24 23:35:41 +00:00
|
|
|
Forward();
|
|
|
|
}
|
|
|
|
}
|
2019-05-04 18:14:48 +00:00
|
|
|
void ForwardBytes(Sci_Position nb) {
|
|
|
|
const Sci_PositionU forwardPos = currentPos + nb;
|
2013-08-28 00:44:27 +00:00
|
|
|
while (forwardPos > currentPos) {
|
2019-05-04 18:14:48 +00:00
|
|
|
const Sci_PositionU currentPosStart = currentPos;
|
2013-08-28 00:44:27 +00:00
|
|
|
Forward();
|
2019-05-04 18:14:48 +00:00
|
|
|
if (currentPos == currentPosStart) {
|
|
|
|
// Reached end
|
|
|
|
return;
|
|
|
|
}
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
|
|
|
}
|
2021-02-21 04:53:09 +00:00
|
|
|
void ChangeState(int state_) noexcept {
|
2009-04-24 23:35:41 +00:00
|
|
|
state = state_;
|
|
|
|
}
|
|
|
|
void SetState(int state_) {
|
2013-08-28 00:44:27 +00:00
|
|
|
styler.ColourTo(currentPos - ((currentPos > lengthDocument) ? 2 : 1), state);
|
2009-04-24 23:35:41 +00:00
|
|
|
state = state_;
|
|
|
|
}
|
|
|
|
void ForwardSetState(int state_) {
|
|
|
|
Forward();
|
2013-08-28 00:44:27 +00:00
|
|
|
styler.ColourTo(currentPos - ((currentPos > lengthDocument) ? 2 : 1), state);
|
2009-04-24 23:35:41 +00:00
|
|
|
state = state_;
|
|
|
|
}
|
2019-05-04 18:14:48 +00:00
|
|
|
Sci_Position LengthCurrent() const {
|
2009-04-24 23:35:41 +00:00
|
|
|
return currentPos - styler.GetStartSegment();
|
|
|
|
}
|
2021-02-21 04:53:09 +00:00
|
|
|
int GetRelative(Sci_Position n, char chDefault='\0') {
|
|
|
|
return static_cast<unsigned char>(styler.SafeGetCharAt(currentPos+n, chDefault));
|
2013-08-28 00:44:27 +00:00
|
|
|
}
|
2019-05-04 18:14:48 +00:00
|
|
|
int GetRelativeCharacter(Sci_Position n) {
|
2013-08-28 00:44:27 +00:00
|
|
|
if (n == 0)
|
|
|
|
return ch;
|
|
|
|
if (multiByteAccess) {
|
|
|
|
if ((currentPosLastRelative != currentPos) ||
|
|
|
|
((n > 0) && ((offsetRelative < 0) || (n < offsetRelative))) ||
|
|
|
|
((n < 0) && ((offsetRelative > 0) || (n > offsetRelative)))) {
|
|
|
|
posRelative = currentPos;
|
|
|
|
offsetRelative = 0;
|
|
|
|
}
|
2019-05-04 18:14:48 +00:00
|
|
|
const Sci_Position diffRelative = n - offsetRelative;
|
|
|
|
const Sci_Position posNew = multiByteAccess->GetRelativePosition(posRelative, diffRelative);
|
|
|
|
const int chReturn = multiByteAccess->GetCharacterAndWidth(posNew, nullptr);
|
2013-08-28 00:44:27 +00:00
|
|
|
posRelative = posNew;
|
|
|
|
currentPosLastRelative = currentPos;
|
|
|
|
offsetRelative = n;
|
2015-06-07 21:19:26 +00:00
|
|
|
return chReturn;
|
2013-08-28 00:44:27 +00:00
|
|
|
} else {
|
|
|
|
// fast version for single byte encodings
|
|
|
|
return static_cast<unsigned char>(styler.SafeGetCharAt(currentPos + n, 0));
|
|
|
|
}
|
2009-04-24 23:35:41 +00:00
|
|
|
}
|
2010-07-12 22:19:51 +00:00
|
|
|
bool Match(char ch0) const {
|
2009-04-24 23:35:41 +00:00
|
|
|
return ch == static_cast<unsigned char>(ch0);
|
|
|
|
}
|
2010-07-12 22:19:51 +00:00
|
|
|
bool Match(char ch0, char ch1) const {
|
2009-04-24 23:35:41 +00:00
|
|
|
return (ch == static_cast<unsigned char>(ch0)) && (chNext == static_cast<unsigned char>(ch1));
|
|
|
|
}
|
|
|
|
bool Match(const char *s) {
|
|
|
|
if (ch != static_cast<unsigned char>(*s))
|
|
|
|
return false;
|
|
|
|
s++;
|
|
|
|
if (!*s)
|
|
|
|
return true;
|
|
|
|
if (chNext != static_cast<unsigned char>(*s))
|
|
|
|
return false;
|
|
|
|
s++;
|
|
|
|
for (int n=2; *s; n++) {
|
2013-08-28 00:44:27 +00:00
|
|
|
if (*s != styler.SafeGetCharAt(currentPos+n, 0))
|
2009-04-24 23:35:41 +00:00
|
|
|
return false;
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// Non-inline
|
2019-05-04 18:14:48 +00:00
|
|
|
bool MatchIgnoreCase(const char *s);
|
|
|
|
bool MatchIgnoreCase2(const char *s);
|
|
|
|
void GetCurrent(char *s, Sci_PositionU len);
|
|
|
|
void GetCurrentLowered(char *s, Sci_PositionU len);
|
2009-04-24 23:35:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-08-21 23:59:56 +00:00
|
|
|
#endif
|