如何在当前光标位置的textarea中插入文本?

IT技术 javascript textarea
2021-01-28 13:57:46

我想创建一个简单的函数,将文本添加到用户光标位置的文本区域中。它需要是一个干净的功能。只是基础知识。我可以弄清楚其余的。

6个回答

输入元素的使用selectionStart/selectionEnd 属性(也适用<textarea>

function insertAtCursor(myField, myValue) {
    //IE support
    if (document.selection) {
        myField.focus();
        sel = document.selection.createRange();
        sel.text = myValue;
    }
    //MOZILLA and others
    else if (myField.selectionStart || myField.selectionStart == '0') {
        var startPos = myField.selectionStart;
        var endPos = myField.selectionEnd;
        myField.value = myField.value.substring(0, startPos)
            + myValue
            + myField.value.substring(endPos, myField.value.length);
    } else {
        myField.value += myValue;
    }
}
感谢 Rab 的回答和 @user340140 的修复。这是一个工作示例
2021-03-19 13:57:46
@ user340140,您的“丢失插入符号”修复程序,仅当我在您建议的行之前将焦点放在输入上时才有效。至少在 Chrome(当前版本 62.0)中更改非重点字段的选择似乎是不可能的
2021-03-19 13:57:46
修复“丢失插入符号位置”:在之前添加插入这些行 } else { myField.selectionStart = startPos + myValue.length; myField.selectionEnd = startPos + myValue.length;
2021-04-02 13:57:46
这段代码有一个小问题:selectionStart是一个数值,因此应该与0和 not进行比较'0',并且可能应该使用===
2021-04-10 13:57:46

此代码段可以在几行 jQuery 1.9+ 中为您提供帮助:http : //jsfiddle.net/4MBUG/2/

$('input[type=button]').on('click', function() {
    var cursorPos = $('#text').prop('selectionStart');
    var v = $('#text').val();
    var textBefore = v.substring(0,  cursorPos);
    var textAfter  = v.substring(cursorPos, v.length);

    $('#text').val(textBefore + $(this).val() + textAfter);
});
@mparkuk:它仍然受到 user340140 上面提到的“丢失插入符位置”问题的困扰。(抱歉,我应该修复它,但我没时间了。)
2021-03-15 13:57:46
这有效,但光标最终位于错误的位置。
2021-03-20 13:57:46
但它不能替换选定的文本
2021-03-26 13:57:46
感谢您提供工作小提琴。我已经更新它以重置插入符号位置并使其成为 jquery 插件:jsfiddle.net/70gqn153
2021-04-09 13:57:46
伟大的!也适用于 1.6,稍作修改。
2021-04-11 13:57:46

为了正确的 Javascript

HTMLTextAreaElement.prototype.insertAtCaret = function (text) {
  text = text || '';
  if (document.selection) {
    // IE
    this.focus();
    var sel = document.selection.createRange();
    sel.text = text;
  } else if (this.selectionStart || this.selectionStart === 0) {
    // Others
    var startPos = this.selectionStart;
    var endPos = this.selectionEnd;
    this.value = this.value.substring(0, startPos) +
      text +
      this.value.substring(endPos, this.value.length);
    this.selectionStart = startPos + text.length;
    this.selectionEnd = startPos + text.length;
  } else {
    this.value += text;
  }
};
非常好的扩展!工作正常。谢谢!
2021-03-29 13:57:46
@ErikAigner 那不对。在 ES6 之前,A.prototype.fn = X是拥有“类”/继承的唯一方法。仅仅因为您可以扩展您的对象,并不意味着您应该扩展本机对象。想象一下 10 年前,您实现了Array#map,然后Array#map变成了本机 API,但与您的不兼容。现在有人打开你的代码库,看到[].map()并假设它是原生 API。你好头痛和错误。
2021-04-06 13:57:46
扩展您不拥有的对象的原型并不是一个好主意。只需将其设为常规功能,它也能正常工作。
2021-04-07 13:57:46
这会在设置后清除编辑元素的撤消缓冲区this.value = ...有没有办法保存?
2021-04-07 13:57:46
最佳解决方案!谢谢
2021-04-11 13:57:46

新答案:

https://developer.mozilla.org/en-US/docs/Web/API/HTMLInputElement/setRangeText

不过,我不确定浏览器对此的支持。

在 Chrome 81 中测试。

function typeInTextarea(newText, el = document.activeElement) {
  const [start, end] = [el.selectionStart, el.selectionEnd];
  el.setRangeText(newText, start, end, 'select');
}

document.getElementById("input").onkeydown = e => {
  if (e.key === "Enter") typeInTextarea("lol");
}
<input id="input" />
<br/><br/>
<div>Press Enter to insert "lol" at caret.</div>
<div>It'll replace a selection with the given text.</div>

旧答案:

Erik Pukinskis 答案的纯 JS 修改:

function typeInTextarea(newText, el = document.activeElement) {
  const start = el.selectionStart
  const end = el.selectionEnd
  const text = el.value
  const before = text.substring(0, start)
  const after  = text.substring(end, text.length)
  el.value = (before + newText + after)
  el.selectionStart = el.selectionEnd = start + newText.length
  el.focus()
}

document.getElementById("input").onkeydown = e => {
  if (e.key === "Enter") typeInTextarea("lol");
}
<input id="input" />
<br/><br/>
<div>Press Enter to insert "lol" at caret.</div>

在 Chrome 47、81 和 Firefox 76 中测试。

如果要在同一字段中键入时更改当前选定文本的值(用于自动完成或类似效果),document.activeElement请将其作为第一个参数传递

这不是最优雅的方法,但它非常简单。

示例用法:

typeInTextarea('hello');
typeInTextarea('haha', document.getElementById('some-id'));
嘿@ErikAigner!我的错,没有意识到这个问题有两个 Erik 的答案。我的意思是Erik Pukinskis我会更新答案以更好地反映这一点。
2021-03-18 13:57:46
你没有用 >> 关闭行;<<
2021-03-26 13:57:46
我在 JSFiddle 上做了一个演示。它也可以使用Version 54.0.2813.0 canary (64-bit),基本上是 Chrome Canary 54.0.2813.0。最后,如果你想让它通过 ID 插入文本框document.getElementById('insertyourIDhere')el在函数中使用代替
2021-03-28 13:57:46
@Phoenix 分号在 Javascript 中是可选的。没有它们也能工作。不过,如果需要,您可以用分号进行编辑。没什么大不了的。
2021-03-30 13:57:46
我的答案的哪一部分不是“纯”JS?我是不是忘记了一些 C++?
2021-04-09 13:57:46

一个简单的解决方案,适用于 firefox、chrome、opera、safari 和 edge,但可能不适用于旧的 IE 浏览器。

var target = document.getElementById("mytextarea_id")

if (target.setRangeText) {
    //if setRangeText function is supported by current browser
    target.setRangeText(data)
} else {
    target.focus()
    document.execCommand('insertText', false /*no UI*/, data);
}

setRangeText功能允许您用提供的文本替换当前选择,或者如果没有选择则在光标位置插入文本。据我所知,它仅受 Firefox 支持。

对于其他浏览器,有“insertText”命令,它只影响当前聚焦的 html 元素,并且具有相同的行为 setRangeText

部分灵感来自这篇文章

不幸的是,execCommand被 MDN 认为已过时:developer.mozilla.org/en-US/docs/Web/API/Document/execCommand不知道为什么,它似乎真的很有用!
2021-03-18 13:57:46
Ramast,那不是你的代码所做的。对于定义它的任何浏览器(大多数),它将使用 setRangeText 而不是 execCommand 。对于你描述的行为,你需要先调用document.execCommand,然后再检查返回值。如果为 false,则使用 target.setRangeText。
2021-03-19 13:57:46
@Jools 如果支持 setRangeText 那么为什么不使用它而不是 execCommand?为什么我需要先尝试 execCommand?
2021-03-21 13:57:46
是的,execCommand 用于其他浏览器,而对于 firefox,则使用 setRangeText 函数代替。
2021-04-04 13:57:46
这几乎是正确的方法。您链接的文章提供了一个完整的解决方案包:insert-text-at-cursor但是我更喜欢,execCommand因为它支持undo并制作了insert-text-textarea不支持 IE,但更小
2021-04-09 13:57:46