sublime.js 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704
  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 cmds = CodeMirror.commands;
  15. var Pos = CodeMirror.Pos;
  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.goSubwordLeft = function(cm) { moveSubword(cm, -1); };
  47. cmds.goSubwordRight = function(cm) { moveSubword(cm, 1); };
  48. cmds.scrollLineUp = function(cm) {
  49. var info = cm.getScrollInfo();
  50. if (!cm.somethingSelected()) {
  51. var visibleBottomLine = cm.lineAtHeight(info.top + info.clientHeight, "local");
  52. if (cm.getCursor().line >= visibleBottomLine)
  53. cm.execCommand("goLineUp");
  54. }
  55. cm.scrollTo(null, info.top - cm.defaultTextHeight());
  56. };
  57. cmds.scrollLineDown = function(cm) {
  58. var info = cm.getScrollInfo();
  59. if (!cm.somethingSelected()) {
  60. var visibleTopLine = cm.lineAtHeight(info.top, "local")+1;
  61. if (cm.getCursor().line <= visibleTopLine)
  62. cm.execCommand("goLineDown");
  63. }
  64. cm.scrollTo(null, info.top + cm.defaultTextHeight());
  65. };
  66. cmds.splitSelectionByLine = function(cm) {
  67. var ranges = cm.listSelections(), lineRanges = [];
  68. for (var i = 0; i < ranges.length; i++) {
  69. var from = ranges[i].from(), to = ranges[i].to();
  70. for (var line = from.line; line <= to.line; ++line)
  71. if (!(to.line > from.line && line == to.line && to.ch == 0))
  72. lineRanges.push({anchor: line == from.line ? from : Pos(line, 0),
  73. head: line == to.line ? to : Pos(line)});
  74. }
  75. cm.setSelections(lineRanges, 0);
  76. };
  77. cmds.singleSelectionTop = function(cm) {
  78. var range = cm.listSelections()[0];
  79. cm.setSelection(range.anchor, range.head, {scroll: false});
  80. };
  81. cmds.selectLine = function(cm) {
  82. var ranges = cm.listSelections(), extended = [];
  83. for (var i = 0; i < ranges.length; i++) {
  84. var range = ranges[i];
  85. extended.push({anchor: Pos(range.from().line, 0),
  86. head: Pos(range.to().line + 1, 0)});
  87. }
  88. cm.setSelections(extended);
  89. };
  90. function insertLine(cm, above) {
  91. if (cm.isReadOnly()) return CodeMirror.Pass
  92. cm.operation(function() {
  93. var len = cm.listSelections().length, newSelection = [], last = -1;
  94. for (var i = 0; i < len; i++) {
  95. var head = cm.listSelections()[i].head;
  96. if (head.line <= last) continue;
  97. var at = Pos(head.line + (above ? 0 : 1), 0);
  98. cm.replaceRange("\n", at, null, "+insertLine");
  99. cm.indentLine(at.line, null, true);
  100. newSelection.push({head: at, anchor: at});
  101. last = head.line + 1;
  102. }
  103. cm.setSelections(newSelection);
  104. });
  105. cm.execCommand("indentAuto");
  106. }
  107. cmds.insertLineAfter = function(cm) { return insertLine(cm, false); };
  108. cmds.insertLineBefore = function(cm) { return insertLine(cm, true); };
  109. function wordAt(cm, pos) {
  110. var start = pos.ch, end = start, line = cm.getLine(pos.line);
  111. while (start && CodeMirror.isWordChar(line.charAt(start - 1))) --start;
  112. while (end < line.length && CodeMirror.isWordChar(line.charAt(end))) ++end;
  113. return {from: Pos(pos.line, start), to: Pos(pos.line, end), word: line.slice(start, end)};
  114. }
  115. cmds.selectNextOccurrence = function(cm) {
  116. var from = cm.getCursor("from"), to = cm.getCursor("to");
  117. var fullWord = cm.state.sublimeFindFullWord == cm.doc.sel;
  118. if (CodeMirror.cmpPos(from, to) == 0) {
  119. var word = wordAt(cm, from);
  120. if (!word.word) return;
  121. cm.setSelection(word.from, word.to);
  122. fullWord = true;
  123. } else {
  124. var text = cm.getRange(from, to);
  125. var query = fullWord ? new RegExp("\\b" + text + "\\b") : text;
  126. var cur = cm.getSearchCursor(query, to);
  127. var found = cur.findNext();
  128. if (!found) {
  129. cur = cm.getSearchCursor(query, Pos(cm.firstLine(), 0));
  130. found = cur.findNext();
  131. }
  132. if (!found || isSelectedRange(cm.listSelections(), cur.from(), cur.to()))
  133. return CodeMirror.Pass
  134. cm.addSelection(cur.from(), cur.to());
  135. }
  136. if (fullWord)
  137. cm.state.sublimeFindFullWord = cm.doc.sel;
  138. };
  139. function addCursorToSelection(cm, dir) {
  140. var ranges = cm.listSelections(), newRanges = [];
  141. for (var i = 0; i < ranges.length; i++) {
  142. var range = ranges[i];
  143. var newAnchor = cm.findPosV(range.anchor, dir, "line");
  144. var newHead = cm.findPosV(range.head, dir, "line");
  145. var newRange = {anchor: newAnchor, head: newHead};
  146. newRanges.push(range);
  147. newRanges.push(newRange);
  148. }
  149. cm.setSelections(newRanges);
  150. }
  151. cmds.addCursorToPrevLine = function(cm) { addCursorToSelection(cm, -1); };
  152. cmds.addCursorToNextLine = function(cm) { addCursorToSelection(cm, 1); };
  153. function isSelectedRange(ranges, from, to) {
  154. for (var i = 0; i < ranges.length; i++)
  155. if (ranges[i].from() == from && ranges[i].to() == to) return true
  156. return false
  157. }
  158. var mirror = "(){}[]";
  159. function selectBetweenBrackets(cm) {
  160. var ranges = cm.listSelections(), newRanges = []
  161. for (var i = 0; i < ranges.length; i++) {
  162. var range = ranges[i], pos = range.head, opening = cm.scanForBracket(pos, -1);
  163. if (!opening) return false;
  164. for (;;) {
  165. var closing = cm.scanForBracket(pos, 1);
  166. if (!closing) return false;
  167. if (closing.ch == mirror.charAt(mirror.indexOf(opening.ch) + 1)) {
  168. newRanges.push({anchor: Pos(opening.pos.line, opening.pos.ch + 1),
  169. head: closing.pos});
  170. break;
  171. }
  172. pos = Pos(closing.pos.line, closing.pos.ch + 1);
  173. }
  174. }
  175. cm.setSelections(newRanges);
  176. return true;
  177. }
  178. cmds.selectScope = function(cm) {
  179. selectBetweenBrackets(cm) || cm.execCommand("selectAll");
  180. };
  181. cmds.selectBetweenBrackets = function(cm) {
  182. if (!selectBetweenBrackets(cm)) return CodeMirror.Pass;
  183. };
  184. cmds.goToBracket = function(cm) {
  185. cm.extendSelectionsBy(function(range) {
  186. var next = cm.scanForBracket(range.head, 1);
  187. if (next && CodeMirror.cmpPos(next.pos, range.head) != 0) return next.pos;
  188. var prev = cm.scanForBracket(range.head, -1);
  189. return prev && Pos(prev.pos.line, prev.pos.ch + 1) || range.head;
  190. });
  191. };
  192. cmds.swapLineUp = function(cm) {
  193. if (cm.isReadOnly()) return CodeMirror.Pass
  194. var ranges = cm.listSelections(), linesToMove = [], at = cm.firstLine() - 1, newSels = [];
  195. for (var i = 0; i < ranges.length; i++) {
  196. var range = ranges[i], from = range.from().line - 1, to = range.to().line;
  197. newSels.push({anchor: Pos(range.anchor.line - 1, range.anchor.ch),
  198. head: Pos(range.head.line - 1, range.head.ch)});
  199. if (range.to().ch == 0 && !range.empty()) --to;
  200. if (from > at) linesToMove.push(from, to);
  201. else if (linesToMove.length) linesToMove[linesToMove.length - 1] = to;
  202. at = to;
  203. }
  204. cm.operation(function() {
  205. for (var i = 0; i < linesToMove.length; i += 2) {
  206. var from = linesToMove[i], to = linesToMove[i + 1];
  207. var line = cm.getLine(from);
  208. cm.replaceRange("", Pos(from, 0), Pos(from + 1, 0), "+swapLine");
  209. if (to > cm.lastLine())
  210. cm.replaceRange("\n" + line, Pos(cm.lastLine()), null, "+swapLine");
  211. else
  212. cm.replaceRange(line + "\n", Pos(to, 0), null, "+swapLine");
  213. }
  214. cm.setSelections(newSels);
  215. cm.scrollIntoView();
  216. });
  217. };
  218. cmds.swapLineDown = function(cm) {
  219. if (cm.isReadOnly()) return CodeMirror.Pass
  220. var ranges = cm.listSelections(), linesToMove = [], at = cm.lastLine() + 1;
  221. for (var i = ranges.length - 1; i >= 0; i--) {
  222. var range = ranges[i], from = range.to().line + 1, to = range.from().line;
  223. if (range.to().ch == 0 && !range.empty()) from--;
  224. if (from < at) linesToMove.push(from, to);
  225. else if (linesToMove.length) linesToMove[linesToMove.length - 1] = to;
  226. at = to;
  227. }
  228. cm.operation(function() {
  229. for (var i = linesToMove.length - 2; i >= 0; i -= 2) {
  230. var from = linesToMove[i], to = linesToMove[i + 1];
  231. var line = cm.getLine(from);
  232. if (from == cm.lastLine())
  233. cm.replaceRange("", Pos(from - 1), Pos(from), "+swapLine");
  234. else
  235. cm.replaceRange("", Pos(from, 0), Pos(from + 1, 0), "+swapLine");
  236. cm.replaceRange(line + "\n", Pos(to, 0), null, "+swapLine");
  237. }
  238. cm.scrollIntoView();
  239. });
  240. };
  241. cmds.toggleCommentIndented = function(cm) {
  242. cm.toggleComment({ indent: true });
  243. }
  244. cmds.joinLines = function(cm) {
  245. var ranges = cm.listSelections(), joined = [];
  246. for (var i = 0; i < ranges.length; i++) {
  247. var range = ranges[i], from = range.from();
  248. var start = from.line, end = range.to().line;
  249. while (i < ranges.length - 1 && ranges[i + 1].from().line == end)
  250. end = ranges[++i].to().line;
  251. joined.push({start: start, end: end, anchor: !range.empty() && from});
  252. }
  253. cm.operation(function() {
  254. var offset = 0, ranges = [];
  255. for (var i = 0; i < joined.length; i++) {
  256. var obj = joined[i];
  257. var anchor = obj.anchor && Pos(obj.anchor.line - offset, obj.anchor.ch), head;
  258. for (var line = obj.start; line <= obj.end; line++) {
  259. var actual = line - offset;
  260. if (line == obj.end) head = Pos(actual, cm.getLine(actual).length + 1);
  261. if (actual < cm.lastLine()) {
  262. cm.replaceRange(" ", Pos(actual), Pos(actual + 1, /^\s*/.exec(cm.getLine(actual + 1))[0].length));
  263. ++offset;
  264. }
  265. }
  266. ranges.push({anchor: anchor || head, head: head});
  267. }
  268. cm.setSelections(ranges, 0);
  269. });
  270. };
  271. cmds.duplicateLine = function(cm) {
  272. cm.operation(function() {
  273. var rangeCount = cm.listSelections().length;
  274. for (var i = 0; i < rangeCount; i++) {
  275. var range = cm.listSelections()[i];
  276. if (range.empty())
  277. cm.replaceRange(cm.getLine(range.head.line) + "\n", Pos(range.head.line, 0));
  278. else
  279. cm.replaceRange(cm.getRange(range.from(), range.to()), range.from());
  280. }
  281. cm.scrollIntoView();
  282. });
  283. };
  284. function sortLines(cm, caseSensitive) {
  285. if (cm.isReadOnly()) return CodeMirror.Pass
  286. var ranges = cm.listSelections(), toSort = [], selected;
  287. for (var i = 0; i < ranges.length; i++) {
  288. var range = ranges[i];
  289. if (range.empty()) continue;
  290. var from = range.from().line, to = range.to().line;
  291. while (i < ranges.length - 1 && ranges[i + 1].from().line == to)
  292. to = ranges[++i].to().line;
  293. if (!ranges[i].to().ch) to--;
  294. toSort.push(from, to);
  295. }
  296. if (toSort.length) selected = true;
  297. else toSort.push(cm.firstLine(), cm.lastLine());
  298. cm.operation(function() {
  299. var ranges = [];
  300. for (var i = 0; i < toSort.length; i += 2) {
  301. var from = toSort[i], to = toSort[i + 1];
  302. var start = Pos(from, 0), end = Pos(to);
  303. var lines = cm.getRange(start, end, false);
  304. if (caseSensitive)
  305. lines.sort();
  306. else
  307. lines.sort(function(a, b) {
  308. var au = a.toUpperCase(), bu = b.toUpperCase();
  309. if (au != bu) { a = au; b = bu; }
  310. return a < b ? -1 : a == b ? 0 : 1;
  311. });
  312. cm.replaceRange(lines, start, end);
  313. if (selected) ranges.push({anchor: start, head: Pos(to + 1, 0)});
  314. }
  315. if (selected) cm.setSelections(ranges, 0);
  316. });
  317. }
  318. cmds.sortLines = function(cm) { sortLines(cm, true); };
  319. cmds.sortLinesInsensitive = function(cm) { sortLines(cm, false); };
  320. cmds.nextBookmark = function(cm) {
  321. var marks = cm.state.sublimeBookmarks;
  322. if (marks) while (marks.length) {
  323. var current = marks.shift();
  324. var found = current.find();
  325. if (found) {
  326. marks.push(current);
  327. return cm.setSelection(found.from, found.to);
  328. }
  329. }
  330. };
  331. cmds.prevBookmark = function(cm) {
  332. var marks = cm.state.sublimeBookmarks;
  333. if (marks) while (marks.length) {
  334. marks.unshift(marks.pop());
  335. var found = marks[marks.length - 1].find();
  336. if (!found)
  337. marks.pop();
  338. else
  339. return cm.setSelection(found.from, found.to);
  340. }
  341. };
  342. cmds.toggleBookmark = function(cm) {
  343. var ranges = cm.listSelections();
  344. var marks = cm.state.sublimeBookmarks || (cm.state.sublimeBookmarks = []);
  345. for (var i = 0; i < ranges.length; i++) {
  346. var from = ranges[i].from(), to = ranges[i].to();
  347. var found = cm.findMarks(from, to);
  348. for (var j = 0; j < found.length; j++) {
  349. if (found[j].sublimeBookmark) {
  350. found[j].clear();
  351. for (var k = 0; k < marks.length; k++)
  352. if (marks[k] == found[j])
  353. marks.splice(k--, 1);
  354. break;
  355. }
  356. }
  357. if (j == found.length)
  358. marks.push(cm.markText(from, to, {sublimeBookmark: true, clearWhenEmpty: false}));
  359. }
  360. };
  361. cmds.clearBookmarks = function(cm) {
  362. var marks = cm.state.sublimeBookmarks;
  363. if (marks) for (var i = 0; i < marks.length; i++) marks[i].clear();
  364. marks.length = 0;
  365. };
  366. cmds.selectBookmarks = function(cm) {
  367. var marks = cm.state.sublimeBookmarks, ranges = [];
  368. if (marks) for (var i = 0; i < marks.length; i++) {
  369. var found = marks[i].find();
  370. if (!found)
  371. marks.splice(i--, 0);
  372. else
  373. ranges.push({anchor: found.from, head: found.to});
  374. }
  375. if (ranges.length)
  376. cm.setSelections(ranges, 0);
  377. };
  378. function modifyWordOrSelection(cm, mod) {
  379. cm.operation(function() {
  380. var ranges = cm.listSelections(), indices = [], replacements = [];
  381. for (var i = 0; i < ranges.length; i++) {
  382. var range = ranges[i];
  383. if (range.empty()) { indices.push(i); replacements.push(""); }
  384. else replacements.push(mod(cm.getRange(range.from(), range.to())));
  385. }
  386. cm.replaceSelections(replacements, "around", "case");
  387. for (var i = indices.length - 1, at; i >= 0; i--) {
  388. var range = ranges[indices[i]];
  389. if (at && CodeMirror.cmpPos(range.head, at) > 0) continue;
  390. var word = wordAt(cm, range.head);
  391. at = word.from;
  392. cm.replaceRange(mod(word.word), word.from, word.to);
  393. }
  394. });
  395. }
  396. cmds.smartBackspace = function(cm) {
  397. if (cm.somethingSelected()) return CodeMirror.Pass;
  398. cm.operation(function() {
  399. var cursors = cm.listSelections();
  400. var indentUnit = cm.getOption("indentUnit");
  401. for (var i = cursors.length - 1; i >= 0; i--) {
  402. var cursor = cursors[i].head;
  403. var toStartOfLine = cm.getRange({line: cursor.line, ch: 0}, cursor);
  404. var column = CodeMirror.countColumn(toStartOfLine, null, cm.getOption("tabSize"));
  405. // Delete by one character by default
  406. var deletePos = cm.findPosH(cursor, -1, "char", false);
  407. if (toStartOfLine && !/\S/.test(toStartOfLine) && column % indentUnit == 0) {
  408. var prevIndent = new Pos(cursor.line,
  409. CodeMirror.findColumn(toStartOfLine, column - indentUnit, indentUnit));
  410. // Smart delete only if we found a valid prevIndent location
  411. if (prevIndent.ch != cursor.ch) deletePos = prevIndent;
  412. }
  413. cm.replaceRange("", deletePos, cursor, "+delete");
  414. }
  415. });
  416. };
  417. cmds.delLineRight = function(cm) {
  418. cm.operation(function() {
  419. var ranges = cm.listSelections();
  420. for (var i = ranges.length - 1; i >= 0; i--)
  421. cm.replaceRange("", ranges[i].anchor, Pos(ranges[i].to().line), "+delete");
  422. cm.scrollIntoView();
  423. });
  424. };
  425. cmds.upcaseAtCursor = function(cm) {
  426. modifyWordOrSelection(cm, function(str) { return str.toUpperCase(); });
  427. };
  428. cmds.downcaseAtCursor = function(cm) {
  429. modifyWordOrSelection(cm, function(str) { return str.toLowerCase(); });
  430. };
  431. cmds.setSublimeMark = function(cm) {
  432. if (cm.state.sublimeMark) cm.state.sublimeMark.clear();
  433. cm.state.sublimeMark = cm.setBookmark(cm.getCursor());
  434. };
  435. cmds.selectToSublimeMark = function(cm) {
  436. var found = cm.state.sublimeMark && cm.state.sublimeMark.find();
  437. if (found) cm.setSelection(cm.getCursor(), found);
  438. };
  439. cmds.deleteToSublimeMark = function(cm) {
  440. var found = cm.state.sublimeMark && cm.state.sublimeMark.find();
  441. if (found) {
  442. var from = cm.getCursor(), to = found;
  443. if (CodeMirror.cmpPos(from, to) > 0) { var tmp = to; to = from; from = tmp; }
  444. cm.state.sublimeKilled = cm.getRange(from, to);
  445. cm.replaceRange("", from, to);
  446. }
  447. };
  448. cmds.swapWithSublimeMark = function(cm) {
  449. var found = cm.state.sublimeMark && cm.state.sublimeMark.find();
  450. if (found) {
  451. cm.state.sublimeMark.clear();
  452. cm.state.sublimeMark = cm.setBookmark(cm.getCursor());
  453. cm.setCursor(found);
  454. }
  455. };
  456. cmds.sublimeYank = function(cm) {
  457. if (cm.state.sublimeKilled != null)
  458. cm.replaceSelection(cm.state.sublimeKilled, null, "paste");
  459. };
  460. cmds.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. cmds.selectLinesUpward = function(cm) {
  465. cm.operation(function() {
  466. var ranges = cm.listSelections();
  467. for (var i = 0; i < ranges.length; i++) {
  468. var range = ranges[i];
  469. if (range.head.line > cm.firstLine())
  470. cm.addSelection(Pos(range.head.line - 1, range.head.ch));
  471. }
  472. });
  473. };
  474. cmds.selectLinesDownward = function(cm) {
  475. cm.operation(function() {
  476. var ranges = cm.listSelections();
  477. for (var i = 0; i < ranges.length; i++) {
  478. var range = ranges[i];
  479. if (range.head.line < cm.lastLine())
  480. cm.addSelection(Pos(range.head.line + 1, range.head.ch));
  481. }
  482. });
  483. };
  484. function getTarget(cm) {
  485. var from = cm.getCursor("from"), to = cm.getCursor("to");
  486. if (CodeMirror.cmpPos(from, to) == 0) {
  487. var word = wordAt(cm, from);
  488. if (!word.word) return;
  489. from = word.from;
  490. to = word.to;
  491. }
  492. return {from: from, to: to, query: cm.getRange(from, to), word: word};
  493. }
  494. function findAndGoTo(cm, forward) {
  495. var target = getTarget(cm);
  496. if (!target) return;
  497. var query = target.query;
  498. var cur = cm.getSearchCursor(query, forward ? target.to : target.from);
  499. if (forward ? cur.findNext() : cur.findPrevious()) {
  500. cm.setSelection(cur.from(), cur.to());
  501. } else {
  502. cur = cm.getSearchCursor(query, forward ? Pos(cm.firstLine(), 0)
  503. : cm.clipPos(Pos(cm.lastLine())));
  504. if (forward ? cur.findNext() : cur.findPrevious())
  505. cm.setSelection(cur.from(), cur.to());
  506. else if (target.word)
  507. cm.setSelection(target.from, target.to);
  508. }
  509. };
  510. cmds.findUnder = function(cm) { findAndGoTo(cm, true); };
  511. cmds.findUnderPrevious = function(cm) { findAndGoTo(cm,false); };
  512. cmds.findAllUnder = function(cm) {
  513. var target = getTarget(cm);
  514. if (!target) return;
  515. var cur = cm.getSearchCursor(target.query);
  516. var matches = [];
  517. var primaryIndex = -1;
  518. while (cur.findNext()) {
  519. matches.push({anchor: cur.from(), head: cur.to()});
  520. if (cur.from().line <= target.from.line && cur.from().ch <= target.from.ch)
  521. primaryIndex++;
  522. }
  523. cm.setSelections(matches, primaryIndex);
  524. };
  525. var keyMap = CodeMirror.keyMap;
  526. keyMap.macSublime = {
  527. "Cmd-Left": "goLineStartSmart",
  528. "Shift-Tab": "indentLess",
  529. "Shift-Ctrl-K": "deleteLine",
  530. "Alt-Q": "wrapLines",
  531. "Ctrl-Left": "goSubwordLeft",
  532. "Ctrl-Right": "goSubwordRight",
  533. "Ctrl-Alt-Up": "scrollLineUp",
  534. "Ctrl-Alt-Down": "scrollLineDown",
  535. "Cmd-L": "selectLine",
  536. "Shift-Cmd-L": "splitSelectionByLine",
  537. "Esc": "singleSelectionTop",
  538. "Cmd-Enter": "insertLineAfter",
  539. "Shift-Cmd-Enter": "insertLineBefore",
  540. "Cmd-D": "selectNextOccurrence",
  541. "Shift-Cmd-Up": "addCursorToPrevLine",
  542. "Shift-Cmd-Down": "addCursorToNextLine",
  543. "Shift-Cmd-Space": "selectScope",
  544. "Shift-Cmd-M": "selectBetweenBrackets",
  545. "Cmd-M": "goToBracket",
  546. "Cmd-Ctrl-Up": "swapLineUp",
  547. "Cmd-Ctrl-Down": "swapLineDown",
  548. "Cmd-/": "toggleCommentIndented",
  549. "Cmd-J": "joinLines",
  550. "Shift-Cmd-D": "duplicateLine",
  551. "F9": "sortLines",
  552. "Cmd-F9": "sortLinesInsensitive",
  553. "F2": "nextBookmark",
  554. "Shift-F2": "prevBookmark",
  555. "Cmd-F2": "toggleBookmark",
  556. "Shift-Cmd-F2": "clearBookmarks",
  557. "Alt-F2": "selectBookmarks",
  558. "Backspace": "smartBackspace",
  559. "Cmd-K Cmd-K": "delLineRight",
  560. "Cmd-K Cmd-U": "upcaseAtCursor",
  561. "Cmd-K Cmd-L": "downcaseAtCursor",
  562. "Cmd-K Cmd-Space": "setSublimeMark",
  563. "Cmd-K Cmd-A": "selectToSublimeMark",
  564. "Cmd-K Cmd-W": "deleteToSublimeMark",
  565. "Cmd-K Cmd-X": "swapWithSublimeMark",
  566. "Cmd-K Cmd-Y": "sublimeYank",
  567. "Cmd-K Cmd-C": "showInCenter",
  568. "Cmd-K Cmd-G": "clearBookmarks",
  569. "Cmd-K Cmd-Backspace": "delLineLeft",
  570. "Cmd-K Cmd-0": "unfoldAll",
  571. "Cmd-K Cmd-J": "unfoldAll",
  572. "Ctrl-Shift-Up": "selectLinesUpward",
  573. "Ctrl-Shift-Down": "selectLinesDownward",
  574. "Cmd-F3": "findUnder",
  575. "Shift-Cmd-F3": "findUnderPrevious",
  576. "Alt-F3": "findAllUnder",
  577. "Shift-Cmd-[": "fold",
  578. "Shift-Cmd-]": "unfold",
  579. "Cmd-I": "findIncremental",
  580. "Shift-Cmd-I": "findIncrementalReverse",
  581. "Cmd-H": "replace",
  582. "F3": "findNext",
  583. "Shift-F3": "findPrev",
  584. "fallthrough": "macDefault"
  585. };
  586. CodeMirror.normalizeKeyMap(keyMap.macSublime);
  587. keyMap.pcSublime = {
  588. "Shift-Tab": "indentLess",
  589. "Shift-Ctrl-K": "deleteLine",
  590. "Alt-Q": "wrapLines",
  591. "Ctrl-T": "transposeChars",
  592. "Alt-Left": "goSubwordLeft",
  593. "Alt-Right": "goSubwordRight",
  594. "Ctrl-Up": "scrollLineUp",
  595. "Ctrl-Down": "scrollLineDown",
  596. "Ctrl-L": "selectLine",
  597. "Shift-Ctrl-L": "splitSelectionByLine",
  598. "Esc": "singleSelectionTop",
  599. "Ctrl-Enter": "insertLineAfter",
  600. "Shift-Ctrl-Enter": "insertLineBefore",
  601. "Ctrl-D": "selectNextOccurrence",
  602. "Alt-CtrlUp": "addCursorToPrevLine",
  603. "Alt-CtrlDown": "addCursorToNextLine",
  604. "Shift-Ctrl-Space": "selectScope",
  605. "Shift-Ctrl-M": "selectBetweenBrackets",
  606. "Ctrl-M": "goToBracket",
  607. "Shift-Ctrl-Up": "swapLineUp",
  608. "Shift-Ctrl-Down": "swapLineDown",
  609. "Ctrl-/": "toggleCommentIndented",
  610. "Ctrl-J": "joinLines",
  611. "Shift-Ctrl-D": "duplicateLine",
  612. "F9": "sortLines",
  613. "Ctrl-F9": "sortLinesInsensitive",
  614. "F2": "nextBookmark",
  615. "Shift-F2": "prevBookmark",
  616. "Ctrl-F2": "toggleBookmark",
  617. "Shift-Ctrl-F2": "clearBookmarks",
  618. "Alt-F2": "selectBookmarks",
  619. "Backspace": "smartBackspace",
  620. "Ctrl-K Ctrl-K": "delLineRight",
  621. "Ctrl-K Ctrl-U": "upcaseAtCursor",
  622. "Ctrl-K Ctrl-L": "downcaseAtCursor",
  623. "Ctrl-K Ctrl-Space": "setSublimeMark",
  624. "Ctrl-K Ctrl-A": "selectToSublimeMark",
  625. "Ctrl-K Ctrl-W": "deleteToSublimeMark",
  626. "Ctrl-K Ctrl-X": "swapWithSublimeMark",
  627. "Ctrl-K Ctrl-Y": "sublimeYank",
  628. "Ctrl-K Ctrl-C": "showInCenter",
  629. "Ctrl-K Ctrl-G": "clearBookmarks",
  630. "Ctrl-K Ctrl-Backspace": "delLineLeft",
  631. "Ctrl-K Ctrl-0": "unfoldAll",
  632. "Ctrl-K Ctrl-J": "unfoldAll",
  633. "Ctrl-Alt-Up": "selectLinesUpward",
  634. "Ctrl-Alt-Down": "selectLinesDownward",
  635. "Ctrl-F3": "findUnder",
  636. "Shift-Ctrl-F3": "findUnderPrevious",
  637. "Alt-F3": "findAllUnder",
  638. "Shift-Ctrl-[": "fold",
  639. "Shift-Ctrl-]": "unfold",
  640. "Ctrl-I": "findIncremental",
  641. "Shift-Ctrl-I": "findIncrementalReverse",
  642. "Ctrl-H": "replace",
  643. "F3": "findNext",
  644. "Shift-F3": "findPrev",
  645. "fallthrough": "pcDefault"
  646. };
  647. CodeMirror.normalizeKeyMap(keyMap.pcSublime);
  648. var mac = keyMap.default == keyMap.macDefault;
  649. keyMap.sublime = mac ? keyMap.macSublime : keyMap.pcSublime;
  650. });