diff options
author | nyamatongwe <unknown> | 2010-07-13 21:24:26 +1000 |
---|---|---|
committer | nyamatongwe <unknown> | 2010-07-13 21:24:26 +1000 |
commit | c0247be1cde4c927b987edff2243524cea28d547 (patch) | |
tree | 482fac70e504ac105d36898d359c5992cea8e4c8 /lexlib/LexAccessor.h | |
parent | 27a22f2c85e3aa0f540c61a0a245a0d759e706a9 (diff) | |
download | scintilla-mirror-c0247be1cde4c927b987edff2243524cea28d547.tar.gz |
New files for new lexer implementation.
Diffstat (limited to 'lexlib/LexAccessor.h')
-rw-r--r-- | lexlib/LexAccessor.h | 175 |
1 files changed, 175 insertions, 0 deletions
diff --git a/lexlib/LexAccessor.h b/lexlib/LexAccessor.h new file mode 100644 index 000000000..dccf31e33 --- /dev/null +++ b/lexlib/LexAccessor.h @@ -0,0 +1,175 @@ +// 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 + +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]; + int startPos; + int endPos; + int codePage; + int lenDoc; + int mask; + char styleBuf[bufferSize]; + int validLen; + char chFlags; + char chWhile; + unsigned int startSeg; + int startPosStyling; + + void Fill(int 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: + LexAccessor(IDocument *pAccess_) : + pAccess(pAccess_), startPos(extremePosition), endPos(0), + codePage(pAccess->CodePage()), lenDoc(pAccess->Length()), + mask(127), validLen(0), chFlags(0), chWhile(0), + startSeg(0), startPosStyling(0) { + } + char operator[](int position) { + if (position < startPos || position >= endPos) { + Fill(position); + } + return buf[position - startPos]; + } + /** Safe version of operator[], returning a defined value for invalid position. */ + char SafeGetCharAt(int 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]; + } + bool IsLeadByte(char ch) { + return pAccess->IsDBCSLeadByte(ch); + } + + bool Match(int pos, const char *s) { + for (int i=0; *s; i++) { + if (*s != SafeGetCharAt(pos+i)) + return false; + s++; + } + return true; + } + char StyleAt(int position) { + return static_cast<char>(pAccess->StyleAt(position) & mask); + } + int GetLine(int position) { + return pAccess->LineFromPosition(position); + } + int LineStart(int line) { + return pAccess->LineStart(line); + } + int LevelAt(int line) { + return pAccess->GetLevel(line); + } + int Length() const { + return lenDoc; + } + void Flush() { + startPos = extremePosition; + if (validLen > 0) { + pAccess->SetStyles(validLen, styleBuf); + startPosStyling += validLen; + validLen = 0; + } + } + int GetLineState(int line) { + return pAccess->GetLineState(line); + } + int SetLineState(int line, int state) { + return pAccess->SetLineState(line, state); + } + // Style setting + void StartAt(unsigned int start, char chMask=31) { + // Store the mask specified for use with StyleAt. + mask = chMask; + pAccess->StartStyling(start, chMask); + startPosStyling = start; + } + void SetFlags(char chFlags_, char chWhile_) { + chFlags = chFlags_; + chWhile = chWhile_; + } + unsigned int GetStartSegment() const { + return startSeg; + } + void StartSegment(unsigned int pos) { + startSeg = pos; + } + void ColourTo(unsigned int 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 { + if (chAttr != chWhile) + chFlags = 0; + chAttr |= chFlags; + for (unsigned int i = startSeg; i <= pos; i++) { + assert((startPosStyling + validLen) < Length()); + styleBuf[validLen++] = static_cast<char>(chAttr); + } + } + } + startSeg = pos+1; + } + void SetLevel(int line, int level) { + pAccess->SetLevel(line, level); + } + void IndicatorFill(int start, int end, int indicator, int value) { + pAccess->DecorationSetCurrentIndicator(indicator); + pAccess->DecorationFillRange(start, value, end - start); + } + + void ChangeLexerState(int start, int end) { + pAccess->ChangeLexerState(start, end); + } +}; + +#ifdef SCI_NAMESPACE +} +#endif + +#endif |