这周碰到一个东西,
contentEditable
,它是用来指定一个元素是否是可编辑的,这也是富文本编辑器实现的底层支持,网上关于这部分东西的资料比较少或者不全,所以我来整理下关于这个属性,和可编辑区域的一些操作吧,
比如获取光标位置,设置光标,往可编辑区域光标处插入内容等等
HTML中的contentEditable的属性可以打开某些元素的可编辑状态.也许你没用过contentEditable属性.甚至从未听说过.contentEditable的作用相当神奇.可以让div或整个网页,以及span等等元素设置为可写。我们最常用的输入文本内容便是input与textarea,使用contentEditable属性后,可以在div,table,p,span,body,等等很多元素中输入内容.
设置一个容器为可编辑区域:
设置contentEditable属性
<p contentEditable style="border: 1px solid red;">grsd</p>
<div contentEditable style="border: 1px solid red;">rsdgsdg </div>
</body>
设置placeholder:
这个不像表单一样,给属性加上placeholder属性指定值就可以了,还需要配合特殊的样式
:empty
做设置
<style type="text/css">
.content:empty::before {
content: attr(placeholder);
font-size: 14px;
color: #CCC;
line-height: 21px;
padding-top: 20px;
.content {
border: 1px solid black;
</style>
<div contentEditable class="content" placeholder="请输入不少于150字"></div>
</body>
因为各个浏览器在标记生成上的不同,因此跨浏览器使用 contenteditable 一直以来都是痛点,例如一些看起来十分简单的事情,如: 当你按下Enter/Return键在可编辑区域中创建一个新的文本行时,不同主流浏览器对此有不同处理(Firefox 插入<br>
、IE/Opera将使用<p>
、 Chrome/Safari 将使用 <div>
)幸运的是在现代浏览器中,这些不同都趋于一致了。截止到Firefox 60,火狐开始使用<div>
元素来包裹新生成的文本行,以与Chrome, modern Opera, Edge, and Safari.的行为趋于一致
改变回车时插入的标签:
如果你想使用不同的方式创建新的段落,上面所有浏览器都支持document.execCommand
方法,该方法提供的 defaultParagraphSeparator
命令能够让你以不同的方式创建新的段落例如, 使用 <p>
元素:
document.execCommand("defaultParagraphSeparator", false, "p");
现在在编辑器中按回车就变成p
标签了
上面只是document.execCommand
的一个命令演示
当一个HTML文档切换到设计模式时,document暴露 execCommand 方法
,该方法允许运行命令来操纵可编辑内容区域的元素。
大多数命令影响document的 selection(粗体,斜体等),当其他命令插入新元素(添加链接)或影响整行(缩进)。当使用contentEditable时,调用 execCommand() 将影响当前活动的可编辑元素
。
命令很多,这里就不列举出来,具体可以参考官方文档document.execCommand命令大全
基于document.execCommand的这些命令
,我们可以做一个小型的富文本编辑器:
<!doctype html>
<title>Rich Text Editor</title>
<script type="text/javascript">
var oDoc, sDefTxt;
function initDoc() {
oDoc = document.getElementById("textBox");
sDefTxt = oDoc.innerHTML;
if(document.compForm.switchMode.checked) {
setDocMode(true);
function formatDoc(sCmd, sValue) {
if(validateMode()) {
document.execCommand(sCmd, false, sValue);
oDoc.focus();
function validateMode() {
if(!document.compForm.switchMode.checked) {
return true;
alert("Uncheck \"Show HTML\".");
oDoc.focus();
return false;
function setDocMode(bToSource) {
var oContent;
if(bToSource) {
oContent = document.createTextNode(oDoc.innerHTML);
oDoc.innerHTML = "";
var oPre = document.createElement("pre");
oDoc.contentEditable = false;
oPre.id = "sourceText";
oPre.contentEditable = true;
oPre.appendChild(oContent);
oDoc.appendChild(oPre);
document.execCommand("defaultParagraphSeparator", false, "div");
} else {
if(document.all) {
oDoc.innerHTML = oDoc.innerText;
} else {
oContent = document.createRange();
oContent.selectNodeContents(oDoc.firstChild);
oDoc.innerHTML = oContent.toString();
oDoc.contentEditable = true;
oDoc.focus();
function printDoc() {
if(!validateMode()) {
return;
var oPrntWin = window.open("", "_blank", "width=450,height=470,left=400,top=100,menubar=yes,toolbar=no,location=no,scrollbars=yes");
oPrntWin.document.open();
oPrntWin.document.write("<!doctype html><html><head><title>Print<\/title><\/head><body οnlοad=\"print();\">" + oDoc.innerHTML + "<\/body><\/html>");
oPrntWin.document.close();
</script>
<style type="text/css">
.intLink {
cursor: pointer;
img.intLink {
border: 0;
#toolBar1 select {
font-size: 10px;
#textBox {
width: 540px;
height: 200px;
border: 1px #000000 solid;
padding: 12px;
overflow: scroll;
#textBox #sourceText {
padding: 0;
margin: 0;
min-width: 498px;
min-height: 200px;
#editMode label {
cursor: pointer;
</style>
</head>
<body onload="initDoc();">
<form name="compForm" method="post" action="sample.php" onsubmit="if(validateMode()){this.myDoc.value=oDoc.innerHTML;return true;}return false;">
<input type="hidden" name="myDoc">
<div id="toolBar1">
<select onchange="formatDoc('formatblock',this[this.selectedIndex].value);this.selectedIndex=0;">
<option selected>- formatting -</option>
<option value="h1">Title 1 <h1></option>
<option value="h2">Title 2 <h2></option>
<option value="h3">Title 3 <h3></option>
<option value="h4">Title 4 <h4></option>
<option value="h5">Title 5 <h5></option>
<option value="h6">Subtitle <h6></option>
<option value="p">Paragraph <p></option>
<option value="pre">Preformatted <pre></option>
</select>
<select onchange="formatDoc('fontname',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- font -</option>
<option>Arial</option>
<option>Arial Black</option>
<option>Courier New</option>
<option>Times New Roman</option>
</select>
<select onchange="formatDoc('fontsize',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- size -</option>
<option value="1">Very small</option>
<option value="2">A bit small</option>
<option value="3">Normal</option>
<option value="4">Medium-large</option>
<option value="5">Big</option>
<option value="6">Very big</option>
<option value="7">Maximum</option>
</select>
<select onchange="formatDoc('forecolor',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- color -</option>
<option value="red">Red</option>
<option value="blue">Blue</option>
<option value="green">Green</option>
<option value="black">Black</option>
</select>
<select onchange="formatDoc('backcolor',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- background -</option>
<option value="red">Red</option>
<option value="green">Green</option>
<option value="black">Black</option>
</select>
</div>
<div id="toolBar2">
<img class="intLink" title="Clean" onclick="if(validateMode()&&confirm('Are you sure?')){oDoc.innerHTML=sDefTxt};" src="" />
<img class="intLink" title="Print" onclick="printDoc();" src="">
<img class="intLink" title="Undo" onclick="formatDoc('undo');" src="" />
<img class="intLink" title="Redo" onclick="formatDoc('redo');" src="" />
<img class="intLink" title="Remove formatting" onclick="formatDoc('removeFormat')" src="">
<img class="intLink" title="Bold" onclick="formatDoc('bold');" src="" />
<img class="intLink" title="Italic" onclick="formatDoc('italic');" src="" />
<img class="intLink" title="Underline" onclick="formatDoc('underline');" src="" />
<img class="intLink" title="Left align" onclick="formatDoc('justifyleft');" src="" />
<img class="intLink" title="Center align" onclick="formatDoc('justifycenter');" src="" />
<img class="intLink" title="Right align" onclick="formatDoc('justifyright');" src="" />
<img class="intLink" title="Numbered list" onclick="formatDoc('insertorderedlist');" src="" />
<img class="intLink" title="Dotted list" onclick="formatDoc('insertunorderedlist');" src="" />
<img class="intLink" title="Quote" onclick="formatDoc('formatblock','blockquote');" src="" />
<img class="intLink" title="Delete indentation" onclick="formatDoc('outdent');" src="" />
<img class="intLink" title="Add indentation" onclick="formatDoc('indent');" src="" />
<img class="intLink" title="Hyperlink" onclick="var sLnk=prompt('Write the URL here','http:\/\/');if(sLnk&&sLnk!=''&&sLnk!='http://'){formatDoc('createlink',sLnk)}" src="" />
<img class="intLink" title="Cut" onclick="formatDoc('cut');" src="" />
<img class="intLink" title="Copy" onclick="formatDoc('copy');" src="" />
<img class="intLink" title="Paste" onclick="formatDoc('paste');" src="" />
</div>
<div id="textBox" contenteditable="true">
<p>Lorem ipsum</p>
</div>
<p id="editMode"><input type="checkbox" name="switchMode" id="switchBox" onchange="setDocMode(this.checked);" /> <label for="switchBox">Show HTML</label></p>
<p><input type="submit" value="Send" /></p>
</form>
</body>
</html>
是不是很酷?
Selection
对象表示用户选择的文本范围或插入符号的当前位置。它代表页面中的文本选区,可能横跨多个元素。文本选区由用户拖拽鼠标经过文字而产生。要获取用于检查或修改的 Selection
对象,请调用 window.getSelection()
。
Selection 对象所对应的是用户所选择的 ranges (区域),俗称拖蓝。默认情况下,该函数只针对一个区域,我们可以这样使用这个函数:
var selObj = window.getSelection();
var range = selObj.getRangeAt(0);
例子: 我写了如下代码来一起探究下这两个是什么东西吧
每次我拖动完,或者点击完鼠标的时候,都会打印selObj
,查看selObj
对象,看看里面有什么属性
<div>我是p标签的内容</div>
<p>我是p标签的内容</p>
<li>li1</li>
<li>li2</li>
</body>
<script type="text/javascript">
document.onmouseup=function(){
var selObj = window.getSelection();
console.log(selObj)
var range = selObj.getRangeAt(0);
console.log(range)
</script>
可以看到window.getSelection()打印的对象就是我们选中区域的对象,对象里面的属性值,我来介绍以下吧
而range对象
里面的信息就简洁很多,这个我就不介绍了,和selection对象
是一样的
当然range对象
和selection对象
原型上也有很多的方法提供给我们使用,可以自行查阅文档
selection对象的方法
range对象的方法
我们要说的是基于这些方法,我们如何做到在可编辑的div或容器中往光标处插入内容,返回光标位置
方法一:较完整的,考虑兼容,插入后,光标自动设置到插入后的位置
<!DOCTYPE html>
<html lang="zh">
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<meta http-equiv="X-UA-Compatible" content="ie=edge" />
<title>Document</title>
</head>
<div class="box" contenteditable style="border: 1px solid red;">
我是可编辑的div
</div>
<button onclick="pasteHtmlAtCaret('我是插入的内容')">插入</button>
</body>
<script type="text/javascript">
function pasteHtmlAtCaret(html) {
var sel, range;
if (window.getSelection) {
sel = window.getSelection();
if (sel.getRangeAt && sel.rangeCount) {
range = sel.getRangeAt(0);
range.deleteContents();
var el = document.createElement("div");
el.innerHTML = html;
var frag = document.createDocumentFragment(), node, lastNode;
while ( (node = el.firstChild) ) {
lastNode = frag.appendChild(node);
range.insertNode(frag);
if (lastNode) {
range = range.cloneRange();
range.setStartAfter(lastNode);
range.collapse(true);
sel.removeAllRanges();
sel.addRange(range);
} else if (document.selection && document.selection.type != "Control") {
document.selection.createRange().pasteHTML(html);
</script>
</html>
方法二:可插入一段html字符串,可被解析
<div id="box" contentEditable>我是p标签的内容</div>
</body>
<script type="text/javascript">
document.getElementById("box").onmouseup = function() {
insertHtmlAtCursor("<strong>666</strong>")
function insertHtmlAtCursor(html) {
var range, node;
range = window.getSelection().getRangeAt(0);
node = range.createContextualFragment(html);
range.insertNode(node);
</script>
方法三: 插入一段文本
<div id="box" contentEditable>我是p标签的内容</div>
</body>
<script type="text/javascript">
document.getElementById("box").onmouseup = function() {
insertTextAtCursor("<strong>666</strong>")
function insertTextAtCursor(txt) {
var sel = window.getSelection();
var iEnd = sel.anchorOffset;
var htmldata = sel.anchorNode.data;
if(htmldata) {
var finaldata = htmldata.substring(0, iEnd) + txt + htmldata.substring(iEnd);
sel.anchorNode.textContent = finaldata
} else {
sel.anchorNode.textContent = txt
</script>
方法四: 插入一个节点
<div id="box" contentEditable>我是p标签的内容</div>
</body>
<script type="text/javascript">
document.getElementById("box").onmouseup = function() {
insertTextAtCursor("<strong>666</strong>")
function insertTextAtCursor(text) {
var sel = window.getSelection();
if(sel.getRangeAt && sel.rangeCount) {
var range = sel.getRangeAt(0);
range.deleteContents();
range.insertNode(document.createTextNode(text));
</script>
总结了以上方式,大概能满足你插入的基本需求了
console.log(window.getSelection().anchorOffset)
console.log(window.getSelection().focusOffset)
document.onmouseup=function(){
console.log(window.getSelection().toString())
<div id="box" contentEditable>我是p标签的内容</div>
</body>
<script type="text/javascript">
document.getElementById("box").onmouseup = function() {
set_focus(this)
function set_focus(el) {
var range = document.createRange();
range.selectNodeContents(el);
range.collapse(false);
var sel = window.getSelection();
sel.removeAllRanges();
sel.addRange(range);
</script>
<div id="box" contentEditable>我是p标签的内容</div>
<button id="saveBtn">记录光标位置</button>
<button id="restoreBtn">恢复光标位置</button>
</body>
<script type="text/javascript">
let range = null
document.getElementById("saveBtn").onclick=function(){
range = saveSelection()
document.getElementById("restoreBtn").onclick=function(){
restoreSelection(range)
function saveSelection() {
if(window.getSelection) {
sel = window.getSelection();
if(sel.getRangeAt && sel.rangeCount) {
return sel.getRangeAt(0);
} else if(document.selection && document.selection.createRange) {
return document.selection.createRange();
return null;
function restoreSelection(range) {
if(range) {
if(window.getSelection) {
sel = window.getSelection();
sel.removeAllRanges();
sel.addRange(range);
} else if(document.selection && range.select) {
range.select();
</script>
先这么多,有新的需求或者想起来再补
这周碰到一个骚东西,contentEditable ,它是用来指定一个元素是否是可编辑的,这也是富文本编辑器实现的底层支持,网上关于这部分东西的资料比较少或者不全,所以我来整理下关于这个属性,和可编辑区域的一些操作吧,比如获取光标位置,设置光标,往可编辑区域光标处插入内容等等HTML中的contentEditable的属性可以打开某些元素的可编辑状态.也许你没用过contentEditable属性.甚至从未听说过.contentEditable的作用相当神奇.可以让div或整个网页,以及span等等元素
对于最新版本:
< script src =" //cdn.jsdelivr.net/npm/medium-editor@latest/dist/js/medium-editor.min.js " > </ script >
< link rel =" stylesheet " href =" //cdn.jsdelivr.net/npm/medium-editor@latest/dist/css/medium-editor.min.css " type =" text/css " media =" screen "
vue-input-contenteditable
输入不受input[type='text']限制的精美输入。 Vue组件包装器,可通过您期望的所有功能进行contenteditable :
npm i vue-input-contenteditable
下面是单个文件组件中使用的组件示例。 导入后,请确保将其添加到您的components属性中,然后它将在您的模板中可用。
<template>
<input-contenteditable
v-model="myModel"
_is="p"
:placeholder="myPlaceHolder"
:maxlength="25" />
</template>
[removed]
import InputContentedit
var _div = document.querySelector('.discuss_area');
var range = document.createRange();
range.selectNodeContents(_div);
range.collapse(
文章目录简易 div 输入框1.获取光标位置2.设置光标位置3.设置选中区域4.获取选中内容5.在光标处插入内容
简易 div 输入框
HTML5规范引入了 contenteditable 属性,它几乎可以用在任何元素上,只要添加这一属性 即可变为可编译区域。查看详情
一个简单的 <div> 输入框:
<style>
.editdiv{
min-heigh...
文章目录selection对象术语属性方法
window.getSelection(),返回一个Selection对象,表示用户选择的文本范围或光标的当前位置。
selection对象
先来看下面两个selection结果:
selection对象一:
selection对象二:
很奇怪,为什么同样选中一段文字,为什么selection对象的属性会不一样呢?
那就需要明确一下Selection...
开发中,时常会遇到一个开发场景。当我们在编辑一个富文本的时候。当用户选择了某些文字。我们需要弹出一个窗,让用户选择是否把它变成可粗体,比如下面这个图
当然,今天不会直接敲出这个效果,而是解析下而已。毕竟敲出来对新手不友好,需要给予更多的时间思考
通过测试多次。我们发现了几个问题,
1.触发这事件的效果,是需要在鼠标起来的时候触发。
2.鼠标起来的时候,如果被选中的包含太多了。比如多个元素,则不显示操作
简单版本就只有这两点,更复杂的效果是还要考虑类似‘ctrl’+‘A’时的情况
对于上述问题,第一点,使用
依赖于emoji 库,主要是雪碧图实现表情的统一。
采用div标签的contenteditable属性实现编辑功能。 难点在于选择表情时的光标位置保存。实现任意光标位置插入表情。
实现一个小小的demo供大家参考, 共同学习!
可以使用 JavaScript 中的 selectionStart 和 selectionEnd 属性来指定光标的位置。
这些属性属于 HTMLInputElement 对象,因此你需要首先将你的输入元素转换为这种类型,然后就可以使用这些属性了。
const input = document.getElementById('myInput');
input.selectionStart = 2;
input.selectionEnd = 4;
这样就会将光标移动到输入字段的第 2 到第 4 个字符的位置。
注意:在使用这些属性之前,你应该确保输入字段是可编辑的(即具有 "contenteditable" 属性)。