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");
 | |
| });
 | |
| 
 | |
| //}}}
 |