// Scintilla source code edit control /** @file LexEiffel.cxx ** Lexer for Eiffel. **/ // Copyright 1998-2001 by Neil Hodgson // The License.txt file describes the conditions under which this software may be distributed. #include #include #include #include #include #include #include "Platform.h" #include "PropSet.h" #include "Accessor.h" #include "KeyWords.h" #include "Scintilla.h" #include "SciLexer.h" inline bool isEiffelOperator(unsigned int ch) { // '.' left out as it is used to make up numbers return ch == '*' || ch == '/' || ch == '\\' || ch == '-' || ch == '+' || ch == '(' || ch == ')' || ch == '=' || ch == '{' || ch == '}' || ch == '~' || ch == '[' || ch == ']' || ch == ';' || ch == '<' || ch == '>' || ch == ',' || ch == '.' || ch == '^' || ch == '%' || ch == ':' || ch == '!' || ch == '@' || ch == '?'; } static void getRangeLowered(unsigned int start, unsigned int end, Accessor &styler, char *s, unsigned int len) { unsigned int i = 0; while ((i < end - start + 1) && (i < len-1)) { s[i] = static_cast(tolower(styler[start + i])); i++; } s[i] = '\0'; } inline bool IsASpace(unsigned int ch) { return (ch == ' ') || ((ch >= 0x09) && (ch <= 0x0d)); } inline bool IsAWordChar(unsigned int ch) { return (ch < 0x80) && (isalnum(ch) || ch == '.' || ch == '_'); } inline bool IsAWordStart(unsigned int ch) { return (ch < 0x80) && (isalnum(ch) || ch == '_'); } inline bool IsADigit(unsigned int ch) { return (ch >= '0') && (ch <= '9'); } // 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 xColouriseContext { Accessor &styler; int lengthDoc; int currentPos; xColouriseContext& operator=(const xColouriseContext&) { return *this; } public: int state; unsigned int chPrev; unsigned int ch; unsigned int chNext; xColouriseContext(unsigned int startPos, int length, int initStyle, Accessor &styler_) : styler(styler_), lengthDoc(startPos + length), currentPos(startPos), state(initStyle), chPrev(0), ch(0), chNext(0) { styler.StartAt(startPos); styler.StartSegment(startPos); int pos = currentPos; ch = static_cast(styler.SafeGetCharAt(pos)); if (styler.IsLeadByte(static_cast(ch))) { pos++; ch = ch << 8; ch |= static_cast(styler.SafeGetCharAt(pos)); } chNext = static_cast(styler.SafeGetCharAt(pos+1)); if (styler.IsLeadByte(static_cast(chNext))) { chNext = chNext << 8; chNext |= static_cast(styler.SafeGetCharAt(pos+2)); } } void Complete() { styler.ColourTo(currentPos - 1, state); } bool More() { return currentPos <= lengthDoc; } void Forward() { chPrev = ch; currentPos++; if (ch >= 0x100) currentPos++; ch = chNext; chNext = static_cast(styler.SafeGetCharAt(currentPos+1)); if (styler.IsLeadByte(static_cast(chNext))) { chNext = chNext << 8; chNext |= static_cast(styler.SafeGetCharAt(currentPos + 2)); } } void ChangeState(int state_) { state = state_; } void SetState(int state_) { styler.ColourTo(currentPos - 1, state); state = state_; } void GetCurrentLowered(char *s, int len) { getRangeLowered(styler.GetStartSegment(), currentPos - 1, styler, s, len); } }; static void ColouriseEiffelDoc(unsigned int startPos, int length, int initStyle, WordList *keywordlists[], Accessor &styler) { WordList &keywords = *keywordlists[0]; xColouriseContext lc(startPos, length, initStyle, styler); for (; lc.More(); lc.Forward()) { if (lc.state == SCE_EIFFEL_STRINGEOL) { if (lc.ch != '\r' && lc.ch != '\n') { lc.SetState(SCE_EIFFEL_DEFAULT); } } else if (lc.state == SCE_EIFFEL_OPERATOR) { lc.SetState(SCE_EIFFEL_DEFAULT); } else if (lc.state == SCE_EIFFEL_WORD) { if (!IsAWordChar(lc.ch)) { char s[100]; lc.GetCurrentLowered(s, sizeof(s)); if (!keywords.InList(s)) { lc.ChangeState(SCE_EIFFEL_IDENTIFIER); } lc.SetState(SCE_EIFFEL_DEFAULT); } } else if (lc.state == SCE_EIFFEL_NUMBER) { if (!IsAWordChar(lc.ch)) { lc.SetState(SCE_EIFFEL_DEFAULT); } } else if (lc.state == SCE_EIFFEL_COMMENTLINE) { if (lc.ch == '\r' || lc.ch == '\n') { lc.SetState(SCE_EIFFEL_DEFAULT); } } else if (lc.state == SCE_EIFFEL_STRING) { if (lc.ch == '%') { lc.Forward(); } else if (lc.ch == '\"') { lc.Forward(); lc.SetState(SCE_EIFFEL_DEFAULT); } } else if (lc.state == SCE_EIFFEL_CHARACTER) { if (lc.ch == '\r' || lc.ch == '\n') { lc.SetState(SCE_EIFFEL_STRINGEOL); } else if (lc.ch == '%') { lc.Forward(); } else if (lc.ch == '\'') { lc.Forward(); lc.SetState(SCE_EIFFEL_DEFAULT); } } if (lc.state == SCE_EIFFEL_DEFAULT) { if (lc.ch == '-' && lc.chNext == '-') { lc.SetState(SCE_EIFFEL_COMMENTLINE); } else if (lc.ch == '\"') { lc.SetState(SCE_EIFFEL_STRING); } else if (lc.ch == '\'') { lc.SetState(SCE_EIFFEL_CHARACTER); } else if (IsADigit(lc.ch) || (lc.ch == '.')) { lc.SetState(SCE_EIFFEL_NUMBER); } else if (IsAWordStart(lc.ch)) { lc.SetState(SCE_EIFFEL_WORD); } else if (isEiffelOperator(lc.ch)) { lc.SetState(SCE_EIFFEL_OPERATOR); } } } lc.Complete(); } static bool IsEiffelComment(Accessor &styler, int pos, int len) { return len>1 && styler[pos]=='-' && styler[pos+1]=='-'; } static void FoldEiffelDocIndent(unsigned int startPos, int length, int, WordList *[], Accessor &styler) { int lengthDoc = startPos + length; // Backtrack to previous line in case need to fix its fold status int lineCurrent = styler.GetLine(startPos); if (startPos > 0) { if (lineCurrent > 0) { lineCurrent--; startPos = styler.LineStart(lineCurrent); } } int spaceFlags = 0; int indentCurrent = styler.IndentAmount(lineCurrent, &spaceFlags, IsEiffelComment); char chNext = styler[startPos]; for (int i = startPos; i < lengthDoc; i++) { char ch = chNext; chNext = styler.SafeGetCharAt(i + 1); if ((ch == '\r' && chNext != '\n') || (ch == '\n') || (i == lengthDoc)) { int lev = indentCurrent; int indentNext = styler.IndentAmount(lineCurrent + 1, &spaceFlags, IsEiffelComment); if (!(indentCurrent & SC_FOLDLEVELWHITEFLAG)) { // Only non whitespace lines can be headers if ((indentCurrent & SC_FOLDLEVELNUMBERMASK) < (indentNext & SC_FOLDLEVELNUMBERMASK)) { lev |= SC_FOLDLEVELHEADERFLAG; } else if (indentNext & SC_FOLDLEVELWHITEFLAG) { // Line after is blank so check the next - maybe should continue further? int spaceFlags2 = 0; int indentNext2 = styler.IndentAmount(lineCurrent + 2, &spaceFlags2, IsEiffelComment); if ((indentCurrent & SC_FOLDLEVELNUMBERMASK) < (indentNext2 & SC_FOLDLEVELNUMBERMASK)) { lev |= SC_FOLDLEVELHEADERFLAG; } } } indentCurrent = indentNext; styler.SetLevel(lineCurrent, lev); lineCurrent++; } } } static void FoldEiffelDocKeyWords(unsigned int startPos, int length, int /* initStyle */, WordList *[], Accessor &styler) { unsigned int lengthDoc = startPos + length; int visibleChars = 0; int lineCurrent = styler.GetLine(startPos); int levelPrev = styler.LevelAt(lineCurrent) & SC_FOLDLEVELNUMBERMASK; int levelCurrent = levelPrev; char chNext = styler[startPos]; int stylePrev = 0; int styleNext = styler.StyleAt(startPos); // lastDeferred should be determined by looking back to last keyword in case // the "deferred" is on a line before "class" bool lastDeferred = false; for (unsigned int i = startPos; i < lengthDoc; i++) { char ch = chNext; chNext = styler.SafeGetCharAt(i + 1); int style = styleNext; styleNext = styler.StyleAt(i + 1); bool atEOL = (ch == '\r' && chNext != '\n') || (ch == '\n'); if ((stylePrev != SCE_EIFFEL_WORD) && (style == SCE_EIFFEL_WORD)) { char s[20]; unsigned int j = 0; while ((j < (sizeof(s) - 1)) && (iswordchar(styler[i + j]))) { s[j] = styler[i + j]; j++; } s[j] = '\0'; if ( (strcmp(s, "check") == 0) || (strcmp(s, "debug") == 0) || (strcmp(s, "deferred") == 0) || (strcmp(s, "do") == 0) || (strcmp(s, "from") == 0) || (strcmp(s, "if") == 0) || (strcmp(s, "inspect") == 0) || (strcmp(s, "once") == 0) ) levelCurrent++; if (!lastDeferred && (strcmp(s, "class") == 0)) levelCurrent++; if (strcmp(s, "end") == 0) levelCurrent--; lastDeferred = strcmp(s, "deferred") == 0; } if (atEOL) { int lev = levelPrev; if (visibleChars == 0) lev |= SC_FOLDLEVELWHITEFLAG; if ((levelCurrent > levelPrev) && (visibleChars > 0)) lev |= SC_FOLDLEVELHEADERFLAG; if (lev != styler.LevelAt(lineCurrent)) { styler.SetLevel(lineCurrent, lev); } lineCurrent++; levelPrev = levelCurrent; visibleChars = 0; } if (!isspacechar(ch)) visibleChars++; stylePrev = style; } // Fill in the real level of the next line, keeping the current flags as they will be filled in later int flagsNext = styler.LevelAt(lineCurrent) & ~SC_FOLDLEVELNUMBERMASK; styler.SetLevel(lineCurrent, levelPrev | flagsNext); } LexerModule lmEiffel(SCLEX_EIFFEL, ColouriseEiffelDoc, "eiffel", FoldEiffelDocIndent); LexerModule lmEiffelkw(SCLEX_EIFFELKW, ColouriseEiffelDoc, "eiffelkw", FoldEiffelDocKeyWords);