如何让contenteditable元素只能输入纯文本

本文介绍如何利用CSS属性user-modify及contenteditable特性实现纯文本编辑功能,有效避免富文本内容的粘贴,提升用户体验。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

虽然说,利用全浏览器都支持的 contenteditable 模拟文本域可以实现体验相当不错的高度跟随内容自动撑开的效果,但是呢,有个很大的问题就是HTML内容可以直接被粘贴进去,如下图所示:

20180103151496235170685.png

之前的文章提到过过滤HTML的方法,保证内容都是纯文本。然而,这种方法的问题在于:

粘贴完毕到过滤结束有时间差,用户很看到内容一闪而过的糟糕体验;
光标的位置会发生变化,不是之前focus的位置了;
当年的我图样图森破,所以,只有上面这种程度。实际上,控制 contenteditable 元素只能输入纯文本是有体验比较好的方法的。

user-modify

一个div元素,要让其可编辑,也就是可读写, contenteditable 属性是最常用方法,做前端的基本上都知道。但是,知道CSS中有属性可以让普通元素可读写的的同学怕是就少多了。

主角亮相: user-modify .

支持属性值如下:

user-modify: read-only;
user-modify: read-write;user-modify: write-only;
user-modify: read-write-plaintext-only;

其中, write-only 不用在意,当下这个年代,基本上没有浏览器支持,以后估计也不会有。 read-only 表示只读,就是普通元素的默认状态啦。然后, read-write 和 read-write-plaintext-only 会让元素表现得像个文本域一样,可以 focus 以及输入内容。

会发现,设置了 read-write 和 read-write-plaintext-only 值的两个

标签元素是可以被focus的:
20180103151496150250544.png
而这两者的区别就在于,一个可以输入富文本,而下面一个只能输入纯文本,例如,我们从某网页同时复制一段内容粘贴进去看看:
2018010315149619229357.png
好了,至此,本文标题的答案实际上就已经有了。也就是给元素设置:

user-modify: read-write-plaintext-only

就可以让元素既可以编辑,也只能输入纯文本,表现得就跟 textarea 文本域一样。

是不是很酷啊!然而,抱歉地跟大家讲下,目前只有webkit内核浏览器才支持 read-write-plaintext-only 这个值,因此,我们的使用其实是:

-webkit-user-modify: read-write-plaintext-only

我们可以在移动端使用,以及,只需要兼顾webkit内容的桌面网页项目。

contenteditable属性控制

使用标准contenteditable属性值的HTML控制法
咳咳,提问:在HTML中, contenteditable 支持的属性值是?

图样图森破时候的我,脑中就只有 contenteditable=”true” 和 contenteditable=”false” ,科科,后来我发现自己太天真了, 新的草案 中明确表示还有多个其他属性值:

The contenteditable attribute is an enumerated attribute whose keywords are the empty string (“”), “events”, “caret”, “typing”, “plaintext-only”, “true”, and “false”. There is one additional state, the inherit state, which is the missing value default (and the invalid value default).

垂直展示下就是(不包括默认的inherit继承):

contenteditable=""
contenteditable="events"
contenteditable="caret"
contenteditable="plaintext-only"
contenteditable="true"
contenteditable="false"

别问我,我也不知道 “events” 和 “caret” 是干什么用的,嘿,但是 “plaintext-only” 我是知道的,可以让编辑区域只能键入纯文本。这里就不需要demo了,直接下面的框框,大家可以试试,看看能不能搞富文本。

<div contenteditable="plaintext-only"></div>

如果您发现,居然出乎意料,可以弄进去富文本,那说明你使用的是非Chrome之流的浏览器。

换句话说, contenteditable=”plaintext-only” 和CSS只的 -webkit-user-modify: read-write-plaintext-only 一样,目前仅仅是Chrome浏览器支持比较好的。

所以,您的项目如果还有很多IE8浏览器的用户,我只能替你惋惜,美妙的东西无法立即用上,不得已,寻求下面的方法。

JS控制粘贴事件

控制粘贴paste事件的JS控制法
如果我们单纯地敲击键盘,输入的内容实际上都是纯文本。除了一些特殊情况,例如IE浏览器下的编辑框会自动把合乎条件的url地址自动加上链接。富文本污染的情况主要出现在复制粘贴的时候,于是,如果我们能在粘贴的时候,对剪切板中的内容进行HTML过滤,再手动插入内容,岂不就可以完美解决无法输入富文本的问题了吗!?

于是,鄙人不才,一番折腾,弄出了下面的代码:

$('[contenteditable]').each(function() {
    // 干掉IE http之类地址自动加链接
    try {
        document.execCommand("AutoUrlDetect", false, false);
    } catch (e) {}

    $(this).on('paste', function(e) {
        e.preventDefault();
        var text = null;

        if(window.clipboardData && clipboardData.setData) {
            // IE
            text = window.clipboardData.getData('text');
        } else {
            text = (e.originalEvent || e).clipboardData.getData('text/plain') || prompt('在这里输入文本');
        }
        if (document.body.createTextRange) {    
            if (document.selection) {
                textRange = document.selection.createRange();
            } else if (window.getSelection) {
                sel = window.getSelection();
                var range = sel.getRangeAt(0);

                // 创建临时元素,使得TextRange可以移动到正确的位置
                var tempEl = document.createElement("span");
                tempEl.innerHTML = "&#FEFF;";
                range.deleteContents();
                range.insertNode(tempEl);
                textRange = document.body.createTextRange();
                textRange.moveToElementText(tempEl);
                tempEl.parentNode.removeChild(tempEl);
            }
            textRange.text = text;
            textRange.collapse(false);
            textRange.select();
        } else {
            // Chrome之类浏览器
            document.execCommand("insertText", false, text);
        }
    });
});

IE浏览器的 contenteditable 框有个问题,会自动添加链接,我们需要的是纯文本,显然这种自以为是的行为不是我们要的,可以使用 document.execCommand(“AutoUrlDetect”, false, false) 来进行处理。
理想情况应该直接使用 document.execCommand(“insertText”) 命令插入内容。但是,但是,IE浏览器虽然运行这段代码没有出错,也是支持 document.execCommand 的,但是,却没有插入内容的表现。也不知道是不是我打开的方式不对,后来,我就寻求更传统的方法,创建文本选区与插入,正好,就IE支持 document.body.createTextRange
document.selection IE浏览器一直是支持的,直到IE11浏览器,直接废弃了,好在 window.getSelection 还活着,于是,又一次分情况处理。

### 如何让 `contenteditable` 元素获得焦点或解决相关问题 #### 使用原生方法使 `contenteditable` 元素聚焦 可以通过 JavaScript 的 `.focus()` 方法直接让 `contenteditable="true"` 的元素获得焦点。然而,在实际开发中,特别是在移动端(如 iOS),由于浏览器的安全机制或其他限制,可能需要额外的处理逻辑。 以下是基本实现方式: ```javascript const editableElement = document.querySelector('[contenteditable]'); if (editableElement) { editableElement.focus(); // 让 contenteditable 元素获得焦点 } ``` 需要注意的是,部分浏览器会默认给获得焦点的 `contenteditable` 元素添加外边框样式(即 outline)。如果希望去除该效果,可以在 CSS 中定义如下规则[^1]: ```css [contenteditable]:focus { outline: none; } ``` #### 处理 iOS 上的特殊行为 在 iOS 平台上,存在一些已知的问题会影响 `contenteditable` 元素的行为。例如: 1. **ScrollView 不随插入符滚动** 如果页面中有动态调整高度的内容区域被设为 `contenteditable=true`,当输入内容超过可视范围时,iOS 可能无法正确更新 ScrollView 的状态,导致插入符脱离视图可见范围[^2]。 解决方案之一是监听键盘事件并手动调整容器的高度或滚动位置。例如: ```javascript const editableDiv = document.querySelector('[contenteditable]'); editableDiv.addEventListener('input', () => { editableDiv.scrollTop = editableDiv.scrollHeight; // 自动滚到底部 }); ``` 2. **自动聚焦失效** iOS 存在一个安全策略:只有由用户交互触发的事件链才能成功调用 `.focus()` 方法。这意味着程序化地尝试设置焦点通常不起作用[^3]。 下面是一个可行的工作流程示例: ```javascript document.addEventListener('touchstart', (event) => { const target = event.target.closest('[contenteditable]'); if (target) { requestAnimationFrame(() => { target.focus(); }); } }); ``` #### 高级需求下的光标管理 对于更复杂的场景,比如自定义光标的显示或者阻止系统键盘弹出的同时保持文本编辑功能正常运作,可以考虑以下技术手段[^4]: - 利用伪类选择器配合绝对定位创建视觉上的“光标”,并通过鼠标点击坐标计算其应处的位置; - 借助 Range API 和 Selection 对象精确操控选区边界及其关联属性; 下面给出一段简单的代码片段用于演示如何强制将光标放置于特定 `<span>` 后方: ```javascript function setCursorAfterSpan(spanElm) { const range = document.createRange(), sel = window.getSelection(); range.setStartAfter(spanElm); range.collapse(true); sel.removeAllRanges(); sel.addRange(range); } // Example usage: setCursorAfterSpan(document.querySelector('#my-span')); ``` 不过要注意,这种方法并非总能在所有主流浏览器间一致表现良好,尤其是 WebKit 内核家族成员往往会对非法的选择范围作出修正动作[^5]。 ---
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

谢泽的网络日志

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值