geany/scintilla/lexlib/LexAccessor.h

205 lines
5.4 KiB
C
Raw Normal View History

// Scintilla source code edit control
/** @file LexAccessor.h
** Interfaces between Scintilla and lexers.
**/
// Copyright 1998-2010 by Neil Hodgson <neilh@scintilla.org>
// The License.txt file describes the conditions under which this software may be distributed.
#ifndef LEXACCESSOR_H
#define LEXACCESSOR_H
#ifdef SCI_NAMESPACE
namespace Scintilla {
#endif
2013-04-27 16:09:29 +02:00
enum EncodingType { enc8bit, encUnicode, encDBCS };
class LexAccessor {
private:
IDocument *pAccess;
enum {extremePosition=0x7FFFFFFF};
/** @a bufferSize is a trade off between time taken to copy the characters
* and retrieval overhead.
* @a slopSize positions the buffer before the desired position
* in case there is some backtracking. */
enum {bufferSize=4000, slopSize=bufferSize/8};
char buf[bufferSize+1];
2015-09-20 18:37:11 +02:00
Sci_Position startPos;
Sci_Position endPos;
int codePage;
2013-04-27 16:09:29 +02:00
enum EncodingType encodingType;
2015-09-20 18:37:11 +02:00
Sci_Position lenDoc;
char styleBuf[bufferSize];
2015-09-20 18:37:11 +02:00
Sci_Position validLen;
Sci_PositionU startSeg;
Sci_Position startPosStyling;
2013-04-27 16:09:29 +02:00
int documentVersion;
2015-09-20 18:37:11 +02:00
void Fill(Sci_Position position) {
startPos = position - slopSize;
if (startPos + bufferSize > lenDoc)
startPos = lenDoc - bufferSize;
if (startPos < 0)
startPos = 0;
endPos = startPos + bufferSize;
if (endPos > lenDoc)
endPos = lenDoc;
pAccess->GetCharRange(buf, startPos, endPos-startPos);
buf[endPos-startPos] = '\0';
}
public:
2014-04-14 16:25:01 +02:00
explicit LexAccessor(IDocument *pAccess_) :
pAccess(pAccess_), startPos(extremePosition), endPos(0),
2014-04-14 16:25:01 +02:00
codePage(pAccess->CodePage()),
2013-04-27 16:09:29 +02:00
encodingType(enc8bit),
lenDoc(pAccess->Length()),
2014-07-04 03:06:17 +02:00
validLen(0),
2014-04-14 16:25:01 +02:00
startSeg(0), startPosStyling(0),
2013-04-27 16:09:29 +02:00
documentVersion(pAccess->Version()) {
// Prevent warnings by static analyzers about uninitialized buf and styleBuf.
buf[0] = 0;
styleBuf[0] = 0;
2013-04-27 16:09:29 +02:00
switch (codePage) {
case 65001:
encodingType = encUnicode;
break;
case 932:
case 936:
case 949:
case 950:
case 1361:
encodingType = encDBCS;
}
}
2015-09-20 18:37:11 +02:00
char operator[](Sci_Position position) {
if (position < startPos || position >= endPos) {
Fill(position);
}
return buf[position - startPos];
}
2013-07-21 11:52:17 +02:00
IDocumentWithLineEnd *MultiByteAccess() const {
if (documentVersion >= dvLineEnd) {
return static_cast<IDocumentWithLineEnd *>(pAccess);
}
return 0;
}
/** Safe version of operator[], returning a defined value for invalid position. */
2015-09-20 18:37:11 +02:00
char SafeGetCharAt(Sci_Position position, char chDefault=' ') {
if (position < startPos || position >= endPos) {
Fill(position);
if (position < startPos || position >= endPos) {
// Position is outside range of document
return chDefault;
}
}
return buf[position - startPos];
}
2013-06-06 16:31:26 +02:00
bool IsLeadByte(char ch) const {
return pAccess->IsDBCSLeadByte(ch);
}
2013-04-27 16:09:29 +02:00
EncodingType Encoding() const {
return encodingType;
}
2015-09-20 18:37:11 +02:00
bool Match(Sci_Position pos, const char *s) {
for (int i=0; *s; i++) {
if (*s != SafeGetCharAt(pos+i))
return false;
s++;
}
return true;
}
2015-09-20 18:37:11 +02:00
char StyleAt(Sci_Position position) const {
2014-07-04 03:06:17 +02:00
return static_cast<char>(pAccess->StyleAt(position));
}
2015-09-20 18:37:11 +02:00
Sci_Position GetLine(Sci_Position position) const {
return pAccess->LineFromPosition(position);
}
2015-09-20 18:37:11 +02:00
Sci_Position LineStart(Sci_Position line) const {
return pAccess->LineStart(line);
}
2015-09-20 18:37:11 +02:00
Sci_Position LineEnd(Sci_Position line) {
2013-04-27 16:09:29 +02:00
if (documentVersion >= dvLineEnd) {
return (static_cast<IDocumentWithLineEnd *>(pAccess))->LineEnd(line);
} else {
// Old interface means only '\r', '\n' and '\r\n' line ends.
2015-09-20 18:37:11 +02:00
Sci_Position startNext = pAccess->LineStart(line+1);
2013-04-27 16:09:29 +02:00
char chLineEnd = SafeGetCharAt(startNext-1);
if (chLineEnd == '\n' && (SafeGetCharAt(startNext-2) == '\r'))
return startNext - 2;
else
return startNext - 1;
}
}
2015-09-20 18:37:11 +02:00
int LevelAt(Sci_Position line) const {
return pAccess->GetLevel(line);
}
2015-09-20 18:37:11 +02:00
Sci_Position Length() const {
return lenDoc;
}
void Flush() {
if (validLen > 0) {
pAccess->SetStyles(validLen, styleBuf);
startPosStyling += validLen;
validLen = 0;
}
}
2015-09-20 18:37:11 +02:00
int GetLineState(Sci_Position line) const {
return pAccess->GetLineState(line);
}
2015-09-20 18:37:11 +02:00
int SetLineState(Sci_Position line, int state) {
return pAccess->SetLineState(line, state);
}
// Style setting
2015-09-20 18:37:11 +02:00
void StartAt(Sci_PositionU start) {
2014-07-04 03:06:17 +02:00
pAccess->StartStyling(start, '\377');
startPosStyling = start;
}
2015-09-20 18:37:11 +02:00
Sci_PositionU GetStartSegment() const {
return startSeg;
}
2015-09-20 18:37:11 +02:00
void StartSegment(Sci_PositionU pos) {
startSeg = pos;
}
2015-09-20 18:37:11 +02:00
void ColourTo(Sci_PositionU pos, int chAttr) {
// Only perform styling if non empty range
if (pos != startSeg - 1) {
assert(pos >= startSeg);
if (pos < startSeg) {
return;
}
if (validLen + (pos - startSeg + 1) >= bufferSize)
Flush();
if (validLen + (pos - startSeg + 1) >= bufferSize) {
// Too big for buffer so send directly
pAccess->SetStyleFor(pos - startSeg + 1, static_cast<char>(chAttr));
} else {
2015-09-20 18:37:11 +02:00
for (Sci_PositionU i = startSeg; i <= pos; i++) {
assert((startPosStyling + validLen) < Length());
styleBuf[validLen++] = static_cast<char>(chAttr);
}
}
}
startSeg = pos+1;
}
2015-09-20 18:37:11 +02:00
void SetLevel(Sci_Position line, int level) {
pAccess->SetLevel(line, level);
}
2015-09-20 18:37:11 +02:00
void IndicatorFill(Sci_Position start, Sci_Position end, int indicator, int value) {
pAccess->DecorationSetCurrentIndicator(indicator);
pAccess->DecorationFillRange(start, value, end - start);
}
2015-09-20 18:37:11 +02:00
void ChangeLexerState(Sci_Position start, Sci_Position end) {
pAccess->ChangeLexerState(start, end);
}
};
#ifdef SCI_NAMESPACE
}
#endif
#endif