velocity.js 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. CodeMirror.defineMode("velocity", function() {
  2. function parseWords(str) {
  3. var obj = {}, words = str.split(" ");
  4. for (var i = 0; i < words.length; ++i) obj[words[i]] = true;
  5. return obj;
  6. }
  7. var keywords = parseWords("#end #else #break #stop #[[ #]] " +
  8. "#{end} #{else} #{break} #{stop}");
  9. var functions = parseWords("#if #elseif #foreach #set #include #parse #macro #define #evaluate " +
  10. "#{if} #{elseif} #{foreach} #{set} #{include} #{parse} #{macro} #{define} #{evaluate}");
  11. var specials = parseWords("$foreach.count $foreach.hasNext $foreach.first $foreach.last $foreach.topmost $foreach.parent.count $foreach.parent.hasNext $foreach.parent.first $foreach.parent.last $foreach.parent $velocityCount $!bodyContent $bodyContent");
  12. var isOperatorChar = /[+\-*&%=<>!?:\/|]/;
  13. function chain(stream, state, f) {
  14. state.tokenize = f;
  15. return f(stream, state);
  16. }
  17. function tokenBase(stream, state) {
  18. var beforeParams = state.beforeParams;
  19. state.beforeParams = false;
  20. var ch = stream.next();
  21. // start of unparsed string?
  22. if ((ch == "'") && state.inParams) {
  23. state.lastTokenWasBuiltin = false;
  24. return chain(stream, state, tokenString(ch));
  25. }
  26. // start of parsed string?
  27. else if ((ch == '"')) {
  28. state.lastTokenWasBuiltin = false;
  29. if (state.inString) {
  30. state.inString = false;
  31. return "string";
  32. }
  33. else if (state.inParams)
  34. return chain(stream, state, tokenString(ch));
  35. }
  36. // is it one of the special signs []{}().,;? Seperator?
  37. else if (/[\[\]{}\(\),;\.]/.test(ch)) {
  38. if (ch == "(" && beforeParams)
  39. state.inParams = true;
  40. else if (ch == ")") {
  41. state.inParams = false;
  42. state.lastTokenWasBuiltin = true;
  43. }
  44. return null;
  45. }
  46. // start of a number value?
  47. else if (/\d/.test(ch)) {
  48. state.lastTokenWasBuiltin = false;
  49. stream.eatWhile(/[\w\.]/);
  50. return "number";
  51. }
  52. // multi line comment?
  53. else if (ch == "#" && stream.eat("*")) {
  54. state.lastTokenWasBuiltin = false;
  55. return chain(stream, state, tokenComment);
  56. }
  57. // unparsed content?
  58. else if (ch == "#" && stream.match(/ *\[ *\[/)) {
  59. state.lastTokenWasBuiltin = false;
  60. return chain(stream, state, tokenUnparsed);
  61. }
  62. // single line comment?
  63. else if (ch == "#" && stream.eat("#")) {
  64. state.lastTokenWasBuiltin = false;
  65. stream.skipToEnd();
  66. return "comment";
  67. }
  68. // variable?
  69. else if (ch == "$") {
  70. stream.eatWhile(/[\w\d\$_\.{}]/);
  71. // is it one of the specials?
  72. if (specials && specials.propertyIsEnumerable(stream.current())) {
  73. return "keyword";
  74. }
  75. else {
  76. state.lastTokenWasBuiltin = true;
  77. state.beforeParams = true;
  78. return "builtin";
  79. }
  80. }
  81. // is it a operator?
  82. else if (isOperatorChar.test(ch)) {
  83. state.lastTokenWasBuiltin = false;
  84. stream.eatWhile(isOperatorChar);
  85. return "operator";
  86. }
  87. else {
  88. // get the whole word
  89. stream.eatWhile(/[\w\$_{}@]/);
  90. var word = stream.current();
  91. // is it one of the listed keywords?
  92. if (keywords && keywords.propertyIsEnumerable(word))
  93. return "keyword";
  94. // is it one of the listed functions?
  95. if (functions && functions.propertyIsEnumerable(word) ||
  96. (stream.current().match(/^#@?[a-z0-9_]+ *$/i) && stream.peek()=="(") &&
  97. !(functions && functions.propertyIsEnumerable(word.toLowerCase()))) {
  98. state.beforeParams = true;
  99. state.lastTokenWasBuiltin = false;
  100. return "keyword";
  101. }
  102. if (state.inString) {
  103. state.lastTokenWasBuiltin = false;
  104. return "string";
  105. }
  106. if (stream.pos > word.length && stream.string.charAt(stream.pos-word.length-1)=="." && state.lastTokenWasBuiltin)
  107. return "builtin";
  108. // default: just a "word"
  109. state.lastTokenWasBuiltin = false;
  110. return null;
  111. }
  112. }
  113. function tokenString(quote) {
  114. return function(stream, state) {
  115. var escaped = false, next, end = false;
  116. while ((next = stream.next()) != null) {
  117. if ((next == quote) && !escaped) {
  118. end = true;
  119. break;
  120. }
  121. if (quote=='"' && stream.peek() == '$' && !escaped) {
  122. state.inString = true;
  123. end = true;
  124. break;
  125. }
  126. escaped = !escaped && next == "\\";
  127. }
  128. if (end) state.tokenize = tokenBase;
  129. return "string";
  130. };
  131. }
  132. function tokenComment(stream, state) {
  133. var maybeEnd = false, ch;
  134. while (ch = stream.next()) {
  135. if (ch == "#" && maybeEnd) {
  136. state.tokenize = tokenBase;
  137. break;
  138. }
  139. maybeEnd = (ch == "*");
  140. }
  141. return "comment";
  142. }
  143. function tokenUnparsed(stream, state) {
  144. var maybeEnd = 0, ch;
  145. while (ch = stream.next()) {
  146. if (ch == "#" && maybeEnd == 2) {
  147. state.tokenize = tokenBase;
  148. break;
  149. }
  150. if (ch == "]")
  151. maybeEnd++;
  152. else if (ch != " ")
  153. maybeEnd = 0;
  154. }
  155. return "meta";
  156. }
  157. // Interface
  158. return {
  159. startState: function() {
  160. return {
  161. tokenize: tokenBase,
  162. beforeParams: false,
  163. inParams: false,
  164. inString: false,
  165. lastTokenWasBuiltin: false
  166. };
  167. },
  168. token: function(stream, state) {
  169. if (stream.eatSpace()) return null;
  170. return state.tokenize(stream, state);
  171. },
  172. blockCommentStart: "#*",
  173. blockCommentEnd: "*#",
  174. lineComment: "##",
  175. fold: "velocity"
  176. };
  177. });
  178. CodeMirror.defineMIME("text/velocity", "velocity");