codemirror-rails 2.22 → 2.23
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/lib/codemirror/rails/version.rb +2 -2
- data/vendor/assets/javascripts/codemirror.js +164 -83
- data/vendor/assets/javascripts/codemirror/modes/clojure.js +13 -13
- data/vendor/assets/javascripts/codemirror/modes/css.js +1 -1
- data/vendor/assets/javascripts/codemirror/modes/htmlmixed.js +1 -1
- data/vendor/assets/javascripts/codemirror/modes/javascript.js +3 -3
- data/vendor/assets/javascripts/codemirror/modes/less.js +86 -57
- data/vendor/assets/javascripts/codemirror/modes/markdown.js +29 -60
- data/vendor/assets/javascripts/codemirror/modes/properties.js +17 -11
- data/vendor/assets/javascripts/codemirror/modes/smarty.js +148 -0
- data/vendor/assets/javascripts/codemirror/modes/stex.js +15 -2
- data/vendor/assets/javascripts/codemirror/modes/tiddlywiki.js +41 -31
- data/vendor/assets/javascripts/codemirror/modes/vbscript.js +26 -0
- data/vendor/assets/javascripts/codemirror/modes/xml.js +2 -1
- data/vendor/assets/javascripts/codemirror/modes/xquery.js +448 -0
- data/vendor/assets/javascripts/codemirror/utils/closetag.js +174 -0
- data/vendor/assets/javascripts/codemirror/utils/foldcode.js +1 -1
- data/vendor/assets/javascripts/codemirror/utils/formatting.js +1 -1
- data/vendor/assets/javascripts/codemirror/utils/javascript-hint.js +4 -2
- data/vendor/assets/javascripts/codemirror/utils/simple-hint.js +6 -0
- data/vendor/assets/stylesheets/codemirror.css +2 -0
- data/vendor/assets/stylesheets/codemirror/modes/tiddlywiki.css +1 -1
- data/vendor/assets/stylesheets/codemirror/themes/eclipse.css +1 -1
- data/vendor/assets/stylesheets/codemirror/themes/elegant.css +2 -2
- data/vendor/assets/stylesheets/codemirror/themes/lesser-dark.css +45 -0
- data/vendor/assets/stylesheets/codemirror/themes/neat.css +3 -3
- data/vendor/assets/stylesheets/codemirror/themes/rubyblue.css +1 -1
- data/vendor/assets/stylesheets/codemirror/themes/xq-dark.css +46 -0
- metadata +10 -6
- data/vendor/assets/javascripts/codemirror/modes/rpm-spec.css +0 -5
- data/vendor/assets/stylesheets/codemirror/modes/properties.css +0 -3
@@ -1,21 +1,23 @@
|
|
1
1
|
CodeMirror.defineMode("properties", function() {
|
2
2
|
return {
|
3
3
|
token: function(stream, state) {
|
4
|
-
var sol = stream.sol();
|
4
|
+
var sol = stream.sol() || state.afterSection;
|
5
5
|
var eol = stream.eol();
|
6
6
|
|
7
|
+
state.afterSection = false;
|
8
|
+
|
7
9
|
if (sol) {
|
8
10
|
if (state.nextMultiline) {
|
9
11
|
state.inMultiline = true;
|
10
12
|
state.nextMultiline = false;
|
11
13
|
} else {
|
12
|
-
state.position = "
|
14
|
+
state.position = "def";
|
13
15
|
}
|
14
16
|
}
|
15
17
|
|
16
18
|
if (eol && ! state.nextMultiline) {
|
17
19
|
state.inMultiline = false;
|
18
|
-
state.position = "
|
20
|
+
state.position = "def";
|
19
21
|
}
|
20
22
|
|
21
23
|
if (sol) {
|
@@ -24,16 +26,18 @@ CodeMirror.defineMode("properties", function() {
|
|
24
26
|
|
25
27
|
var ch = stream.next();
|
26
28
|
|
27
|
-
if (sol && (ch === "#" || ch === "!")) {
|
29
|
+
if (sol && (ch === "#" || ch === "!" || ch === ";")) {
|
28
30
|
state.position = "comment";
|
29
31
|
stream.skipToEnd();
|
30
32
|
return "comment";
|
31
|
-
|
33
|
+
} else if (sol && ch === "[") {
|
34
|
+
state.afterSection = true;
|
35
|
+
stream.skipTo("]"); stream.eat("]");
|
36
|
+
return "header";
|
32
37
|
} else if (ch === "=" || ch === ":") {
|
33
|
-
state.position = "
|
34
|
-
return
|
35
|
-
|
36
|
-
} else if (ch === "\\" && state.position === "value") {
|
38
|
+
state.position = "quote";
|
39
|
+
return null;
|
40
|
+
} else if (ch === "\\" && state.position === "quote") {
|
37
41
|
if (stream.next() !== "u") { // u = Unicode sequence \u1234
|
38
42
|
// Multiline value
|
39
43
|
state.nextMultiline = true;
|
@@ -45,9 +49,10 @@ CodeMirror.defineMode("properties", function() {
|
|
45
49
|
|
46
50
|
startState: function() {
|
47
51
|
return {
|
48
|
-
position : "
|
52
|
+
position : "def", // Current position, "def", "quote" or "comment"
|
49
53
|
nextMultiline : false, // Is the next line multiline value
|
50
|
-
inMultiline : false
|
54
|
+
inMultiline : false, // Is the current line a multiline value
|
55
|
+
afterSection : false // Did we just open a section
|
51
56
|
};
|
52
57
|
}
|
53
58
|
|
@@ -55,3 +60,4 @@ CodeMirror.defineMode("properties", function() {
|
|
55
60
|
});
|
56
61
|
|
57
62
|
CodeMirror.defineMIME("text/x-properties", "properties");
|
63
|
+
CodeMirror.defineMIME("text/x-ini", "properties");
|
@@ -0,0 +1,148 @@
|
|
1
|
+
CodeMirror.defineMode("smarty", function(config, parserConfig) {
|
2
|
+
var keyFuncs = ["debug", "extends", "function", "include", "literal"];
|
3
|
+
var last;
|
4
|
+
var regs = {
|
5
|
+
operatorChars: /[+\-*&%=<>!?]/,
|
6
|
+
validIdentifier: /[a-zA-Z0-9\_]/,
|
7
|
+
stringChar: /[\'\"]/
|
8
|
+
}
|
9
|
+
var leftDelim = (typeof config.mode.leftDelimiter != 'undefined') ? config.mode.leftDelimiter : "{";
|
10
|
+
var rightDelim = (typeof config.mode.rightDelimiter != 'undefined') ? config.mode.rightDelimiter : "}";
|
11
|
+
function ret(style, lst) { last = lst; return style; }
|
12
|
+
|
13
|
+
|
14
|
+
function tokenizer(stream, state) {
|
15
|
+
function chain(parser) {
|
16
|
+
state.tokenize = parser;
|
17
|
+
return parser(stream, state);
|
18
|
+
}
|
19
|
+
|
20
|
+
if (stream.match(leftDelim, true)) {
|
21
|
+
if (stream.eat("*")) {
|
22
|
+
return chain(inBlock("comment", "*" + rightDelim));
|
23
|
+
}
|
24
|
+
else {
|
25
|
+
state.tokenize = inSmarty;
|
26
|
+
return "tag";
|
27
|
+
}
|
28
|
+
}
|
29
|
+
else {
|
30
|
+
// I'd like to do an eatWhile() here, but I can't get it to eat only up to the rightDelim string/char
|
31
|
+
stream.next();
|
32
|
+
return null;
|
33
|
+
}
|
34
|
+
}
|
35
|
+
|
36
|
+
function inSmarty(stream, state) {
|
37
|
+
if (stream.match(rightDelim, true)) {
|
38
|
+
state.tokenize = tokenizer;
|
39
|
+
return ret("tag", null);
|
40
|
+
}
|
41
|
+
|
42
|
+
var ch = stream.next();
|
43
|
+
if (ch == "$") {
|
44
|
+
stream.eatWhile(regs.validIdentifier);
|
45
|
+
return ret("variable-2", "variable");
|
46
|
+
}
|
47
|
+
else if (ch == ".") {
|
48
|
+
return ret("operator", "property");
|
49
|
+
}
|
50
|
+
else if (regs.stringChar.test(ch)) {
|
51
|
+
state.tokenize = inAttribute(ch);
|
52
|
+
return ret("string", "string");
|
53
|
+
}
|
54
|
+
else if (regs.operatorChars.test(ch)) {
|
55
|
+
stream.eatWhile(regs.operatorChars);
|
56
|
+
return ret("operator", "operator");
|
57
|
+
}
|
58
|
+
else if (ch == "[" || ch == "]") {
|
59
|
+
return ret("bracket", "bracket");
|
60
|
+
}
|
61
|
+
else if (/\d/.test(ch)) {
|
62
|
+
stream.eatWhile(/\d/);
|
63
|
+
return ret("number", "number");
|
64
|
+
}
|
65
|
+
else {
|
66
|
+
if (state.last == "variable") {
|
67
|
+
if (ch == "@") {
|
68
|
+
stream.eatWhile(regs.validIdentifier);
|
69
|
+
return ret("property", "property");
|
70
|
+
}
|
71
|
+
else if (ch == "|") {
|
72
|
+
stream.eatWhile(regs.validIdentifier);
|
73
|
+
return ret("qualifier", "modifier");
|
74
|
+
}
|
75
|
+
}
|
76
|
+
else if (state.last == "whitespace") {
|
77
|
+
stream.eatWhile(regs.validIdentifier);
|
78
|
+
return ret("attribute", "modifier");
|
79
|
+
}
|
80
|
+
else if (state.last == "property") {
|
81
|
+
stream.eatWhile(regs.validIdentifier);
|
82
|
+
return ret("property", null);
|
83
|
+
}
|
84
|
+
else if (/\s/.test(ch)) {
|
85
|
+
last = "whitespace";
|
86
|
+
return null;
|
87
|
+
}
|
88
|
+
|
89
|
+
var str = "";
|
90
|
+
if (ch != "/") {
|
91
|
+
str += ch;
|
92
|
+
}
|
93
|
+
var c = "";
|
94
|
+
while ((c = stream.eat(regs.validIdentifier))) {
|
95
|
+
str += c;
|
96
|
+
}
|
97
|
+
var i, j;
|
98
|
+
for (i=0, j=keyFuncs.length; i<j; i++) {
|
99
|
+
if (keyFuncs[i] == str) {
|
100
|
+
return ret("keyword", "keyword");
|
101
|
+
}
|
102
|
+
}
|
103
|
+
if (/\s/.test(ch)) {
|
104
|
+
return null;
|
105
|
+
}
|
106
|
+
return ret("tag", "tag");
|
107
|
+
}
|
108
|
+
}
|
109
|
+
|
110
|
+
function inAttribute(quote) {
|
111
|
+
return function(stream, state) {
|
112
|
+
while (!stream.eol()) {
|
113
|
+
if (stream.next() == quote) {
|
114
|
+
state.tokenize = inSmarty;
|
115
|
+
break;
|
116
|
+
}
|
117
|
+
}
|
118
|
+
return "string";
|
119
|
+
};
|
120
|
+
}
|
121
|
+
|
122
|
+
function inBlock(style, terminator) {
|
123
|
+
return function(stream, state) {
|
124
|
+
while (!stream.eol()) {
|
125
|
+
if (stream.match(terminator)) {
|
126
|
+
state.tokenize = tokenizer;
|
127
|
+
break;
|
128
|
+
}
|
129
|
+
stream.next();
|
130
|
+
}
|
131
|
+
return style;
|
132
|
+
};
|
133
|
+
}
|
134
|
+
|
135
|
+
return {
|
136
|
+
startState: function() {
|
137
|
+
return { tokenize: tokenizer, mode: "smarty", last: null };
|
138
|
+
},
|
139
|
+
token: function(stream, state) {
|
140
|
+
var style = state.tokenize(stream, state);
|
141
|
+
state.last = last;
|
142
|
+
return style;
|
143
|
+
},
|
144
|
+
electricChars: ""
|
145
|
+
}
|
146
|
+
});
|
147
|
+
|
148
|
+
CodeMirror.defineMIME("text/x-smarty", "smarty");
|
@@ -82,7 +82,7 @@ CodeMirror.defineMode("stex", function(cmCfg, modeCfg)
|
|
82
82
|
}
|
83
83
|
|
84
84
|
function normal(source, state) {
|
85
|
-
if (source.match(/^\\[a-
|
85
|
+
if (source.match(/^\\[a-zA-Z@]+/)) {
|
86
86
|
var cmdName = source.current();
|
87
87
|
cmdName = cmdName.substr(1, cmdName.length-1);
|
88
88
|
var plug = plugins[cmdName];
|
@@ -95,9 +95,22 @@ CodeMirror.defineMode("stex", function(cmCfg, modeCfg)
|
|
95
95
|
return plug.style;
|
96
96
|
}
|
97
97
|
|
98
|
+
// escape characters
|
99
|
+
if (source.match(/^\\[$&%#{}_]/)) {
|
100
|
+
return "tag";
|
101
|
+
}
|
102
|
+
|
103
|
+
// white space control characters
|
104
|
+
if (source.match(/^\\[,;!\/]/)) {
|
105
|
+
return "tag";
|
106
|
+
}
|
107
|
+
|
98
108
|
var ch = source.next();
|
99
109
|
if (ch == "%") {
|
100
|
-
|
110
|
+
// special case: % at end of its own line; stay in same state
|
111
|
+
if (!source.eol()) {
|
112
|
+
setState(state, inCComment);
|
113
|
+
}
|
101
114
|
return "comment";
|
102
115
|
}
|
103
116
|
else if (ch=='}' || ch==']') {
|
@@ -1,18 +1,18 @@
|
|
1
1
|
/***
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
2
|
+
|''Name''|tiddlywiki.js|
|
3
|
+
|''Description''|Enables TiddlyWikiy syntax highlighting using CodeMirror|
|
4
|
+
|''Author''|PMario|
|
5
|
+
|''Version''|0.1.7|
|
6
|
+
|''Status''|''stable''|
|
7
|
+
|''Source''|[[GitHub|https://github.com/pmario/CodeMirror2/blob/tw-syntax/mode/tiddlywiki]]|
|
8
|
+
|''Documentation''|http://codemirror.tiddlyspace.com/|
|
9
|
+
|''License''|[[MIT License|http://www.opensource.org/licenses/mit-license.php]]|
|
10
|
+
|''CoreVersion''|2.5.0|
|
11
|
+
|''Requires''|codemirror.js|
|
12
|
+
|''Keywords''|syntax highlighting color code mirror codemirror|
|
13
|
+
! Info
|
14
|
+
CoreVersion parameter is needed for TiddlyWiki only!
|
15
|
+
***/
|
16
16
|
//{{{
|
17
17
|
CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
18
18
|
var indentUnit = config.indentUnit;
|
@@ -47,20 +47,20 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
47
47
|
}();
|
48
48
|
|
49
49
|
var isSpaceName = /[\w_\-]/i,
|
50
|
-
reHR = /^\-\-\-\-+$/,
|
50
|
+
reHR = /^\-\-\-\-+$/, // <hr>
|
51
51
|
reWikiCommentStart = /^\/\*\*\*$/, // /***
|
52
52
|
reWikiCommentStop = /^\*\*\*\/$/, // ***/
|
53
53
|
reBlockQuote = /^<<<$/,
|
54
54
|
|
55
|
-
reJsCodeStart = /^\/\/\{\{\{$/, // //{{{
|
56
|
-
reJsCodeStop = /^\/\/\}\}\}$/, // //}}}
|
57
|
-
reXmlCodeStart = /^<!--\{\{\{-->$/,
|
58
|
-
reXmlCodeStop = /^<!--\}\}\}-->$/,
|
55
|
+
reJsCodeStart = /^\/\/\{\{\{$/, // //{{{ js block start
|
56
|
+
reJsCodeStop = /^\/\/\}\}\}$/, // //}}} js stop
|
57
|
+
reXmlCodeStart = /^<!--\{\{\{-->$/, // xml block start
|
58
|
+
reXmlCodeStop = /^<!--\}\}\}-->$/, // xml stop
|
59
59
|
|
60
|
-
reCodeBlockStart = /^\{\{\{$/,
|
61
|
-
reCodeBlockStop = /^\}\}\}$/,
|
60
|
+
reCodeBlockStart = /^\{\{\{$/, // {{{ TW text div block start
|
61
|
+
reCodeBlockStop = /^\}\}\}$/, // }}} TW text stop
|
62
62
|
|
63
|
-
reCodeStart = /\{\{\{/,
|
63
|
+
reCodeStart = /\{\{\{/, // {{{ code span start
|
64
64
|
reUntilCodeStop = /.*?\}\}\}/;
|
65
65
|
|
66
66
|
function chain(stream, state, f) {
|
@@ -95,9 +95,9 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
95
95
|
|
96
96
|
state.block = false; // indicates the start of a code block.
|
97
97
|
|
98
|
-
ch = stream.peek();
|
98
|
+
ch = stream.peek(); // don't eat, to make matching simpler
|
99
99
|
|
100
|
-
// check start of blocks
|
100
|
+
// check start of blocks
|
101
101
|
if (sol && /[<\/\*{}\-]/.test(ch)) {
|
102
102
|
if (stream.match(reCodeBlockStart)) {
|
103
103
|
state.block = true;
|
@@ -116,7 +116,7 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
116
116
|
return ret('hr', 'hr');
|
117
117
|
}
|
118
118
|
} // sol
|
119
|
-
|
119
|
+
ch = stream.next();
|
120
120
|
|
121
121
|
if (sol && /[\/\*!#;:>|]/.test(ch)) {
|
122
122
|
if (ch == "!") { // tw header
|
@@ -131,11 +131,11 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
131
131
|
stream.eatWhile('#');
|
132
132
|
return ret("list", "list");
|
133
133
|
}
|
134
|
-
if (ch == ";") { //
|
134
|
+
if (ch == ";") { // definition list, term
|
135
135
|
stream.eatWhile(';');
|
136
136
|
return ret("list", "list");
|
137
137
|
}
|
138
|
-
if (ch == ":") { //
|
138
|
+
if (ch == ":") { // definition list, description
|
139
139
|
stream.eatWhile(':');
|
140
140
|
return ret("list", "list");
|
141
141
|
}
|
@@ -162,6 +162,9 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
162
162
|
if (ch == '"') {
|
163
163
|
return ret('string', 'string');
|
164
164
|
}
|
165
|
+
if (ch == '~') { // _no_ CamelCase indicator should be bold
|
166
|
+
return ret('text', 'brace');
|
167
|
+
}
|
165
168
|
if (/[\[\]]/.test(ch)) { // check for [[..]]
|
166
169
|
if (stream.peek() == ch) {
|
167
170
|
stream.next();
|
@@ -189,9 +192,15 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
189
192
|
return chain(stream, state, twTokenUnderline);
|
190
193
|
}
|
191
194
|
}
|
192
|
-
|
195
|
+
// strikethrough and mdash handling
|
196
|
+
if (ch == "-") {
|
193
197
|
if (stream.eat("-")) {
|
194
|
-
|
198
|
+
// if strikethrough looks ugly, change CSS.
|
199
|
+
if (stream.peek() != ' ')
|
200
|
+
return chain(stream, state, twTokenStrike);
|
201
|
+
// mdash
|
202
|
+
if (stream.peek() == ' ')
|
203
|
+
return ret('text', 'brace');
|
195
204
|
}
|
196
205
|
}
|
197
206
|
if (ch == "'") { // tw bold
|
@@ -208,6 +217,7 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
208
217
|
return ret(ch);
|
209
218
|
}
|
210
219
|
|
220
|
+
// core macro handling
|
211
221
|
stream.eatWhile(/[\w\$_]/);
|
212
222
|
var word = stream.current(),
|
213
223
|
known = textwords.propertyIsEnumerable(word) && textwords[word];
|
@@ -301,8 +311,8 @@ CodeMirror.defineMode("tiddlywiki", function (config, parserConfig) {
|
|
301
311
|
return ret("text", "underlined");
|
302
312
|
}
|
303
313
|
|
304
|
-
// tw strike through text looks ugly
|
305
|
-
//
|
314
|
+
// tw strike through text looks ugly
|
315
|
+
// change CSS if needed
|
306
316
|
function twTokenStrike(stream, state) {
|
307
317
|
var maybeEnd = false,
|
308
318
|
ch, nr;
|
@@ -0,0 +1,26 @@
|
|
1
|
+
CodeMirror.defineMode("vbscript", function() {
|
2
|
+
var regexVBScriptKeyword = /^(?:Call|Case|CDate|Clear|CInt|CLng|Const|CStr|Description|Dim|Do|Each|Else|ElseIf|End|Err|Error|Exit|False|For|Function|If|LCase|Loop|LTrim|Next|Nothing|Now|Number|On|Preserve|Quit|ReDim|Resume|RTrim|Select|Set|Sub|Then|To|Trim|True|UBound|UCase|Until|VbCr|VbCrLf|VbLf|VbTab)$/im;
|
3
|
+
|
4
|
+
return {
|
5
|
+
token: function(stream) {
|
6
|
+
if (stream.eatSpace()) return null;
|
7
|
+
var ch = stream.next();
|
8
|
+
if (ch == "'") {
|
9
|
+
stream.skipToEnd();
|
10
|
+
return "comment";
|
11
|
+
}
|
12
|
+
if (ch == '"') {
|
13
|
+
stream.skipTo('"');
|
14
|
+
return "string";
|
15
|
+
}
|
16
|
+
|
17
|
+
if (/\w/.test(ch)) {
|
18
|
+
stream.eatWhile(/\w/);
|
19
|
+
if (regexVBScriptKeyword.test(stream.current())) return "keyword";
|
20
|
+
}
|
21
|
+
return null;
|
22
|
+
}
|
23
|
+
};
|
24
|
+
});
|
25
|
+
|
26
|
+
CodeMirror.defineMIME("text/vbscript", "vbscript");
|
@@ -264,4 +264,5 @@ CodeMirror.defineMode("xml", function(config, parserConfig) {
|
|
264
264
|
});
|
265
265
|
|
266
266
|
CodeMirror.defineMIME("application/xml", "xml");
|
267
|
-
CodeMirror.
|
267
|
+
if (!CodeMirror.mimeModes.hasOwnProperty("text/html"))
|
268
|
+
CodeMirror.defineMIME("text/html", {name: "xml", htmlMode: true});
|
@@ -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");
|