mirror of
https://github.com/boxgaming/qbjs.git
synced 2024-09-20 04:24:45 +00:00
363 lines
16 KiB
JavaScript
363 lines
16 KiB
JavaScript
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
|
// Distributed under an MIT license: https://codemirror.net/LICENSE
|
|
|
|
// TODO: remove vestiges of vbscript
|
|
|
|
(function(mod) {
|
|
if (typeof exports == "object" && typeof module == "object") // CommonJS
|
|
mod(require("../../lib/codemirror"));
|
|
else if (typeof define == "function" && define.amd) // AMD
|
|
define(["../../lib/codemirror"], mod);
|
|
else // Plain browser env
|
|
mod(CodeMirror);
|
|
})(function(CodeMirror) {
|
|
"use strict";
|
|
|
|
CodeMirror.defineMode("qbjs", function(conf, parserConf) {
|
|
var ERRORCLASS = 'error';
|
|
|
|
function wordRegexp(words) {
|
|
return new RegExp("^((" + words.join(")|(") + "))\\b", "i");
|
|
}
|
|
|
|
var singleOperators = new RegExp("^[\\+\\-\\*/&\\\\\\^<>=]");
|
|
var doubleOperators = new RegExp("^((<>)|(<=)|(>=))");
|
|
var singleDelimiters = new RegExp('^[\\.,]');
|
|
var brakets = new RegExp('^[\\(\\)]');
|
|
var identifiers = new RegExp("^[A-Za-z][_A-Za-z0-9]*");
|
|
|
|
var openingKeywords = ['sub','select','while','if','function', 'property', 'with', 'for', 'type'];
|
|
var middleKeywords = ['else','elseif','case'];
|
|
var endKeywords = ['next','loop','wend'];
|
|
|
|
var wordOperators = wordRegexp(['and', 'or', 'not', 'xor', 'is', 'mod', 'eqv', 'imp']);
|
|
var commonkeywords = ['dim', 'as', 'redim', 'then', 'until', 'randomize',
|
|
'byval','byref','new','property', 'exit', 'in',
|
|
'const', 'integer', 'single', 'double', '_unsigned', 'string', '_byte',
|
|
'let', 'option explicit', 'call'];
|
|
|
|
//This list was from: http://msdn.microsoft.com/en-us/library/f8tbc79x(v=vs.84).aspx
|
|
var atomWords = ['true', 'false', 'nothing', 'empty', 'null'];
|
|
|
|
// QB64 Keywords
|
|
var builtinFuncsWords = ['_delay', '_fontwidth', '_height', '_limit', '_keyhit', '_newimage', '_pi', '_printstring', '_printwidth', '_rgb',
|
|
'_rgb32', '_round', '_title', '_trim', '_width', 'abs', 'asc', 'atn', 'chr\\$', 'cls', 'color', 'cos', 'fix', 'input',
|
|
'instr', 'int', 'left\\$', 'lcase\\$', 'len', 'line', 'locate', 'ltrim\\$', 'mid\\$', 'print', 'right\\$', 'rtrim\\$', 'rnd',
|
|
'screen', 'sgn', 'sin', 'sleep', 'sqr', 'str\\$', 'tan', 'ubound', 'ucase\\$', 'val', '_screenexists', 'exp', 'timer',
|
|
'pset', '_red32', '_green32', '_blue32', '_red', '_green', '_blue', 'circle', '_alpha32', '_keydown', 'swap', '_instrrev',
|
|
'command\\$', '_mouseinput', '_mousex', '_mousey', '_mousebutton', 'log', '_atan2', 'inkey\\$'];
|
|
|
|
//This list was from: http://msdn.microsoft.com/en-us/library/ydz4cfk3(v=vs.84).aspx
|
|
var builtinConsts = ['gx_true','vbBlack', 'vbRed', 'vbGreen', 'vbYellow', 'vbBlue', 'vbMagenta', 'vbCyan', 'vbWhite', 'vbBinaryCompare', 'vbTextCompare',
|
|
'vbSunday', 'vbMonday', 'vbTuesday', 'vbWednesday', 'vbThursday', 'vbFriday', 'vbSaturday', 'vbUseSystemDayOfWeek', 'vbFirstJan1', 'vbFirstFourDays', 'vbFirstFullWeek',
|
|
'vbGeneralDate', 'vbLongDate', 'vbShortDate', 'vbLongTime', 'vbShortTime', 'vbObjectError',
|
|
'vbOKOnly', 'vbOKCancel', 'vbAbortRetryIgnore', 'vbYesNoCancel', 'vbYesNo', 'vbRetryCancel', 'vbCritical', 'vbQuestion', 'vbExclamation', 'vbInformation', 'vbDefaultButton1', 'vbDefaultButton2',
|
|
'vbDefaultButton3', 'vbDefaultButton4', 'vbApplicationModal', 'vbSystemModal', 'vbOK', 'vbCancel', 'vbAbort', 'vbRetry', 'vbIgnore', 'vbYes', 'vbNo',
|
|
'vbCr', 'VbCrLf', 'vbFormFeed', 'vbLf', 'vbNewLine', 'vbNullChar', 'vbNullString', 'vbTab', 'vbVerticalTab', 'vbUseDefault', 'vbTrue', 'vbFalse',
|
|
'vbEmpty', 'vbNull', 'vbInteger', 'vbLong', 'vbSingle', 'vbDouble', 'vbCurrency', 'vbDate', 'vbString', 'vbObject', 'vbError', 'vbBoolean', 'vbVariant', 'vbDataObject', 'vbDecimal', 'vbByte', 'vbArray'];
|
|
//This list was from: http://msdn.microsoft.com/en-us/library/hkc375ea(v=vs.84).aspx
|
|
var builtinObjsWords = ['WScript', 'err', 'debug', 'RegExp', '\\$if', '\\$end if'];
|
|
var knownProperties = ['description', 'firstindex', 'global', 'helpcontext', 'helpfile', 'ignorecase', 'length', 'number', 'pattern', 'source', 'value', 'count'];
|
|
//var knownMethods = ['clear', 'execute', 'raise', 'replace', 'test', 'write', 'writeline', 'close', 'open', 'state', 'eof', 'update', 'addnew', 'end', 'createobject', 'quit', 'gxscenecreate'];
|
|
var knownMethods = ['gxongameevent', 'gxmousex', 'gxmousey', 'gxsoundload', 'gxsoundplay', 'gxsoundrepeat', 'gxsoundvolume', 'gxsoundpause', 'gxsoundstop',
|
|
'gxsoundmuted', 'gxsoundmuted', 'gxentityanimate', 'gxentityanimatestop', 'gxentityanimatemode', 'gxentityanimatemode',
|
|
'gxscreenentitycreate', 'gxentitycreate', 'gxentityvisible', 'gxentitymove', 'gxentitypos', 'gxentityvx', 'gxentityvy',
|
|
'gxentityx', 'gxentityy', 'gxentitywidth', 'gxentityheight', 'gxentityframenext', 'gxentityframeset', 'gxentitytype',
|
|
'gxentitytype', 'gxentityuid', 'gxfontuid', 'gxentityapplygravity', 'gxentitycollisionoffset',
|
|
'gxentitycollisionoffsetleft', 'gxentitycollisionoffsettop', 'gxentitycollisionoffsetright',
|
|
'gxentitycollisionoffsetbottom', 'gxfullscreen', 'gxbackgroundadd', 'gxbackgroundy', 'gxbackgroundheight',
|
|
'gxbackgroundclear', 'gxsceneembedded', 'gxscenecreate', 'gxscenewindowsize', 'gxscenescale', 'gxsceneresize',
|
|
'gxscenedestroy', 'gxcustomdraw', 'gxframerate', 'gxframe', 'gxscenedraw', 'gxscenemove', 'gxscenepos', 'gxscenex',
|
|
'gxsceney', 'gxscenewidth', 'gxsceneheight', 'gxscenecolumns', 'gxscenerows', 'gxscenestart', 'gxsceneupdate',
|
|
'gxscenefollowentity', 'gxsceneconstrain', 'gxscenestop', 'gxmapcreate', 'gxmapcolumns', 'gxmaprows', 'gxmaplayers',
|
|
'gxmaplayervisible', 'gxmaplayeradd', 'gxmaplayerinsert', 'gxmaplayerremove', 'gxmapresize', 'gxmapdraw',
|
|
'gxmaptileposat', 'gxmaptile', 'gxmaptile', 'gxmaptiledepth', 'gxmaptileadd', 'gxmaptileremove', 'gxmapversion',
|
|
'gxmapsave', 'gxmapload', 'gxmapisometric', 'gxspritedraw', 'gxspritedrawscaled', 'gxtilesetcreate',
|
|
'gxtilesetreplaceimage', 'gxtilesetload', 'gxtilesetsave', 'gxtilesetpos', 'gxtilesetwidth', 'gxtilesetheight',
|
|
'gxtilesetcolumns', 'gxtilesetrows', 'gxtilesetfilename', 'gxtilesetimage', 'gxtilesetanimationcreate',
|
|
'gxtilesetanimationadd', 'gxtilesetanimationremove', 'gxtilesetanimationframes', 'gxtilesetanimationspeed',
|
|
'gxfontcreate', 'gxfontwidth', 'gxfontheight', 'gxfontcharspacing', 'gxfontlinespacing',
|
|
'gxdrawtext', 'gxdebug', 'gxdebug', 'gxdebugscreenentities', 'gxdebugfont', 'gxdebugtilebordercolor',
|
|
'gxdebugentitybordercolor', 'gxdebugentitycollisioncolor', 'gxkeyinput', 'gxkeydown', 'gxdeviceinputdetect',
|
|
'gxdeviceinputtest', 'gxdevicename', 'gxdevicetypename', 'gxinputtypename', 'gxkeybuttonname'];
|
|
|
|
var aspBuiltinObjsWords = ['server', 'response', 'request', 'session', 'application'];
|
|
var aspKnownProperties = ['buffer', 'cachecontrol', 'charset', 'contenttype', 'expires', 'expiresabsolute', 'isclientconnected', 'pics', 'status', //response
|
|
'clientcertificate', 'cookies', 'form', 'querystring', 'servervariables', 'totalbytes', //request
|
|
'contents', 'staticobjects', //application
|
|
'codepage', 'lcid', 'sessionid', 'timeout', //session
|
|
'scripttimeout']; //server
|
|
var aspKnownMethods = ['addheader', 'appendtolog', 'binarywrite', 'end', 'flush', 'redirect', //response
|
|
'binaryread', //request
|
|
'remove', 'removeall', 'lock', 'unlock', //application
|
|
'abandon', //session
|
|
'getlasterror', 'htmlencode', 'mappath', 'transfer', 'urlencode']; //server
|
|
|
|
var knownWords = knownMethods.concat(knownProperties);
|
|
|
|
builtinObjsWords = builtinObjsWords.concat(builtinConsts);
|
|
|
|
if (conf.isASP){
|
|
builtinObjsWords = builtinObjsWords.concat(aspBuiltinObjsWords);
|
|
knownWords = knownWords.concat(aspKnownMethods, aspKnownProperties);
|
|
};
|
|
|
|
var keywords = wordRegexp(commonkeywords);
|
|
var atoms = wordRegexp(atomWords);
|
|
var builtinFuncs = wordRegexp(builtinFuncsWords);
|
|
var builtinObjs = wordRegexp(builtinObjsWords);
|
|
var known = wordRegexp(knownWords);
|
|
var stringPrefixes = '"';
|
|
|
|
var opening = wordRegexp(openingKeywords);
|
|
var middle = wordRegexp(middleKeywords);
|
|
var closing = wordRegexp(endKeywords);
|
|
var doubleClosing = wordRegexp(['end']);
|
|
var doOpening = wordRegexp(['do']);
|
|
var noIndentWords = wordRegexp(['on error resume next', 'exit']);
|
|
var comment = wordRegexp(['rem']);
|
|
|
|
|
|
function indent(_stream, state) {
|
|
state.currentIndent++;
|
|
}
|
|
|
|
function dedent(_stream, state) {
|
|
state.currentIndent--;
|
|
}
|
|
// tokenizers
|
|
function tokenBase(stream, state) {
|
|
if (stream.eatSpace()) {
|
|
return 'space';
|
|
//return null;
|
|
}
|
|
|
|
var ch = stream.peek();
|
|
|
|
// Handle Comments
|
|
if (ch === "'") {
|
|
stream.skipToEnd();
|
|
return 'comment';
|
|
}
|
|
if (stream.match(comment)){
|
|
stream.skipToEnd();
|
|
return 'comment';
|
|
}
|
|
|
|
|
|
// Handle Number Literals
|
|
if (stream.match(/^((&H)|(&O))?[0-9\.]/i, false) && !stream.match(/^((&H)|(&O))?[0-9\.]+[a-z_]/i, false)) {
|
|
var floatLiteral = false;
|
|
// Floats
|
|
if (stream.match(/^\d*\.\d+/i)) { floatLiteral = true; }
|
|
else if (stream.match(/^\d+\.\d*/)) { floatLiteral = true; }
|
|
else if (stream.match(/^\.\d+/)) { floatLiteral = true; }
|
|
|
|
if (floatLiteral) {
|
|
// Float literals may be "imaginary"
|
|
stream.eat(/J/i);
|
|
return 'number';
|
|
}
|
|
// Integers
|
|
var intLiteral = false;
|
|
// Hex
|
|
if (stream.match(/^&H[0-9a-f]+/i)) { intLiteral = true; }
|
|
// Octal
|
|
else if (stream.match(/^&O[0-7]+/i)) { intLiteral = true; }
|
|
// Decimal
|
|
else if (stream.match(/^[1-9]\d*F?/)) {
|
|
// Decimal literals may be "imaginary"
|
|
stream.eat(/J/i);
|
|
// TODO - Can you have imaginary longs?
|
|
intLiteral = true;
|
|
}
|
|
// Zero by itself with no other piece of number.
|
|
else if (stream.match(/^0(?![\dx])/i)) { intLiteral = true; }
|
|
if (intLiteral) {
|
|
// Integer literals may be "long"
|
|
stream.eat(/L/i);
|
|
return 'number';
|
|
}
|
|
}
|
|
|
|
// Handle Strings
|
|
if (stream.match(stringPrefixes)) {
|
|
state.tokenize = tokenStringFactory(stream.current());
|
|
return state.tokenize(stream, state);
|
|
}
|
|
|
|
// Handle operators and Delimiters
|
|
if (stream.match(doubleOperators)
|
|
|| stream.match(singleOperators)
|
|
|| stream.match(wordOperators)) {
|
|
return 'operator';
|
|
}
|
|
if (stream.match(singleDelimiters)) {
|
|
return null;
|
|
}
|
|
|
|
if (stream.match(brakets)) {
|
|
return "bracket";
|
|
}
|
|
|
|
if (stream.match(noIndentWords)) {
|
|
state.doInCurrentLine = true;
|
|
|
|
return 'keyword';
|
|
}
|
|
|
|
if (stream.match(doOpening)) {
|
|
indent(stream,state);
|
|
state.doInCurrentLine = true;
|
|
|
|
return 'keyword';
|
|
}
|
|
if (stream.match(opening)) {
|
|
if (! state.doInCurrentLine)
|
|
indent(stream,state);
|
|
else
|
|
state.doInCurrentLine = false;
|
|
|
|
return 'keyword';
|
|
}
|
|
if (stream.match(middle)) {
|
|
return 'keyword';
|
|
}
|
|
|
|
|
|
if (stream.match(doubleClosing)) {
|
|
dedent(stream,state);
|
|
dedent(stream,state);
|
|
|
|
return 'keyword';
|
|
}
|
|
if (stream.match(closing)) {
|
|
if (! state.doInCurrentLine)
|
|
dedent(stream,state);
|
|
else
|
|
state.doInCurrentLine = false;
|
|
|
|
return 'keyword';
|
|
}
|
|
|
|
if (stream.match(keywords)) {
|
|
return 'keyword';
|
|
}
|
|
|
|
if (stream.match(atoms)) {
|
|
return 'atom';
|
|
}
|
|
|
|
if (stream.match(known)) {
|
|
return 'variable-2';
|
|
}
|
|
|
|
if (stream.match(builtinFuncs)) {
|
|
return 'builtin';
|
|
}
|
|
|
|
if (stream.match(builtinObjs)){
|
|
return 'variable-2';
|
|
}
|
|
|
|
if (stream.match(identifiers)) {
|
|
return 'variable';
|
|
}
|
|
|
|
// Handle non-detected items
|
|
stream.next();
|
|
return ERRORCLASS;
|
|
}
|
|
|
|
function tokenStringFactory(delimiter) {
|
|
var singleline = delimiter.length == 1;
|
|
var OUTCLASS = 'string';
|
|
|
|
return function(stream, state) {
|
|
while (!stream.eol()) {
|
|
stream.eatWhile(/[^'"]/);
|
|
if (stream.match(delimiter)) {
|
|
state.tokenize = tokenBase;
|
|
return OUTCLASS;
|
|
} else {
|
|
stream.eat(/['"]/);
|
|
}
|
|
}
|
|
if (singleline) {
|
|
if (parserConf.singleLineStringErrors) {
|
|
return ERRORCLASS;
|
|
} else {
|
|
state.tokenize = tokenBase;
|
|
}
|
|
}
|
|
return OUTCLASS;
|
|
};
|
|
}
|
|
|
|
|
|
function tokenLexer(stream, state) {
|
|
var style = state.tokenize(stream, state);
|
|
var current = stream.current();
|
|
|
|
// Handle '.' connected identifiers
|
|
if (current === '.') {
|
|
style = state.tokenize(stream, state);
|
|
|
|
current = stream.current();
|
|
if (style && (style.substr(0, 8) === 'variable' || style==='builtin' || style==='keyword')){//|| knownWords.indexOf(current.substring(1)) > -1) {
|
|
if (style === 'builtin' || style === 'keyword') style='variable';
|
|
if (knownWords.indexOf(current.substr(1)) > -1) style='variable-2';
|
|
|
|
return style;
|
|
} else {
|
|
return ERRORCLASS;
|
|
}
|
|
}
|
|
|
|
return style;
|
|
}
|
|
|
|
var external = {
|
|
electricChars:"dDpPtTfFeE ",
|
|
startState: function() {
|
|
return {
|
|
tokenize: tokenBase,
|
|
lastToken: null,
|
|
currentIndent: 0,
|
|
nextLineIndent: 0,
|
|
doInCurrentLine: false,
|
|
ignoreKeyword: false
|
|
|
|
|
|
};
|
|
},
|
|
|
|
token: function(stream, state) {
|
|
if (stream.sol()) {
|
|
state.currentIndent += state.nextLineIndent;
|
|
state.nextLineIndent = 0;
|
|
state.doInCurrentLine = 0;
|
|
}
|
|
var style = tokenLexer(stream, state);
|
|
|
|
state.lastToken = {style:style, content: stream.current()};
|
|
|
|
if (style==='space') style=null;
|
|
|
|
return style;
|
|
},
|
|
|
|
indent: function(state, textAfter) {
|
|
var trueText = textAfter.replace(/^\s+|\s+$/g, '') ;
|
|
if (trueText.match(closing) || trueText.match(doubleClosing) || trueText.match(middle)) return conf.indentUnit*(state.currentIndent-1);
|
|
if(state.currentIndent < 0) return 0;
|
|
return state.currentIndent * conf.indentUnit;
|
|
}
|
|
|
|
};
|
|
return external;
|
|
});
|
|
|
|
CodeMirror.defineMIME("text/qbjs", "qbjs");
|
|
|
|
});
|