feather_cms 0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- data/.gitignore +4 -0
- data/Gemfile +4 -0
- data/README.md +46 -0
- data/Rakefile +1 -0
- data/feather_cms.gemspec +24 -0
- data/lib/feather_cms/config.rb +54 -0
- data/lib/feather_cms/model.rb +59 -0
- data/lib/feather_cms/railtie.rb +13 -0
- data/lib/feather_cms/template_cache.rb +52 -0
- data/lib/feather_cms/version.rb +3 -0
- data/lib/feather_cms/view_helper.rb +20 -0
- data/lib/feather_cms.rb +19 -0
- data/lib/generators/feather_cms/USAGE +10 -0
- data/lib/generators/feather_cms/feather_cms_generator.rb +63 -0
- data/lib/generators/feather_cms/templates/bootstrap.css +3992 -0
- data/lib/generators/feather_cms/templates/codemirror/codemirror.css +117 -0
- data/lib/generators/feather_cms/templates/codemirror/codemirror.js +2909 -0
- data/lib/generators/feather_cms/templates/codemirror/feather_cms.js +13 -0
- data/lib/generators/feather_cms/templates/codemirror/modes/css.js +124 -0
- data/lib/generators/feather_cms/templates/codemirror/modes/htmlmixed.js +83 -0
- data/lib/generators/feather_cms/templates/codemirror/modes/javascript.js +360 -0
- data/lib/generators/feather_cms/templates/codemirror/modes/xml.js +267 -0
- data/lib/generators/feather_cms/templates/codemirror/util/dialog.css +23 -0
- data/lib/generators/feather_cms/templates/codemirror/util/dialog.js +63 -0
- data/lib/generators/feather_cms/templates/codemirror/util/foldcode.js +186 -0
- data/lib/generators/feather_cms/templates/codemirror/util/formatting.js +294 -0
- data/lib/generators/feather_cms/templates/codemirror/util/javascript-hint.js +134 -0
- data/lib/generators/feather_cms/templates/codemirror/util/match-highlighter.js +44 -0
- data/lib/generators/feather_cms/templates/codemirror/util/overlay.js +51 -0
- data/lib/generators/feather_cms/templates/codemirror/util/runmode.js +49 -0
- data/lib/generators/feather_cms/templates/codemirror/util/search.js +114 -0
- data/lib/generators/feather_cms/templates/codemirror/util/searchcursor.js +117 -0
- data/lib/generators/feather_cms/templates/codemirror/util/simple-hint.css +16 -0
- data/lib/generators/feather_cms/templates/codemirror/util/simple-hint.js +68 -0
- data/lib/generators/feather_cms/templates/controller.rb +34 -0
- data/lib/generators/feather_cms/templates/form.html.erb +40 -0
- data/lib/generators/feather_cms/templates/index.html.erb +11 -0
- data/lib/generators/feather_cms/templates/initializer.rb +5 -0
- data/lib/generators/feather_cms/templates/layout.html.erb +52 -0
- data/lib/generators/feather_cms/templates/migration.rb +13 -0
- data/lib/generators/feather_cms/templates/model.rb +3 -0
- metadata +98 -0
@@ -0,0 +1,294 @@
|
|
1
|
+
// ============== Formatting extensions ============================
|
2
|
+
// A common storage for all mode-specific formatting features
|
3
|
+
if (!CodeMirror.modeExtensions) CodeMirror.modeExtensions = {};
|
4
|
+
|
5
|
+
// Returns the extension of the editor's current mode
|
6
|
+
CodeMirror.defineExtension("getModeExt", function () {
|
7
|
+
var mname = CodeMirror.resolveMode(this.getOption("mode")).name;
|
8
|
+
var ext = CodeMirror.modeExtensions[mname];
|
9
|
+
if (!ext) throw new Error("No extensions found for mode " + mname);
|
10
|
+
return ext;
|
11
|
+
});
|
12
|
+
|
13
|
+
// If the current mode is 'htmlmixed', returns the extension of a mode located at
|
14
|
+
// the specified position (can be htmlmixed, css or javascript). Otherwise, simply
|
15
|
+
// returns the extension of the editor's current mode.
|
16
|
+
CodeMirror.defineExtension("getModeExtAtPos", function (pos) {
|
17
|
+
var token = this.getTokenAt(pos);
|
18
|
+
if (token && token.state && token.state.mode)
|
19
|
+
return CodeMirror.modeExtensions[token.state.mode == "html" ? "htmlmixed" : token.state.mode];
|
20
|
+
else
|
21
|
+
return this.getModeExt();
|
22
|
+
});
|
23
|
+
|
24
|
+
// Comment/uncomment the specified range
|
25
|
+
CodeMirror.defineExtension("commentRange", function (isComment, from, to) {
|
26
|
+
var curMode = this.getModeExtAtPos(this.getCursor());
|
27
|
+
if (isComment) { // Comment range
|
28
|
+
var commentedText = this.getRange(from, to);
|
29
|
+
this.replaceRange(curMode.commentStart + this.getRange(from, to) + curMode.commentEnd
|
30
|
+
, from, to);
|
31
|
+
if (from.line == to.line && from.ch == to.ch) { // An empty comment inserted - put cursor inside
|
32
|
+
this.setCursor(from.line, from.ch + curMode.commentStart.length);
|
33
|
+
}
|
34
|
+
}
|
35
|
+
else { // Uncomment range
|
36
|
+
var selText = this.getRange(from, to);
|
37
|
+
var startIndex = selText.indexOf(curMode.commentStart);
|
38
|
+
var endIndex = selText.lastIndexOf(curMode.commentEnd);
|
39
|
+
if (startIndex > -1 && endIndex > -1 && endIndex > startIndex) {
|
40
|
+
// Take string till comment start
|
41
|
+
selText = selText.substr(0, startIndex)
|
42
|
+
// From comment start till comment end
|
43
|
+
+ selText.substring(startIndex + curMode.commentStart.length, endIndex)
|
44
|
+
// From comment end till string end
|
45
|
+
+ selText.substr(endIndex + curMode.commentEnd.length);
|
46
|
+
}
|
47
|
+
this.replaceRange(selText, from, to);
|
48
|
+
}
|
49
|
+
});
|
50
|
+
|
51
|
+
// Applies automatic mode-aware indentation to the specified range
|
52
|
+
CodeMirror.defineExtension("autoIndentRange", function (from, to) {
|
53
|
+
var cmInstance = this;
|
54
|
+
this.operation(function () {
|
55
|
+
for (var i = from.line; i <= to.line; i++) {
|
56
|
+
cmInstance.indentLine(i, "smart");
|
57
|
+
}
|
58
|
+
});
|
59
|
+
});
|
60
|
+
|
61
|
+
// Applies automatic formatting to the specified range
|
62
|
+
CodeMirror.defineExtension("autoFormatRange", function (from, to) {
|
63
|
+
var absStart = this.indexFromPos(from);
|
64
|
+
var absEnd = this.indexFromPos(to);
|
65
|
+
// Insert additional line breaks where necessary according to the
|
66
|
+
// mode's syntax
|
67
|
+
var res = this.getModeExt().autoFormatLineBreaks(this.getValue(), absStart, absEnd);
|
68
|
+
var cmInstance = this;
|
69
|
+
|
70
|
+
// Replace and auto-indent the range
|
71
|
+
this.operation(function () {
|
72
|
+
cmInstance.replaceRange(res, from, to);
|
73
|
+
var startLine = cmInstance.posFromIndex(absStart).line;
|
74
|
+
var endLine = cmInstance.posFromIndex(absStart + res.length).line;
|
75
|
+
for (var i = startLine; i <= endLine; i++) {
|
76
|
+
cmInstance.indentLine(i, "smart");
|
77
|
+
}
|
78
|
+
});
|
79
|
+
});
|
80
|
+
|
81
|
+
// Define extensions for a few modes
|
82
|
+
|
83
|
+
CodeMirror.modeExtensions["css"] = {
|
84
|
+
commentStart: "/*",
|
85
|
+
commentEnd: "*/",
|
86
|
+
wordWrapChars: [";", "\\{", "\\}"],
|
87
|
+
autoFormatLineBreaks: function (text) {
|
88
|
+
return text.replace(new RegExp("(;|\\{|\\})([^\r\n])", "g"), "$1\n$2");
|
89
|
+
}
|
90
|
+
};
|
91
|
+
|
92
|
+
CodeMirror.modeExtensions["javascript"] = {
|
93
|
+
commentStart: "/*",
|
94
|
+
commentEnd: "*/",
|
95
|
+
wordWrapChars: [";", "\\{", "\\}"],
|
96
|
+
|
97
|
+
getNonBreakableBlocks: function (text) {
|
98
|
+
var nonBreakableRegexes = [
|
99
|
+
new RegExp("for\\s*?\\(([\\s\\S]*?)\\)"),
|
100
|
+
new RegExp("'([\\s\\S]*?)('|$)"),
|
101
|
+
new RegExp("\"([\\s\\S]*?)(\"|$)"),
|
102
|
+
new RegExp("//.*([\r\n]|$)")
|
103
|
+
];
|
104
|
+
var nonBreakableBlocks = new Array();
|
105
|
+
for (var i = 0; i < nonBreakableRegexes.length; i++) {
|
106
|
+
var curPos = 0;
|
107
|
+
while (curPos < text.length) {
|
108
|
+
var m = text.substr(curPos).match(nonBreakableRegexes[i]);
|
109
|
+
if (m != null) {
|
110
|
+
nonBreakableBlocks.push({
|
111
|
+
start: curPos + m.index,
|
112
|
+
end: curPos + m.index + m[0].length
|
113
|
+
});
|
114
|
+
curPos += m.index + Math.max(1, m[0].length);
|
115
|
+
}
|
116
|
+
else { // No more matches
|
117
|
+
break;
|
118
|
+
}
|
119
|
+
}
|
120
|
+
}
|
121
|
+
nonBreakableBlocks.sort(function (a, b) {
|
122
|
+
return a.start - b.start;
|
123
|
+
});
|
124
|
+
|
125
|
+
return nonBreakableBlocks;
|
126
|
+
},
|
127
|
+
|
128
|
+
autoFormatLineBreaks: function (text) {
|
129
|
+
var curPos = 0;
|
130
|
+
var reLinesSplitter = new RegExp("(;|\\{|\\})([^\r\n])", "g");
|
131
|
+
var nonBreakableBlocks = this.getNonBreakableBlocks(text);
|
132
|
+
if (nonBreakableBlocks != null) {
|
133
|
+
var res = "";
|
134
|
+
for (var i = 0; i < nonBreakableBlocks.length; i++) {
|
135
|
+
if (nonBreakableBlocks[i].start > curPos) { // Break lines till the block
|
136
|
+
res += text.substring(curPos, nonBreakableBlocks[i].start).replace(reLinesSplitter, "$1\n$2");
|
137
|
+
curPos = nonBreakableBlocks[i].start;
|
138
|
+
}
|
139
|
+
if (nonBreakableBlocks[i].start <= curPos
|
140
|
+
&& nonBreakableBlocks[i].end >= curPos) { // Skip non-breakable block
|
141
|
+
res += text.substring(curPos, nonBreakableBlocks[i].end);
|
142
|
+
curPos = nonBreakableBlocks[i].end;
|
143
|
+
}
|
144
|
+
}
|
145
|
+
if (curPos < text.length - 1) {
|
146
|
+
res += text.substr(curPos).replace(reLinesSplitter, "$1\n$2");
|
147
|
+
}
|
148
|
+
return res;
|
149
|
+
}
|
150
|
+
else {
|
151
|
+
return text.replace(reLinesSplitter, "$1\n$2");
|
152
|
+
}
|
153
|
+
}
|
154
|
+
};
|
155
|
+
|
156
|
+
CodeMirror.modeExtensions["xml"] = {
|
157
|
+
commentStart: "<!--",
|
158
|
+
commentEnd: "-->",
|
159
|
+
wordWrapChars: [">"],
|
160
|
+
|
161
|
+
autoFormatLineBreaks: function (text) {
|
162
|
+
var lines = text.split("\n");
|
163
|
+
var reProcessedPortion = new RegExp("(^\\s*?<|^[^<]*?)(.+)(>\\s*?$|[^>]*?$)");
|
164
|
+
var reOpenBrackets = new RegExp("<", "g");
|
165
|
+
var reCloseBrackets = new RegExp("(>)([^\r\n])", "g");
|
166
|
+
for (var i = 0; i < lines.length; i++) {
|
167
|
+
var mToProcess = lines[i].match(reProcessedPortion);
|
168
|
+
if (mToProcess != null && mToProcess.length > 3) { // The line starts with whitespaces and ends with whitespaces
|
169
|
+
lines[i] = mToProcess[1]
|
170
|
+
+ mToProcess[2].replace(reOpenBrackets, "\n$&").replace(reCloseBrackets, "$1\n$2")
|
171
|
+
+ mToProcess[3];
|
172
|
+
continue;
|
173
|
+
}
|
174
|
+
}
|
175
|
+
|
176
|
+
return lines.join("\n");
|
177
|
+
}
|
178
|
+
};
|
179
|
+
|
180
|
+
CodeMirror.modeExtensions["htmlmixed"] = {
|
181
|
+
commentStart: "<!--",
|
182
|
+
commentEnd: "-->",
|
183
|
+
wordWrapChars: [">", ";", "\\{", "\\}"],
|
184
|
+
|
185
|
+
getModeInfos: function (text, absPos) {
|
186
|
+
var modeInfos = new Array();
|
187
|
+
modeInfos[0] =
|
188
|
+
{
|
189
|
+
pos: 0,
|
190
|
+
modeExt: CodeMirror.modeExtensions["xml"],
|
191
|
+
modeName: "xml"
|
192
|
+
};
|
193
|
+
|
194
|
+
var modeMatchers = new Array();
|
195
|
+
modeMatchers[0] =
|
196
|
+
{
|
197
|
+
regex: new RegExp("<style[^>]*>([\\s\\S]*?)(</style[^>]*>|$)", "i"),
|
198
|
+
modeExt: CodeMirror.modeExtensions["css"],
|
199
|
+
modeName: "css"
|
200
|
+
};
|
201
|
+
modeMatchers[1] =
|
202
|
+
{
|
203
|
+
regex: new RegExp("<script[^>]*>([\\s\\S]*?)(</script[^>]*>|$)", "i"),
|
204
|
+
modeExt: CodeMirror.modeExtensions["javascript"],
|
205
|
+
modeName: "javascript"
|
206
|
+
};
|
207
|
+
|
208
|
+
var lastCharPos = (typeof (absPos) !== "undefined" ? absPos : text.length - 1);
|
209
|
+
// Detect modes for the entire text
|
210
|
+
for (var i = 0; i < modeMatchers.length; i++) {
|
211
|
+
var curPos = 0;
|
212
|
+
while (curPos <= lastCharPos) {
|
213
|
+
var m = text.substr(curPos).match(modeMatchers[i].regex);
|
214
|
+
if (m != null) {
|
215
|
+
if (m.length > 1 && m[1].length > 0) {
|
216
|
+
// Push block begin pos
|
217
|
+
var blockBegin = curPos + m.index + m[0].indexOf(m[1]);
|
218
|
+
modeInfos.push(
|
219
|
+
{
|
220
|
+
pos: blockBegin,
|
221
|
+
modeExt: modeMatchers[i].modeExt,
|
222
|
+
modeName: modeMatchers[i].modeName
|
223
|
+
});
|
224
|
+
// Push block end pos
|
225
|
+
modeInfos.push(
|
226
|
+
{
|
227
|
+
pos: blockBegin + m[1].length,
|
228
|
+
modeExt: modeInfos[0].modeExt,
|
229
|
+
modeName: modeInfos[0].modeName
|
230
|
+
});
|
231
|
+
curPos += m.index + m[0].length;
|
232
|
+
continue;
|
233
|
+
}
|
234
|
+
else {
|
235
|
+
curPos += m.index + Math.max(m[0].length, 1);
|
236
|
+
}
|
237
|
+
}
|
238
|
+
else { // No more matches
|
239
|
+
break;
|
240
|
+
}
|
241
|
+
}
|
242
|
+
}
|
243
|
+
// Sort mode infos
|
244
|
+
modeInfos.sort(function sortModeInfo(a, b) {
|
245
|
+
return a.pos - b.pos;
|
246
|
+
});
|
247
|
+
|
248
|
+
return modeInfos;
|
249
|
+
},
|
250
|
+
|
251
|
+
autoFormatLineBreaks: function (text, startPos, endPos) {
|
252
|
+
var modeInfos = this.getModeInfos(text);
|
253
|
+
var reBlockStartsWithNewline = new RegExp("^\\s*?\n");
|
254
|
+
var reBlockEndsWithNewline = new RegExp("\n\\s*?$");
|
255
|
+
var res = "";
|
256
|
+
// Use modes info to break lines correspondingly
|
257
|
+
if (modeInfos.length > 1) { // Deal with multi-mode text
|
258
|
+
for (var i = 1; i <= modeInfos.length; i++) {
|
259
|
+
var selStart = modeInfos[i - 1].pos;
|
260
|
+
var selEnd = (i < modeInfos.length ? modeInfos[i].pos : endPos);
|
261
|
+
|
262
|
+
if (selStart >= endPos) { // The block starts later than the needed fragment
|
263
|
+
break;
|
264
|
+
}
|
265
|
+
if (selStart < startPos) {
|
266
|
+
if (selEnd <= startPos) { // The block starts earlier than the needed fragment
|
267
|
+
continue;
|
268
|
+
}
|
269
|
+
selStart = startPos;
|
270
|
+
}
|
271
|
+
if (selEnd > endPos) {
|
272
|
+
selEnd = endPos;
|
273
|
+
}
|
274
|
+
var textPortion = text.substring(selStart, selEnd);
|
275
|
+
if (modeInfos[i - 1].modeName != "xml") { // Starting a CSS or JavaScript block
|
276
|
+
if (!reBlockStartsWithNewline.test(textPortion)
|
277
|
+
&& selStart > 0) { // The block does not start with a line break
|
278
|
+
textPortion = "\n" + textPortion;
|
279
|
+
}
|
280
|
+
if (!reBlockEndsWithNewline.test(textPortion)
|
281
|
+
&& selEnd < text.length - 1) { // The block does not end with a line break
|
282
|
+
textPortion += "\n";
|
283
|
+
}
|
284
|
+
}
|
285
|
+
res += modeInfos[i - 1].modeExt.autoFormatLineBreaks(textPortion);
|
286
|
+
}
|
287
|
+
}
|
288
|
+
else { // Single-mode text
|
289
|
+
res = modeInfos[0].modeExt.autoFormatLineBreaks(text.substring(startPos, endPos));
|
290
|
+
}
|
291
|
+
|
292
|
+
return res;
|
293
|
+
}
|
294
|
+
};
|
@@ -0,0 +1,134 @@
|
|
1
|
+
(function () {
|
2
|
+
function forEach(arr, f) {
|
3
|
+
for (var i = 0, e = arr.length; i < e; ++i) f(arr[i]);
|
4
|
+
}
|
5
|
+
|
6
|
+
function arrayContains(arr, item) {
|
7
|
+
if (!Array.prototype.indexOf) {
|
8
|
+
var i = arr.length;
|
9
|
+
while (i--) {
|
10
|
+
if (arr[i] === item) {
|
11
|
+
return true;
|
12
|
+
}
|
13
|
+
}
|
14
|
+
return false;
|
15
|
+
}
|
16
|
+
return arr.indexOf(item) != -1;
|
17
|
+
}
|
18
|
+
|
19
|
+
function scriptHint(editor, keywords, getToken) {
|
20
|
+
// Find the token at the cursor
|
21
|
+
var cur = editor.getCursor(), token = getToken(editor, cur), tprop = token;
|
22
|
+
// If it's not a 'word-style' token, ignore the token.
|
23
|
+
if (!/^[\w$_]*$/.test(token.string)) {
|
24
|
+
token = tprop = {start: cur.ch, end: cur.ch, string: "", state: token.state,
|
25
|
+
className: token.string == "." ? "property" : null};
|
26
|
+
}
|
27
|
+
// If it is a property, find out what it is a property of.
|
28
|
+
while (tprop.className == "property") {
|
29
|
+
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
30
|
+
if (tprop.string != ".") return;
|
31
|
+
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
32
|
+
if (tprop.string == ')') {
|
33
|
+
var level = 1;
|
34
|
+
do {
|
35
|
+
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
36
|
+
switch (tprop.string) {
|
37
|
+
case ')': level++; break;
|
38
|
+
case '(': level--; break;
|
39
|
+
default: break;
|
40
|
+
}
|
41
|
+
} while (level > 0)
|
42
|
+
tprop = getToken(editor, {line: cur.line, ch: tprop.start});
|
43
|
+
if (tprop.className == 'variable')
|
44
|
+
tprop.className = 'function';
|
45
|
+
else return; // no clue
|
46
|
+
}
|
47
|
+
if (!context) var context = [];
|
48
|
+
context.push(tprop);
|
49
|
+
}
|
50
|
+
return {list: getCompletions(token, context, keywords),
|
51
|
+
from: {line: cur.line, ch: token.start},
|
52
|
+
to: {line: cur.line, ch: token.end}};
|
53
|
+
}
|
54
|
+
|
55
|
+
CodeMirror.javascriptHint = function(editor) {
|
56
|
+
return scriptHint(editor, javascriptKeywords,
|
57
|
+
function (e, cur) {return e.getTokenAt(cur);});
|
58
|
+
}
|
59
|
+
|
60
|
+
function getCoffeeScriptToken(editor, cur) {
|
61
|
+
// This getToken, it is for coffeescript, imitates the behavior of
|
62
|
+
// getTokenAt method in javascript.js, that is, returning "property"
|
63
|
+
// type and treat "." as indepenent token.
|
64
|
+
var token = editor.getTokenAt(cur);
|
65
|
+
if (cur.ch == token.start + 1 && token.string.charAt(0) == '.') {
|
66
|
+
token.end = token.start;
|
67
|
+
token.string = '.';
|
68
|
+
token.className = "property";
|
69
|
+
}
|
70
|
+
else if (/^\.[\w$_]*$/.test(token.string)) {
|
71
|
+
token.className = "property";
|
72
|
+
token.start++;
|
73
|
+
token.string = token.string.replace(/\./, '');
|
74
|
+
}
|
75
|
+
return token;
|
76
|
+
}
|
77
|
+
|
78
|
+
CodeMirror.coffeescriptHint = function(editor) {
|
79
|
+
return scriptHint(editor, coffeescriptKeywords, getCoffeeScriptToken);
|
80
|
+
}
|
81
|
+
|
82
|
+
var stringProps = ("charAt charCodeAt indexOf lastIndexOf substring substr slice trim trimLeft trimRight " +
|
83
|
+
"toUpperCase toLowerCase split concat match replace search").split(" ");
|
84
|
+
var arrayProps = ("length concat join splice push pop shift unshift slice reverse sort indexOf " +
|
85
|
+
"lastIndexOf every some filter forEach map reduce reduceRight ").split(" ");
|
86
|
+
var funcProps = "prototype apply call bind".split(" ");
|
87
|
+
var javascriptKeywords = ("break case catch continue debugger default delete do else false finally for function " +
|
88
|
+
"if in instanceof new null return switch throw true try typeof var void while with").split(" ");
|
89
|
+
var coffeescriptKeywords = ("and break catch class continue delete do else extends false finally for " +
|
90
|
+
"if in instanceof isnt new no not null of off on or return switch then throw true try typeof until void while with yes").split(" ");
|
91
|
+
|
92
|
+
function getCompletions(token, context, keywords) {
|
93
|
+
var found = [], start = token.string;
|
94
|
+
function maybeAdd(str) {
|
95
|
+
if (str.indexOf(start) == 0 && !arrayContains(found, str)) found.push(str);
|
96
|
+
}
|
97
|
+
function gatherCompletions(obj) {
|
98
|
+
if (typeof obj == "string") forEach(stringProps, maybeAdd);
|
99
|
+
else if (obj instanceof Array) forEach(arrayProps, maybeAdd);
|
100
|
+
else if (obj instanceof Function) forEach(funcProps, maybeAdd);
|
101
|
+
for (var name in obj) maybeAdd(name);
|
102
|
+
}
|
103
|
+
|
104
|
+
if (context) {
|
105
|
+
// If this is a property, see if it belongs to some object we can
|
106
|
+
// find in the current environment.
|
107
|
+
var obj = context.pop(), base;
|
108
|
+
if (obj.className == "variable")
|
109
|
+
base = window[obj.string];
|
110
|
+
else if (obj.className == "string")
|
111
|
+
base = "";
|
112
|
+
else if (obj.className == "atom")
|
113
|
+
base = 1;
|
114
|
+
else if (obj.className == "function") {
|
115
|
+
if (window.jQuery != null && (obj.string == '$' || obj.string == 'jQuery') &&
|
116
|
+
(typeof window.jQuery == 'function'))
|
117
|
+
base = window.jQuery();
|
118
|
+
else if (window._ != null && (obj.string == '_') && (typeof window._ == 'function'))
|
119
|
+
base = window._();
|
120
|
+
}
|
121
|
+
while (base != null && context.length)
|
122
|
+
base = base[context.pop().string];
|
123
|
+
if (base != null) gatherCompletions(base);
|
124
|
+
}
|
125
|
+
else {
|
126
|
+
// If not, just look in the window object and any local scope
|
127
|
+
// (reading into JS mode internals to get at the local variables)
|
128
|
+
for (var v = token.state.localVars; v; v = v.next) maybeAdd(v.name);
|
129
|
+
gatherCompletions(window);
|
130
|
+
forEach(keywords, maybeAdd);
|
131
|
+
}
|
132
|
+
return found;
|
133
|
+
}
|
134
|
+
})();
|
@@ -0,0 +1,44 @@
|
|
1
|
+
// Define match-highlighter commands. Depends on searchcursor.js
|
2
|
+
// Use by attaching the following function call to the onCursorActivity event:
|
3
|
+
//myCodeMirror.matchHighlight(minChars);
|
4
|
+
// And including a special span.CodeMirror-matchhighlight css class (also optionally a separate one for .CodeMirror-focused -- see demo matchhighlighter.html)
|
5
|
+
|
6
|
+
(function() {
|
7
|
+
var DEFAULT_MIN_CHARS = 2;
|
8
|
+
|
9
|
+
function MatchHighlightState() {
|
10
|
+
this.marked = [];
|
11
|
+
}
|
12
|
+
function getMatchHighlightState(cm) {
|
13
|
+
return cm._matchHighlightState || (cm._matchHighlightState = new MatchHighlightState());
|
14
|
+
}
|
15
|
+
|
16
|
+
function clearMarks(cm) {
|
17
|
+
var state = getMatchHighlightState(cm);
|
18
|
+
for (var i = 0; i < state.marked.length; ++i)
|
19
|
+
state.marked[i].clear();
|
20
|
+
state.marked = [];
|
21
|
+
}
|
22
|
+
|
23
|
+
function markDocument(cm, className, minChars) {
|
24
|
+
clearMarks(cm);
|
25
|
+
minChars = (typeof minChars !== 'undefined' ? minChars : DEFAULT_MIN_CHARS);
|
26
|
+
if (cm.somethingSelected() && cm.getSelection().length >= minChars) {
|
27
|
+
var state = getMatchHighlightState(cm);
|
28
|
+
var query = cm.getSelection();
|
29
|
+
cm.operation(function() {
|
30
|
+
if (cm.lineCount() < 2000) { // This is too expensive on big documents.
|
31
|
+
for (var cursor = cm.getSearchCursor(query); cursor.findNext();) {
|
32
|
+
//Only apply matchhighlight to the matches other than the one actually selected
|
33
|
+
if (!(cursor.from().line === cm.getCursor(true).line && cursor.from().ch === cm.getCursor(true).ch))
|
34
|
+
state.marked.push(cm.markText(cursor.from(), cursor.to(), className));
|
35
|
+
}
|
36
|
+
}
|
37
|
+
});
|
38
|
+
}
|
39
|
+
}
|
40
|
+
|
41
|
+
CodeMirror.defineExtension("matchHighlight", function(className, minChars) {
|
42
|
+
markDocument(this, className, minChars);
|
43
|
+
});
|
44
|
+
})();
|
@@ -0,0 +1,51 @@
|
|
1
|
+
// Utility function that allows modes to be combined. The mode given
|
2
|
+
// as the base argument takes care of most of the normal mode
|
3
|
+
// functionality, but a second (typically simple) mode is used, which
|
4
|
+
// can override the style of text. Both modes get to parse all of the
|
5
|
+
// text, but when both assign a non-null style to a piece of code, the
|
6
|
+
// overlay wins, unless the combine argument was true, in which case
|
7
|
+
// the styles are combined.
|
8
|
+
|
9
|
+
CodeMirror.overlayParser = function(base, overlay, combine) {
|
10
|
+
return {
|
11
|
+
startState: function() {
|
12
|
+
return {
|
13
|
+
base: CodeMirror.startState(base),
|
14
|
+
overlay: CodeMirror.startState(overlay),
|
15
|
+
basePos: 0, baseCur: null,
|
16
|
+
overlayPos: 0, overlayCur: null
|
17
|
+
};
|
18
|
+
},
|
19
|
+
copyState: function(state) {
|
20
|
+
return {
|
21
|
+
base: CodeMirror.copyState(base, state.base),
|
22
|
+
overlay: CodeMirror.copyState(overlay, state.overlay),
|
23
|
+
basePos: state.basePos, baseCur: null,
|
24
|
+
overlayPos: state.overlayPos, overlayCur: null
|
25
|
+
};
|
26
|
+
},
|
27
|
+
|
28
|
+
token: function(stream, state) {
|
29
|
+
if (stream.start == state.basePos) {
|
30
|
+
state.baseCur = base.token(stream, state.base);
|
31
|
+
state.basePos = stream.pos;
|
32
|
+
}
|
33
|
+
if (stream.start == state.overlayPos) {
|
34
|
+
stream.pos = stream.start;
|
35
|
+
state.overlayCur = overlay.token(stream, state.overlay);
|
36
|
+
state.overlayPos = stream.pos;
|
37
|
+
}
|
38
|
+
stream.pos = Math.min(state.basePos, state.overlayPos);
|
39
|
+
if (stream.eol()) state.basePos = state.overlayPos = 0;
|
40
|
+
|
41
|
+
if (state.overlayCur == null) return state.baseCur;
|
42
|
+
if (state.baseCur != null && combine) return state.baseCur + " " + state.overlayCur;
|
43
|
+
else return state.overlayCur;
|
44
|
+
},
|
45
|
+
|
46
|
+
indent: function(state, textAfter) {
|
47
|
+
return base.indent(state.base, textAfter);
|
48
|
+
},
|
49
|
+
electricChars: base.electricChars
|
50
|
+
};
|
51
|
+
};
|
@@ -0,0 +1,49 @@
|
|
1
|
+
CodeMirror.runMode = function(string, modespec, callback, options) {
|
2
|
+
var mode = CodeMirror.getMode(CodeMirror.defaults, modespec);
|
3
|
+
var isNode = callback.nodeType == 1;
|
4
|
+
var tabSize = (options && options.tabSize) || CodeMirror.defaults.tabSize;
|
5
|
+
if (isNode) {
|
6
|
+
var node = callback, accum = [], col = 0;
|
7
|
+
callback = function(text, style) {
|
8
|
+
if (text == "\n") {
|
9
|
+
accum.push("<br>");
|
10
|
+
col = 0;
|
11
|
+
return;
|
12
|
+
}
|
13
|
+
var escaped = "";
|
14
|
+
// HTML-escape and replace tabs
|
15
|
+
for (var pos = 0;;) {
|
16
|
+
var idx = text.indexOf("\t", pos);
|
17
|
+
if (idx == -1) {
|
18
|
+
escaped += CodeMirror.htmlEscape(text.slice(pos));
|
19
|
+
col += text.length - pos;
|
20
|
+
break;
|
21
|
+
} else {
|
22
|
+
col += idx - pos;
|
23
|
+
escaped += CodeMirror.htmlEscape(text.slice(pos, idx));
|
24
|
+
var size = tabSize - col % tabSize;
|
25
|
+
col += size;
|
26
|
+
for (var i = 0; i < size; ++i) escaped += " ";
|
27
|
+
pos = idx + 1;
|
28
|
+
}
|
29
|
+
}
|
30
|
+
|
31
|
+
if (style)
|
32
|
+
accum.push("<span class=\"cm-" + CodeMirror.htmlEscape(style) + "\">" + escaped + "</span>");
|
33
|
+
else
|
34
|
+
accum.push(escaped);
|
35
|
+
}
|
36
|
+
}
|
37
|
+
var lines = CodeMirror.splitLines(string), state = CodeMirror.startState(mode);
|
38
|
+
for (var i = 0, e = lines.length; i < e; ++i) {
|
39
|
+
if (i) callback("\n");
|
40
|
+
var stream = new CodeMirror.StringStream(lines[i]);
|
41
|
+
while (!stream.eol()) {
|
42
|
+
var style = mode.token(stream, state);
|
43
|
+
callback(stream.current(), style, i, stream.start);
|
44
|
+
stream.start = stream.pos;
|
45
|
+
}
|
46
|
+
}
|
47
|
+
if (isNode)
|
48
|
+
node.innerHTML = accum.join("");
|
49
|
+
};
|
@@ -0,0 +1,114 @@
|
|
1
|
+
// Define search commands. Depends on dialog.js or another
|
2
|
+
// implementation of the openDialog method.
|
3
|
+
|
4
|
+
// Replace works a little oddly -- it will do the replace on the next
|
5
|
+
// Ctrl-G (or whatever is bound to findNext) press. You prevent a
|
6
|
+
// replace by making sure the match is no longer selected when hitting
|
7
|
+
// Ctrl-G.
|
8
|
+
|
9
|
+
(function() {
|
10
|
+
function SearchState() {
|
11
|
+
this.posFrom = this.posTo = this.query = null;
|
12
|
+
this.marked = [];
|
13
|
+
}
|
14
|
+
function getSearchState(cm) {
|
15
|
+
return cm._searchState || (cm._searchState = new SearchState());
|
16
|
+
}
|
17
|
+
function dialog(cm, text, shortText, f) {
|
18
|
+
if (cm.openDialog) cm.openDialog(text, f);
|
19
|
+
else f(prompt(shortText, ""));
|
20
|
+
}
|
21
|
+
function confirmDialog(cm, text, shortText, fs) {
|
22
|
+
if (cm.openConfirm) cm.openConfirm(text, fs);
|
23
|
+
else if (confirm(shortText)) fs[0]();
|
24
|
+
}
|
25
|
+
function parseQuery(query) {
|
26
|
+
var isRE = query.match(/^\/(.*)\/$/);
|
27
|
+
return isRE ? new RegExp(isRE[1]) : query;
|
28
|
+
}
|
29
|
+
var queryDialog =
|
30
|
+
'Search: <input type="text" style="width: 10em"> <span style="color: #888">(Use /re/ syntax for regexp search)</span>';
|
31
|
+
function doSearch(cm, rev) {
|
32
|
+
var state = getSearchState(cm);
|
33
|
+
if (state.query) return findNext(cm, rev);
|
34
|
+
dialog(cm, queryDialog, "Search for:", function(query) {
|
35
|
+
cm.operation(function() {
|
36
|
+
if (!query || state.query) return;
|
37
|
+
state.query = parseQuery(query);
|
38
|
+
if (cm.lineCount() < 2000) { // This is too expensive on big documents.
|
39
|
+
for (var cursor = cm.getSearchCursor(query); cursor.findNext();)
|
40
|
+
state.marked.push(cm.markText(cursor.from(), cursor.to(), "CodeMirror-searching"));
|
41
|
+
}
|
42
|
+
state.posFrom = state.posTo = cm.getCursor();
|
43
|
+
findNext(cm, rev);
|
44
|
+
});
|
45
|
+
});
|
46
|
+
}
|
47
|
+
function findNext(cm, rev) {cm.operation(function() {
|
48
|
+
var state = getSearchState(cm);
|
49
|
+
var cursor = cm.getSearchCursor(state.query, rev ? state.posFrom : state.posTo);
|
50
|
+
if (!cursor.find(rev)) {
|
51
|
+
cursor = cm.getSearchCursor(state.query, rev ? {line: cm.lineCount() - 1} : {line: 0, ch: 0});
|
52
|
+
if (!cursor.find(rev)) return;
|
53
|
+
}
|
54
|
+
cm.setSelection(cursor.from(), cursor.to());
|
55
|
+
state.posFrom = cursor.from(); state.posTo = cursor.to();
|
56
|
+
})}
|
57
|
+
function clearSearch(cm) {cm.operation(function() {
|
58
|
+
var state = getSearchState(cm);
|
59
|
+
if (!state.query) return;
|
60
|
+
state.query = null;
|
61
|
+
for (var i = 0; i < state.marked.length; ++i) state.marked[i].clear();
|
62
|
+
state.marked.length = 0;
|
63
|
+
})}
|
64
|
+
|
65
|
+
var replaceQueryDialog =
|
66
|
+
'Replace: <input type="text" style="width: 10em"> <span style="color: #888">(Use /re/ syntax for regexp search)</span>';
|
67
|
+
var replacementQueryDialog = 'With: <input type="text" style="width: 10em">';
|
68
|
+
var doReplaceConfirm = "Replace? <button>Yes</button> <button>No</button> <button>Stop</button>";
|
69
|
+
function replace(cm, all) {
|
70
|
+
dialog(cm, replaceQueryDialog, "Replace:", function(query) {
|
71
|
+
if (!query) return;
|
72
|
+
query = parseQuery(query);
|
73
|
+
dialog(cm, replacementQueryDialog, "Replace with:", function(text) {
|
74
|
+
if (all) {
|
75
|
+
cm.operation(function() {
|
76
|
+
for (var cursor = cm.getSearchCursor(query); cursor.findNext();) {
|
77
|
+
if (typeof query != "string") {
|
78
|
+
var match = cm.getRange(cursor.from(), cursor.to()).match(query);
|
79
|
+
cursor.replace(text.replace(/\$(\d)/, function(w, i) {return match[i];}));
|
80
|
+
} else cursor.replace(text);
|
81
|
+
}
|
82
|
+
});
|
83
|
+
} else {
|
84
|
+
clearSearch(cm);
|
85
|
+
var cursor = cm.getSearchCursor(query, cm.getCursor());
|
86
|
+
function advance() {
|
87
|
+
var start = cursor.from(), match;
|
88
|
+
if (!(match = cursor.findNext())) {
|
89
|
+
cursor = cm.getSearchCursor(query);
|
90
|
+
if (!(match = cursor.findNext()) ||
|
91
|
+
(cursor.from().line == start.line && cursor.from().ch == start.ch)) return;
|
92
|
+
}
|
93
|
+
cm.setSelection(cursor.from(), cursor.to());
|
94
|
+
confirmDialog(cm, doReplaceConfirm, "Replace?",
|
95
|
+
[function() {doReplace(match);}, advance]);
|
96
|
+
}
|
97
|
+
function doReplace(match) {
|
98
|
+
cursor.replace(typeof query == "string" ? text :
|
99
|
+
text.replace(/\$(\d)/, function(w, i) {return match[i];}));
|
100
|
+
advance();
|
101
|
+
}
|
102
|
+
advance();
|
103
|
+
}
|
104
|
+
});
|
105
|
+
});
|
106
|
+
}
|
107
|
+
|
108
|
+
CodeMirror.commands.find = function(cm) {clearSearch(cm); doSearch(cm);};
|
109
|
+
CodeMirror.commands.findNext = doSearch;
|
110
|
+
CodeMirror.commands.findPrev = function(cm) {doSearch(cm, true);};
|
111
|
+
CodeMirror.commands.clearSearch = clearSearch;
|
112
|
+
CodeMirror.commands.replace = replace;
|
113
|
+
CodeMirror.commands.replaceAll = function(cm) {replace(cm, true);};
|
114
|
+
})();
|