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