Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

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