mirror of
https://github.com/renbaoshuo/S2OJ.git
synced 2025-01-25 22:10:00 +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.
370 lines
10 KiB
JavaScript
370 lines
10 KiB
JavaScript
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
|
// Distributed under an MIT license: http://codemirror.net/LICENSE
|
|
|
|
/***
|
|
|''Name''|tiddlywiki.js|
|
|
|''Description''|Enables TiddlyWikiy syntax highlighting using CodeMirror|
|
|
|''Author''|PMario|
|
|
|''Version''|0.1.7|
|
|
|''Status''|''stable''|
|
|
|''Source''|[[GitHub|https://github.com/pmario/CodeMirror2/blob/tw-syntax/mode/tiddlywiki]]|
|
|
|''Documentation''|http://codemirror.tiddlyspace.com/|
|
|
|''License''|[[MIT License|http://www.opensource.org/licenses/mit-license.php]]|
|
|
|''CoreVersion''|2.5.0|
|
|
|''Requires''|codemirror.js|
|
|
|''Keywords''|syntax highlighting color code mirror codemirror|
|
|
! Info
|
|
CoreVersion parameter is needed for TiddlyWiki only!
|
|
***/
|
|
//{{{
|
|
|
|
(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("tiddlywiki", function () {
|
|
// Tokenizer
|
|
var textwords = {};
|
|
|
|
var keywords = function () {
|
|
function kw(type) {
|
|
return { type: type, style: "macro"};
|
|
}
|
|
return {
|
|
"allTags": kw('allTags'), "closeAll": kw('closeAll'), "list": kw('list'),
|
|
"newJournal": kw('newJournal'), "newTiddler": kw('newTiddler'),
|
|
"permaview": kw('permaview'), "saveChanges": kw('saveChanges'),
|
|
"search": kw('search'), "slider": kw('slider'), "tabs": kw('tabs'),
|
|
"tag": kw('tag'), "tagging": kw('tagging'), "tags": kw('tags'),
|
|
"tiddler": kw('tiddler'), "timeline": kw('timeline'),
|
|
"today": kw('today'), "version": kw('version'), "option": kw('option'),
|
|
|
|
"with": kw('with'),
|
|
"filter": kw('filter')
|
|
};
|
|
}();
|
|
|
|
var isSpaceName = /[\w_\-]/i,
|
|
reHR = /^\-\-\-\-+$/, // <hr>
|
|
reWikiCommentStart = /^\/\*\*\*$/, // /***
|
|
reWikiCommentStop = /^\*\*\*\/$/, // ***/
|
|
reBlockQuote = /^<<<$/,
|
|
|
|
reJsCodeStart = /^\/\/\{\{\{$/, // //{{{ js block start
|
|
reJsCodeStop = /^\/\/\}\}\}$/, // //}}} js stop
|
|
reXmlCodeStart = /^<!--\{\{\{-->$/, // xml block start
|
|
reXmlCodeStop = /^<!--\}\}\}-->$/, // xml stop
|
|
|
|
reCodeBlockStart = /^\{\{\{$/, // {{{ TW text div block start
|
|
reCodeBlockStop = /^\}\}\}$/, // }}} TW text stop
|
|
|
|
reUntilCodeStop = /.*?\}\}\}/;
|
|
|
|
function chain(stream, state, f) {
|
|
state.tokenize = f;
|
|
return f(stream, state);
|
|
}
|
|
|
|
// Used as scratch variables to communicate multiple values without
|
|
// consing up tons of objects.
|
|
var type, content;
|
|
|
|
function ret(tp, style, cont) {
|
|
type = tp;
|
|
content = cont;
|
|
return style;
|
|
}
|
|
|
|
function jsTokenBase(stream, state) {
|
|
var sol = stream.sol(), ch;
|
|
|
|
state.block = false; // indicates the start of a code block.
|
|
|
|
ch = stream.peek(); // don't eat, to make matching simpler
|
|
|
|
// check start of blocks
|
|
if (sol && /[<\/\*{}\-]/.test(ch)) {
|
|
if (stream.match(reCodeBlockStart)) {
|
|
state.block = true;
|
|
return chain(stream, state, twTokenCode);
|
|
}
|
|
if (stream.match(reBlockQuote)) {
|
|
return ret('quote', 'quote');
|
|
}
|
|
if (stream.match(reWikiCommentStart) || stream.match(reWikiCommentStop)) {
|
|
return ret('code', 'comment');
|
|
}
|
|
if (stream.match(reJsCodeStart) || stream.match(reJsCodeStop) || stream.match(reXmlCodeStart) || stream.match(reXmlCodeStop)) {
|
|
return ret('code', 'comment');
|
|
}
|
|
if (stream.match(reHR)) {
|
|
return ret('hr', 'hr');
|
|
}
|
|
} // sol
|
|
ch = stream.next();
|
|
|
|
if (sol && /[\/\*!#;:>|]/.test(ch)) {
|
|
if (ch == "!") { // tw header
|
|
stream.skipToEnd();
|
|
return ret("header", "header");
|
|
}
|
|
if (ch == "*") { // tw list
|
|
stream.eatWhile('*');
|
|
return ret("list", "comment");
|
|
}
|
|
if (ch == "#") { // tw numbered list
|
|
stream.eatWhile('#');
|
|
return ret("list", "comment");
|
|
}
|
|
if (ch == ";") { // definition list, term
|
|
stream.eatWhile(';');
|
|
return ret("list", "comment");
|
|
}
|
|
if (ch == ":") { // definition list, description
|
|
stream.eatWhile(':');
|
|
return ret("list", "comment");
|
|
}
|
|
if (ch == ">") { // single line quote
|
|
stream.eatWhile(">");
|
|
return ret("quote", "quote");
|
|
}
|
|
if (ch == '|') {
|
|
return ret('table', 'header');
|
|
}
|
|
}
|
|
|
|
if (ch == '{' && stream.match(/\{\{/)) {
|
|
return chain(stream, state, twTokenCode);
|
|
}
|
|
|
|
// rudimentary html:// file:// link matching. TW knows much more ...
|
|
if (/[hf]/i.test(ch)) {
|
|
if (/[ti]/i.test(stream.peek()) && stream.match(/\b(ttps?|tp|ile):\/\/[\-A-Z0-9+&@#\/%?=~_|$!:,.;]*[A-Z0-9+&@#\/%=~_|$]/i)) {
|
|
return ret("link", "link");
|
|
}
|
|
}
|
|
// just a little string indicator, don't want to have the whole string covered
|
|
if (ch == '"') {
|
|
return ret('string', 'string');
|
|
}
|
|
if (ch == '~') { // _no_ CamelCase indicator should be bold
|
|
return ret('text', 'brace');
|
|
}
|
|
if (/[\[\]]/.test(ch)) { // check for [[..]]
|
|
if (stream.peek() == ch) {
|
|
stream.next();
|
|
return ret('brace', 'brace');
|
|
}
|
|
}
|
|
if (ch == "@") { // check for space link. TODO fix @@...@@ highlighting
|
|
stream.eatWhile(isSpaceName);
|
|
return ret("link", "link");
|
|
}
|
|
if (/\d/.test(ch)) { // numbers
|
|
stream.eatWhile(/\d/);
|
|
return ret("number", "number");
|
|
}
|
|
if (ch == "/") { // tw invisible comment
|
|
if (stream.eat("%")) {
|
|
return chain(stream, state, twTokenComment);
|
|
}
|
|
else if (stream.eat("/")) { //
|
|
return chain(stream, state, twTokenEm);
|
|
}
|
|
}
|
|
if (ch == "_") { // tw underline
|
|
if (stream.eat("_")) {
|
|
return chain(stream, state, twTokenUnderline);
|
|
}
|
|
}
|
|
// strikethrough and mdash handling
|
|
if (ch == "-") {
|
|
if (stream.eat("-")) {
|
|
// if strikethrough looks ugly, change CSS.
|
|
if (stream.peek() != ' ')
|
|
return chain(stream, state, twTokenStrike);
|
|
// mdash
|
|
if (stream.peek() == ' ')
|
|
return ret('text', 'brace');
|
|
}
|
|
}
|
|
if (ch == "'") { // tw bold
|
|
if (stream.eat("'")) {
|
|
return chain(stream, state, twTokenStrong);
|
|
}
|
|
}
|
|
if (ch == "<") { // tw macro
|
|
if (stream.eat("<")) {
|
|
return chain(stream, state, twTokenMacro);
|
|
}
|
|
}
|
|
else {
|
|
return ret(ch);
|
|
}
|
|
|
|
// core macro handling
|
|
stream.eatWhile(/[\w\$_]/);
|
|
var word = stream.current(),
|
|
known = textwords.propertyIsEnumerable(word) && textwords[word];
|
|
|
|
return known ? ret(known.type, known.style, word) : ret("text", null, word);
|
|
|
|
} // jsTokenBase()
|
|
|
|
// tw invisible comment
|
|
function twTokenComment(stream, state) {
|
|
var maybeEnd = false,
|
|
ch;
|
|
while (ch = stream.next()) {
|
|
if (ch == "/" && maybeEnd) {
|
|
state.tokenize = jsTokenBase;
|
|
break;
|
|
}
|
|
maybeEnd = (ch == "%");
|
|
}
|
|
return ret("comment", "comment");
|
|
}
|
|
|
|
// tw strong / bold
|
|
function twTokenStrong(stream, state) {
|
|
var maybeEnd = false,
|
|
ch;
|
|
while (ch = stream.next()) {
|
|
if (ch == "'" && maybeEnd) {
|
|
state.tokenize = jsTokenBase;
|
|
break;
|
|
}
|
|
maybeEnd = (ch == "'");
|
|
}
|
|
return ret("text", "strong");
|
|
}
|
|
|
|
// tw code
|
|
function twTokenCode(stream, state) {
|
|
var ch, sb = state.block;
|
|
|
|
if (sb && stream.current()) {
|
|
return ret("code", "comment");
|
|
}
|
|
|
|
if (!sb && stream.match(reUntilCodeStop)) {
|
|
state.tokenize = jsTokenBase;
|
|
return ret("code", "comment");
|
|
}
|
|
|
|
if (sb && stream.sol() && stream.match(reCodeBlockStop)) {
|
|
state.tokenize = jsTokenBase;
|
|
return ret("code", "comment");
|
|
}
|
|
|
|
ch = stream.next();
|
|
return (sb) ? ret("code", "comment") : ret("code", "comment");
|
|
}
|
|
|
|
// tw em / italic
|
|
function twTokenEm(stream, state) {
|
|
var maybeEnd = false,
|
|
ch;
|
|
while (ch = stream.next()) {
|
|
if (ch == "/" && maybeEnd) {
|
|
state.tokenize = jsTokenBase;
|
|
break;
|
|
}
|
|
maybeEnd = (ch == "/");
|
|
}
|
|
return ret("text", "em");
|
|
}
|
|
|
|
// tw underlined text
|
|
function twTokenUnderline(stream, state) {
|
|
var maybeEnd = false,
|
|
ch;
|
|
while (ch = stream.next()) {
|
|
if (ch == "_" && maybeEnd) {
|
|
state.tokenize = jsTokenBase;
|
|
break;
|
|
}
|
|
maybeEnd = (ch == "_");
|
|
}
|
|
return ret("text", "underlined");
|
|
}
|
|
|
|
// tw strike through text looks ugly
|
|
// change CSS if needed
|
|
function twTokenStrike(stream, state) {
|
|
var maybeEnd = false, ch;
|
|
|
|
while (ch = stream.next()) {
|
|
if (ch == "-" && maybeEnd) {
|
|
state.tokenize = jsTokenBase;
|
|
break;
|
|
}
|
|
maybeEnd = (ch == "-");
|
|
}
|
|
return ret("text", "strikethrough");
|
|
}
|
|
|
|
// macro
|
|
function twTokenMacro(stream, state) {
|
|
var ch, word, known;
|
|
|
|
if (stream.current() == '<<') {
|
|
return ret('brace', 'macro');
|
|
}
|
|
|
|
ch = stream.next();
|
|
if (!ch) {
|
|
state.tokenize = jsTokenBase;
|
|
return ret(ch);
|
|
}
|
|
if (ch == ">") {
|
|
if (stream.peek() == '>') {
|
|
stream.next();
|
|
state.tokenize = jsTokenBase;
|
|
return ret("brace", "macro");
|
|
}
|
|
}
|
|
|
|
stream.eatWhile(/[\w\$_]/);
|
|
word = stream.current();
|
|
known = keywords.propertyIsEnumerable(word) && keywords[word];
|
|
|
|
if (known) {
|
|
return ret(known.type, known.style, word);
|
|
}
|
|
else {
|
|
return ret("macro", null, word);
|
|
}
|
|
}
|
|
|
|
// Interface
|
|
return {
|
|
startState: function () {
|
|
return {
|
|
tokenize: jsTokenBase,
|
|
indented: 0,
|
|
level: 0
|
|
};
|
|
},
|
|
|
|
token: function (stream, state) {
|
|
if (stream.eatSpace()) return null;
|
|
var style = state.tokenize(stream, state);
|
|
return style;
|
|
},
|
|
|
|
electricChars: ""
|
|
};
|
|
});
|
|
|
|
CodeMirror.defineMIME("text/x-tiddlywiki", "tiddlywiki");
|
|
});
|
|
|
|
//}}}
|