mirror of
https://github.com/hedgedoc/hedgedoc.git
synced 2025-06-05 01:04:25 -04:00
First commit, version 0.2.7
This commit is contained in:
parent
61eb11d23c
commit
4b0ca55eb7
1379 changed files with 173000 additions and 0 deletions
575
public/vendor/codemirror/mode/slim/slim.js
vendored
Executable file
575
public/vendor/codemirror/mode/slim/slim.js
vendored
Executable file
|
@ -0,0 +1,575 @@
|
|||
// 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");
|
||||
});
|
Loading…
Add table
Add a link
Reference in a new issue