codemirror-rails 2.2.1 → 2.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (81) hide show
  1. data/README.md +8 -0
  2. data/codemirror-rails.gemspec +1 -1
  3. data/lib/codemirror/rails/version.rb +2 -2
  4. data/vendor/assets/javascripts/codemirror.js +842 -422
  5. data/vendor/assets/javascripts/codemirror/keymaps/emacs.js +2 -2
  6. data/vendor/assets/javascripts/codemirror/keymaps/vim.js +722 -32
  7. data/vendor/assets/javascripts/codemirror/modes/clike.js +31 -9
  8. data/vendor/assets/javascripts/codemirror/modes/clojure.js +14 -14
  9. data/vendor/assets/javascripts/codemirror/modes/coffeescript.js +59 -37
  10. data/vendor/assets/javascripts/codemirror/modes/css.js +1 -1
  11. data/vendor/assets/javascripts/codemirror/modes/diff.js +24 -5
  12. data/vendor/assets/javascripts/codemirror/modes/ecl.js +203 -0
  13. data/vendor/assets/javascripts/codemirror/modes/erlang.js +251 -0
  14. data/vendor/assets/javascripts/codemirror/modes/gfm.js +40 -4
  15. data/vendor/assets/javascripts/codemirror/modes/go.js +170 -0
  16. data/vendor/assets/javascripts/codemirror/modes/htmlembedded.js +1 -1
  17. data/vendor/assets/javascripts/codemirror/modes/htmlmixed.js +4 -2
  18. data/vendor/assets/javascripts/codemirror/modes/javascript.js +10 -9
  19. data/vendor/assets/javascripts/codemirror/modes/less.js +232 -0
  20. data/vendor/assets/javascripts/codemirror/modes/markdown.js +52 -49
  21. data/vendor/assets/javascripts/codemirror/modes/mysql.js +188 -0
  22. data/vendor/assets/javascripts/codemirror/modes/pascal.js +2 -46
  23. data/vendor/assets/javascripts/codemirror/modes/perl.js +1 -1
  24. data/vendor/assets/javascripts/codemirror/modes/php.js +55 -25
  25. data/vendor/assets/javascripts/codemirror/modes/pig.js +172 -0
  26. data/vendor/assets/javascripts/codemirror/modes/properties.js +63 -0
  27. data/vendor/assets/javascripts/codemirror/modes/python.js +37 -32
  28. data/vendor/assets/javascripts/codemirror/modes/rpm-spec.js +1 -1
  29. data/vendor/assets/javascripts/codemirror/modes/rst.js +1 -1
  30. data/vendor/assets/javascripts/codemirror/modes/ruby.js +14 -14
  31. data/vendor/assets/javascripts/codemirror/modes/rust.js +36 -15
  32. data/vendor/assets/javascripts/codemirror/modes/scheme.js +74 -46
  33. data/vendor/assets/javascripts/codemirror/modes/shell.js +103 -0
  34. data/vendor/assets/javascripts/codemirror/modes/smalltalk.js +16 -16
  35. data/vendor/assets/javascripts/codemirror/modes/smarty.js +148 -0
  36. data/vendor/assets/javascripts/codemirror/modes/stex.js +21 -6
  37. data/vendor/assets/javascripts/codemirror/modes/tiddlywiki.js +55 -45
  38. data/vendor/assets/javascripts/codemirror/modes/tiki.js +316 -0
  39. data/vendor/assets/javascripts/codemirror/modes/vbscript.js +26 -0
  40. data/vendor/assets/javascripts/codemirror/modes/verilog.js +194 -0
  41. data/vendor/assets/javascripts/codemirror/modes/xml.js +89 -16
  42. data/vendor/assets/javascripts/codemirror/modes/xmlpure.js +5 -0
  43. data/vendor/assets/javascripts/codemirror/modes/xquery.js +448 -0
  44. data/vendor/assets/javascripts/codemirror/utils/closetag.js +146 -0
  45. data/vendor/assets/javascripts/codemirror/utils/dialog.js +63 -0
  46. data/vendor/assets/javascripts/codemirror/utils/foldcode.js +196 -0
  47. data/vendor/assets/javascripts/codemirror/utils/formatting.js +297 -0
  48. data/vendor/assets/javascripts/codemirror/utils/javascript-hint.js +134 -0
  49. data/vendor/assets/javascripts/codemirror/utils/loadmode.js +51 -0
  50. data/vendor/assets/javascripts/codemirror/utils/match-highlighter.js +44 -0
  51. data/vendor/assets/javascripts/codemirror/utils/multiplex.js +72 -0
  52. data/vendor/assets/javascripts/codemirror/{overlay.js → utils/overlay.js} +3 -2
  53. data/vendor/assets/javascripts/codemirror/utils/pig-hint.js +123 -0
  54. data/vendor/assets/javascripts/codemirror/utils/runmode.js +49 -0
  55. data/vendor/assets/javascripts/codemirror/utils/search.js +118 -0
  56. data/vendor/assets/javascripts/codemirror/utils/searchcursor.js +117 -0
  57. data/vendor/assets/javascripts/codemirror/utils/simple-hint.js +72 -0
  58. data/vendor/assets/stylesheets/codemirror.css +69 -5
  59. data/vendor/assets/stylesheets/codemirror/modes/tiddlywiki.css +14 -21
  60. data/vendor/assets/stylesheets/codemirror/modes/tiki.css +26 -0
  61. data/vendor/assets/stylesheets/codemirror/themes/ambiance.css +81 -0
  62. data/vendor/assets/stylesheets/codemirror/themes/blackboard.css +25 -0
  63. data/vendor/assets/stylesheets/codemirror/themes/cobalt.css +1 -1
  64. data/vendor/assets/stylesheets/codemirror/themes/eclipse.css +1 -1
  65. data/vendor/assets/stylesheets/codemirror/themes/elegant.css +2 -2
  66. data/vendor/assets/stylesheets/codemirror/themes/erlang-dark.css +21 -0
  67. data/vendor/assets/stylesheets/codemirror/themes/lesser-dark.css +44 -0
  68. data/vendor/assets/stylesheets/codemirror/themes/monokai.css +1 -1
  69. data/vendor/assets/stylesheets/codemirror/themes/neat.css +3 -3
  70. data/vendor/assets/stylesheets/codemirror/themes/night.css +1 -1
  71. data/vendor/assets/stylesheets/codemirror/themes/rubyblue.css +2 -2
  72. data/vendor/assets/stylesheets/codemirror/themes/vibrant-ink.css +27 -0
  73. data/vendor/assets/stylesheets/codemirror/themes/xq-dark.css +46 -0
  74. data/vendor/assets/stylesheets/codemirror/utils/dialog.css +23 -0
  75. data/vendor/assets/stylesheets/codemirror/utils/simple-hint.css +16 -0
  76. metadata +41 -10
  77. data/vendor/assets/javascripts/codemirror/runmode.js +0 -27
  78. data/vendor/assets/stylesheets/codemirror/modes/clike.css +0 -7
  79. data/vendor/assets/stylesheets/codemirror/modes/markdown.css +0 -10
  80. data/vendor/assets/stylesheets/codemirror/modes/rst.css +0 -75
  81. data/vendor/assets/stylesheets/codemirror/themes/default.css +0 -19
@@ -1,11 +1,44 @@
1
1
  CodeMirror.defineMode("xml", function(config, parserConfig) {
2
2
  var indentUnit = config.indentUnit;
3
3
  var Kludges = parserConfig.htmlMode ? {
4
- autoSelfClosers: {"br": true, "img": true, "hr": true, "link": true, "input": true,
5
- "meta": true, "col": true, "frame": true, "base": true, "area": true},
4
+ autoSelfClosers: {'area': true, 'base': true, 'br': true, 'col': true, 'command': true,
5
+ 'embed': true, 'frame': true, 'hr': true, 'img': true, 'input': true,
6
+ 'keygen': true, 'link': true, 'meta': true, 'param': true, 'source': true,
7
+ 'track': true, 'wbr': true},
8
+ implicitlyClosed: {'dd': true, 'li': true, 'optgroup': true, 'option': true, 'p': true,
9
+ 'rp': true, 'rt': true, 'tbody': true, 'td': true, 'tfoot': true,
10
+ 'th': true, 'tr': true},
11
+ contextGrabbers: {
12
+ 'dd': {'dd': true, 'dt': true},
13
+ 'dt': {'dd': true, 'dt': true},
14
+ 'li': {'li': true},
15
+ 'option': {'option': true, 'optgroup': true},
16
+ 'optgroup': {'optgroup': true},
17
+ 'p': {'address': true, 'article': true, 'aside': true, 'blockquote': true, 'dir': true,
18
+ 'div': true, 'dl': true, 'fieldset': true, 'footer': true, 'form': true,
19
+ 'h1': true, 'h2': true, 'h3': true, 'h4': true, 'h5': true, 'h6': true,
20
+ 'header': true, 'hgroup': true, 'hr': true, 'menu': true, 'nav': true, 'ol': true,
21
+ 'p': true, 'pre': true, 'section': true, 'table': true, 'ul': true},
22
+ 'rp': {'rp': true, 'rt': true},
23
+ 'rt': {'rp': true, 'rt': true},
24
+ 'tbody': {'tbody': true, 'tfoot': true},
25
+ 'td': {'td': true, 'th': true},
26
+ 'tfoot': {'tbody': true},
27
+ 'th': {'td': true, 'th': true},
28
+ 'thead': {'tbody': true, 'tfoot': true},
29
+ 'tr': {'tr': true}
30
+ },
6
31
  doNotIndent: {"pre": true},
7
- allowUnquoted: true
8
- } : {autoSelfClosers: {}, doNotIndent: {}, allowUnquoted: false};
32
+ allowUnquoted: true,
33
+ allowMissing: false
34
+ } : {
35
+ autoSelfClosers: {},
36
+ implicitlyClosed: {},
37
+ contextGrabbers: {},
38
+ doNotIndent: {},
39
+ allowUnquoted: false,
40
+ allowMissing: false
41
+ };
9
42
  var alignCDATA = parserConfig.alignCDATA;
10
43
 
11
44
  // Return variables for tokenizers
@@ -47,9 +80,17 @@ CodeMirror.defineMode("xml", function(config, parserConfig) {
47
80
  }
48
81
  }
49
82
  else if (ch == "&") {
50
- stream.eatWhile(/[^;]/);
51
- stream.eat(";");
52
- return "atom";
83
+ var ok;
84
+ if (stream.eat("#")) {
85
+ if (stream.eat("x")) {
86
+ ok = stream.eatWhile(/[a-fA-F\d]/) && stream.eat(";");
87
+ } else {
88
+ ok = stream.eatWhile(/[\d]/) && stream.eat(";");
89
+ }
90
+ } else {
91
+ ok = stream.eatWhile(/[\w\.\-:]/) && stream.eat(";");
92
+ }
93
+ return ok ? "atom" : "error";
53
94
  }
54
95
  else {
55
96
  stream.eatWhile(/[^&<]/);
@@ -153,7 +194,12 @@ CodeMirror.defineMode("xml", function(config, parserConfig) {
153
194
  } else if (type == "closeTag") {
154
195
  var err = false;
155
196
  if (curState.context) {
156
- err = curState.context.tagName != tagName;
197
+ if (curState.context.tagName != tagName) {
198
+ if (Kludges.implicitlyClosed.hasOwnProperty(curState.context.tagName.toLowerCase())) {
199
+ popContext();
200
+ }
201
+ err = !curState.context || curState.context.tagName != tagName;
202
+ }
157
203
  } else {
158
204
  err = true;
159
205
  }
@@ -165,9 +211,15 @@ CodeMirror.defineMode("xml", function(config, parserConfig) {
165
211
  function endtag(startOfLine) {
166
212
  return function(type) {
167
213
  if (type == "selfcloseTag" ||
168
- (type == "endTag" && Kludges.autoSelfClosers.hasOwnProperty(curState.tagName.toLowerCase())))
214
+ (type == "endTag" && Kludges.autoSelfClosers.hasOwnProperty(curState.tagName.toLowerCase()))) {
215
+ maybePopContext(curState.tagName.toLowerCase());
216
+ return cont();
217
+ }
218
+ if (type == "endTag") {
219
+ maybePopContext(curState.tagName.toLowerCase());
220
+ pushContext(curState.tagName, startOfLine);
169
221
  return cont();
170
- if (type == "endTag") {pushContext(curState.tagName, startOfLine); return cont();}
222
+ }
171
223
  return cont();
172
224
  };
173
225
  }
@@ -179,17 +231,37 @@ CodeMirror.defineMode("xml", function(config, parserConfig) {
179
231
  return cont(arguments.callee);
180
232
  }
181
233
  }
234
+ function maybePopContext(nextTagName) {
235
+ var parentTagName;
236
+ while (true) {
237
+ if (!curState.context) {
238
+ return;
239
+ }
240
+ parentTagName = curState.context.tagName.toLowerCase();
241
+ if (!Kludges.contextGrabbers.hasOwnProperty(parentTagName) ||
242
+ !Kludges.contextGrabbers[parentTagName].hasOwnProperty(nextTagName)) {
243
+ return;
244
+ }
245
+ popContext();
246
+ }
247
+ }
182
248
 
183
249
  function attributes(type) {
184
- if (type == "word") {setStyle = "attribute"; return cont(attributes);}
250
+ if (type == "word") {setStyle = "attribute"; return cont(attribute, attributes);}
251
+ if (type == "endTag" || type == "selfcloseTag") return pass();
252
+ setStyle = "error";
253
+ return cont(attributes);
254
+ }
255
+ function attribute(type) {
185
256
  if (type == "equals") return cont(attvalue, attributes);
186
- if (type == "string") {setStyle = "error"; return cont(attributes);}
187
- return pass();
257
+ if (!Kludges.allowMissing) setStyle = "error";
258
+ return (type == "endTag" || type == "selfcloseTag") ? pass() : cont();
188
259
  }
189
260
  function attvalue(type) {
190
- if (type == "word" && Kludges.allowUnquoted) {setStyle = "string"; return cont();}
191
261
  if (type == "string") return cont(attvaluemaybe);
192
- return pass();
262
+ if (type == "word" && Kludges.allowUnquoted) {setStyle = "string"; return cont();}
263
+ setStyle = "error";
264
+ return (type == "endTag" || type == "selfCloseTag") ? pass() : cont();
193
265
  }
194
266
  function attvaluemaybe(type) {
195
267
  if (type == "string") return cont(attvaluemaybe);
@@ -249,4 +321,5 @@ CodeMirror.defineMode("xml", function(config, parserConfig) {
249
321
  });
250
322
 
251
323
  CodeMirror.defineMIME("application/xml", "xml");
252
- CodeMirror.defineMIME("text/html", {name: "xml", htmlMode: true});
324
+ if (!CodeMirror.mimeModes.hasOwnProperty("text/html"))
325
+ CodeMirror.defineMIME("text/html", {name: "xml", htmlMode: true});
@@ -15,6 +15,7 @@ CodeMirror.defineMode("xmlpure", function(config, parserConfig) {
15
15
  var STYLE_ATTRIBUTE = "attribute";
16
16
  var STYLE_WORD = "string";
17
17
  var STYLE_TEXT = "atom";
18
+ var STYLE_ENTITIES = "string";
18
19
 
19
20
  var TAG_INSTRUCTION = "!instruction";
20
21
  var TAG_CDATA = "!cdata";
@@ -290,6 +291,10 @@ CodeMirror.defineMode("xmlpure", function(config, parserConfig) {
290
291
  state.tokenize = parseElementTagName;
291
292
  return STYLE_ELEMENT_NAME;
292
293
  }
294
+ } else if(stream.eat("&")) {
295
+ stream.eatWhile(/[^;]/);
296
+ stream.eat(";");
297
+ return STYLE_ENTITIES;
293
298
  } else {
294
299
  // new context: text
295
300
  pushContext(state, TAG_TEXT);
@@ -0,0 +1,448 @@
1
+ /*
2
+ Copyright (C) 2011 by MarkLogic Corporation
3
+ Author: Mike Brevoort <mike@brevoort.com>
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
22
+ */
23
+ CodeMirror.defineMode("xquery", function(config, parserConfig) {
24
+
25
+ // The keywords object is set to the result of this self executing
26
+ // function. Each keyword is a property of the keywords object whose
27
+ // value is {type: atype, style: astyle}
28
+ var keywords = function(){
29
+ // conveinence functions used to build keywords object
30
+ function kw(type) {return {type: type, style: "keyword"};}
31
+ var A = kw("keyword a")
32
+ , B = kw("keyword b")
33
+ , C = kw("keyword c")
34
+ , operator = kw("operator")
35
+ , atom = {type: "atom", style: "atom"}
36
+ , punctuation = {type: "punctuation", style: ""}
37
+ , qualifier = {type: "axis_specifier", style: "qualifier"};
38
+
39
+ // kwObj is what is return from this function at the end
40
+ var kwObj = {
41
+ 'if': A, 'switch': A, 'while': A, 'for': A,
42
+ 'else': B, 'then': B, 'try': B, 'finally': B, 'catch': B,
43
+ 'element': C, 'attribute': C, 'let': C, 'implements': C, 'import': C, 'module': C, 'namespace': C,
44
+ 'return': C, 'super': C, 'this': C, 'throws': C, 'where': C, 'private': C,
45
+ ',': punctuation,
46
+ 'null': atom, 'fn:false()': atom, 'fn:true()': atom
47
+ };
48
+
49
+ // a list of 'basic' keywords. For each add a property to kwObj with the value of
50
+ // {type: basic[i], style: "keyword"} e.g. 'after' --> {type: "after", style: "keyword"}
51
+ var basic = ['after','ancestor','ancestor-or-self','and','as','ascending','assert','attribute','before',
52
+ 'by','case','cast','child','comment','declare','default','define','descendant','descendant-or-self',
53
+ 'descending','document','document-node','element','else','eq','every','except','external','following',
54
+ 'following-sibling','follows','for','function','if','import','in','instance','intersect','item',
55
+ 'let','module','namespace','node','node','of','only','or','order','parent','precedes','preceding',
56
+ 'preceding-sibling','processing-instruction','ref','return','returns','satisfies','schema','schema-element',
57
+ 'self','some','sortby','stable','text','then','to','treat','typeswitch','union','variable','version','where',
58
+ 'xquery', 'empty-sequence'];
59
+ for(var i=0, l=basic.length; i < l; i++) { kwObj[basic[i]] = kw(basic[i])};
60
+
61
+ // a list of types. For each add a property to kwObj with the value of
62
+ // {type: "atom", style: "atom"}
63
+ var types = ['xs:string', 'xs:float', 'xs:decimal', 'xs:double', 'xs:integer', 'xs:boolean', 'xs:date', 'xs:dateTime',
64
+ 'xs:time', 'xs:duration', 'xs:dayTimeDuration', 'xs:time', 'xs:yearMonthDuration', 'numeric', 'xs:hexBinary',
65
+ 'xs:base64Binary', 'xs:anyURI', 'xs:QName', 'xs:byte','xs:boolean','xs:anyURI','xf:yearMonthDuration'];
66
+ for(var i=0, l=types.length; i < l; i++) { kwObj[types[i]] = atom;};
67
+
68
+ // each operator will add a property to kwObj with value of {type: "operator", style: "keyword"}
69
+ var operators = ['eq', 'ne', 'lt', 'le', 'gt', 'ge', ':=', '=', '>', '>=', '<', '<=', '.', '|', '?', 'and', 'or', 'div', 'idiv', 'mod', '*', '/', '+', '-'];
70
+ for(var i=0, l=operators.length; i < l; i++) { kwObj[operators[i]] = operator;};
71
+
72
+ // each axis_specifiers will add a property to kwObj with value of {type: "axis_specifier", style: "qualifier"}
73
+ var axis_specifiers = ["self::", "attribute::", "child::", "descendant::", "descendant-or-self::", "parent::",
74
+ "ancestor::", "ancestor-or-self::", "following::", "preceding::", "following-sibling::", "preceding-sibling::"];
75
+ for(var i=0, l=axis_specifiers.length; i < l; i++) { kwObj[axis_specifiers[i]] = qualifier; };
76
+
77
+ return kwObj;
78
+ }();
79
+
80
+ // Used as scratch variables to communicate multiple values without
81
+ // consing up tons of objects.
82
+ var type, content;
83
+
84
+ function ret(tp, style, cont) {
85
+ type = tp; content = cont;
86
+ return style;
87
+ }
88
+
89
+ function chain(stream, state, f) {
90
+ state.tokenize = f;
91
+ return f(stream, state);
92
+ }
93
+
94
+ // the primary mode tokenizer
95
+ function tokenBase(stream, state) {
96
+ var ch = stream.next(),
97
+ mightBeFunction = false,
98
+ isEQName = isEQNameAhead(stream);
99
+
100
+ // an XML tag (if not in some sub, chained tokenizer)
101
+ if (ch == "<") {
102
+ if(stream.match("!--", true))
103
+ return chain(stream, state, tokenXMLComment);
104
+
105
+ if(stream.match("![CDATA", false)) {
106
+ state.tokenize = tokenCDATA;
107
+ return ret("tag", "tag");
108
+ }
109
+
110
+ if(stream.match("?", false)) {
111
+ return chain(stream, state, tokenPreProcessing);
112
+ }
113
+
114
+ var isclose = stream.eat("/");
115
+ stream.eatSpace();
116
+ var tagName = "", c;
117
+ while ((c = stream.eat(/[^\s\u00a0=<>\"\'\/?]/))) tagName += c;
118
+
119
+ return chain(stream, state, tokenTag(tagName, isclose));
120
+ }
121
+ // start code block
122
+ else if(ch == "{") {
123
+ pushStateStack(state,{ type: "codeblock"});
124
+ return ret("", "");
125
+ }
126
+ // end code block
127
+ else if(ch == "}") {
128
+ popStateStack(state);
129
+ return ret("", "");
130
+ }
131
+ // if we're in an XML block
132
+ else if(isInXmlBlock(state)) {
133
+ if(ch == ">")
134
+ return ret("tag", "tag");
135
+ else if(ch == "/" && stream.eat(">")) {
136
+ popStateStack(state);
137
+ return ret("tag", "tag");
138
+ }
139
+ else
140
+ return ret("word", "word");
141
+ }
142
+ // if a number
143
+ else if (/\d/.test(ch)) {
144
+ stream.match(/^\d*(?:\.\d*)?(?:E[+\-]?\d+)?/);
145
+ return ret("number", "atom");
146
+ }
147
+ // comment start
148
+ else if (ch === "(" && stream.eat(":")) {
149
+ pushStateStack(state, { type: "comment"});
150
+ return chain(stream, state, tokenComment);
151
+ }
152
+ // quoted string
153
+ else if ( !isEQName && (ch === '"' || ch === "'"))
154
+ return chain(stream, state, tokenString(ch));
155
+ // variable
156
+ else if(ch === "$") {
157
+ return chain(stream, state, tokenVariable);
158
+ }
159
+ // assignment
160
+ else if(ch ===":" && stream.eat("=")) {
161
+ return ret("operator", "keyword");
162
+ }
163
+ // open paren
164
+ else if(ch === "(") {
165
+ pushStateStack(state, { type: "paren"});
166
+ return ret("", "");
167
+ }
168
+ // close paren
169
+ else if(ch === ")") {
170
+ popStateStack(state);
171
+ return ret("", "");
172
+ }
173
+ // open paren
174
+ else if(ch === "[") {
175
+ pushStateStack(state, { type: "bracket"});
176
+ return ret("", "");
177
+ }
178
+ // close paren
179
+ else if(ch === "]") {
180
+ popStateStack(state);
181
+ return ret("", "");
182
+ }
183
+ else {
184
+ var known = keywords.propertyIsEnumerable(ch) && keywords[ch];
185
+
186
+ // if there's a EQName ahead, consume the rest of the string portion, it's likely a function
187
+ if(isEQName && ch === '\"') while(stream.next() !== '"'){}
188
+ if(isEQName && ch === '\'') while(stream.next() !== '\''){}
189
+
190
+ // gobble up a word if the character is not known
191
+ if(!known) stream.eatWhile(/[\w\$_-]/);
192
+
193
+ // gobble a colon in the case that is a lib func type call fn:doc
194
+ var foundColon = stream.eat(":")
195
+
196
+ // if there's not a second colon, gobble another word. Otherwise, it's probably an axis specifier
197
+ // which should get matched as a keyword
198
+ if(!stream.eat(":") && foundColon) {
199
+ stream.eatWhile(/[\w\$_-]/);
200
+ }
201
+ // if the next non whitespace character is an open paren, this is probably a function (if not a keyword of other sort)
202
+ if(stream.match(/^[ \t]*\(/, false)) {
203
+ mightBeFunction = true;
204
+ }
205
+ // is the word a keyword?
206
+ var word = stream.current();
207
+ known = keywords.propertyIsEnumerable(word) && keywords[word];
208
+
209
+ // if we think it's a function call but not yet known,
210
+ // set style to variable for now for lack of something better
211
+ if(mightBeFunction && !known) known = {type: "function_call", style: "variable def"};
212
+
213
+ // if the previous word was element, attribute, axis specifier, this word should be the name of that
214
+ if(isInXmlConstructor(state)) {
215
+ popStateStack(state);
216
+ return ret("word", "word", word);
217
+ }
218
+ // as previously checked, if the word is element,attribute, axis specifier, call it an "xmlconstructor" and
219
+ // push the stack so we know to look for it on the next word
220
+ if(word == "element" || word == "attribute" || known.type == "axis_specifier") pushStateStack(state, {type: "xmlconstructor"});
221
+
222
+ // if the word is known, return the details of that else just call this a generic 'word'
223
+ return known ? ret(known.type, known.style, word) :
224
+ ret("word", "word", word);
225
+ }
226
+ }
227
+
228
+ // handle comments, including nested
229
+ function tokenComment(stream, state) {
230
+ var maybeEnd = false, maybeNested = false, nestedCount = 0, ch;
231
+ while (ch = stream.next()) {
232
+ if (ch == ")" && maybeEnd) {
233
+ if(nestedCount > 0)
234
+ nestedCount--;
235
+ else {
236
+ popStateStack(state);
237
+ break;
238
+ }
239
+ }
240
+ else if(ch == ":" && maybeNested) {
241
+ nestedCount++;
242
+ }
243
+ maybeEnd = (ch == ":");
244
+ maybeNested = (ch == "(");
245
+ }
246
+
247
+ return ret("comment", "comment");
248
+ }
249
+
250
+ // tokenizer for string literals
251
+ // optionally pass a tokenizer function to set state.tokenize back to when finished
252
+ function tokenString(quote, f) {
253
+ return function(stream, state) {
254
+ var ch;
255
+
256
+ if(isInString(state) && stream.current() == quote) {
257
+ popStateStack(state);
258
+ if(f) state.tokenize = f;
259
+ return ret("string", "string");
260
+ }
261
+
262
+ pushStateStack(state, { type: "string", name: quote, tokenize: tokenString(quote, f) });
263
+
264
+ // if we're in a string and in an XML block, allow an embedded code block
265
+ if(stream.match("{", false) && isInXmlAttributeBlock(state)) {
266
+ state.tokenize = tokenBase;
267
+ return ret("string", "string");
268
+ }
269
+
270
+
271
+ while (ch = stream.next()) {
272
+ if (ch == quote) {
273
+ popStateStack(state);
274
+ if(f) state.tokenize = f;
275
+ break;
276
+ }
277
+ else {
278
+ // if we're in a string and in an XML block, allow an embedded code block in an attribute
279
+ if(stream.match("{", false) && isInXmlAttributeBlock(state)) {
280
+ state.tokenize = tokenBase;
281
+ return ret("string", "string");
282
+ }
283
+
284
+ }
285
+ }
286
+
287
+ return ret("string", "string");
288
+ };
289
+ }
290
+
291
+ // tokenizer for variables
292
+ function tokenVariable(stream, state) {
293
+ var isVariableChar = /[\w\$_-]/;
294
+
295
+ // a variable may start with a quoted EQName so if the next character is quote, consume to the next quote
296
+ if(stream.eat("\"")) {
297
+ while(stream.next() !== '\"'){};
298
+ stream.eat(":");
299
+ } else {
300
+ stream.eatWhile(isVariableChar);
301
+ if(!stream.match(":=", false)) stream.eat(":");
302
+ }
303
+ stream.eatWhile(isVariableChar);
304
+ state.tokenize = tokenBase;
305
+ return ret("variable", "variable");
306
+ }
307
+
308
+ // tokenizer for XML tags
309
+ function tokenTag(name, isclose) {
310
+ return function(stream, state) {
311
+ stream.eatSpace();
312
+ if(isclose && stream.eat(">")) {
313
+ popStateStack(state);
314
+ state.tokenize = tokenBase;
315
+ return ret("tag", "tag");
316
+ }
317
+ // self closing tag without attributes?
318
+ if(!stream.eat("/"))
319
+ pushStateStack(state, { type: "tag", name: name, tokenize: tokenBase});
320
+ if(!stream.eat(">")) {
321
+ state.tokenize = tokenAttribute;
322
+ return ret("tag", "tag");
323
+ }
324
+ else {
325
+ state.tokenize = tokenBase;
326
+ }
327
+ return ret("tag", "tag");
328
+ }
329
+ }
330
+
331
+ // tokenizer for XML attributes
332
+ function tokenAttribute(stream, state) {
333
+ var ch = stream.next();
334
+
335
+ if(ch == "/" && stream.eat(">")) {
336
+ if(isInXmlAttributeBlock(state)) popStateStack(state);
337
+ if(isInXmlBlock(state)) popStateStack(state);
338
+ return ret("tag", "tag");
339
+ }
340
+ if(ch == ">") {
341
+ if(isInXmlAttributeBlock(state)) popStateStack(state);
342
+ return ret("tag", "tag");
343
+ }
344
+ if(ch == "=")
345
+ return ret("", "");
346
+ // quoted string
347
+ if (ch == '"' || ch == "'")
348
+ return chain(stream, state, tokenString(ch, tokenAttribute));
349
+
350
+ if(!isInXmlAttributeBlock(state))
351
+ pushStateStack(state, { type: "attribute", name: name, tokenize: tokenAttribute});
352
+
353
+ stream.eat(/[a-zA-Z_:]/);
354
+ stream.eatWhile(/[-a-zA-Z0-9_:.]/);
355
+ stream.eatSpace();
356
+
357
+ // the case where the attribute has not value and the tag was closed
358
+ if(stream.match(">", false) || stream.match("/", false)) {
359
+ popStateStack(state);
360
+ state.tokenize = tokenBase;
361
+ }
362
+
363
+ return ret("attribute", "attribute");
364
+ }
365
+
366
+ // handle comments, including nested
367
+ function tokenXMLComment(stream, state) {
368
+ while (ch = stream.next()) {
369
+ if (ch == "-" && stream.match("->", true)) {
370
+ state.tokenize = tokenBase;
371
+ return ret("comment", "comment");
372
+ }
373
+ }
374
+ }
375
+
376
+
377
+ // handle CDATA
378
+ function tokenCDATA(stream, state) {
379
+ while (ch = stream.next()) {
380
+ if (ch == "]" && stream.match("]", true)) {
381
+ state.tokenize = tokenBase;
382
+ return ret("comment", "comment");
383
+ }
384
+ }
385
+ }
386
+
387
+ // handle preprocessing instructions
388
+ function tokenPreProcessing(stream, state) {
389
+ while (ch = stream.next()) {
390
+ if (ch == "?" && stream.match(">", true)) {
391
+ state.tokenize = tokenBase;
392
+ return ret("comment", "comment meta");
393
+ }
394
+ }
395
+ }
396
+
397
+
398
+ // functions to test the current context of the state
399
+ function isInXmlBlock(state) { return isIn(state, "tag"); }
400
+ function isInXmlAttributeBlock(state) { return isIn(state, "attribute"); }
401
+ function isInCodeBlock(state) { return isIn(state, "codeblock"); }
402
+ function isInXmlConstructor(state) { return isIn(state, "xmlconstructor"); }
403
+ function isInString(state) { return isIn(state, "string"); }
404
+
405
+ function isEQNameAhead(stream) {
406
+ // assume we've already eaten a quote (")
407
+ if(stream.current() === '"')
408
+ return stream.match(/^[^\"]+\"\:/, false);
409
+ else if(stream.current() === '\'')
410
+ return stream.match(/^[^\"]+\'\:/, false);
411
+ else
412
+ return false;
413
+ }
414
+
415
+ function isIn(state, type) {
416
+ return (state.stack.length && state.stack[state.stack.length - 1].type == type);
417
+ }
418
+
419
+ function pushStateStack(state, newState) {
420
+ state.stack.push(newState);
421
+ }
422
+
423
+ function popStateStack(state) {
424
+ var popped = state.stack.pop();
425
+ var reinstateTokenize = state.stack.length && state.stack[state.stack.length-1].tokenize
426
+ state.tokenize = reinstateTokenize || tokenBase;
427
+ }
428
+
429
+ // the interface for the mode API
430
+ return {
431
+ startState: function(basecolumn) {
432
+ return {
433
+ tokenize: tokenBase,
434
+ cc: [],
435
+ stack: []
436
+ };
437
+ },
438
+
439
+ token: function(stream, state) {
440
+ if (stream.eatSpace()) return null;
441
+ var style = state.tokenize(stream, state);
442
+ return style;
443
+ }
444
+ };
445
+
446
+ });
447
+
448
+ CodeMirror.defineMIME("application/xquery", "xquery");