mirror of
				https://github.com/TriliumNext/Notes.git
				synced 2025-10-27 01:51:34 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			357 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			357 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| // CodeMirror, copyright (c) by Marijn Haverbeke and others
 | |
| // Distributed under an MIT license: https://codemirror.net/5/LICENSE
 | |
| 
 | |
| (function(mod) {
 | |
|   if (typeof exports == "object" && typeof module == "object") // CommonJS
 | |
|     mod(require("../../lib/codemirror"), require("../htmlmixed/htmlmixed"),
 | |
|         require("../../addon/mode/overlay"));
 | |
|   else if (typeof define == "function" && define.amd) // AMD
 | |
|     define(["../../lib/codemirror", "../htmlmixed/htmlmixed",
 | |
|             "../../addon/mode/overlay"], mod);
 | |
|   else // Plain browser env
 | |
|     mod(CodeMirror);
 | |
| })(function(CodeMirror) {
 | |
|   "use strict";
 | |
| 
 | |
|   CodeMirror.defineMode("django:inner", function() {
 | |
|     var keywords = ["block", "endblock", "for", "endfor", "true", "false", "filter", "endfilter",
 | |
|                     "loop", "none", "self", "super", "if", "elif", "endif", "as", "else", "import",
 | |
|                     "with", "endwith", "without", "context", "ifequal", "endifequal", "ifnotequal",
 | |
|                     "endifnotequal", "extends", "include", "load", "comment", "endcomment",
 | |
|                     "empty", "url", "static", "trans", "blocktrans", "endblocktrans", "now",
 | |
|                     "regroup", "lorem", "ifchanged", "endifchanged", "firstof", "debug", "cycle",
 | |
|                     "csrf_token", "autoescape", "endautoescape", "spaceless", "endspaceless",
 | |
|                     "ssi", "templatetag", "verbatim", "endverbatim", "widthratio"],
 | |
|         filters = ["add", "addslashes", "capfirst", "center", "cut", "date",
 | |
|                    "default", "default_if_none", "dictsort",
 | |
|                    "dictsortreversed", "divisibleby", "escape", "escapejs",
 | |
|                    "filesizeformat", "first", "floatformat", "force_escape",
 | |
|                    "get_digit", "iriencode", "join", "last", "length",
 | |
|                    "length_is", "linebreaks", "linebreaksbr", "linenumbers",
 | |
|                    "ljust", "lower", "make_list", "phone2numeric", "pluralize",
 | |
|                    "pprint", "random", "removetags", "rjust", "safe",
 | |
|                    "safeseq", "slice", "slugify", "stringformat", "striptags",
 | |
|                    "time", "timesince", "timeuntil", "title", "truncatechars",
 | |
|                    "truncatechars_html", "truncatewords", "truncatewords_html",
 | |
|                    "unordered_list", "upper", "urlencode", "urlize",
 | |
|                    "urlizetrunc", "wordcount", "wordwrap", "yesno"],
 | |
|         operators = ["==", "!=", "<", ">", "<=", ">="],
 | |
|         wordOperators = ["in", "not", "or", "and"];
 | |
| 
 | |
|     keywords = new RegExp("^\\b(" + keywords.join("|") + ")\\b");
 | |
|     filters = new RegExp("^\\b(" + filters.join("|") + ")\\b");
 | |
|     operators = new RegExp("^\\b(" + operators.join("|") + ")\\b");
 | |
|     wordOperators = new RegExp("^\\b(" + wordOperators.join("|") + ")\\b");
 | |
| 
 | |
|     // We have to return "null" instead of null, in order to avoid string
 | |
|     // styling as the default, when using Django templates inside HTML
 | |
|     // element attributes
 | |
|     function tokenBase (stream, state) {
 | |
|       // Attempt to identify a variable, template or comment tag respectively
 | |
|       if (stream.match("{{")) {
 | |
|         state.tokenize = inVariable;
 | |
|         return "tag";
 | |
|       } else if (stream.match("{%")) {
 | |
|         state.tokenize = inTag;
 | |
|         return "tag";
 | |
|       } else if (stream.match("{#")) {
 | |
|         state.tokenize = inComment;
 | |
|         return "comment";
 | |
|       }
 | |
| 
 | |
|       // Ignore completely any stream series that do not match the
 | |
|       // Django template opening tags.
 | |
|       while (stream.next() != null && !stream.match(/\{[{%#]/, false)) {}
 | |
|       return null;
 | |
|     }
 | |
| 
 | |
|     // A string can be included in either single or double quotes (this is
 | |
|     // the delimiter). Mark everything as a string until the start delimiter
 | |
|     // occurs again.
 | |
|     function inString (delimiter, previousTokenizer) {
 | |
|       return function (stream, state) {
 | |
|         if (!state.escapeNext && stream.eat(delimiter)) {
 | |
|           state.tokenize = previousTokenizer;
 | |
|         } else {
 | |
|           if (state.escapeNext) {
 | |
|             state.escapeNext = false;
 | |
|           }
 | |
| 
 | |
|           var ch = stream.next();
 | |
| 
 | |
|           // Take into account the backslash for escaping characters, such as
 | |
|           // the string delimiter.
 | |
|           if (ch == "\\") {
 | |
|             state.escapeNext = true;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         return "string";
 | |
|       };
 | |
|     }
 | |
| 
 | |
|     // Apply Django template variable syntax highlighting
 | |
|     function inVariable (stream, state) {
 | |
|       // Attempt to match a dot that precedes a property
 | |
|       if (state.waitDot) {
 | |
|         state.waitDot = false;
 | |
| 
 | |
|         if (stream.peek() != ".") {
 | |
|           return "null";
 | |
|         }
 | |
| 
 | |
|         // Dot followed by a non-word character should be considered an error.
 | |
|         if (stream.match(/\.\W+/)) {
 | |
|           return "error";
 | |
|         } else if (stream.eat(".")) {
 | |
|           state.waitProperty = true;
 | |
|           return "null";
 | |
|         } else {
 | |
|           throw Error ("Unexpected error while waiting for property.");
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Attempt to match a pipe that precedes a filter
 | |
|       if (state.waitPipe) {
 | |
|         state.waitPipe = false;
 | |
| 
 | |
|         if (stream.peek() != "|") {
 | |
|           return "null";
 | |
|         }
 | |
| 
 | |
|         // Pipe followed by a non-word character should be considered an error.
 | |
|         if (stream.match(/\.\W+/)) {
 | |
|           return "error";
 | |
|         } else if (stream.eat("|")) {
 | |
|           state.waitFilter = true;
 | |
|           return "null";
 | |
|         } else {
 | |
|           throw Error ("Unexpected error while waiting for filter.");
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Highlight properties
 | |
|       if (state.waitProperty) {
 | |
|         state.waitProperty = false;
 | |
|         if (stream.match(/\b(\w+)\b/)) {
 | |
|           state.waitDot = true;  // A property can be followed by another property
 | |
|           state.waitPipe = true;  // A property can be followed by a filter
 | |
|           return "property";
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Highlight filters
 | |
|       if (state.waitFilter) {
 | |
|           state.waitFilter = false;
 | |
|         if (stream.match(filters)) {
 | |
|           return "variable-2";
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Ignore all white spaces
 | |
|       if (stream.eatSpace()) {
 | |
|         state.waitProperty = false;
 | |
|         return "null";
 | |
|       }
 | |
| 
 | |
|       // Identify numbers
 | |
|       if (stream.match(/\b\d+(\.\d+)?\b/)) {
 | |
|         return "number";
 | |
|       }
 | |
| 
 | |
|       // Identify strings
 | |
|       if (stream.match("'")) {
 | |
|         state.tokenize = inString("'", state.tokenize);
 | |
|         return "string";
 | |
|       } else if (stream.match('"')) {
 | |
|         state.tokenize = inString('"', state.tokenize);
 | |
|         return "string";
 | |
|       }
 | |
| 
 | |
|       // Attempt to find the variable
 | |
|       if (stream.match(/\b(\w+)\b/) && !state.foundVariable) {
 | |
|         state.waitDot = true;
 | |
|         state.waitPipe = true;  // A property can be followed by a filter
 | |
|         return "variable";
 | |
|       }
 | |
| 
 | |
|       // If found closing tag reset
 | |
|       if (stream.match("}}")) {
 | |
|         state.waitProperty = null;
 | |
|         state.waitFilter = null;
 | |
|         state.waitDot = null;
 | |
|         state.waitPipe = null;
 | |
|         state.tokenize = tokenBase;
 | |
|         return "tag";
 | |
|       }
 | |
| 
 | |
|       // If nothing was found, advance to the next character
 | |
|       stream.next();
 | |
|       return "null";
 | |
|     }
 | |
| 
 | |
|     function inTag (stream, state) {
 | |
|       // Attempt to match a dot that precedes a property
 | |
|       if (state.waitDot) {
 | |
|         state.waitDot = false;
 | |
| 
 | |
|         if (stream.peek() != ".") {
 | |
|           return "null";
 | |
|         }
 | |
| 
 | |
|         // Dot followed by a non-word character should be considered an error.
 | |
|         if (stream.match(/\.\W+/)) {
 | |
|           return "error";
 | |
|         } else if (stream.eat(".")) {
 | |
|           state.waitProperty = true;
 | |
|           return "null";
 | |
|         } else {
 | |
|           throw Error ("Unexpected error while waiting for property.");
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Attempt to match a pipe that precedes a filter
 | |
|       if (state.waitPipe) {
 | |
|         state.waitPipe = false;
 | |
| 
 | |
|         if (stream.peek() != "|") {
 | |
|           return "null";
 | |
|         }
 | |
| 
 | |
|         // Pipe followed by a non-word character should be considered an error.
 | |
|         if (stream.match(/\.\W+/)) {
 | |
|           return "error";
 | |
|         } else if (stream.eat("|")) {
 | |
|           state.waitFilter = true;
 | |
|           return "null";
 | |
|         } else {
 | |
|           throw Error ("Unexpected error while waiting for filter.");
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Highlight properties
 | |
|       if (state.waitProperty) {
 | |
|         state.waitProperty = false;
 | |
|         if (stream.match(/\b(\w+)\b/)) {
 | |
|           state.waitDot = true;  // A property can be followed by another property
 | |
|           state.waitPipe = true;  // A property can be followed by a filter
 | |
|           return "property";
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Highlight filters
 | |
|       if (state.waitFilter) {
 | |
|           state.waitFilter = false;
 | |
|         if (stream.match(filters)) {
 | |
|           return "variable-2";
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // Ignore all white spaces
 | |
|       if (stream.eatSpace()) {
 | |
|         state.waitProperty = false;
 | |
|         return "null";
 | |
|       }
 | |
| 
 | |
|       // Identify numbers
 | |
|       if (stream.match(/\b\d+(\.\d+)?\b/)) {
 | |
|         return "number";
 | |
|       }
 | |
| 
 | |
|       // Identify strings
 | |
|       if (stream.match("'")) {
 | |
|         state.tokenize = inString("'", state.tokenize);
 | |
|         return "string";
 | |
|       } else if (stream.match('"')) {
 | |
|         state.tokenize = inString('"', state.tokenize);
 | |
|         return "string";
 | |
|       }
 | |
| 
 | |
|       // Attempt to match an operator
 | |
|       if (stream.match(operators)) {
 | |
|         return "operator";
 | |
|       }
 | |
| 
 | |
|       // Attempt to match a word operator
 | |
|       if (stream.match(wordOperators)) {
 | |
|         return "keyword";
 | |
|       }
 | |
| 
 | |
|       // Attempt to match a keyword
 | |
|       var keywordMatch = stream.match(keywords);
 | |
|       if (keywordMatch) {
 | |
|         if (keywordMatch[0] == "comment") {
 | |
|           state.blockCommentTag = true;
 | |
|         }
 | |
|         return "keyword";
 | |
|       }
 | |
| 
 | |
|       // Attempt to match a variable
 | |
|       if (stream.match(/\b(\w+)\b/)) {
 | |
|         state.waitDot = true;
 | |
|         state.waitPipe = true;  // A property can be followed by a filter
 | |
|         return "variable";
 | |
|       }
 | |
| 
 | |
|       // If found closing tag reset
 | |
|       if (stream.match("%}")) {
 | |
|         state.waitProperty = null;
 | |
|         state.waitFilter = null;
 | |
|         state.waitDot = null;
 | |
|         state.waitPipe = null;
 | |
|         // If the tag that closes is a block comment tag, we want to mark the
 | |
|         // following code as comment, until the tag closes.
 | |
|         if (state.blockCommentTag) {
 | |
|           state.blockCommentTag = false;  // Release the "lock"
 | |
|           state.tokenize = inBlockComment;
 | |
|         } else {
 | |
|           state.tokenize = tokenBase;
 | |
|         }
 | |
|         return "tag";
 | |
|       }
 | |
| 
 | |
|       // If nothing was found, advance to the next character
 | |
|       stream.next();
 | |
|       return "null";
 | |
|     }
 | |
| 
 | |
|     // Mark everything as comment inside the tag and the tag itself.
 | |
|     function inComment (stream, state) {
 | |
|       if (stream.match(/^.*?#\}/)) state.tokenize = tokenBase
 | |
|       else stream.skipToEnd()
 | |
|       return "comment";
 | |
|     }
 | |
| 
 | |
|     // Mark everything as a comment until the `blockcomment` tag closes.
 | |
|     function inBlockComment (stream, state) {
 | |
|       if (stream.match(/\{%\s*endcomment\s*%\}/, false)) {
 | |
|         state.tokenize = inTag;
 | |
|         stream.match("{%");
 | |
|         return "tag";
 | |
|       } else {
 | |
|         stream.next();
 | |
|         return "comment";
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return {
 | |
|       startState: function () {
 | |
|         return {tokenize: tokenBase};
 | |
|       },
 | |
|       token: function (stream, state) {
 | |
|         return state.tokenize(stream, state);
 | |
|       },
 | |
|       blockCommentStart: "{% comment %}",
 | |
|       blockCommentEnd: "{% endcomment %}"
 | |
|     };
 | |
|   });
 | |
| 
 | |
|   CodeMirror.defineMode("django", function(config) {
 | |
|     var htmlBase = CodeMirror.getMode(config, "text/html");
 | |
|     var djangoInner = CodeMirror.getMode(config, "django:inner");
 | |
|     return CodeMirror.overlayMode(htmlBase, djangoInner);
 | |
|   });
 | |
| 
 | |
|   CodeMirror.defineMIME("text/x-django", "django");
 | |
| });
 | 
