contenteditable 插入及粘貼純文本內容

来源:https://www.cnblogs.com/_error/archive/2018/04/18/8872996.html
-Advertisement-
Play Games

本文主要介紹 div 標簽設置 contenteditable = ' true ' 時,在游標位置插入輸入的內容,或在游標位置粘貼純文本內容。文中涉及知識,可參考以下: http://www.zhangxinxu.com/wordpress/2016/01/contenteditable-plai ...


  本文主要介紹 div 標簽設置  contenteditable = ' true ' 時,在游標位置插入輸入的內容,或在游標位置粘貼純文本內容。文中涉及知識,可參考以下: http://www.zhangxinxu.com/wordpress/2016/01/contenteditable-plaintext-only/ 

http://www.jb51.net/article/57650.htm 

https://www.cnblogs.com/rainman/archive/2011/02/27/1966482.html

http://www.zhangxinxu.com/wordpress/2011/04/js-range-html%E6%96%87%E6%A1%A3%E6%96%87%E5%AD%97%E5%86%85%E5%AE%B9%E9%80%89%E4%B8%AD%E3%80%81%E5%BA%93%E5%8F%8A%E5%BA%94%E7%94%A8%E4%BB%8B%E7%BB%8D/

http://kjah.iteye.com/blog/422509

  先搭好 html ,一個按鈕用於插入操作,一個 div 實現 contenteditable 功能,及主要邏輯

<button type="button" id='insert'>插入標題</button>
<div contentEditable="true" id="editor">
    <h3>副標題</h3>
    <p>文本</p>
</div>

  接著是實現 js 邏輯功能

document.getElementById('insert').onclick=function(){
    // 點擊插入按鈕時,系統彈框輸入內容
    var content = prompt('請輸入內容');

    // 防止編輯框沒有獲取焦點時點擊,故加一個操作,使編輯框獲取焦點
    document.getElementById('editor').focus();

    // 執行插入方法
    if(!!content){
        insertHtml('<h4>'+content+'</h4>');
    }
}

  主要邏輯在 insertHtml 方法中

function insertHtmlAtCaret(html) {
    //Selection 對象,表示用戶選擇的文本範圍或游標的當前位置。
    //在非IE瀏覽器(Firefox、Safari、Chrome、Opera)下可以使用window.getSelection()獲得selection對象
    //anchor 選中區域的“起點”。
    //focus 選中區域的“結束點”。
    //range 是一種fragment(HTML片斷),它包含了節點或文本節點的一部分。一般情況下,同一時刻頁面中只可能有一個range,也有可能是多個range(使用Ctrl健進行多選,不過有的瀏覽器不允許,例如Chrome)。可以從selection中獲得range對象,也可以使用document.createRange()方法獲得。
    var sel = window.getSelection(), 
        range;

    if (sel.getRangeAt && sel.rangeCount) {
        //getRangeAt(index) 從當前selection對象中獲得一個range對象。
        range = sel.getRangeAt(0);
        //deleteContents()方法,range內容會被刪除
        range.deleteContents();
        //將輸入的內容寫入並載入到 dom 中
        var el = document.createElement("div");
        el.innerHTML = html;
        var frag = document.createDocumentFragment(), node, lastNode;
        while ( (node = el.firstChild) ) {
            lastNode = frag.appendChild(node);
        }
        //insertNode,在range的開始位置插入一 個節點
        range.insertNode(frag);
        //收尾
        if (lastNode) {
            range = range.cloneRange();
            range.setStartAfter(lastNode);
            range.collapse(true);
            sel.removeAllRanges();
            sel.addRange(range);
        }
    }
    
}

  此時往編輯框里粘貼內容,會帶上原本樣式,顯然不是我們要的結果,需對粘貼文本進行更改

document.getElementById('editor').onpaste=function(event){
    var e = event || window.event
    // 阻止預設粘貼
    e.preventDefault();
    // 粘貼事件有一個clipboardData的屬性,提供了對剪貼板的訪問
    // clipboardData的getData(fomat) 從剪貼板獲取指定格式的數據
    var text =  (e.originalEvent || e).clipboardData.getData('text/plain') || prompt('在這裡輸入文本');
    // 插入
    document.execCommand("insertText", false, text);
};

  最後附上完整代碼

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>contenteditable</title>
    <style>
        #insert{
            width: 90px;
            height: 30px;
            border: 1px solid #ccc;
            background-color: #fff;
            margin: 10px;
        }
        #editor{
            padding: 10px;
            overflow-y: auto;
            min-height:200px; 
            border:1px solid #f33;
            outline: 0;
        }
        #editor h4{
            margin: 10px 0;
        }
    </style>
</head>
<body>

<button type="button" id='insert'>插入標題</button>
<div contentEditable="true" id="editor">
    <h3>副標題</h3>
    <p>文本</p>
</div>
    
</body>
<script>

document.getElementById('insert').onclick=function(){
    var content = prompt('請輸入內容');
    document.getElementById('editor').focus();
    if(!!content){
        insertHtmlAtCaret('<h4>'+content+'</h4>');
    }
}

document.getElementById('editor').onpaste=function(event){
    var e = event || window.event
    e.preventDefault();
    var text =  (e.originalEvent || e).clipboardData.getData('text/plain') || prompt('在這裡輸入文本');

    document.execCommand("insertText", false, text);
};

function insertHtmlAtCaret(html) {
    var sel = window.getSelection(), 
        range;

    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);
        }
    }
}
</script>
</html>

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這一節繼續深入Router模塊,首先從最常用的use開始。 router.use 方法源碼如下: 前半部分十分熟悉,根本就是app.use的翻版。 當然,最後遍歷中間件函數處理的時候就不一樣了,引入了新的本地模塊Layer。 Layer 不太理解這個層的意義,無論是app.use還是router.u ...
  • 這篇文章收藏在筆記裡面有幾年了,原文作者無從考究了,侵刪! 最近再翻出來看,依然覺得受益匪淺。這次整理分享出來,希望對大家有所啟示。 背景 如果你是剛進入WEB前端研發領域,想試試這潭水有多深,看這篇文章吧; 如果你是做了兩三年WEB產品前端研發,迷茫找不著提高之路,看這篇文章吧; 如果你是四五年的 ...
  • 1. 想要使用Node.js,必須先配置開發環境。進入Node.js官網下載指定平臺安裝包(nodejs.org),Windows系統下載安裝紅框的安裝包,macOS 下載安裝藍框的安裝包,看清楚自己電腦是32位還是64位。一直下一步直到完成就好。 Node.js的開發非常活躍,更新時以兩個版本更新 ...
  • 要實現的頁面效果: 1.顯示歷史搜索, 2.最近搜索的排在最前, 2.最多顯示8條歷史 4.清空歷史記錄 思路: 1.首先顯示歷史記錄需要一個數組searchItems,通過ng-repeat顯示每一個<li> 註:關於搜索實現,見:angularjs+ionic的app端分頁和條件 2.js實現 ...
  • Array.prototype.uniq = function () { var arr = []; var flag = true; this.forEach(function(item) { // 排除 NaN (重要!!!) NaN 和自身不相等 // 除了NaN 其他數據 和 自己都 相等 ...
  • <div class="upload_box"> <div id="preview"> <img id="imghead" src="0.jpg" width="190" height="130"/> <!--圖片顯示位置--> </div> <b>上傳圖片</b> <input type="fil ...
  • 原因 : Vue 的組件作用域都是孤立的,不允許在子組件的模板內直接引用父組件的數據。必須使用特定的方法才能實現組件之間的數據傳遞。 props 父組件給子組件傳遞數據 props:作用是父組件給子組件傳遞數據。 語法:參考《vue(二)-父子組件語法》。 註意要點: 1: 子組件要顯式聲明需要哪些 ...
  • 最近剛辭了原來的那家公司,準備新找一份工作。其中有個公司要求會Openlayers3。一看到這個要求,就知道公司業務涉及地圖圖表比較多。 Openlayers本身是一個基於GIS地圖相關的功能豐富的JS組件庫,功能和方法很多,學起來也需要一點點慢慢學習,邊學邊用。這裡先簡單介紹一個入門實例。 代碼見 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...