Implement CSS `display: flow-root` (modern clearfix)
[WebKit-https.git] / Source / WebInspectorUI / UserInterface / External / CodeMirror / sublime.js
1 // CodeMirror, copyright (c) by Marijn Haverbeke and others
2 // Distributed under an MIT license: http://codemirror.net/LICENSE
3
4 // A rough approximation of Sublime Text's keybindings
5 // Depends on addon/search/searchcursor.js and optionally addon/dialog/dialogs.js
6
7 (function(mod) {
8   if (typeof exports == "object" && typeof module == "object") // CommonJS
9     mod(require("../lib/codemirror"), require("../addon/search/searchcursor"), require("../addon/edit/matchbrackets"));
10   else if (typeof define == "function" && define.amd) // AMD
11     define(["../lib/codemirror", "../addon/search/searchcursor", "../addon/edit/matchbrackets"], mod);
12   else // Plain browser env
13     mod(CodeMirror);
14 })(function(CodeMirror) {
15   "use strict";
16
17   var map = CodeMirror.keyMap.sublime = {fallthrough: "default"};
18   var cmds = CodeMirror.commands;
19   var Pos = CodeMirror.Pos;
20   var mac = CodeMirror.keyMap["default"] == CodeMirror.keyMap.macDefault;
21   var ctrl = mac ? "Cmd-" : "Ctrl-";
22
23   // This is not exactly Sublime's algorithm. I couldn't make heads or tails of that.
24   function findPosSubword(doc, start, dir) {
25     if (dir < 0 && start.ch == 0) return doc.clipPos(Pos(start.line - 1));
26     var line = doc.getLine(start.line);
27     if (dir > 0 && start.ch >= line.length) return doc.clipPos(Pos(start.line + 1, 0));
28     var state = "start", type;
29     for (var pos = start.ch, e = dir < 0 ? 0 : line.length, i = 0; pos != e; pos += dir, i++) {
30       var next = line.charAt(dir < 0 ? pos - 1 : pos);
31       var cat = next != "_" && CodeMirror.isWordChar(next) ? "w" : "o";
32       if (cat == "w" && next.toUpperCase() == next) cat = "W";
33       if (state == "start") {
34         if (cat != "o") { state = "in"; type = cat; }
35       } else if (state == "in") {
36         if (type != cat) {
37           if (type == "w" && cat == "W" && dir < 0) pos--;
38           if (type == "W" && cat == "w" && dir > 0) { type = "w"; continue; }
39           break;
40         }
41       }
42     }
43     return Pos(start.line, pos);
44   }
45
46   function moveSubword(cm, dir) {
47     cm.extendSelectionsBy(function(range) {
48       if (cm.display.shift || cm.doc.extend || range.empty())
49         return findPosSubword(cm.doc, range.head, dir);
50       else
51         return dir < 0 ? range.from() : range.to();
52     });
53   }
54
55   var goSubwordCombo = mac ? "Ctrl-" : "Alt-";
56
57   cmds[map[goSubwordCombo + "Left"] = "goSubwordLeft"] = function(cm) { moveSubword(cm, -1); };
58   cmds[map[goSubwordCombo + "Right"] = "goSubwordRight"] = function(cm) { moveSubword(cm, 1); };
59
60   if (mac) map["Cmd-Left"] = "goLineStartSmart";
61
62   var scrollLineCombo = mac ? "Ctrl-Alt-" : "Ctrl-";
63
64   cmds[map[scrollLineCombo + "Up"] = "scrollLineUp"] = function(cm) {
65     var info = cm.getScrollInfo();
66     if (!cm.somethingSelected()) {
67       var visibleBottomLine = cm.lineAtHeight(info.top + info.clientHeight, "local");
68       if (cm.getCursor().line >= visibleBottomLine)
69         cm.execCommand("goLineUp");
70     }
71     cm.scrollTo(null, info.top - cm.defaultTextHeight());
72   };
73   cmds[map[scrollLineCombo + "Down"] = "scrollLineDown"] = function(cm) {
74     var info = cm.getScrollInfo();
75     if (!cm.somethingSelected()) {
76       var visibleTopLine = cm.lineAtHeight(info.top, "local")+1;
77       if (cm.getCursor().line <= visibleTopLine)
78         cm.execCommand("goLineDown");
79     }
80     cm.scrollTo(null, info.top + cm.defaultTextHeight());
81   };
82
83   cmds[map["Shift-" + ctrl + "L"] = "splitSelectionByLine"] = function(cm) {
84     var ranges = cm.listSelections(), lineRanges = [];
85     for (var i = 0; i < ranges.length; i++) {
86       var from = ranges[i].from(), to = ranges[i].to();
87       for (var line = from.line; line <= to.line; ++line)
88         if (!(to.line > from.line && line == to.line && to.ch == 0))
89           lineRanges.push({anchor: line == from.line ? from : Pos(line, 0),
90                            head: line == to.line ? to : Pos(line)});
91     }
92     cm.setSelections(lineRanges, 0);
93   };
94
95   map["Shift-Tab"] = "indentLess";
96
97   cmds[map["Esc"] = "singleSelectionTop"] = function(cm) {
98     var range = cm.listSelections()[0];
99     cm.setSelection(range.anchor, range.head, {scroll: false});
100   };
101
102   cmds[map[ctrl + "L"] = "selectLine"] = function(cm) {
103     var ranges = cm.listSelections(), extended = [];
104     for (var i = 0; i < ranges.length; i++) {
105       var range = ranges[i];
106       extended.push({anchor: Pos(range.from().line, 0),
107                      head: Pos(range.to().line + 1, 0)});
108     }
109     cm.setSelections(extended);
110   };
111
112   map["Shift-Ctrl-K"] = "deleteLine";
113
114   function insertLine(cm, above) {
115     if (cm.isReadOnly()) return CodeMirror.Pass
116     cm.operation(function() {
117       var len = cm.listSelections().length, newSelection = [], last = -1;
118       for (var i = 0; i < len; i++) {
119         var head = cm.listSelections()[i].head;
120         if (head.line <= last) continue;
121         var at = Pos(head.line + (above ? 0 : 1), 0);
122         cm.replaceRange("\n", at, null, "+insertLine");
123         cm.indentLine(at.line, null, true);
124         newSelection.push({head: at, anchor: at});
125         last = head.line + 1;
126       }
127       cm.setSelections(newSelection);
128     });
129     cm.execCommand("indentAuto");
130   }
131
132   cmds[map[ctrl + "Enter"] = "insertLineAfter"] = function(cm) { return insertLine(cm, false); };
133
134   cmds[map["Shift-" + ctrl + "Enter"] = "insertLineBefore"] = function(cm) { return insertLine(cm, true); };
135
136   function wordAt(cm, pos) {
137     var start = pos.ch, end = start, line = cm.getLine(pos.line);
138     while (start && CodeMirror.isWordChar(line.charAt(start - 1))) --start;
139     while (end < line.length && CodeMirror.isWordChar(line.charAt(end))) ++end;
140     return {from: Pos(pos.line, start), to: Pos(pos.line, end), word: line.slice(start, end)};
141   }
142
143   cmds[map[ctrl + "D"] = "selectNextOccurrence"] = function(cm) {
144     var from = cm.getCursor("from"), to = cm.getCursor("to");
145     var fullWord = cm.state.sublimeFindFullWord == cm.doc.sel;
146     if (CodeMirror.cmpPos(from, to) == 0) {
147       var word = wordAt(cm, from);
148       if (!word.word) return;
149       cm.setSelection(word.from, word.to);
150       fullWord = true;
151     } else {
152       var text = cm.getRange(from, to);
153       var query = fullWord ? new RegExp("\\b" + text + "\\b") : text;
154       var cur = cm.getSearchCursor(query, to);
155       if (cur.findNext()) {
156         cm.addSelection(cur.from(), cur.to());
157       } else {
158         cur = cm.getSearchCursor(query, Pos(cm.firstLine(), 0));
159         if (cur.findNext())
160           cm.addSelection(cur.from(), cur.to());
161       }
162     }
163     if (fullWord)
164       cm.state.sublimeFindFullWord = cm.doc.sel;
165   };
166
167   var mirror = "(){}[]";
168   function selectBetweenBrackets(cm) {
169     var ranges = cm.listSelections(), newRanges = []
170     for (var i = 0; i < ranges.length; i++) {
171       var range = ranges[i], pos = range.head, opening = cm.scanForBracket(pos, -1);
172       if (!opening) return false;
173       for (;;) {
174         var closing = cm.scanForBracket(pos, 1);
175         if (!closing) return false;
176         if (closing.ch == mirror.charAt(mirror.indexOf(opening.ch) + 1)) {
177           newRanges.push({anchor: Pos(opening.pos.line, opening.pos.ch + 1),
178                           head: closing.pos});
179           break;
180         }
181         pos = Pos(closing.pos.line, closing.pos.ch + 1);
182       }
183     }
184     cm.setSelections(newRanges);
185     return true;
186   }
187
188   cmds[map["Shift-" + ctrl + "Space"] = "selectScope"] = function(cm) {
189     selectBetweenBrackets(cm) || cm.execCommand("selectAll");
190   };
191   cmds[map["Shift-" + ctrl + "M"] = "selectBetweenBrackets"] = function(cm) {
192     if (!selectBetweenBrackets(cm)) return CodeMirror.Pass;
193   };
194
195   cmds[map[ctrl + "M"] = "goToBracket"] = function(cm) {
196     cm.extendSelectionsBy(function(range) {
197       var next = cm.scanForBracket(range.head, 1);
198       if (next && CodeMirror.cmpPos(next.pos, range.head) != 0) return next.pos;
199       var prev = cm.scanForBracket(range.head, -1);
200       return prev && Pos(prev.pos.line, prev.pos.ch + 1) || range.head;
201     });
202   };
203
204   var swapLineCombo = mac ? "Cmd-Ctrl-" : "Shift-Ctrl-";
205
206   cmds[map[swapLineCombo + "Up"] = "swapLineUp"] = function(cm) {
207     if (cm.isReadOnly()) return CodeMirror.Pass
208     var ranges = cm.listSelections(), linesToMove = [], at = cm.firstLine() - 1, newSels = [];
209     for (var i = 0; i < ranges.length; i++) {
210       var range = ranges[i], from = range.from().line - 1, to = range.to().line;
211       newSels.push({anchor: Pos(range.anchor.line - 1, range.anchor.ch),
212                     head: Pos(range.head.line - 1, range.head.ch)});
213       if (range.to().ch == 0 && !range.empty()) --to;
214       if (from > at) linesToMove.push(from, to);
215       else if (linesToMove.length) linesToMove[linesToMove.length - 1] = to;
216       at = to;
217     }
218     cm.operation(function() {
219       for (var i = 0; i < linesToMove.length; i += 2) {
220         var from = linesToMove[i], to = linesToMove[i + 1];
221         var line = cm.getLine(from);
222         cm.replaceRange("", Pos(from, 0), Pos(from + 1, 0), "+swapLine");
223         if (to > cm.lastLine())
224           cm.replaceRange("\n" + line, Pos(cm.lastLine()), null, "+swapLine");
225         else
226           cm.replaceRange(line + "\n", Pos(to, 0), null, "+swapLine");
227       }
228       cm.setSelections(newSels);
229       cm.scrollIntoView();
230     });
231   };
232
233   cmds[map[swapLineCombo + "Down"] = "swapLineDown"] = function(cm) {
234     if (cm.isReadOnly()) return CodeMirror.Pass
235     var ranges = cm.listSelections(), linesToMove = [], at = cm.lastLine() + 1;
236     for (var i = ranges.length - 1; i >= 0; i--) {
237       var range = ranges[i], from = range.to().line + 1, to = range.from().line;
238       if (range.to().ch == 0 && !range.empty()) from--;
239       if (from < at) linesToMove.push(from, to);
240       else if (linesToMove.length) linesToMove[linesToMove.length - 1] = to;
241       at = to;
242     }
243     cm.operation(function() {
244       for (var i = linesToMove.length - 2; i >= 0; i -= 2) {
245         var from = linesToMove[i], to = linesToMove[i + 1];
246         var line = cm.getLine(from);
247         if (from == cm.lastLine())
248           cm.replaceRange("", Pos(from - 1), Pos(from), "+swapLine");
249         else
250           cm.replaceRange("", Pos(from, 0), Pos(from + 1, 0), "+swapLine");
251         cm.replaceRange(line + "\n", Pos(to, 0), null, "+swapLine");
252       }
253       cm.scrollIntoView();
254     });
255   };
256
257   cmds[map[ctrl + "/"] = "toggleCommentIndented"] = function(cm) {
258     cm.toggleComment({ indent: true });
259   }
260
261   cmds[map[ctrl + "J"] = "joinLines"] = function(cm) {
262     var ranges = cm.listSelections(), joined = [];
263     for (var i = 0; i < ranges.length; i++) {
264       var range = ranges[i], from = range.from();
265       var start = from.line, end = range.to().line;
266       while (i < ranges.length - 1 && ranges[i + 1].from().line == end)
267         end = ranges[++i].to().line;
268       joined.push({start: start, end: end, anchor: !range.empty() && from});
269     }
270     cm.operation(function() {
271       var offset = 0, ranges = [];
272       for (var i = 0; i < joined.length; i++) {
273         var obj = joined[i];
274         var anchor = obj.anchor && Pos(obj.anchor.line - offset, obj.anchor.ch), head;
275         for (var line = obj.start; line <= obj.end; line++) {
276           var actual = line - offset;
277           if (line == obj.end) head = Pos(actual, cm.getLine(actual).length + 1);
278           if (actual < cm.lastLine()) {
279             cm.replaceRange(" ", Pos(actual), Pos(actual + 1, /^\s*/.exec(cm.getLine(actual + 1))[0].length));
280             ++offset;
281           }
282         }
283         ranges.push({anchor: anchor || head, head: head});
284       }
285       cm.setSelections(ranges, 0);
286     });
287   };
288
289   cmds[map["Shift-" + ctrl + "D"] = "duplicateLine"] = function(cm) {
290     cm.operation(function() {
291       var rangeCount = cm.listSelections().length;
292       for (var i = 0; i < rangeCount; i++) {
293         var range = cm.listSelections()[i];
294         if (range.empty())
295           cm.replaceRange(cm.getLine(range.head.line) + "\n", Pos(range.head.line, 0));
296         else
297           cm.replaceRange(cm.getRange(range.from(), range.to()), range.from());
298       }
299       cm.scrollIntoView();
300     });
301   };
302
303   if (!mac) map[ctrl + "T"] = "transposeChars";
304
305   function sortLines(cm, caseSensitive) {
306     if (cm.isReadOnly()) return CodeMirror.Pass
307     var ranges = cm.listSelections(), toSort = [], selected;
308     for (var i = 0; i < ranges.length; i++) {
309       var range = ranges[i];
310       if (range.empty()) continue;
311       var from = range.from().line, to = range.to().line;
312       while (i < ranges.length - 1 && ranges[i + 1].from().line == to)
313         to = range[++i].to().line;
314       toSort.push(from, to);
315     }
316     if (toSort.length) selected = true;
317     else toSort.push(cm.firstLine(), cm.lastLine());
318
319     cm.operation(function() {
320       var ranges = [];
321       for (var i = 0; i < toSort.length; i += 2) {
322         var from = toSort[i], to = toSort[i + 1];
323         var start = Pos(from, 0), end = Pos(to);
324         var lines = cm.getRange(start, end, false);
325         if (caseSensitive)
326           lines.sort();
327         else
328           lines.sort(function(a, b) {
329             var au = a.toUpperCase(), bu = b.toUpperCase();
330             if (au != bu) { a = au; b = bu; }
331             return a < b ? -1 : a == b ? 0 : 1;
332           });
333         cm.replaceRange(lines, start, end);
334         if (selected) ranges.push({anchor: start, head: end});
335       }
336       if (selected) cm.setSelections(ranges, 0);
337     });
338   }
339
340   cmds[map["F9"] = "sortLines"] = function(cm) { sortLines(cm, true); };
341   cmds[map[ctrl + "F9"] = "sortLinesInsensitive"] = function(cm) { sortLines(cm, false); };
342
343   cmds[map["F2"] = "nextBookmark"] = function(cm) {
344     var marks = cm.state.sublimeBookmarks;
345     if (marks) while (marks.length) {
346       var current = marks.shift();
347       var found = current.find();
348       if (found) {
349         marks.push(current);
350         return cm.setSelection(found.from, found.to);
351       }
352     }
353   };
354
355   cmds[map["Shift-F2"] = "prevBookmark"] = function(cm) {
356     var marks = cm.state.sublimeBookmarks;
357     if (marks) while (marks.length) {
358       marks.unshift(marks.pop());
359       var found = marks[marks.length - 1].find();
360       if (!found)
361         marks.pop();
362       else
363         return cm.setSelection(found.from, found.to);
364     }
365   };
366
367   cmds[map[ctrl + "F2"] = "toggleBookmark"] = function(cm) {
368     var ranges = cm.listSelections();
369     var marks = cm.state.sublimeBookmarks || (cm.state.sublimeBookmarks = []);
370     for (var i = 0; i < ranges.length; i++) {
371       var from = ranges[i].from(), to = ranges[i].to();
372       var found = cm.findMarks(from, to);
373       for (var j = 0; j < found.length; j++) {
374         if (found[j].sublimeBookmark) {
375           found[j].clear();
376           for (var k = 0; k < marks.length; k++)
377             if (marks[k] == found[j])
378               marks.splice(k--, 1);
379           break;
380         }
381       }
382       if (j == found.length)
383         marks.push(cm.markText(from, to, {sublimeBookmark: true, clearWhenEmpty: false}));
384     }
385   };
386
387   cmds[map["Shift-" + ctrl + "F2"] = "clearBookmarks"] = function(cm) {
388     var marks = cm.state.sublimeBookmarks;
389     if (marks) for (var i = 0; i < marks.length; i++) marks[i].clear();
390     marks.length = 0;
391   };
392
393   cmds[map["Alt-F2"] = "selectBookmarks"] = function(cm) {
394     var marks = cm.state.sublimeBookmarks, ranges = [];
395     if (marks) for (var i = 0; i < marks.length; i++) {
396       var found = marks[i].find();
397       if (!found)
398         marks.splice(i--, 0);
399       else
400         ranges.push({anchor: found.from, head: found.to});
401     }
402     if (ranges.length)
403       cm.setSelections(ranges, 0);
404   };
405
406   map["Alt-Q"] = "wrapLines";
407
408   var cK = ctrl + "K ";
409
410   function modifyWordOrSelection(cm, mod) {
411     cm.operation(function() {
412       var ranges = cm.listSelections(), indices = [], replacements = [];
413       for (var i = 0; i < ranges.length; i++) {
414         var range = ranges[i];
415         if (range.empty()) { indices.push(i); replacements.push(""); }
416         else replacements.push(mod(cm.getRange(range.from(), range.to())));
417       }
418       cm.replaceSelections(replacements, "around", "case");
419       for (var i = indices.length - 1, at; i >= 0; i--) {
420         var range = ranges[indices[i]];
421         if (at && CodeMirror.cmpPos(range.head, at) > 0) continue;
422         var word = wordAt(cm, range.head);
423         at = word.from;
424         cm.replaceRange(mod(word.word), word.from, word.to);
425       }
426     });
427   }
428
429   map[cK + ctrl + "Backspace"] = "delLineLeft";
430
431   cmds[map["Backspace"] = "smartBackspace"] = function(cm) {
432     if (cm.somethingSelected()) return CodeMirror.Pass;
433
434     cm.operation(function() {
435       var cursors = cm.listSelections();
436       var indentUnit = cm.getOption("indentUnit");
437
438       for (var i = cursors.length - 1; i >= 0; i--) {
439         var cursor = cursors[i].head;
440         var toStartOfLine = cm.getRange({line: cursor.line, ch: 0}, cursor);
441         var column = CodeMirror.countColumn(toStartOfLine, null, cm.getOption("tabSize"));
442
443         // Delete by one character by default
444         var deletePos = cm.findPosH(cursor, -1, "char", false);
445
446         if (toStartOfLine && !/\S/.test(toStartOfLine) && column % indentUnit == 0) {
447           var prevIndent = new Pos(cursor.line,
448             CodeMirror.findColumn(toStartOfLine, column - indentUnit, indentUnit));
449
450           // Smart delete only if we found a valid prevIndent location
451           if (prevIndent.ch != cursor.ch) deletePos = prevIndent;
452         }
453
454         cm.replaceRange("", deletePos, cursor, "+delete");
455       }
456     });
457   };
458
459   cmds[map[cK + ctrl + "K"] = "delLineRight"] = function(cm) {
460     cm.operation(function() {
461       var ranges = cm.listSelections();
462       for (var i = ranges.length - 1; i >= 0; i--)
463         cm.replaceRange("", ranges[i].anchor, Pos(ranges[i].to().line), "+delete");
464       cm.scrollIntoView();
465     });
466   };
467
468   cmds[map[cK + ctrl + "U"] = "upcaseAtCursor"] = function(cm) {
469     modifyWordOrSelection(cm, function(str) { return str.toUpperCase(); });
470   };
471   cmds[map[cK + ctrl + "L"] = "downcaseAtCursor"] = function(cm) {
472     modifyWordOrSelection(cm, function(str) { return str.toLowerCase(); });
473   };
474
475   cmds[map[cK + ctrl + "Space"] = "setSublimeMark"] = function(cm) {
476     if (cm.state.sublimeMark) cm.state.sublimeMark.clear();
477     cm.state.sublimeMark = cm.setBookmark(cm.getCursor());
478   };
479   cmds[map[cK + ctrl + "A"] = "selectToSublimeMark"] = function(cm) {
480     var found = cm.state.sublimeMark && cm.state.sublimeMark.find();
481     if (found) cm.setSelection(cm.getCursor(), found);
482   };
483   cmds[map[cK + ctrl + "W"] = "deleteToSublimeMark"] = function(cm) {
484     var found = cm.state.sublimeMark && cm.state.sublimeMark.find();
485     if (found) {
486       var from = cm.getCursor(), to = found;
487       if (CodeMirror.cmpPos(from, to) > 0) { var tmp = to; to = from; from = tmp; }
488       cm.state.sublimeKilled = cm.getRange(from, to);
489       cm.replaceRange("", from, to);
490     }
491   };
492   cmds[map[cK + ctrl + "X"] = "swapWithSublimeMark"] = function(cm) {
493     var found = cm.state.sublimeMark && cm.state.sublimeMark.find();
494     if (found) {
495       cm.state.sublimeMark.clear();
496       cm.state.sublimeMark = cm.setBookmark(cm.getCursor());
497       cm.setCursor(found);
498     }
499   };
500   cmds[map[cK + ctrl + "Y"] = "sublimeYank"] = function(cm) {
501     if (cm.state.sublimeKilled != null)
502       cm.replaceSelection(cm.state.sublimeKilled, null, "paste");
503   };
504
505   map[cK + ctrl + "G"] = "clearBookmarks";
506   cmds[map[cK + ctrl + "C"] = "showInCenter"] = function(cm) {
507     var pos = cm.cursorCoords(null, "local");
508     cm.scrollTo(null, (pos.top + pos.bottom) / 2 - cm.getScrollInfo().clientHeight / 2);
509   };
510
511   var selectLinesCombo = mac ? "Ctrl-Shift-" : "Ctrl-Alt-";
512   cmds[map[selectLinesCombo + "Up"] = "selectLinesUpward"] = function(cm) {
513     cm.operation(function() {
514       var ranges = cm.listSelections();
515       for (var i = 0; i < ranges.length; i++) {
516         var range = ranges[i];
517         if (range.head.line > cm.firstLine())
518           cm.addSelection(Pos(range.head.line - 1, range.head.ch));
519       }
520     });
521   };
522   cmds[map[selectLinesCombo + "Down"] = "selectLinesDownward"] = function(cm) {
523     cm.operation(function() {
524       var ranges = cm.listSelections();
525       for (var i = 0; i < ranges.length; i++) {
526         var range = ranges[i];
527         if (range.head.line < cm.lastLine())
528           cm.addSelection(Pos(range.head.line + 1, range.head.ch));
529       }
530     });
531   };
532
533   function getTarget(cm) {
534     var from = cm.getCursor("from"), to = cm.getCursor("to");
535     if (CodeMirror.cmpPos(from, to) == 0) {
536       var word = wordAt(cm, from);
537       if (!word.word) return;
538       from = word.from;
539       to = word.to;
540     }
541     return {from: from, to: to, query: cm.getRange(from, to), word: word};
542   }
543
544   function findAndGoTo(cm, forward) {
545     var target = getTarget(cm);
546     if (!target) return;
547     var query = target.query;
548     var cur = cm.getSearchCursor(query, forward ? target.to : target.from);
549
550     if (forward ? cur.findNext() : cur.findPrevious()) {
551       cm.setSelection(cur.from(), cur.to());
552     } else {
553       cur = cm.getSearchCursor(query, forward ? Pos(cm.firstLine(), 0)
554                                               : cm.clipPos(Pos(cm.lastLine())));
555       if (forward ? cur.findNext() : cur.findPrevious())
556         cm.setSelection(cur.from(), cur.to());
557       else if (target.word)
558         cm.setSelection(target.from, target.to);
559     }
560   };
561   cmds[map[ctrl + "F3"] = "findUnder"] = function(cm) { findAndGoTo(cm, true); };
562   cmds[map["Shift-" + ctrl + "F3"] = "findUnderPrevious"] = function(cm) { findAndGoTo(cm,false); };
563   cmds[map["Alt-F3"] = "findAllUnder"] = function(cm) {
564     var target = getTarget(cm);
565     if (!target) return;
566     var cur = cm.getSearchCursor(target.query);
567     var matches = [];
568     var primaryIndex = -1;
569     while (cur.findNext()) {
570       matches.push({anchor: cur.from(), head: cur.to()});
571       if (cur.from().line <= target.from.line && cur.from().ch <= target.from.ch)
572         primaryIndex++;
573     }
574     cm.setSelections(matches, primaryIndex);
575   };
576
577   map["Shift-" + ctrl + "["] = "fold";
578   map["Shift-" + ctrl + "]"] = "unfold";
579   map[cK + ctrl + "0"] = map[cK + ctrl + "J"] = "unfoldAll";
580
581   map[ctrl + "I"] = "findIncremental";
582   map["Shift-" + ctrl + "I"] = "findIncrementalReverse";
583   map[ctrl + "H"] = "replace";
584   map["F3"] = "findNext";
585   map["Shift-F3"] = "findPrev";
586
587   CodeMirror.normalizeKeyMap(map);
588 });