| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451 | // CodeMirror, copyright (c) by Marijn Haverbeke and others// Distributed under an MIT license: http://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("rust", function() {  var indentUnit = 4, altIndentUnit = 2;  var valKeywords = {    "if": "if-style", "while": "if-style", "loop": "else-style", "else": "else-style",    "do": "else-style", "ret": "else-style", "fail": "else-style",    "break": "atom", "cont": "atom", "const": "let", "resource": "fn",    "let": "let", "fn": "fn", "for": "for", "alt": "alt", "iface": "iface",    "impl": "impl", "type": "type", "enum": "enum", "mod": "mod",    "as": "op", "true": "atom", "false": "atom", "assert": "op", "check": "op",    "claim": "op", "native": "ignore", "unsafe": "ignore", "import": "else-style",    "export": "else-style", "copy": "op", "log": "op", "log_err": "op",    "use": "op", "bind": "op", "self": "atom", "struct": "enum"  };  var typeKeywords = function() {    var keywords = {"fn": "fn", "block": "fn", "obj": "obj"};    var atoms = "bool uint int i8 i16 i32 i64 u8 u16 u32 u64 float f32 f64 str char".split(" ");    for (var i = 0, e = atoms.length; i < e; ++i) keywords[atoms[i]] = "atom";    return keywords;  }();  var operatorChar = /[+\-*&%=<>!?|\.@]/;  // Tokenizer  // Used as scratch variable to communicate multiple values without  // consing up tons of objects.  var tcat, content;  function r(tc, style) {    tcat = tc;    return style;  }  function tokenBase(stream, state) {    var ch = stream.next();    if (ch == '"') {      state.tokenize = tokenString;      return state.tokenize(stream, state);    }    if (ch == "'") {      tcat = "atom";      if (stream.eat("\\")) {        if (stream.skipTo("'")) { stream.next(); return "string"; }        else { return "error"; }      } else {        stream.next();        return stream.eat("'") ? "string" : "error";      }    }    if (ch == "/") {      if (stream.eat("/")) { stream.skipToEnd(); return "comment"; }      if (stream.eat("*")) {        state.tokenize = tokenComment(1);        return state.tokenize(stream, state);      }    }    if (ch == "#") {      if (stream.eat("[")) { tcat = "open-attr"; return null; }      stream.eatWhile(/\w/);      return r("macro", "meta");    }    if (ch == ":" && stream.match(":<")) {      return r("op", null);    }    if (ch.match(/\d/) || (ch == "." && stream.eat(/\d/))) {      var flp = false;      if (!stream.match(/^x[\da-f]+/i) && !stream.match(/^b[01]+/)) {        stream.eatWhile(/\d/);        if (stream.eat(".")) { flp = true; stream.eatWhile(/\d/); }        if (stream.match(/^e[+\-]?\d+/i)) { flp = true; }      }      if (flp) stream.match(/^f(?:32|64)/);      else stream.match(/^[ui](?:8|16|32|64)/);      return r("atom", "number");    }    if (ch.match(/[()\[\]{}:;,]/)) return r(ch, null);    if (ch == "-" && stream.eat(">")) return r("->", null);    if (ch.match(operatorChar)) {      stream.eatWhile(operatorChar);      return r("op", null);    }    stream.eatWhile(/\w/);    content = stream.current();    if (stream.match(/^::\w/)) {      stream.backUp(1);      return r("prefix", "variable-2");    }    if (state.keywords.propertyIsEnumerable(content))      return r(state.keywords[content], content.match(/true|false/) ? "atom" : "keyword");    return r("name", "variable");  }  function tokenString(stream, state) {    var ch, escaped = false;    while (ch = stream.next()) {      if (ch == '"' && !escaped) {        state.tokenize = tokenBase;        return r("atom", "string");      }      escaped = !escaped && ch == "\\";    }    // Hack to not confuse the parser when a string is split in    // pieces.    return r("op", "string");  }  function tokenComment(depth) {    return function(stream, state) {      var lastCh = null, ch;      while (ch = stream.next()) {        if (ch == "/" && lastCh == "*") {          if (depth == 1) {            state.tokenize = tokenBase;            break;          } else {            state.tokenize = tokenComment(depth - 1);            return state.tokenize(stream, state);          }        }        if (ch == "*" && lastCh == "/") {          state.tokenize = tokenComment(depth + 1);          return state.tokenize(stream, state);        }        lastCh = ch;      }      return "comment";    };  }  // Parser  var cx = {state: null, stream: null, marked: null, cc: null};  function pass() {    for (var i = arguments.length - 1; i >= 0; i--) cx.cc.push(arguments[i]);  }  function cont() {    pass.apply(null, arguments);    return true;  }  function pushlex(type, info) {    var result = function() {      var state = cx.state;      state.lexical = {indented: state.indented, column: cx.stream.column(),                       type: type, prev: state.lexical, info: info};    };    result.lex = true;    return result;  }  function poplex() {    var state = cx.state;    if (state.lexical.prev) {      if (state.lexical.type == ")")        state.indented = state.lexical.indented;      state.lexical = state.lexical.prev;    }  }  function typecx() { cx.state.keywords = typeKeywords; }  function valcx() { cx.state.keywords = valKeywords; }  poplex.lex = typecx.lex = valcx.lex = true;  function commasep(comb, end) {    function more(type) {      if (type == ",") return cont(comb, more);      if (type == end) return cont();      return cont(more);    }    return function(type) {      if (type == end) return cont();      return pass(comb, more);    };  }  function stat_of(comb, tag) {    return cont(pushlex("stat", tag), comb, poplex, block);  }  function block(type) {    if (type == "}") return cont();    if (type == "let") return stat_of(letdef1, "let");    if (type == "fn") return stat_of(fndef);    if (type == "type") return cont(pushlex("stat"), tydef, endstatement, poplex, block);    if (type == "enum") return stat_of(enumdef);    if (type == "mod") return stat_of(mod);    if (type == "iface") return stat_of(iface);    if (type == "impl") return stat_of(impl);    if (type == "open-attr") return cont(pushlex("]"), commasep(expression, "]"), poplex);    if (type == "ignore" || type.match(/[\]\);,]/)) return cont(block);    return pass(pushlex("stat"), expression, poplex, endstatement, block);  }  function endstatement(type) {    if (type == ";") return cont();    return pass();  }  function expression(type) {    if (type == "atom" || type == "name") return cont(maybeop);    if (type == "{") return cont(pushlex("}"), exprbrace, poplex);    if (type.match(/[\[\(]/)) return matchBrackets(type, expression);    if (type.match(/[\]\)\};,]/)) return pass();    if (type == "if-style") return cont(expression, expression);    if (type == "else-style" || type == "op") return cont(expression);    if (type == "for") return cont(pattern, maybetype, inop, expression, expression);    if (type == "alt") return cont(expression, altbody);    if (type == "fn") return cont(fndef);    if (type == "macro") return cont(macro);    return cont();  }  function maybeop(type) {    if (content == ".") return cont(maybeprop);    if (content == "::<"){return cont(typarams, maybeop);}    if (type == "op" || content == ":") return cont(expression);    if (type == "(" || type == "[") return matchBrackets(type, expression);    return pass();  }  function maybeprop() {    if (content.match(/^\w+$/)) {cx.marked = "variable"; return cont(maybeop);}    return pass(expression);  }  function exprbrace(type) {    if (type == "op") {      if (content == "|") return cont(blockvars, poplex, pushlex("}", "block"), block);      if (content == "||") return cont(poplex, pushlex("}", "block"), block);    }    if (content == "mutable" || (content.match(/^\w+$/) && cx.stream.peek() == ":"                                 && !cx.stream.match("::", false)))      return pass(record_of(expression));    return pass(block);  }  function record_of(comb) {    function ro(type) {      if (content == "mutable" || content == "with") {cx.marked = "keyword"; return cont(ro);}      if (content.match(/^\w*$/)) {cx.marked = "variable"; return cont(ro);}      if (type == ":") return cont(comb, ro);      if (type == "}") return cont();      return cont(ro);    }    return ro;  }  function blockvars(type) {    if (type == "name") {cx.marked = "def"; return cont(blockvars);}    if (type == "op" && content == "|") return cont();    return cont(blockvars);  }  function letdef1(type) {    if (type.match(/[\]\)\};]/)) return cont();    if (content == "=") return cont(expression, letdef2);    if (type == ",") return cont(letdef1);    return pass(pattern, maybetype, letdef1);  }  function letdef2(type) {    if (type.match(/[\]\)\};,]/)) return pass(letdef1);    else return pass(expression, letdef2);  }  function maybetype(type) {    if (type == ":") return cont(typecx, rtype, valcx);    return pass();  }  function inop(type) {    if (type == "name" && content == "in") {cx.marked = "keyword"; return cont();}    return pass();  }  function fndef(type) {    if (content == "@" || content == "~") {cx.marked = "keyword"; return cont(fndef);}    if (type == "name") {cx.marked = "def"; return cont(fndef);}    if (content == "<") return cont(typarams, fndef);    if (type == "{") return pass(expression);    if (type == "(") return cont(pushlex(")"), commasep(argdef, ")"), poplex, fndef);    if (type == "->") return cont(typecx, rtype, valcx, fndef);    if (type == ";") return cont();    return cont(fndef);  }  function tydef(type) {    if (type == "name") {cx.marked = "def"; return cont(tydef);}    if (content == "<") return cont(typarams, tydef);    if (content == "=") return cont(typecx, rtype, valcx);    return cont(tydef);  }  function enumdef(type) {    if (type == "name") {cx.marked = "def"; return cont(enumdef);}    if (content == "<") return cont(typarams, enumdef);    if (content == "=") return cont(typecx, rtype, valcx, endstatement);    if (type == "{") return cont(pushlex("}"), typecx, enumblock, valcx, poplex);    return cont(enumdef);  }  function enumblock(type) {    if (type == "}") return cont();    if (type == "(") return cont(pushlex(")"), commasep(rtype, ")"), poplex, enumblock);    if (content.match(/^\w+$/)) cx.marked = "def";    return cont(enumblock);  }  function mod(type) {    if (type == "name") {cx.marked = "def"; return cont(mod);}    if (type == "{") return cont(pushlex("}"), block, poplex);    return pass();  }  function iface(type) {    if (type == "name") {cx.marked = "def"; return cont(iface);}    if (content == "<") return cont(typarams, iface);    if (type == "{") return cont(pushlex("}"), block, poplex);    return pass();  }  function impl(type) {    if (content == "<") return cont(typarams, impl);    if (content == "of" || content == "for") {cx.marked = "keyword"; return cont(rtype, impl);}    if (type == "name") {cx.marked = "def"; return cont(impl);}    if (type == "{") return cont(pushlex("}"), block, poplex);    return pass();  }  function typarams() {    if (content == ">") return cont();    if (content == ",") return cont(typarams);    if (content == ":") return cont(rtype, typarams);    return pass(rtype, typarams);  }  function argdef(type) {    if (type == "name") {cx.marked = "def"; return cont(argdef);}    if (type == ":") return cont(typecx, rtype, valcx);    return pass();  }  function rtype(type) {    if (type == "name") {cx.marked = "variable-3"; return cont(rtypemaybeparam); }    if (content == "mutable") {cx.marked = "keyword"; return cont(rtype);}    if (type == "atom") return cont(rtypemaybeparam);    if (type == "op" || type == "obj") return cont(rtype);    if (type == "fn") return cont(fntype);    if (type == "{") return cont(pushlex("{"), record_of(rtype), poplex);    return matchBrackets(type, rtype);  }  function rtypemaybeparam() {    if (content == "<") return cont(typarams);    return pass();  }  function fntype(type) {    if (type == "(") return cont(pushlex("("), commasep(rtype, ")"), poplex, fntype);    if (type == "->") return cont(rtype);    return pass();  }  function pattern(type) {    if (type == "name") {cx.marked = "def"; return cont(patternmaybeop);}    if (type == "atom") return cont(patternmaybeop);    if (type == "op") return cont(pattern);    if (type.match(/[\]\)\};,]/)) return pass();    return matchBrackets(type, pattern);  }  function patternmaybeop(type) {    if (type == "op" && content == ".") return cont();    if (content == "to") {cx.marked = "keyword"; return cont(pattern);}    else return pass();  }  function altbody(type) {    if (type == "{") return cont(pushlex("}", "alt"), altblock1, poplex);    return pass();  }  function altblock1(type) {    if (type == "}") return cont();    if (type == "|") return cont(altblock1);    if (content == "when") {cx.marked = "keyword"; return cont(expression, altblock2);}    if (type.match(/[\]\);,]/)) return cont(altblock1);    return pass(pattern, altblock2);  }  function altblock2(type) {    if (type == "{") return cont(pushlex("}", "alt"), block, poplex, altblock1);    else return pass(altblock1);  }  function macro(type) {    if (type.match(/[\[\(\{]/)) return matchBrackets(type, expression);    return pass();  }  function matchBrackets(type, comb) {    if (type == "[") return cont(pushlex("]"), commasep(comb, "]"), poplex);    if (type == "(") return cont(pushlex(")"), commasep(comb, ")"), poplex);    if (type == "{") return cont(pushlex("}"), commasep(comb, "}"), poplex);    return cont();  }  function parse(state, stream, style) {    var cc = state.cc;    // Communicate our context to the combinators.    // (Less wasteful than consing up a hundred closures on every call.)    cx.state = state; cx.stream = stream; cx.marked = null, cx.cc = cc;    while (true) {      var combinator = cc.length ? cc.pop() : block;      if (combinator(tcat)) {        while(cc.length && cc[cc.length - 1].lex)          cc.pop()();        return cx.marked || style;      }    }  }  return {    startState: function() {      return {        tokenize: tokenBase,        cc: [],        lexical: {indented: -indentUnit, column: 0, type: "top", align: false},        keywords: valKeywords,        indented: 0      };    },    token: function(stream, state) {      if (stream.sol()) {        if (!state.lexical.hasOwnProperty("align"))          state.lexical.align = false;        state.indented = stream.indentation();      }      if (stream.eatSpace()) return null;      tcat = content = null;      var style = state.tokenize(stream, state);      if (style == "comment") return style;      if (!state.lexical.hasOwnProperty("align"))        state.lexical.align = true;      if (tcat == "prefix") return style;      if (!content) content = stream.current();      return parse(state, stream, style);    },    indent: function(state, textAfter) {      if (state.tokenize != tokenBase) return 0;      var firstChar = textAfter && textAfter.charAt(0), lexical = state.lexical,          type = lexical.type, closing = firstChar == type;      if (type == "stat") return lexical.indented + indentUnit;      if (lexical.align) return lexical.column + (closing ? 0 : 1);      return lexical.indented + (closing ? 0 : (lexical.info == "alt" ? altIndentUnit : indentUnit));    },    electricChars: "{}",    blockCommentStart: "/*",    blockCommentEnd: "*/",    lineComment: "//",    fold: "brace"  };});CodeMirror.defineMIME("text/x-rustsrc", "rust");});
 |