文章原创作者:codingWeb 链接:https://blog.csdn.net/fesfsefgs/article/details/109540259
这周碰到一个东西,contentEditable
,它是用来指定一个元素是否是可编辑的,这也是富文本编辑器实现的底层支持,网上关于这部分东西的资料比较少或者不全,所以我来整理下关于这个属性,和可编辑区域的一些操作吧,比如获取光标位置,设置光标,往可编辑区域光标处插入内容等等
HTML中的contentEditable的属性可以打开某些元素的可编辑状态.也许你没用过contentEditable属性.甚至从未听说过.contentEditable的作用相当神奇.可以让div或整个网页,以及span等等元素设置为可写。我们最常用的输入文本内容便是input与textarea,使用contentEditable属性后,可以在div,table,p,span,body,等等很多元素中输入内容.
设置一个容器为可编辑区域: 设置contentEditable属性
<body>
<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>
< body>
< 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的强大:
上面只是document.execCommand
的一个命令演示
当一个HTML文档切换到设计模式时,document暴露 execCommand 方法
,该方法允许运行命令来操纵可编辑内容区域的元素。 大多数命令影响document的 selection(粗体,斜体等),当其他命令插入新元素(添加链接)或影响整行(缩进)。当使用contentEditable时,调用 execCommand() 将影响当前活动的可编辑元素
。
命令很多,这里就不列举出来,具体可以参考官方文档document.execCommand命令大全
基于document.execCommand的这些命令
,我们可以做一个小型的富文本编辑器:
<! doctype html >
< html>
< head>
< 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>
是不是很酷?
window.getSelection():文档地址
Selection
对象表示用户选择的文本范围或插入符号的当前位置。它代表页面中的文本选区,可能横跨多个元素。文本选区由用户拖拽鼠标经过文字而产生。要获取用于检查或修改的 Selection
对象,请调用 window.getSelection()
。
Selection 对象所对应的是用户所选择的 ranges (区域),俗称拖蓝。默认情况下,该函数只针对一个区域,我们可以这样使用这个函数:
var selObj = window.getSelection();
var range = selObj.getRangeAt(0);
例子: 我写了如下代码来一起探究下这两个是什么东西吧 每次我拖动完,或者点击完鼠标的时候,都会打印selObj
,查看selObj
对象,看看里面有什么属性
< body>
< div> 我是p标签的内容</ div>
< p> 我是p标签的内容</ p>
< ul>
< li> li1</ li>
< li> li2</ li>
</ ul>
</ 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或容器中往光标处插入内容,返回光标位置
通过js的方式在可编辑区域的光标处插入文本:
方法一: 较完整的,考虑兼容,插入后,光标自动设置到插入后的位置
<! DOCTYPE html >
< html lang = " zh" >
< head>
< 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>
< body>
< 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字符串,可被解析
< body>
< 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>
方法三: 插入一段文本
< body>
< 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>
方法四: 插入一个节点
< body>
< 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 ( ) )
}
光标移动到最后:
< body>
< 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>
记录光标和恢复光标:通过一个例子演示
< body>
< 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>
所有评论(0)