⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 lexcaml.cxx

📁 非常好用的可移植的多平台C/C++源代码编辑器
💻 CXX
字号:
// Scintilla source code edit control
/** @file LexCaml.cxx
 ** Lexer for Objective Caml.
 **/
// Copyright 2005 by Robert Roessler <robertr@rftp.com>
// The License.txt file describes the conditions under which this software may be distributed.
/*	Release History
	20050204 Initial release.
	20050205 Quick compiler standards/"cleanliness" adjustment.
	20050206 Added cast for IsLeadByte().
	20050209 Changes to "external" build support.
	20050306 Fix for 1st-char-in-doc "corner" case.
	20050502 Fix for [harmless] one-past-the-end coloring.
	20050515 Refined numeric token recognition logic.
*/

#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <stdio.h>
#include <stdarg.h>

#include "Platform.h"

#include "PropSet.h"
#include "Accessor.h"
#include "StyleContext.h"
#include "KeyWords.h"
#include "Scintilla.h"
#include "SciLexer.h"

//	Since the Microsoft __iscsym[f] funcs are not ANSI...
inline int  iscaml(int c) {return isalnum(c) || c == '_';}
inline int iscamlf(int c) {return isalpha(c) || c == '_';}
inline int iscamld(int c) {return isdigit(c) || c == '_';}

#ifdef BUILD_AS_EXTERNAL_LEXER
/*
	(actually seems to work!)
*/
#include "WindowAccessor.h"
#include "ExternalLexer.h"

#if PLAT_WIN
#include <windows.h>
#endif

static void ColouriseCamlDoc(
	unsigned int startPos, int length,
	int initStyle,
	WordList *keywordlists[],
	Accessor &styler);

static void FoldCamlDoc(
	unsigned int startPos, int length,
	int initStyle,
	WordList *keywordlists[],
	Accessor &styler);

static void InternalLexOrFold(int lexOrFold, unsigned int startPos, int length,
	int initStyle, char *words[], WindowID window, char *props);

static const char* LexerName = "caml";

#ifdef TRACE
void Platform::DebugPrintf(const char *format, ...) {
	char buffer[2000];
	va_list pArguments;
	va_start(pArguments, format);
	vsprintf(buffer,format,pArguments);
	va_end(pArguments);
	Platform::DebugDisplay(buffer);
}
#else
void Platform::DebugPrintf(const char *, ...) {
}
#endif

bool Platform::IsDBCSLeadByte(int codePage, char ch) {
	return ::IsDBCSLeadByteEx(codePage, ch) != 0;
}

long Platform::SendScintilla(WindowID w, unsigned int msg, unsigned long wParam, long lParam) {
	return ::SendMessage(reinterpret_cast<HWND>(w), msg, wParam, lParam);
}

long Platform::SendScintillaPointer(WindowID w, unsigned int msg, unsigned long wParam, void *lParam) {
	return ::SendMessage(reinterpret_cast<HWND>(w), msg, wParam,
		reinterpret_cast<LPARAM>(lParam));
}

void EXT_LEXER_DECL Fold(unsigned int lexer, unsigned int startPos, int length,
	int initStyle, char *words[], WindowID window, char *props)
{
	// below useless evaluation(s) to supress "not used" warnings
	lexer;
	// build expected data structures and do the Fold
	InternalLexOrFold(1, startPos, length, initStyle, words, window, props);

}

int EXT_LEXER_DECL GetLexerCount()
{
	return 1;	// just us [Objective] Caml lexers here!
}

void EXT_LEXER_DECL GetLexerName(unsigned int Index, char *name, int buflength)
{
	// below useless evaluation(s) to supress "not used" warnings
	Index;
	// return as much of our lexer name as will fit (what's up with Index?)
	if (buflength > 0) {
		buflength--;
		int n = strlen(LexerName);
		if (n > buflength)
			n = buflength;
		memcpy(name, LexerName, n), name[n] = '\0';
	}
}

void EXT_LEXER_DECL Lex(unsigned int lexer, unsigned int startPos, int length,
	int initStyle, char *words[], WindowID window, char *props)
{
	// below useless evaluation(s) to supress "not used" warnings
	lexer;
	// build expected data structures and do the Lex
	InternalLexOrFold(0, startPos, length, initStyle, words, window, props);
}

static void InternalLexOrFold(int foldOrLex, unsigned int startPos, int length,
	int initStyle, char *words[], WindowID window, char *props)
{
	// create and initialize a WindowAccessor (including contained PropSet)
	PropSet ps;
	ps.SetMultiple(props);
	WindowAccessor wa(window, ps);
	// create and initialize WordList(s)
	int nWL = 0;
	for (; words[nWL]; nWL++) ;	// count # of WordList PTRs needed
	WordList** wl = new WordList* [nWL + 1];// alloc WordList PTRs
	int i = 0;
	for (; i < nWL; i++) {
		wl[i] = new WordList();	// (works or THROWS bad_alloc EXCEPTION)
		wl[i]->Set(words[i]);
	}
	wl[i] = 0;
	// call our "internal" folder/lexer (... then do Flush!)
	if (foldOrLex)
		FoldCamlDoc(startPos, length, initStyle, wl, wa);
	else
		ColouriseCamlDoc(startPos, length, initStyle, wl, wa);
	wa.Flush();
	// clean up before leaving
	for (i = nWL - 1; i >= 0; i--)
		delete wl[i];
	delete [] wl;
}

static
#endif	/* BUILD_AS_EXTERNAL_LEXER */

void ColouriseCamlDoc(
	unsigned int startPos, int length,
	int initStyle,
	WordList *keywordlists[],
	Accessor &styler)
{
	// initialize styler
	styler.StartAt(startPos);
	styler.StartSegment(startPos);
	// set up [initial] state info (terminating states that shouldn't "bleed")
	int state = initStyle, nesting = 0;
	if (state < SCE_CAML_STRING)
		state = SCE_CAML_DEFAULT;
	if (state >= SCE_CAML_COMMENT)
		nesting = state - SCE_CAML_COMMENT;
	int chLast = startPos? static_cast<unsigned char>(styler[startPos - 1]): ' ';
	int chNext = static_cast<unsigned char>(styler[startPos]);

	int chBase = 'd', chToken = 0, chLit = 0, chSkip;
	WordList& keywords = *keywordlists[0];
	WordList& keywords2 = *keywordlists[1];

	// foreach char in range...
	unsigned int i = startPos;
	const unsigned int endPos = startPos + length;
	for (; i < endPos; i += chSkip) {
		// set up [per-char] state info
		int ch = chNext;
		chNext = static_cast<unsigned char>(styler.SafeGetCharAt(i + 1));
		int state2 = -1;	// (ASSUME no state change)
		int chColor = i - 1;// (ASSUME standard coloring range)
		chSkip = 1;			// (ASSUME scanner "eats" 1 char)

		// this may be the correct thing to do... or not
		if (styler.IsLeadByte(static_cast<char>(ch))) {
			chNext = static_cast<unsigned char>(styler.SafeGetCharAt(i + 2)),
				chSkip++;
			continue;
		}

		// step state machine
		switch (state) {
		case SCE_CAML_DEFAULT:
			// it's wide open; what do we have?
			if (iscamlf(ch))
				state2 = SCE_CAML_IDENTIFIER, chToken = i;
			else if (ch == '`')
				state2 = SCE_CAML_TAGNAME, chToken = i;
			else if (ch == '#' && isdigit(chNext))
				state2 = SCE_CAML_LINENUM, chToken = i;
			else if (isdigit(ch)) {
				state2 = SCE_CAML_NUMBER,
					chBase = strchr("xXoObB", chNext)? chNext: 'd';
				if (chBase != 'd')
					ch = chNext,
						chNext = static_cast<unsigned char>(styler.SafeGetCharAt(i + 2)),
						chSkip++;
			} else if (ch == '\'')	/* (char literal?) */
				state2 = SCE_CAML_CHAR, chToken = i, chLit = 0;
			else if (ch == '\"')
				state2 = SCE_CAML_STRING;
			else if (ch == '(' && chNext == '*')
				state2 = SCE_CAML_COMMENT,
					ch = ' ',	// (make SURE "(*)" isn't seen as a closed comment)
					chNext = static_cast<unsigned char>(styler.SafeGetCharAt(i + 2)),
					chSkip++, nesting = 0;
			else if (strchr("!?~"		/* Caml "prefix-symbol" */
					"=<>@^|&+-*/$%"		/* Caml "infix-symbol" */
					"()[]{};,:.#", ch))	/* Caml "bracket" or ;,:.# */
				state2 = SCE_CAML_OPERATOR, chToken = i;
			break;

		case SCE_CAML_IDENTIFIER:
			// [try to] interpret as [additional] identifier char
			if (!(iscaml(ch) || ch == '\'')) {
				const int n = i - chToken;
				if (n < 24) {
					// length is believable as keyword, [re-]construct token
					char t[24];
					int p = 0;
					for (int q = chToken; p < n; p++, q++)
						t[p] = styler[q];
					t[p] = '\0';
					// special-case "_" token as KEYWORD
					if ((n == 1 && chLast == '_') || keywords.InList(t))
						state = SCE_CAML_KEYWORD;
					else if (keywords2.InList(t))
						state = SCE_CAML_KEYWORD2;
				}
				state2 = SCE_CAML_DEFAULT, chNext = ch, chSkip--;
			}
			break;

		case SCE_CAML_TAGNAME:
			// [try to] interpret as [additional] tagname char
			if (!(iscaml(ch) || ch == '\''))
				state2 = SCE_CAML_DEFAULT, chNext = ch, chSkip--;
			break;

		/*case SCE_CAML_KEYWORD:
		case SCE_CAML_KEYWORD2:
			// [try to] interpret as [additional] keyword char
			if (!iscaml(ch))
				state2 = SCE_CAML_DEFAULT, chNext = ch, chSkip--;
			break;*/

		case SCE_CAML_LINENUM:
			// [try to] interpret as [additional] linenum directive char
			if (!isdigit(ch))
				state2 = SCE_CAML_DEFAULT, chNext = ch, chSkip--;
			break;

		case SCE_CAML_OPERATOR: {
			// [try to] interpret as [additional] operator char
			char* o = 0;
			if (iscaml(ch) || isspace(ch)			/* ident or whitespace */
				|| ((o = strchr(")]};,\'\"`#", ch)) != 0)/* "termination" chars */
				|| !strchr("!$%&*+-./:<=>?@^|~", ch)/* "operator" chars */) {
				// check for INCLUSIVE termination
				if (o && strchr(")]};,", ch)) {
					if ((ch == ')' && chLast == '(') || (ch == ']' && chLast == '['))
						// special-case "()" and "[]" tokens as KEYWORDS
						state = SCE_CAML_KEYWORD;
					chColor++;
				} else
					chNext = ch, chSkip--;
				state2 = SCE_CAML_DEFAULT;
			}
			break;
		}

		case SCE_CAML_NUMBER:
			// [try to] interpret as [additional] numeric literal char
			// N.B. - improperly accepts "extra" digits in base 2 or 8 literals
			if (iscamld(ch) || ((chBase == 'x' || chBase == 'X') && isxdigit(ch)))
				break;
			// how about an integer suffix?
			if ((ch == 'l' || ch == 'L' || ch == 'n')&& (iscamld(chLast)
				|| ((chBase == 'x' || chBase == 'X') && isxdigit(chLast))))
				break;
			// or a floating-point literal?
			if (chBase == 'd') {
				// with a decimal point?
				if (ch == '.' && iscamld(chLast))
					break;
				// with an exponent? (I)
				if ((ch == 'e' || ch == 'E') && (iscamld(chLast) || chLast == '.'))
					break;
				// with an exponent? (II)
				if ((ch == '+' || ch == '-') && (chLast == 'e' || chLast == 'E'))
					break;
			}
			// it looks like we have run out of number
			state2 = SCE_CAML_DEFAULT, chNext = ch, chSkip--;
			break;

		case SCE_CAML_CHAR:
			// [try to] interpret as [additional] char literal char
			if (ch == '\\') {
				chLit = 1;	// (definitely IS a char literal)
				if (chLast == '\\')
					ch = ' ';	// (so termination test isn't fooled)
			// should we be terminating - one way or another?
			} else if ((ch == '\'' && chLast != '\\') || ch == '\r' || ch == '\n') {
				state2 = SCE_CAML_DEFAULT;
				if (ch == '\'')
					chColor++;
				else
					state = SCE_CAML_IDENTIFIER;
			// ... maybe a char literal, maybe not
			} else if (chLit < 1 && i - chToken >= 2)
				state = SCE_CAML_IDENTIFIER, chNext = ch, chSkip--;
			break;

		case SCE_CAML_STRING:
			// [try to] interpret as [additional] string literal char
			if (ch == '\\' && chLast == '\\')
				ch = ' ';	// (so '\\' doesn't cause us trouble)
			else if (ch == '\"' && chLast != '\\')
				state2 = SCE_CAML_DEFAULT, chColor++;
			break;

		case SCE_CAML_COMMENT:
		case SCE_CAML_COMMENT+1:
		case SCE_CAML_COMMENT+2:
		case SCE_CAML_COMMENT+3:
			// we're IN a comment - does this start a NESTED comment?
			if (ch == '(' && chNext == '*')
				state2 = state + 1,
					ch = ' ',	// (make SURE "(*)" isn't seen as a closed comment)
					chNext = static_cast<unsigned char>(styler.SafeGetCharAt(i + 2)),
					chSkip++, nesting++;
			// [try to] interpret as [additional] comment char
			else if (ch == ')' && chLast == '*')
				state2 = nesting? (state - 1): SCE_CAML_DEFAULT, chColor++, nesting--;
			break;
		}

		// handle state change and char coloring as required
		if (state2 >= 0) {
			// (1st char will NOT be colored until AT LEAST 2nd char)
			if (chColor >= 0)
				styler.ColourTo(chColor, state);
			state = state2;
		}
		chLast = ch;
	}

	// do any required terminal char coloring (JIC)
	if (i >= endPos)
		i = endPos - 1;
	styler.ColourTo(i, state);
//	styler.Flush();	// (is this always done by calling code?)
}

#ifdef BUILD_AS_EXTERNAL_LEXER
static
#endif	/* BUILD_AS_EXTERNAL_LEXER */
void FoldCamlDoc(
	unsigned int startPos, int length,
	int initStyle,
	WordList *keywordlists[],
	Accessor &styler)
{
	// below useless evaluation(s) to supress "not used" warnings
	startPos || length || initStyle || keywordlists[0] || styler.Length();
}

static const char * const camlWordListDesc[] = {
	"Keywords",		// primary Objective Caml keywords
	"Keywords2",	// "optional" keywords (typically from Pervasives)
	0
};

#ifndef BUILD_AS_EXTERNAL_LEXER
LexerModule lmCaml(SCLEX_CAML, ColouriseCamlDoc, "caml", FoldCamlDoc, camlWordListDesc);
#endif	/* BUILD_AS_EXTERNAL_LEXER */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -