Replace textarea.value calls

This commit is contained in:
2026-05-04 11:54:58 +02:00
parent 0d0d06d265
commit 0d6aac7844
2 changed files with 40 additions and 23 deletions
+30 -19
View File
@@ -6,44 +6,55 @@
// --- DOM helpers --- // --- DOM helpers ---
// Route every edit through execCommand so the browser's native undo/redo
// stack is preserved. Direct assignment to textarea.value would wipe it.
function replaceRange(start, end, text) {
textarea.focus();
textarea.selectionStart = start;
textarea.selectionEnd = end;
document.execCommand('insertText', false, text);
}
function wrap(before, after, placeholder) { function wrap(before, after, placeholder) {
var start = textarea.selectionStart; var start = textarea.selectionStart;
var end = textarea.selectionEnd; var end = textarea.selectionEnd;
var selected = textarea.value.slice(start, end) || placeholder; var hadSelection = end > start;
var replacement = before + selected + after; var selected = hadSelection ? textarea.value.slice(start, end) : placeholder;
textarea.value = textarea.value.slice(0, start) + replacement + textarea.value.slice(end); replaceRange(start, end, before + selected + after);
if (selected === placeholder) { if (!hadSelection) {
textarea.selectionStart = start + before.length; textarea.selectionStart = start + before.length;
textarea.selectionEnd = start + before.length + placeholder.length; textarea.selectionEnd = start + before.length + placeholder.length;
} else {
textarea.selectionStart = start + replacement.length;
textarea.selectionEnd = start + replacement.length;
} }
textarea.focus();
} }
function linePrefix(prefix) { function linePrefix(prefix) {
var start = textarea.selectionStart; var start = textarea.selectionStart;
var lineStart = textarea.value.lastIndexOf('\n', start - 1) + 1; var lineStart = textarea.value.lastIndexOf('\n', start - 1) + 1;
textarea.value = textarea.value.slice(0, lineStart) + prefix + textarea.value.slice(lineStart); replaceRange(lineStart, lineStart, prefix);
textarea.selectionStart = textarea.selectionEnd = start + prefix.length; textarea.selectionStart = textarea.selectionEnd = start + prefix.length;
textarea.focus();
} }
function insertAtCursor(s) { function insertAtCursor(s) {
var start = textarea.selectionStart; replaceRange(textarea.selectionStart, textarea.selectionEnd, s);
var end = textarea.selectionEnd;
textarea.value = textarea.value.slice(0, start) + s + textarea.value.slice(end);
textarea.selectionStart = textarea.selectionEnd = start + s.length;
textarea.dispatchEvent(new Event('input'));
textarea.focus();
} }
function applyResult(result) { function applyResult(result) {
textarea.value = result.text; var oldText = textarea.value;
var newText = result.text;
var prefixLen = 0;
var maxPrefix = Math.min(oldText.length, newText.length);
while (prefixLen < maxPrefix && oldText.charCodeAt(prefixLen) === newText.charCodeAt(prefixLen)) {
prefixLen++;
}
var oldEnd = oldText.length;
var newEnd = newText.length;
while (oldEnd > prefixLen && newEnd > prefixLen
&& oldText.charCodeAt(oldEnd - 1) === newText.charCodeAt(newEnd - 1)) {
oldEnd--;
newEnd--;
}
replaceRange(prefixLen, oldEnd, newText.slice(prefixLen, newEnd));
textarea.selectionStart = textarea.selectionEnd = result.cursor; textarea.selectionStart = textarea.selectionEnd = result.cursor;
textarea.dispatchEvent(new Event('input'));
textarea.focus();
} }
function applyTableOp(fn, arg) { function applyTableOp(fn, arg) {
+10 -4
View File
@@ -49,22 +49,28 @@ window.EditorMovie = (function () {
return out.join('\n'); return out.join('\n');
} }
function replaceRange(ta, start, end, text) {
ta.focus();
ta.selectionStart = start;
ta.selectionEnd = end;
document.execCommand('insertText', false, text);
}
function insertOrReplace(ta, markup) { function insertOrReplace(ta, markup) {
var t = ta.value || ''; var t = ta.value || '';
var b = t.indexOf(BEGIN); var b = t.indexOf(BEGIN);
var e = t.indexOf(END); var e = t.indexOf(END);
if (b !== -1 && e !== -1 && e > b) { if (b !== -1 && e !== -1 && e > b) {
ta.value = t.slice(0, b) + markup + t.slice(e + END.length); replaceRange(ta, b, e + END.length, markup);
} else { } else {
var h = t.match(/^#{1,6}\s+.+?\s*$/m); var h = t.match(/^#{1,6}\s+.+?\s*$/m);
if (h) { if (h) {
var idx = t.indexOf(h[0]) + h[0].length; var idx = t.indexOf(h[0]) + h[0].length;
ta.value = t.slice(0, idx) + '\n\n' + markup + t.slice(idx); replaceRange(ta, idx, idx, '\n\n' + markup);
} else { } else {
ta.value = markup + (t ? '\n\n' + t : ''); replaceRange(ta, 0, 0, t ? markup + '\n\n' : markup);
} }
} }
ta.dispatchEvent(new Event('input'));
} }
function fetchMovie(title, year) { function fetchMovie(title, year) {