mirror of
https://github.com/renbaoshuo/S2OJ.git
synced 2025-01-15 10:11:53 +00:00
96d4a3ecf7
Due to historical reasons, the code is in subfolder "1". With SVN removal, we place the code back and remove the annoying "1" folder.
576 lines
18 KiB
JavaScript
576 lines
18 KiB
JavaScript
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
|
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
|
|
|
// Slim Highlighting for CodeMirror copyright (c) HicknHack Software Gmbh
|
|
|
|
(function(mod) {
|
|
if (typeof exports == "object" && typeof module == "object") // CommonJS
|
|
mod(require("../../lib/codemirror"), require("../htmlmixed/htmlmixed"), require("../ruby/ruby"));
|
|
else if (typeof define == "function" && define.amd) // AMD
|
|
define(["../../lib/codemirror", "../htmlmixed/htmlmixed", "../ruby/ruby"], mod);
|
|
else // Plain browser env
|
|
mod(CodeMirror);
|
|
})(function(CodeMirror) {
|
|
"use strict";
|
|
|
|
CodeMirror.defineMode("slim", function(config) {
|
|
var htmlMode = CodeMirror.getMode(config, {name: "htmlmixed"});
|
|
var rubyMode = CodeMirror.getMode(config, "ruby");
|
|
var modes = { html: htmlMode, ruby: rubyMode };
|
|
var embedded = {
|
|
ruby: "ruby",
|
|
javascript: "javascript",
|
|
css: "text/css",
|
|
sass: "text/x-sass",
|
|
scss: "text/x-scss",
|
|
less: "text/x-less",
|
|
styl: "text/x-styl", // no highlighting so far
|
|
coffee: "coffeescript",
|
|
asciidoc: "text/x-asciidoc",
|
|
markdown: "text/x-markdown",
|
|
textile: "text/x-textile", // no highlighting so far
|
|
creole: "text/x-creole", // no highlighting so far
|
|
wiki: "text/x-wiki", // no highlighting so far
|
|
mediawiki: "text/x-mediawiki", // no highlighting so far
|
|
rdoc: "text/x-rdoc", // no highlighting so far
|
|
builder: "text/x-builder", // no highlighting so far
|
|
nokogiri: "text/x-nokogiri", // no highlighting so far
|
|
erb: "application/x-erb"
|
|
};
|
|
var embeddedRegexp = function(map){
|
|
var arr = [];
|
|
for(var key in map) arr.push(key);
|
|
return new RegExp("^("+arr.join('|')+"):");
|
|
}(embedded);
|
|
|
|
var styleMap = {
|
|
"commentLine": "comment",
|
|
"slimSwitch": "operator special",
|
|
"slimTag": "tag",
|
|
"slimId": "attribute def",
|
|
"slimClass": "attribute qualifier",
|
|
"slimAttribute": "attribute",
|
|
"slimSubmode": "keyword special",
|
|
"closeAttributeTag": null,
|
|
"slimDoctype": null,
|
|
"lineContinuation": null
|
|
};
|
|
var closing = {
|
|
"{": "}",
|
|
"[": "]",
|
|
"(": ")"
|
|
};
|
|
|
|
var nameStartChar = "_a-zA-Z\xC0-\xD6\xD8-\xF6\xF8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD";
|
|
var nameChar = nameStartChar + "\\-0-9\xB7\u0300-\u036F\u203F-\u2040";
|
|
var nameRegexp = new RegExp("^[:"+nameStartChar+"](?::["+nameChar+"]|["+nameChar+"]*)");
|
|
var attributeNameRegexp = new RegExp("^[:"+nameStartChar+"][:\\."+nameChar+"]*(?=\\s*=)");
|
|
var wrappedAttributeNameRegexp = new RegExp("^[:"+nameStartChar+"][:\\."+nameChar+"]*");
|
|
var classNameRegexp = /^\.-?[_a-zA-Z]+[\w\-]*/;
|
|
var classIdRegexp = /^#[_a-zA-Z]+[\w\-]*/;
|
|
|
|
function backup(pos, tokenize, style) {
|
|
var restore = function(stream, state) {
|
|
state.tokenize = tokenize;
|
|
if (stream.pos < pos) {
|
|
stream.pos = pos;
|
|
return style;
|
|
}
|
|
return state.tokenize(stream, state);
|
|
};
|
|
return function(stream, state) {
|
|
state.tokenize = restore;
|
|
return tokenize(stream, state);
|
|
};
|
|
}
|
|
|
|
function maybeBackup(stream, state, pat, offset, style) {
|
|
var cur = stream.current();
|
|
var idx = cur.search(pat);
|
|
if (idx > -1) {
|
|
state.tokenize = backup(stream.pos, state.tokenize, style);
|
|
stream.backUp(cur.length - idx - offset);
|
|
}
|
|
return style;
|
|
}
|
|
|
|
function continueLine(state, column) {
|
|
state.stack = {
|
|
parent: state.stack,
|
|
style: "continuation",
|
|
indented: column,
|
|
tokenize: state.line
|
|
};
|
|
state.line = state.tokenize;
|
|
}
|
|
function finishContinue(state) {
|
|
if (state.line == state.tokenize) {
|
|
state.line = state.stack.tokenize;
|
|
state.stack = state.stack.parent;
|
|
}
|
|
}
|
|
|
|
function lineContinuable(column, tokenize) {
|
|
return function(stream, state) {
|
|
finishContinue(state);
|
|
if (stream.match(/^\\$/)) {
|
|
continueLine(state, column);
|
|
return "lineContinuation";
|
|
}
|
|
var style = tokenize(stream, state);
|
|
if (stream.eol() && stream.current().match(/(?:^|[^\\])(?:\\\\)*\\$/)) {
|
|
stream.backUp(1);
|
|
}
|
|
return style;
|
|
};
|
|
}
|
|
function commaContinuable(column, tokenize) {
|
|
return function(stream, state) {
|
|
finishContinue(state);
|
|
var style = tokenize(stream, state);
|
|
if (stream.eol() && stream.current().match(/,$/)) {
|
|
continueLine(state, column);
|
|
}
|
|
return style;
|
|
};
|
|
}
|
|
|
|
function rubyInQuote(endQuote, tokenize) {
|
|
// TODO: add multi line support
|
|
return function(stream, state) {
|
|
var ch = stream.peek();
|
|
if (ch == endQuote && state.rubyState.tokenize.length == 1) {
|
|
// step out of ruby context as it seems to complete processing all the braces
|
|
stream.next();
|
|
state.tokenize = tokenize;
|
|
return "closeAttributeTag";
|
|
} else {
|
|
return ruby(stream, state);
|
|
}
|
|
};
|
|
}
|
|
function startRubySplat(tokenize) {
|
|
var rubyState;
|
|
var runSplat = function(stream, state) {
|
|
if (state.rubyState.tokenize.length == 1 && !state.rubyState.context.prev) {
|
|
stream.backUp(1);
|
|
if (stream.eatSpace()) {
|
|
state.rubyState = rubyState;
|
|
state.tokenize = tokenize;
|
|
return tokenize(stream, state);
|
|
}
|
|
stream.next();
|
|
}
|
|
return ruby(stream, state);
|
|
};
|
|
return function(stream, state) {
|
|
rubyState = state.rubyState;
|
|
state.rubyState = rubyMode.startState();
|
|
state.tokenize = runSplat;
|
|
return ruby(stream, state);
|
|
};
|
|
}
|
|
|
|
function ruby(stream, state) {
|
|
return rubyMode.token(stream, state.rubyState);
|
|
}
|
|
|
|
function htmlLine(stream, state) {
|
|
if (stream.match(/^\\$/)) {
|
|
return "lineContinuation";
|
|
}
|
|
return html(stream, state);
|
|
}
|
|
function html(stream, state) {
|
|
if (stream.match(/^#\{/)) {
|
|
state.tokenize = rubyInQuote("}", state.tokenize);
|
|
return null;
|
|
}
|
|
return maybeBackup(stream, state, /[^\\]#\{/, 1, htmlMode.token(stream, state.htmlState));
|
|
}
|
|
|
|
function startHtmlLine(lastTokenize) {
|
|
return function(stream, state) {
|
|
var style = htmlLine(stream, state);
|
|
if (stream.eol()) state.tokenize = lastTokenize;
|
|
return style;
|
|
};
|
|
}
|
|
|
|
function startHtmlMode(stream, state, offset) {
|
|
state.stack = {
|
|
parent: state.stack,
|
|
style: "html",
|
|
indented: stream.column() + offset, // pipe + space
|
|
tokenize: state.line
|
|
};
|
|
state.line = state.tokenize = html;
|
|
return null;
|
|
}
|
|
|
|
function comment(stream, state) {
|
|
stream.skipToEnd();
|
|
return state.stack.style;
|
|
}
|
|
|
|
function commentMode(stream, state) {
|
|
state.stack = {
|
|
parent: state.stack,
|
|
style: "comment",
|
|
indented: state.indented + 1,
|
|
tokenize: state.line
|
|
};
|
|
state.line = comment;
|
|
return comment(stream, state);
|
|
}
|
|
|
|
function attributeWrapper(stream, state) {
|
|
if (stream.eat(state.stack.endQuote)) {
|
|
state.line = state.stack.line;
|
|
state.tokenize = state.stack.tokenize;
|
|
state.stack = state.stack.parent;
|
|
return null;
|
|
}
|
|
if (stream.match(wrappedAttributeNameRegexp)) {
|
|
state.tokenize = attributeWrapperAssign;
|
|
return "slimAttribute";
|
|
}
|
|
stream.next();
|
|
return null;
|
|
}
|
|
function attributeWrapperAssign(stream, state) {
|
|
if (stream.match(/^==?/)) {
|
|
state.tokenize = attributeWrapperValue;
|
|
return null;
|
|
}
|
|
return attributeWrapper(stream, state);
|
|
}
|
|
function attributeWrapperValue(stream, state) {
|
|
var ch = stream.peek();
|
|
if (ch == '"' || ch == "\'") {
|
|
state.tokenize = readQuoted(ch, "string", true, false, attributeWrapper);
|
|
stream.next();
|
|
return state.tokenize(stream, state);
|
|
}
|
|
if (ch == '[') {
|
|
return startRubySplat(attributeWrapper)(stream, state);
|
|
}
|
|
if (stream.match(/^(true|false|nil)\b/)) {
|
|
state.tokenize = attributeWrapper;
|
|
return "keyword";
|
|
}
|
|
return startRubySplat(attributeWrapper)(stream, state);
|
|
}
|
|
|
|
function startAttributeWrapperMode(state, endQuote, tokenize) {
|
|
state.stack = {
|
|
parent: state.stack,
|
|
style: "wrapper",
|
|
indented: state.indented + 1,
|
|
tokenize: tokenize,
|
|
line: state.line,
|
|
endQuote: endQuote
|
|
};
|
|
state.line = state.tokenize = attributeWrapper;
|
|
return null;
|
|
}
|
|
|
|
function sub(stream, state) {
|
|
if (stream.match(/^#\{/)) {
|
|
state.tokenize = rubyInQuote("}", state.tokenize);
|
|
return null;
|
|
}
|
|
var subStream = new CodeMirror.StringStream(stream.string.slice(state.stack.indented), stream.tabSize);
|
|
subStream.pos = stream.pos - state.stack.indented;
|
|
subStream.start = stream.start - state.stack.indented;
|
|
subStream.lastColumnPos = stream.lastColumnPos - state.stack.indented;
|
|
subStream.lastColumnValue = stream.lastColumnValue - state.stack.indented;
|
|
var style = state.subMode.token(subStream, state.subState);
|
|
stream.pos = subStream.pos + state.stack.indented;
|
|
return style;
|
|
}
|
|
function firstSub(stream, state) {
|
|
state.stack.indented = stream.column();
|
|
state.line = state.tokenize = sub;
|
|
return state.tokenize(stream, state);
|
|
}
|
|
|
|
function createMode(mode) {
|
|
var query = embedded[mode];
|
|
var spec = CodeMirror.mimeModes[query];
|
|
if (spec) {
|
|
return CodeMirror.getMode(config, spec);
|
|
}
|
|
var factory = CodeMirror.modes[query];
|
|
if (factory) {
|
|
return factory(config, {name: query});
|
|
}
|
|
return CodeMirror.getMode(config, "null");
|
|
}
|
|
|
|
function getMode(mode) {
|
|
if (!modes.hasOwnProperty(mode)) {
|
|
return modes[mode] = createMode(mode);
|
|
}
|
|
return modes[mode];
|
|
}
|
|
|
|
function startSubMode(mode, state) {
|
|
var subMode = getMode(mode);
|
|
var subState = subMode.startState && subMode.startState();
|
|
|
|
state.subMode = subMode;
|
|
state.subState = subState;
|
|
|
|
state.stack = {
|
|
parent: state.stack,
|
|
style: "sub",
|
|
indented: state.indented + 1,
|
|
tokenize: state.line
|
|
};
|
|
state.line = state.tokenize = firstSub;
|
|
return "slimSubmode";
|
|
}
|
|
|
|
function doctypeLine(stream, _state) {
|
|
stream.skipToEnd();
|
|
return "slimDoctype";
|
|
}
|
|
|
|
function startLine(stream, state) {
|
|
var ch = stream.peek();
|
|
if (ch == '<') {
|
|
return (state.tokenize = startHtmlLine(state.tokenize))(stream, state);
|
|
}
|
|
if (stream.match(/^[|']/)) {
|
|
return startHtmlMode(stream, state, 1);
|
|
}
|
|
if (stream.match(/^\/(!|\[\w+])?/)) {
|
|
return commentMode(stream, state);
|
|
}
|
|
if (stream.match(/^(-|==?[<>]?)/)) {
|
|
state.tokenize = lineContinuable(stream.column(), commaContinuable(stream.column(), ruby));
|
|
return "slimSwitch";
|
|
}
|
|
if (stream.match(/^doctype\b/)) {
|
|
state.tokenize = doctypeLine;
|
|
return "keyword";
|
|
}
|
|
|
|
var m = stream.match(embeddedRegexp);
|
|
if (m) {
|
|
return startSubMode(m[1], state);
|
|
}
|
|
|
|
return slimTag(stream, state);
|
|
}
|
|
|
|
function slim(stream, state) {
|
|
if (state.startOfLine) {
|
|
return startLine(stream, state);
|
|
}
|
|
return slimTag(stream, state);
|
|
}
|
|
|
|
function slimTag(stream, state) {
|
|
if (stream.eat('*')) {
|
|
state.tokenize = startRubySplat(slimTagExtras);
|
|
return null;
|
|
}
|
|
if (stream.match(nameRegexp)) {
|
|
state.tokenize = slimTagExtras;
|
|
return "slimTag";
|
|
}
|
|
return slimClass(stream, state);
|
|
}
|
|
function slimTagExtras(stream, state) {
|
|
if (stream.match(/^(<>?|><?)/)) {
|
|
state.tokenize = slimClass;
|
|
return null;
|
|
}
|
|
return slimClass(stream, state);
|
|
}
|
|
function slimClass(stream, state) {
|
|
if (stream.match(classIdRegexp)) {
|
|
state.tokenize = slimClass;
|
|
return "slimId";
|
|
}
|
|
if (stream.match(classNameRegexp)) {
|
|
state.tokenize = slimClass;
|
|
return "slimClass";
|
|
}
|
|
return slimAttribute(stream, state);
|
|
}
|
|
function slimAttribute(stream, state) {
|
|
if (stream.match(/^([\[\{\(])/)) {
|
|
return startAttributeWrapperMode(state, closing[RegExp.$1], slimAttribute);
|
|
}
|
|
if (stream.match(attributeNameRegexp)) {
|
|
state.tokenize = slimAttributeAssign;
|
|
return "slimAttribute";
|
|
}
|
|
if (stream.peek() == '*') {
|
|
stream.next();
|
|
state.tokenize = startRubySplat(slimContent);
|
|
return null;
|
|
}
|
|
return slimContent(stream, state);
|
|
}
|
|
function slimAttributeAssign(stream, state) {
|
|
if (stream.match(/^==?/)) {
|
|
state.tokenize = slimAttributeValue;
|
|
return null;
|
|
}
|
|
// should never happen, because of forward lookup
|
|
return slimAttribute(stream, state);
|
|
}
|
|
|
|
function slimAttributeValue(stream, state) {
|
|
var ch = stream.peek();
|
|
if (ch == '"' || ch == "\'") {
|
|
state.tokenize = readQuoted(ch, "string", true, false, slimAttribute);
|
|
stream.next();
|
|
return state.tokenize(stream, state);
|
|
}
|
|
if (ch == '[') {
|
|
return startRubySplat(slimAttribute)(stream, state);
|
|
}
|
|
if (ch == ':') {
|
|
return startRubySplat(slimAttributeSymbols)(stream, state);
|
|
}
|
|
if (stream.match(/^(true|false|nil)\b/)) {
|
|
state.tokenize = slimAttribute;
|
|
return "keyword";
|
|
}
|
|
return startRubySplat(slimAttribute)(stream, state);
|
|
}
|
|
function slimAttributeSymbols(stream, state) {
|
|
stream.backUp(1);
|
|
if (stream.match(/^[^\s],(?=:)/)) {
|
|
state.tokenize = startRubySplat(slimAttributeSymbols);
|
|
return null;
|
|
}
|
|
stream.next();
|
|
return slimAttribute(stream, state);
|
|
}
|
|
function readQuoted(quote, style, embed, unescaped, nextTokenize) {
|
|
return function(stream, state) {
|
|
finishContinue(state);
|
|
var fresh = stream.current().length == 0;
|
|
if (stream.match(/^\\$/, fresh)) {
|
|
if (!fresh) return style;
|
|
continueLine(state, state.indented);
|
|
return "lineContinuation";
|
|
}
|
|
if (stream.match(/^#\{/, fresh)) {
|
|
if (!fresh) return style;
|
|
state.tokenize = rubyInQuote("}", state.tokenize);
|
|
return null;
|
|
}
|
|
var escaped = false, ch;
|
|
while ((ch = stream.next()) != null) {
|
|
if (ch == quote && (unescaped || !escaped)) {
|
|
state.tokenize = nextTokenize;
|
|
break;
|
|
}
|
|
if (embed && ch == "#" && !escaped) {
|
|
if (stream.eat("{")) {
|
|
stream.backUp(2);
|
|
break;
|
|
}
|
|
}
|
|
escaped = !escaped && ch == "\\";
|
|
}
|
|
if (stream.eol() && escaped) {
|
|
stream.backUp(1);
|
|
}
|
|
return style;
|
|
};
|
|
}
|
|
function slimContent(stream, state) {
|
|
if (stream.match(/^==?/)) {
|
|
state.tokenize = ruby;
|
|
return "slimSwitch";
|
|
}
|
|
if (stream.match(/^\/$/)) { // tag close hint
|
|
state.tokenize = slim;
|
|
return null;
|
|
}
|
|
if (stream.match(/^:/)) { // inline tag
|
|
state.tokenize = slimTag;
|
|
return "slimSwitch";
|
|
}
|
|
startHtmlMode(stream, state, 0);
|
|
return state.tokenize(stream, state);
|
|
}
|
|
|
|
var mode = {
|
|
// default to html mode
|
|
startState: function() {
|
|
var htmlState = htmlMode.startState();
|
|
var rubyState = rubyMode.startState();
|
|
return {
|
|
htmlState: htmlState,
|
|
rubyState: rubyState,
|
|
stack: null,
|
|
last: null,
|
|
tokenize: slim,
|
|
line: slim,
|
|
indented: 0
|
|
};
|
|
},
|
|
|
|
copyState: function(state) {
|
|
return {
|
|
htmlState : CodeMirror.copyState(htmlMode, state.htmlState),
|
|
rubyState: CodeMirror.copyState(rubyMode, state.rubyState),
|
|
subMode: state.subMode,
|
|
subState: state.subMode && CodeMirror.copyState(state.subMode, state.subState),
|
|
stack: state.stack,
|
|
last: state.last,
|
|
tokenize: state.tokenize,
|
|
line: state.line
|
|
};
|
|
},
|
|
|
|
token: function(stream, state) {
|
|
if (stream.sol()) {
|
|
state.indented = stream.indentation();
|
|
state.startOfLine = true;
|
|
state.tokenize = state.line;
|
|
while (state.stack && state.stack.indented > state.indented && state.last != "slimSubmode") {
|
|
state.line = state.tokenize = state.stack.tokenize;
|
|
state.stack = state.stack.parent;
|
|
state.subMode = null;
|
|
state.subState = null;
|
|
}
|
|
}
|
|
if (stream.eatSpace()) return null;
|
|
var style = state.tokenize(stream, state);
|
|
state.startOfLine = false;
|
|
if (style) state.last = style;
|
|
return styleMap.hasOwnProperty(style) ? styleMap[style] : style;
|
|
},
|
|
|
|
blankLine: function(state) {
|
|
if (state.subMode && state.subMode.blankLine) {
|
|
return state.subMode.blankLine(state.subState);
|
|
}
|
|
},
|
|
|
|
innerMode: function(state) {
|
|
if (state.subMode) return {state: state.subState, mode: state.subMode};
|
|
return {state: state, mode: mode};
|
|
}
|
|
|
|
//indent: function(state) {
|
|
// return state.indented;
|
|
//}
|
|
};
|
|
return mode;
|
|
}, "htmlmixed", "ruby");
|
|
|
|
CodeMirror.defineMIME("text/x-slim", "slim");
|
|
CodeMirror.defineMIME("application/x-slim", "slim");
|
|
});
|