mirror of
				https://github.com/notepad-plus-plus/notepad-plus-plus.git
				synced 2025-10-31 11:34:05 +01:00 
			
		
		
		
	Update with https://www.scintilla.org/scintilla521.zip https://www.scintilla.org/lexilla515.zip - fix setting to bring Scintilla::PositionCR from ScintillaStructures.h inline with Sci_Position.h Sci_PositionCR - add workaround to enable lexer for searchResult commented out SCI_SETILEXER call on searchResult to get one result which is correctly handled by the lexer, added comment about the current problem with property @MarkingsStruct which seems to disappear after call to SCI_SETILEXER or CreateLexer - corrected usage of ObjC lexer - removed unnecessary filter stuff - use own sections for scintilla and lexilla build targets and allow parallel builds - as libscilex is no longer existing, changed to libscintilla - adapt makefiles and cmake - use VS2019 - started simple changes for createlexer adaptations, nullpointercheck missing on return of lexer name from deprecated LexerNameFromID -> undefined behaviour - movement from id -> lexer name, mostly done via LexerNameFromID + switching off corresponding compiler warning - changed to SCI_SETILEXER from SCI_SETLEXER, SCI_SETLEXERLANGUAGE needs to be corrected, see Scintilla5Migration.html - just commented out: SCI_LOADLEXERLIBRARY Fix #10504, close #11419
		
			
				
	
	
		
			379 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			379 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Scintilla source code edit control
 | |
| /** @file LexInno.cxx
 | |
|  ** Lexer for Inno Setup scripts.
 | |
|  **/
 | |
| // Written by Friedrich Vedder <fvedd@t-online.de>, using code from LexOthers.cxx.
 | |
| // Modified by Michael Heath.
 | |
| // The License.txt file describes the conditions under which this software may be distributed.
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <stdio.h>
 | |
| #include <stdarg.h>
 | |
| #include <assert.h>
 | |
| #include <ctype.h>
 | |
| 
 | |
| #include <string>
 | |
| #include <string_view>
 | |
| 
 | |
| #include "ILexer.h"
 | |
| #include "Scintilla.h"
 | |
| #include "SciLexer.h"
 | |
| 
 | |
| #include "WordList.h"
 | |
| #include "LexAccessor.h"
 | |
| #include "Accessor.h"
 | |
| #include "StyleContext.h"
 | |
| #include "CharacterSet.h"
 | |
| #include "LexerModule.h"
 | |
| 
 | |
| using namespace Lexilla;
 | |
| 
 | |
| static bool innoIsBlank(int ch) {
 | |
| 	return (ch == ' ') || (ch == '\t');
 | |
| }
 | |
| 
 | |
| static bool innoNextNotBlankIs(Sci_Position i, Accessor &styler, char needle) {
 | |
| 	char ch;
 | |
| 
 | |
| 	while (i < styler.Length()) {
 | |
| 		ch = styler.SafeGetCharAt(i);
 | |
| 
 | |
| 		if (ch == needle)
 | |
| 			return true;
 | |
| 
 | |
| 		if (!innoIsBlank(ch))
 | |
| 			return false;
 | |
| 
 | |
| 		i++;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| static void ColouriseInnoDoc(Sci_PositionU startPos, Sci_Position length, int, WordList *keywordLists[], Accessor &styler) {
 | |
| 	int state = SCE_INNO_DEFAULT;
 | |
| 	char chPrev;
 | |
| 	char ch = 0;
 | |
| 	char chNext = styler[startPos];
 | |
| 	Sci_Position lengthDoc = startPos + length;
 | |
| 	char *buffer = new char[length + 1];
 | |
| 	Sci_Position bufferCount = 0;
 | |
| 	bool isBOL, isEOL, isWS, isBOLWS = 0;
 | |
| 
 | |
| 	// Save line state later with bitState and bitand with bit... to get line state
 | |
| 	int bitState = 0;
 | |
| 	int const bitCode = 1, bitMessages = 2, bitCommentCurly = 4, bitCommentRound = 8;
 | |
| 
 | |
| 	// Get keyword lists
 | |
| 	WordList §ionKeywords = *keywordLists[0];
 | |
| 	WordList &standardKeywords = *keywordLists[1];
 | |
| 	WordList ¶meterKeywords = *keywordLists[2];
 | |
| 	WordList &preprocessorKeywords = *keywordLists[3];
 | |
| 	WordList &pascalKeywords = *keywordLists[4];
 | |
| 	WordList &userKeywords = *keywordLists[5];
 | |
| 
 | |
| 	// Get line state
 | |
| 	Sci_Position curLine = styler.GetLine(startPos);
 | |
| 	int curLineState = curLine > 0 ? styler.GetLineState(curLine - 1) : 0;
 | |
| 	bool isCode = (curLineState & bitCode);
 | |
| 	bool isMessages = (curLineState & bitMessages);
 | |
| 	bool isCommentCurly = (curLineState & bitCommentCurly);
 | |
| 	bool isCommentRound = (curLineState & bitCommentRound);
 | |
| 	bool isCommentSlash = false;
 | |
| 
 | |
| 	// Continue Pascal multline comment state
 | |
| 	if (isCommentCurly || isCommentRound)
 | |
| 		state = SCE_INNO_COMMENT_PASCAL;
 | |
| 
 | |
| 	// Go through all provided text segment
 | |
| 	// using the hand-written state machine shown below
 | |
| 	styler.StartAt(startPos);
 | |
| 	styler.StartSegment(startPos);
 | |
| 
 | |
| 	for (Sci_Position i = startPos; i < lengthDoc; i++) {
 | |
| 		chPrev = ch;
 | |
| 		ch = chNext;
 | |
| 		chNext = styler.SafeGetCharAt(i + 1);
 | |
| 
 | |
| 		if (styler.IsLeadByte(ch)) {
 | |
| 			chNext = styler.SafeGetCharAt(i + 2);
 | |
| 			i++;
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		isBOL = (chPrev == 0) || (chPrev == '\n') || (chPrev == '\r' && ch != '\n');
 | |
| 		isBOLWS = (isBOL) ? 1 : (isBOLWS && (chPrev == ' ' || chPrev == '\t'));
 | |
| 		isEOL = (ch == '\n' || ch == '\r');
 | |
| 		isWS = (ch == ' ' || ch == '\t');
 | |
| 
 | |
| 		if ((ch == '\r' && chNext != '\n') || (ch == '\n')) {
 | |
| 			// Remember the line state for future incremental lexing
 | |
| 			curLine = styler.GetLine(i);
 | |
| 			bitState = 0;
 | |
| 
 | |
| 			if (isCode)
 | |
| 				bitState |= bitCode;
 | |
| 
 | |
| 			if (isMessages)
 | |
| 				bitState |= bitMessages;
 | |
| 
 | |
| 			if (isCommentCurly)
 | |
| 				bitState |= bitCommentCurly;
 | |
| 
 | |
| 			if (isCommentRound)
 | |
| 				bitState |= bitCommentRound;
 | |
| 
 | |
| 			styler.SetLineState(curLine, bitState);
 | |
| 		}
 | |
| 
 | |
| 		switch(state) {
 | |
| 			case SCE_INNO_DEFAULT:
 | |
| 				if (!isCode && ch == ';' && isBOLWS) {
 | |
| 					// Start of a comment
 | |
| 					state = SCE_INNO_COMMENT;
 | |
| 					styler.ColourTo(i, SCE_INNO_COMMENT);
 | |
| 				} else if (ch == '[' && isBOLWS) {
 | |
| 					// Start of a section name
 | |
| 					state = SCE_INNO_SECTION;
 | |
| 					bufferCount = 0;
 | |
| 				} else if (ch == '#' && isBOLWS) {
 | |
| 					// Start of a preprocessor directive
 | |
| 					state = SCE_INNO_PREPROC;
 | |
| 				} else if (!isCode && ch == '{' && chNext != '{' && chPrev != '{') {
 | |
| 					// Start of an inline expansion
 | |
| 					state = SCE_INNO_INLINE_EXPANSION;
 | |
| 				} else if (isCode && ch == '{') {
 | |
| 					// Start of a Pascal comment
 | |
| 					state = SCE_INNO_COMMENT_PASCAL;
 | |
| 					isCommentCurly = true;
 | |
| 					styler.ColourTo(i, SCE_INNO_COMMENT_PASCAL);
 | |
| 				} else if (isCode && (ch == '(' && chNext == '*')) {
 | |
| 					// Start of a Pascal comment
 | |
| 					state = SCE_INNO_COMMENT_PASCAL;
 | |
| 					isCommentRound = true;
 | |
| 					styler.ColourTo(i + 1, SCE_INNO_COMMENT_PASCAL);
 | |
| 				} else if (isCode && ch == '/' && chNext == '/') {
 | |
| 					// Start of C-style comment
 | |
| 					state = SCE_INNO_COMMENT_PASCAL;
 | |
| 					isCommentSlash = true;
 | |
| 					styler.ColourTo(i + 1, SCE_INNO_COMMENT_PASCAL);
 | |
| 				} else if (!isMessages && ch == '"') {
 | |
| 					// Start of a double-quote string
 | |
| 					state = SCE_INNO_STRING_DOUBLE;
 | |
| 					styler.ColourTo(i, SCE_INNO_STRING_DOUBLE);
 | |
| 				} else if (!isMessages && ch == '\'') {
 | |
| 					// Start of a single-quote string
 | |
| 					state = SCE_INNO_STRING_SINGLE;
 | |
| 					styler.ColourTo(i, SCE_INNO_STRING_SINGLE);
 | |
| 				} else if (!isMessages && IsASCII(ch) && (isalpha(ch) || (ch == '_'))) {
 | |
| 					// Start of an identifier
 | |
| 					state = SCE_INNO_IDENTIFIER;
 | |
| 					bufferCount = 0;
 | |
| 					buffer[bufferCount++] = static_cast<char>(tolower(ch));
 | |
| 				} else {
 | |
| 					// Style it the default style
 | |
| 					styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_COMMENT:
 | |
| 				if (isEOL) {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i - 1, SCE_INNO_COMMENT);
 | |
| 					styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 				} else {
 | |
| 					styler.ColourTo(i, SCE_INNO_COMMENT);
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_IDENTIFIER:
 | |
| 				if (IsASCII(ch) && (isalnum(ch) || (ch == '_'))) {
 | |
| 					buffer[bufferCount++] = static_cast<char>(tolower(ch));
 | |
| 				} else {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					buffer[bufferCount] = '\0';
 | |
| 
 | |
| 					// Check if the buffer contains a keyword
 | |
| 					if (!isCode && standardKeywords.InList(buffer) && innoNextNotBlankIs(i, styler, '=')) {
 | |
| 						styler.ColourTo(i - 1, SCE_INNO_KEYWORD);
 | |
| 					} else if (!isCode && parameterKeywords.InList(buffer) && innoNextNotBlankIs(i, styler, ':')) {
 | |
| 						styler.ColourTo(i - 1, SCE_INNO_PARAMETER);
 | |
| 					} else if (isCode && pascalKeywords.InList(buffer)) {
 | |
| 						styler.ColourTo(i - 1, SCE_INNO_KEYWORD_PASCAL);
 | |
| 					} else if (!isCode && userKeywords.InList(buffer)) {
 | |
| 						styler.ColourTo(i - 1, SCE_INNO_KEYWORD_USER);
 | |
| 					} else {
 | |
| 						styler.ColourTo(i - 1, SCE_INNO_DEFAULT);
 | |
| 					}
 | |
| 
 | |
| 					// Push back the faulty character
 | |
| 					chNext = styler[i--];
 | |
| 					ch = chPrev;
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_SECTION:
 | |
| 				if (ch == ']') {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					buffer[bufferCount] = '\0';
 | |
| 
 | |
| 					// Check if the buffer contains a section name
 | |
| 					if (sectionKeywords.InList(buffer)) {
 | |
| 						styler.ColourTo(i, SCE_INNO_SECTION);
 | |
| 						isCode = !CompareCaseInsensitive(buffer, "code");
 | |
| 
 | |
| 						isMessages = isCode ? false : (
 | |
| 									!CompareCaseInsensitive(buffer, "custommessages")
 | |
| 									|| !CompareCaseInsensitive(buffer, "messages"));
 | |
| 					} else {
 | |
| 						styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 					}
 | |
| 				} else if (IsASCII(ch) && (isalnum(ch) || (ch == '_'))) {
 | |
| 					buffer[bufferCount++] = static_cast<char>(tolower(ch));
 | |
| 				} else {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_PREPROC:
 | |
| 				if (isWS || isEOL) {
 | |
| 					if (IsASCII(chPrev) && isalpha(chPrev)) {
 | |
| 						state = SCE_INNO_DEFAULT;
 | |
| 						buffer[bufferCount] = '\0';
 | |
| 
 | |
| 						// Check if the buffer contains a preprocessor directive
 | |
| 						if (preprocessorKeywords.InList(buffer)) {
 | |
| 							styler.ColourTo(i - 1, SCE_INNO_PREPROC);
 | |
| 						} else {
 | |
| 							styler.ColourTo(i - 1, SCE_INNO_DEFAULT);
 | |
| 						}
 | |
| 
 | |
| 						// Push back the faulty character
 | |
| 						chNext = styler[i--];
 | |
| 						ch = chPrev;
 | |
| 					}
 | |
| 				} else if (IsASCII(ch) && isalpha(ch)) {
 | |
| 					if (chPrev == '#' || chPrev == ' ' || chPrev == '\t')
 | |
| 						bufferCount = 0;
 | |
| 					buffer[bufferCount++] = static_cast<char>(tolower(ch));
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_STRING_DOUBLE:
 | |
| 				if (ch == '"') {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i, SCE_INNO_STRING_DOUBLE);
 | |
| 				} else if (isEOL) {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i - 1, SCE_INNO_STRING_DOUBLE);
 | |
| 					styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 				} else {
 | |
| 					styler.ColourTo(i, SCE_INNO_STRING_DOUBLE);
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_STRING_SINGLE:
 | |
| 				if (ch == '\'') {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i, SCE_INNO_STRING_SINGLE);
 | |
| 				} else if (isEOL) {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i - 1, SCE_INNO_STRING_SINGLE);
 | |
| 					styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 				} else {
 | |
| 					styler.ColourTo(i, SCE_INNO_STRING_SINGLE);
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_INLINE_EXPANSION:
 | |
| 				if (ch == '}') {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i, SCE_INNO_INLINE_EXPANSION);
 | |
| 				} else if (isEOL) {
 | |
| 					state = SCE_INNO_DEFAULT;
 | |
| 					styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case SCE_INNO_COMMENT_PASCAL:
 | |
| 				if (isCommentSlash) {
 | |
| 					if (isEOL) {
 | |
| 						state = SCE_INNO_DEFAULT;
 | |
| 						isCommentSlash = false;
 | |
| 						styler.ColourTo(i - 1, SCE_INNO_COMMENT_PASCAL);
 | |
| 						styler.ColourTo(i, SCE_INNO_DEFAULT);
 | |
| 					} else {
 | |
| 						styler.ColourTo(i, SCE_INNO_COMMENT_PASCAL);
 | |
| 					}
 | |
| 				} else if (isCommentCurly) {
 | |
| 					if (ch == '}') {
 | |
| 						state = SCE_INNO_DEFAULT;
 | |
| 						isCommentCurly = false;
 | |
| 					}
 | |
| 					styler.ColourTo(i, SCE_INNO_COMMENT_PASCAL);
 | |
| 				} else if (isCommentRound) {
 | |
| 					if (ch == ')' && chPrev == '*') {
 | |
| 						state = SCE_INNO_DEFAULT;
 | |
| 						isCommentRound = false;
 | |
| 					}
 | |
| 					styler.ColourTo(i, SCE_INNO_COMMENT_PASCAL);
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 		}
 | |
| 	}
 | |
| 	delete []buffer;
 | |
| }
 | |
| 
 | |
| static const char * const innoWordListDesc[] = {
 | |
| 	"Sections",
 | |
| 	"Keywords",
 | |
| 	"Parameters",
 | |
| 	"Preprocessor directives",
 | |
| 	"Pascal keywords",
 | |
| 	"User defined keywords",
 | |
| 	0
 | |
| };
 | |
| 
 | |
| static void FoldInnoDoc(Sci_PositionU startPos, Sci_Position length, int, WordList *[], Accessor &styler) {
 | |
| 	Sci_PositionU endPos = startPos + length;
 | |
| 	char chNext = styler[startPos];
 | |
| 
 | |
| 	Sci_Position lineCurrent = styler.GetLine(startPos);
 | |
| 
 | |
| 	bool sectionFlag = false;
 | |
| 	int levelPrev = lineCurrent > 0 ? styler.LevelAt(lineCurrent - 1) : SC_FOLDLEVELBASE;
 | |
| 	int level;
 | |
| 
 | |
| 	for (Sci_PositionU i = startPos; i < endPos; i++) {
 | |
| 		char ch = chNext;
 | |
| 		chNext = styler[i + 1];
 | |
| 		bool atEOL = (ch == '\r' && chNext != '\n') || (ch == '\n');
 | |
| 		int style = styler.StyleAt(i);
 | |
| 
 | |
| 		if (style == SCE_INNO_SECTION)
 | |
| 			sectionFlag = true;
 | |
| 
 | |
| 		if (atEOL || i == endPos - 1) {
 | |
| 			if (sectionFlag) {
 | |
| 				level = SC_FOLDLEVELBASE | SC_FOLDLEVELHEADERFLAG;
 | |
| 				if (level == levelPrev)
 | |
| 					styler.SetLevel(lineCurrent - 1, levelPrev & ~SC_FOLDLEVELHEADERFLAG);
 | |
| 			} else {
 | |
| 				level = levelPrev & SC_FOLDLEVELNUMBERMASK;
 | |
| 				if (levelPrev & SC_FOLDLEVELHEADERFLAG)
 | |
| 					level++;
 | |
| 			}
 | |
| 
 | |
| 			styler.SetLevel(lineCurrent, level);
 | |
| 
 | |
| 			levelPrev = level;
 | |
| 			lineCurrent++;
 | |
| 			sectionFlag = false;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| LexerModule lmInno(SCLEX_INNOSETUP, ColouriseInnoDoc, "inno", FoldInnoDoc, innoWordListDesc);
 |