sublime.js 19 KB

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