| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252 | // CodeMirror, copyright (c) by Marijn Haverbeke and others// Distributed under an MIT license: https://codemirror.net/LICENSE(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("oz", function (conf) {  function wordRegexp(words) {    return new RegExp("^((" + words.join(")|(") + "))\\b");  }  var singleOperators = /[\^@!\|<>#~\.\*\-\+\\/,=]/;  var doubleOperators = /(<-)|(:=)|(=<)|(>=)|(<=)|(<:)|(>:)|(=:)|(\\=)|(\\=:)|(!!)|(==)|(::)/;  var tripleOperators = /(:::)|(\.\.\.)|(=<:)|(>=:)/;  var middle = ["in", "then", "else", "of", "elseof", "elsecase", "elseif", "catch",    "finally", "with", "require", "prepare", "import", "export", "define", "do"];  var end = ["end"];  var atoms = wordRegexp(["true", "false", "nil", "unit"]);  var commonKeywords = wordRegexp(["andthen", "at", "attr", "declare", "feat", "from", "lex",    "mod", "div", "mode", "orelse", "parser", "prod", "prop", "scanner", "self", "syn", "token"]);  var openingKeywords = wordRegexp(["local", "proc", "fun", "case", "class", "if", "cond", "or", "dis",    "choice", "not", "thread", "try", "raise", "lock", "for", "suchthat", "meth", "functor"]);  var middleKeywords = wordRegexp(middle);  var endKeywords = wordRegexp(end);  // Tokenizers  function tokenBase(stream, state) {    if (stream.eatSpace()) {      return null;    }    // Brackets    if(stream.match(/[{}]/)) {      return "bracket";    }    // Special [] keyword    if (stream.match(/(\[])/)) {        return "keyword"    }    // Operators    if (stream.match(tripleOperators) || stream.match(doubleOperators)) {      return "operator";    }    // Atoms    if(stream.match(atoms)) {      return 'atom';    }    // Opening keywords    var matched = stream.match(openingKeywords);    if (matched) {      if (!state.doInCurrentLine)        state.currentIndent++;      else        state.doInCurrentLine = false;      // Special matching for signatures      if(matched[0] == "proc" || matched[0] == "fun")        state.tokenize = tokenFunProc;      else if(matched[0] == "class")        state.tokenize = tokenClass;      else if(matched[0] == "meth")        state.tokenize = tokenMeth;      return 'keyword';    }    // Middle and other keywords    if (stream.match(middleKeywords) || stream.match(commonKeywords)) {      return "keyword"    }    // End keywords    if (stream.match(endKeywords)) {      state.currentIndent--;      return 'keyword';    }    // Eat the next char for next comparisons    var ch = stream.next();    // Strings    if (ch == '"' || ch == "'") {      state.tokenize = tokenString(ch);      return state.tokenize(stream, state);    }    // Numbers    if (/[~\d]/.test(ch)) {      if (ch == "~") {        if(! /^[0-9]/.test(stream.peek()))          return null;        else if (( stream.next() == "0" && stream.match(/^[xX][0-9a-fA-F]+/)) || stream.match(/^[0-9]*(\.[0-9]+)?([eE][~+]?[0-9]+)?/))          return "number";      }      if ((ch == "0" && stream.match(/^[xX][0-9a-fA-F]+/)) || stream.match(/^[0-9]*(\.[0-9]+)?([eE][~+]?[0-9]+)?/))        return "number";      return null;    }    // Comments    if (ch == "%") {      stream.skipToEnd();      return 'comment';    }    else if (ch == "/") {      if (stream.eat("*")) {        state.tokenize = tokenComment;        return tokenComment(stream, state);      }    }    // Single operators    if(singleOperators.test(ch)) {      return "operator";    }    // If nothing match, we skip the entire alphanumerical block    stream.eatWhile(/\w/);    return "variable";  }  function tokenClass(stream, state) {    if (stream.eatSpace()) {      return null;    }    stream.match(/([A-Z][A-Za-z0-9_]*)|(`.+`)/);    state.tokenize = tokenBase;    return "variable-3"  }  function tokenMeth(stream, state) {    if (stream.eatSpace()) {      return null;    }    stream.match(/([a-zA-Z][A-Za-z0-9_]*)|(`.+`)/);    state.tokenize = tokenBase;    return "def"  }  function tokenFunProc(stream, state) {    if (stream.eatSpace()) {      return null;    }    if(!state.hasPassedFirstStage && stream.eat("{")) {      state.hasPassedFirstStage = true;      return "bracket";    }    else if(state.hasPassedFirstStage) {      stream.match(/([A-Z][A-Za-z0-9_]*)|(`.+`)|\$/);      state.hasPassedFirstStage = false;      state.tokenize = tokenBase;      return "def"    }    else {      state.tokenize = tokenBase;      return null;    }  }  function tokenComment(stream, state) {    var maybeEnd = false, ch;    while (ch = stream.next()) {      if (ch == "/" && maybeEnd) {        state.tokenize = tokenBase;        break;      }      maybeEnd = (ch == "*");    }    return "comment";  }  function tokenString(quote) {    return function (stream, state) {      var escaped = false, next, end = false;      while ((next = stream.next()) != null) {        if (next == quote && !escaped) {          end = true;          break;        }        escaped = !escaped && next == "\\";      }      if (end || !escaped)        state.tokenize = tokenBase;      return "string";    };  }  function buildElectricInputRegEx() {    // Reindentation should occur on [] or on a match of any of    // the block closing keywords, at the end of a line.    var allClosings = middle.concat(end);    return new RegExp("[\\[\\]]|(" + allClosings.join("|") + ")$");  }  return {    startState: function () {      return {        tokenize: tokenBase,        currentIndent: 0,        doInCurrentLine: false,        hasPassedFirstStage: false      };    },    token: function (stream, state) {      if (stream.sol())        state.doInCurrentLine = 0;      return state.tokenize(stream, state);    },    indent: function (state, textAfter) {      var trueText = textAfter.replace(/^\s+|\s+$/g, '');      if (trueText.match(endKeywords) || trueText.match(middleKeywords) || trueText.match(/(\[])/))        return conf.indentUnit * (state.currentIndent - 1);      if (state.currentIndent < 0)        return 0;      return state.currentIndent * conf.indentUnit;    },    fold: "indent",    electricInput: buildElectricInputRegEx(),    lineComment: "%",    blockCommentStart: "/*",    blockCommentEnd: "*/"  };});CodeMirror.defineMIME("text/x-oz", "oz");});
 |