前端交流圈

可编辑div问题总结(光标,显示等)

2020-06-30  本文已影响0人  lulu_c

背景:需要编写一个简单的文本输入框,要求可以添加超链接,并且超链接可以修改内容,超链接在文本框中只显示标题(类似在html中显示a标签一样)
目前的基于vue框架:

html:

<div 
ref="myeditor" 
contenteditable="true" 
class="myeditor" 
v-html="text.textarea" 
@click="editlink($event)" 
@input="myeditorChange"
@keypress.enter="myeditorenter($event)"  // 监听回车
@blur="myeditorblur">
</div>

contenteditable属性可以把div变成可编辑状态,这是开发文本编辑框的基础。添加该属性后基本是满足一般文本输入。

1.换车键问题:
在可编辑div回车会在文本框中添加div标签(也可能是p标签),所以需要在div中监控回车事件并取消默认事件

 myeditorenter(e) {
            e.preventDefault();
            RangeUtil.insertHtmlAtCaret('<br/>')  // 在当前光标插入文本
            // this.$refs.myeditor.innerHTML = this.$refs.myeditor.innerHTML+'<br/>'
        },

2.获取div的内容
需要使用this.$refs.myeditor.innerHTML来获取,text.textarea虽然绑定了值但实际是获取不到的,除非有进行手动赋值。

3.添加超链接
这里主要讲讲添加的方式,具体可根据需求来。
因为我这里是点击一个按钮打开弹框的方式来添加超链接,所以:

这样能确保超链接是添加在你之前光标定位的地方,不然就会出现超链接只能添加在文本尾部的情况。

涉及光标的方法:

// 光标类方法
export const RangeUtil = {
    // 当前光标位置插入文本
    insertHtmlAtCaret: (html) => {
        let sel, range;
        if (window.getSelection) {
            // IE9 and non-IE
            sel = window.getSelection();
            if (sel.getRangeAt && sel.rangeCount) {
                range = sel.getRangeAt(0);
                range.deleteContents();
                // Range.createContextualFragment() would be useful here but is
                // non-standard and not supported in all browsers (IE9, for one)
                let el = document.createElement("div");
                el.innerHTML = html;
                let frag = document.createDocumentFragment(), node, lastNode;
                while ((node = el.firstChild)) {
                    lastNode = frag.appendChild(node);
                }
                range.insertNode(frag);
                // Preserve the selection
                if (lastNode) {
                    range = range.cloneRange();
                    range.setStartAfter(lastNode);
                    range.collapse(true);
                    sel.removeAllRanges();
                    sel.addRange(range);
                }
            }
        } else if (document.selection && document.selection.type != "Control") {
            // IE < 9
            document.selection.createRange().pasteHTML(html);
        }
    },
    // 保存光标位置
    saveRange: () => {
        let selection = window.getSelection ? window.getSelection() : document.selection;
        if (!selection.rangeCount) return;
        let range = selection.createRange ? selection.createRange() : selection.getRangeAt(0);
        window._range = range;
    },
    // 基于保存的光标插入内容
    insertContent(str) {
        let selection, range = window._range;
        if (!window.getSelection) {
            range.pasteHTML(str);
            range.collapse(false);
            range.select();
        } else {
            selection = window.getSelection ? window.getSelection() : document.selection;
            range.collapse(false);
            let hasR = range.createContextualFragment(str);
            let hasR_lastChild = hasR.lastChild;
            while (hasR_lastChild && hasR_lastChild.nodeName.toLowerCase() == "br" && hasR_lastChild.previousSibling && hasR_lastChild.previousSibling.nodeName.toLowerCase() == "br") {
                let e = hasR_lastChild;
                hasR_lastChild = hasR_lastChild.previousSibling;
                hasR.removeChild(e);
            }
            range.insertNode(hasR);
            if (hasR_lastChild) {
                range.setEndAfter(hasR_lastChild);
                range.setStartAfter(hasR_lastChild);
            }
            selection.removeAllRanges();
            selection.addRange(range);
        }  
    },
上一篇下一篇

猜你喜欢

热点阅读