2017-06-24 00:58:56 +03:00
|
|
|
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
2020-02-26 21:59:06 +03:00
|
|
|
// Distributed under an MIT license: https://codemirror.net/LICENSE
|
2017-06-24 00:58:56 +03:00
|
|
|
|
|
|
|
(function(mod) {
|
|
|
|
if (typeof exports == "object" && typeof module == "object") // CommonJS
|
|
|
|
mod(require("../../lib/codemirror"));
|
|
|
|
else if (typeof define == "function" && define.amd) // AMD
|
|
|
|
define(["../../lib/codemirror"], mod);
|
|
|
|
else // Plain browser env
|
|
|
|
mod(CodeMirror);
|
|
|
|
})(function(CodeMirror) {
|
2020-02-26 21:59:06 +03:00
|
|
|
var nonspace = /\S/g;
|
|
|
|
var repeat = String.prototype.repeat || function (n) { return Array(n + 1).join(this); };
|
2017-06-24 00:58:56 +03:00
|
|
|
function continueComment(cm) {
|
|
|
|
if (cm.getOption("disableInput")) return CodeMirror.Pass;
|
|
|
|
var ranges = cm.listSelections(), mode, inserts = [];
|
|
|
|
for (var i = 0; i < ranges.length; i++) {
|
2017-11-17 12:02:05 +03:00
|
|
|
var pos = ranges[i].head
|
|
|
|
if (!/\bcomment\b/.test(cm.getTokenTypeAt(pos))) return CodeMirror.Pass;
|
|
|
|
var modeHere = cm.getModeAt(pos)
|
2017-06-24 00:58:56 +03:00
|
|
|
if (!mode) mode = modeHere;
|
|
|
|
else if (mode != modeHere) return CodeMirror.Pass;
|
|
|
|
|
2020-02-26 21:59:06 +03:00
|
|
|
var insert = null, line, found;
|
|
|
|
var blockStart = mode.blockCommentStart, lineCmt = mode.lineComment;
|
|
|
|
if (blockStart && mode.blockCommentContinue) {
|
|
|
|
line = cm.getLine(pos.line);
|
|
|
|
var end = line.lastIndexOf(mode.blockCommentEnd, pos.ch - mode.blockCommentEnd.length);
|
|
|
|
// 1. if this block comment ended
|
|
|
|
// 2. if this is actually inside a line comment
|
|
|
|
if (end != -1 && end == pos.ch - mode.blockCommentEnd.length ||
|
|
|
|
lineCmt && (found = line.lastIndexOf(lineCmt, pos.ch - 1)) > -1 &&
|
|
|
|
/\bcomment\b/.test(cm.getTokenTypeAt({line: pos.line, ch: found + 1}))) {
|
|
|
|
// ...then don't continue it
|
|
|
|
} else if (pos.ch >= blockStart.length &&
|
|
|
|
(found = line.lastIndexOf(blockStart, pos.ch - blockStart.length)) > -1 &&
|
|
|
|
found > end) {
|
|
|
|
// reuse the existing leading spaces/tabs/mixed
|
|
|
|
// or build the correct indent using CM's tab/indent options
|
|
|
|
if (nonspaceAfter(0, line) >= found) {
|
|
|
|
insert = line.slice(0, found);
|
|
|
|
} else {
|
|
|
|
var tabSize = cm.options.tabSize, numTabs;
|
|
|
|
found = CodeMirror.countColumn(line, found, tabSize);
|
|
|
|
insert = !cm.options.indentWithTabs ? repeat.call(" ", found) :
|
|
|
|
repeat.call("\t", (numTabs = Math.floor(found / tabSize))) +
|
|
|
|
repeat.call(" ", found - tabSize * numTabs);
|
2017-06-24 00:58:56 +03:00
|
|
|
}
|
2020-02-26 21:59:06 +03:00
|
|
|
} else if ((found = line.indexOf(mode.blockCommentContinue)) > -1 &&
|
|
|
|
found <= pos.ch &&
|
|
|
|
found <= nonspaceAfter(0, line)) {
|
|
|
|
insert = line.slice(0, found);
|
2017-06-24 00:58:56 +03:00
|
|
|
}
|
2017-11-17 12:02:05 +03:00
|
|
|
if (insert != null) insert += mode.blockCommentContinue
|
2017-06-24 00:58:56 +03:00
|
|
|
}
|
2020-02-26 21:59:06 +03:00
|
|
|
if (insert == null && lineCmt && continueLineCommentEnabled(cm)) {
|
|
|
|
if (line == null) line = cm.getLine(pos.line);
|
|
|
|
found = line.indexOf(lineCmt);
|
|
|
|
// cursor at pos 0, line comment also at pos 0 => shift it down, don't continue
|
|
|
|
if (!pos.ch && !found) insert = "";
|
|
|
|
// continue only if the line starts with an optional space + line comment
|
|
|
|
else if (found > -1 && nonspaceAfter(0, line) >= found) {
|
|
|
|
// don't continue if there's only space(s) after cursor or the end of the line
|
|
|
|
insert = nonspaceAfter(pos.ch, line) > -1;
|
|
|
|
// but always continue if the next line starts with a line comment too
|
|
|
|
if (!insert) {
|
|
|
|
var next = cm.getLine(pos.line + 1) || '',
|
|
|
|
nextFound = next.indexOf(lineCmt);
|
|
|
|
insert = nextFound > -1 && nonspaceAfter(0, next) >= nextFound || null;
|
|
|
|
}
|
|
|
|
if (insert) {
|
|
|
|
insert = line.slice(0, found) + lineCmt +
|
|
|
|
line.slice(found + lineCmt.length).match(/^\s*/)[0];
|
|
|
|
}
|
2017-06-24 00:58:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (insert == null) return CodeMirror.Pass;
|
|
|
|
inserts[i] = "\n" + insert;
|
|
|
|
}
|
|
|
|
|
|
|
|
cm.operation(function() {
|
|
|
|
for (var i = ranges.length - 1; i >= 0; i--)
|
|
|
|
cm.replaceRange(inserts[i], ranges[i].from(), ranges[i].to(), "+insert");
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-02-26 21:59:06 +03:00
|
|
|
function nonspaceAfter(ch, str) {
|
|
|
|
nonspace.lastIndex = ch;
|
|
|
|
var m = nonspace.exec(str);
|
|
|
|
return m ? m.index : -1;
|
|
|
|
}
|
|
|
|
|
2017-06-24 00:58:56 +03:00
|
|
|
function continueLineCommentEnabled(cm) {
|
|
|
|
var opt = cm.getOption("continueComments");
|
|
|
|
if (opt && typeof opt == "object")
|
|
|
|
return opt.continueLineComment !== false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
CodeMirror.defineOption("continueComments", null, function(cm, val, prev) {
|
|
|
|
if (prev && prev != CodeMirror.Init)
|
|
|
|
cm.removeKeyMap("continueComment");
|
|
|
|
if (val) {
|
|
|
|
var key = "Enter";
|
|
|
|
if (typeof val == "string")
|
|
|
|
key = val;
|
|
|
|
else if (typeof val == "object" && val.key)
|
|
|
|
key = val.key;
|
|
|
|
var map = {name: "continueComment"};
|
|
|
|
map[key] = continueComment;
|
|
|
|
cm.addKeyMap(map);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|