sublime.js 22 KB

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