geany/scintilla/lexlib/StyleContext.h

232 lines
6.0 KiB
C
Raw Normal View History

// Scintilla source code edit control
2013-07-21 11:52:17 +02:00
/** @file StyleContext.h
** Lexer infrastructure.
**/
// Copyright 1998-2004 by Neil Hodgson <neilh@scintilla.org>
// This file is in the public domain.
#ifndef STYLECONTEXT_H
#define STYLECONTEXT_H
#ifdef SCI_NAMESPACE
namespace Scintilla {
#endif
static inline int MakeLowerCase(int ch) {
if (ch < 'A' || ch > 'Z')
return ch;
else
return ch - 'A' + 'a';
}
// 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 {
LexAccessor &styler;
2013-07-21 11:52:17 +02:00
IDocumentWithLineEnd *multiByteAccess;
unsigned int endPos;
2013-04-27 16:09:29 +02:00
unsigned int lengthDocument;
2013-07-21 11:52:17 +02:00
// Used for optimizing GetRelativeCharacter
unsigned int posRelative;
unsigned int currentPosLastRelative;
int offsetRelative;
StyleContext &operator=(const StyleContext &);
2013-04-27 16:09:29 +02:00
2013-07-21 11:52:17 +02:00
void GetNextChar() {
if (multiByteAccess) {
chNext = multiByteAccess->GetCharacterAndWidth(currentPos+width, &widthNext);
} else {
chNext = static_cast<unsigned char>(styler.SafeGetCharAt(currentPos+width, 0));
widthNext = 1;
}
2013-07-21 11:52:17 +02:00
// End of line determined from line end position, allowing CR, LF,
// CRLF and Unicode line ends as set by document.
2013-05-22 03:23:57 +02:00
if (currentLine < lineDocEnd)
2013-07-21 11:52:17 +02:00
atLineEnd = static_cast<int>(currentPos) >= (lineStartNext-1);
2013-04-27 16:09:29 +02:00
else // Last line
2013-07-21 11:52:17 +02:00
atLineEnd = static_cast<int>(currentPos) >= lineStartNext;
}
public:
unsigned int currentPos;
2013-04-27 16:09:29 +02:00
int currentLine;
2013-05-22 03:23:57 +02:00
int lineDocEnd;
2013-04-27 16:09:29 +02:00
int lineStartNext;
bool atLineStart;
bool atLineEnd;
int state;
int chPrev;
int ch;
2013-07-21 11:52:17 +02:00
int width;
int chNext;
2013-07-21 11:52:17 +02:00
int widthNext;
StyleContext(unsigned int startPos, unsigned int length,
int initStyle, LexAccessor &styler_, char chMask=31) :
styler(styler_),
2013-07-21 11:52:17 +02:00
multiByteAccess(0),
endPos(startPos + length),
2013-07-21 11:52:17 +02:00
posRelative(0),
currentPosLastRelative(0x7FFFFFFF),
offsetRelative(0),
currentPos(startPos),
2013-04-27 16:09:29 +02:00
currentLine(-1),
lineStartNext(-1),
atLineEnd(false),
state(initStyle & chMask), // Mask off all bits which aren't in the chMask.
chPrev(0),
ch(0),
2013-07-21 11:52:17 +02:00
width(0),
chNext(0),
widthNext(1) {
if (styler.Encoding() != enc8bit) {
multiByteAccess = styler.MultiByteAccess();
}
styler.StartAt(startPos, chMask);
styler.StartSegment(startPos);
2013-04-27 16:09:29 +02:00
currentLine = styler.GetLine(startPos);
lineStartNext = styler.LineStart(currentLine+1);
lengthDocument = static_cast<unsigned int>(styler.Length());
if (endPos == lengthDocument)
endPos++;
2013-05-22 03:23:57 +02:00
lineDocEnd = styler.GetLine(lengthDocument);
2013-04-27 16:09:29 +02:00
atLineStart = static_cast<unsigned int>(styler.LineStart(currentLine)) == startPos;
2013-07-21 11:52:17 +02:00
// Variable width is now 0 so GetNextChar gets the char at currentPos into chNext/widthNext
width = 0;
GetNextChar();
ch = chNext;
width = widthNext;
GetNextChar();
}
void Complete() {
2013-04-27 16:09:29 +02:00
styler.ColourTo(currentPos - ((currentPos > lengthDocument) ? 2 : 1), state);
styler.Flush();
}
bool More() const {
return currentPos < endPos;
}
void Forward() {
if (currentPos < endPos) {
atLineStart = atLineEnd;
2013-04-27 16:09:29 +02:00
if (atLineStart) {
currentLine++;
lineStartNext = styler.LineStart(currentLine+1);
}
chPrev = ch;
2013-07-21 11:52:17 +02:00
currentPos += width;
ch = chNext;
2013-07-21 11:52:17 +02:00
width = widthNext;
GetNextChar();
} else {
atLineStart = false;
chPrev = ' ';
ch = ' ';
chNext = ' ';
atLineEnd = true;
}
}
void Forward(int nb) {
for (int i = 0; i < nb; i++) {
Forward();
}
}
2013-07-21 11:52:17 +02:00
void ForwardBytes(int nb) {
size_t forwardPos = currentPos + nb;
while (forwardPos > currentPos) {
Forward();
}
}
void ChangeState(int state_) {
state = state_;
}
void SetState(int state_) {
2013-04-27 16:09:29 +02:00
styler.ColourTo(currentPos - ((currentPos > lengthDocument) ? 2 : 1), state);
state = state_;
}
void ForwardSetState(int state_) {
Forward();
2013-04-27 16:09:29 +02:00
styler.ColourTo(currentPos - ((currentPos > lengthDocument) ? 2 : 1), state);
state = state_;
}
2013-06-06 16:31:26 +02:00
int LengthCurrent() const {
return currentPos - styler.GetStartSegment();
}
int GetRelative(int n) {
2013-04-27 16:09:29 +02:00
return static_cast<unsigned char>(styler.SafeGetCharAt(currentPos+n, 0));
}
2013-07-21 11:52:17 +02:00
int GetRelativeCharacter(int n) {
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;
}
int diffRelative = n - offsetRelative;
int posNew = multiByteAccess->GetRelativePosition(posRelative, diffRelative);
2013-10-16 15:36:57 +02:00
int chReturn = multiByteAccess->GetCharacterAndWidth(posNew, 0);
2013-07-21 11:52:17 +02:00
posRelative = posNew;
currentPosLastRelative = currentPos;
offsetRelative = n;
2013-10-16 15:36:57 +02:00
return chReturn;
2013-07-21 11:52:17 +02:00
} else {
// fast version for single byte encodings
return static_cast<unsigned char>(styler.SafeGetCharAt(currentPos + n, 0));
}
}
bool Match(char ch0) const {
return ch == static_cast<unsigned char>(ch0);
}
bool Match(char ch0, char ch1) const {
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-04-27 16:09:29 +02:00
if (*s != styler.SafeGetCharAt(currentPos+n, 0))
return false;
s++;
}
return true;
}
bool MatchIgnoreCase(const char *s) {
if (MakeLowerCase(ch) != static_cast<unsigned char>(*s))
return false;
s++;
if (MakeLowerCase(chNext) != static_cast<unsigned char>(*s))
return false;
s++;
for (int n=2; *s; n++) {
if (static_cast<unsigned char>(*s) !=
2013-04-27 16:09:29 +02:00
MakeLowerCase(static_cast<unsigned char>(styler.SafeGetCharAt(currentPos+n, 0))))
return false;
s++;
}
return true;
}
// Non-inline
void GetCurrent(char *s, unsigned int len);
void GetCurrentLowered(char *s, unsigned int len);
};
#ifdef SCI_NAMESPACE
}
#endif
#endif