富文本實現@選擇人

来源:https://www.cnblogs.com/gxp69/archive/2022/04/11/16130664.html
-Advertisement-
Play Games

準備工作 npm i wangeditor --save npm i caret-pos --save 組件: <!--富文本--> <div :id="editorEleId" @keydown="onKeyDownInput($event)" @click="onClickEditor" ></ ...


 

 

準備工作

npm i wangeditor --save
npm i caret-pos --save

組件:


<!--富文本-->
<div
    :id="editorEleId"
    @keydown="onKeyDownInput($event)"
    @click="onClickEditor"
></div>
<!--  成員選擇  -->
 <div class="userPopupList" :style="{left: left + 'px', top: top + 'px'}" v-show="show">
      <el-input v-model="userName" ref="input"></el-input>
            <ul>
                <li
                    v-for="user in protectPersons.filter((item) => item.workNick.includes(this.userName))"
                    :key="user.userId"
                    @click="createSelectElement(user.workNick,user.userId)"
                >
                    <el-avatar
                        :size="22"
                        :src="user.icon"
                        class="m-r-10" style="vertical-align: middle"
                    >
                        <img src="../../assets/images/defaultIcon.gif"/>
                    </el-avatar>
                    <span>{{user.workNick}}</span>
                </li>
            </ul>
</div>
.userPopupList{
    position: fixed;
    z-index: 9999;
    /deep/input{
        border:none;
        background: transparent;
    }
    ul{
        max-height: 200px;
        overflow-y: auto;
        border: 1px solid #dbdada;
        background: #fff;
        padding: 10px 10px 0;
        border-radius: 3px;
        li{
            margin-bottom: 5px;
            padding: 5px 10px;
            &:hover{
                background: #f6f5f5;
            }
        }
    }
}
position:any = {};
left:number = 0;
top:number = 0;
show:boolean = false;
isRendering: boolean = false;
userName: string = '';
users:any = [];
  
...富文本初始化
this.editor.config.onchange = () => {
// 生成@的標簽的時候會觸發渲染、此時不要記錄游標坐標
if (this.isRendering === false) {
this.setRecordCoordinates() // 記錄坐標
}else {
this.isRendering = false;
}
};
// 每次點擊獲取更新坐標
onClickEditor() {
this.setRecordCoordinates()
}
// 獲取當前游標坐標
setRecordCoordinates() {
try {
// getSelection() 返回一個 Selection 對象,表示用戶選擇的文本範圍或游標的當前位置。
      if(!this.show){
      const selection:any = window.getSelection();
      this.position = {
      range: selection.getRangeAt(0),
      selection: selection
      }
      }
    } catch (error) {
console.log(error, '游標獲取失敗了~')
}
}
// keydown觸發事件 記錄游標
onKeyDownInput(e:any) {
const isCode = ((e.keyCode === 229 && e.key === '@') || (e.keyCode === 229 && e.code === 'Digit2') || e.keyCode === 50) && e.shiftKey
if (isCode) {
this.setRecordCoordinates(); // 保存坐標
this.getPosition();
// 顯示選擇框,定時原因:1、@會插入到input中,2、游標位置也是input的,會導致插入位置錯誤
setTimeout(()=>{
this.show = true
this.$nextTick(()=>{
(this.$refs.input as any).focus();
})
},200)
}
}
//彈窗列表 - 選人 - 生成@的內容
createSelectElement(name:string, id:string, type = 'default') {
// 獲取當前文本游標的位置。
const { range } = this.position
// 生成需要顯示的內容
let spanNodeFirst:any = document.createElement('span')
spanNodeFirst.className = 'user-node'
spanNodeFirst.style.color = '#409EFF'
spanNodeFirst.innerHTML = `@${name}&nbsp;` // @的文本信息
spanNodeFirst.dataset.id = id // 用戶ID、為後續解析富文本提供
spanNodeFirst.contentEditable = false // 當設置為false時,富文本會把成功文本視為一個節點。

// 需要在字元前插入一個空格否則、在換行與兩個@標簽連續的時候導致無法刪除標簽
let spanNode = document.createElement('span');
spanNode.innerHTML = '&nbsp;';

//創建一個新的空白的文檔片段,拆入對應文本內容
let frag = document.createDocumentFragment()
frag.appendChild(spanNode);
frag.appendChild(spanNodeFirst);
frag.appendChild(spanNode);

// 如果是鍵盤觸發的預設刪除面前的@,前文中我們沒有阻止@的生成所以要刪除@的再插入ps:如果你是數組遍歷的請傳入type 不然會一直刪除你前面的字元。
if (type === 'default') {
const textNode = range.startContainer;
range.setStart(textNode, range.endOffset - 1);
range.setEnd(textNode, range.endOffset);
range.deleteContents();
}
this.isRendering = true;
// 判斷是否有文本、是否有坐標
if ((this.editor.txt.text() || type === 'default')&& this.position && range) {
range.insertNode(frag)
} else {
// 如果沒有內容一開始就插入數據特別處理
this.editor.txt.append(`<span data-id="${id}" style="color: #409EFF" contentEditable="false">@${name}&nbsp;</span>`)
}
this.show = false;
this.userName = '';
}
// 獲取當前游標位置
getPosition () {
const ele:any = this.editor.$textElem.elems[0];
const pos = position(ele)
const off = offset(ele)
const parentW = ele.offsetWidth
// 這個是彈窗列表
const childEle:any = document.getElementsByClassName("userPopupList")
const childW = childEle.offsetWidth
// 彈框偏移超出父元素的寬高
if (parentW - pos.left < childW) {
this.left = off.left - childW
} else {
this.left = off.left
}
this.top = off.top - 4
}

//提交評論
sure(ev: any) {
...
const users = document.querySelectorAll('.user-node');
let userIds:string[] = [];
users.forEach((item:any) => {
userIds.push(item.getAttribute('data-id'))
})
...
}


 


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

-Advertisement-
Play Games
更多相關文章
  • 所有OpenHarmony相關官方直播課程,我們都將在OpenHarmony B站官方賬號“OpenHarmony開發者社區”上彙總發佈。 ...
  • 為了適配更多移動產品形態、提升用戶體驗、以及與合作伙伴協同促進生態建設持續高品質發展,我們本次對《原子化服務上架規範》(後文簡稱:規範)做了更新。 ...
  • 前言 在 $AppClick 事件採集中,還有兩個比較特殊的控制項: UITableView •UICollectionView 這兩個控制項的點擊事件,一般指的是點擊 UITableViewCell 和 UICollectionViewCell。而 UITableViewCell 和 UICollec ...
  • 當用戶卸載App徹底流失時,應用內消息等便捷的用戶觸達交互紐帶將無法再次連通他們,且對於卸載用戶的判定更是技術瓶頸。如何便捷且合理的通過運營策略將卸載用戶召回,是用戶運營同學迫切想知道的答案。 某知名RPG游戲對玩家的活躍度要求極高,通常會判定超過3天未登錄的用戶可能已徹底卸載。對於卸載用戶的召回, ...
  • 經過前面六天的知識學習,對Node.js開發的基礎知識,有了一個初步的掌握,今天繼續學習Node.js後端web開發的相關知識,本篇文章作為Node.js服務端程式開發的基礎入門知識,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 前言 每天網上的博客各個領域都會涌現新文章,有時候看到感興趣的知識就想把某段文字 copy下來 摘錄下來,等有時間後慢慢品味 在部分網站上,如果只是複製少量文字,並沒有什麼不同。但是當我們複製的文字多的話會發現多了一個小尾巴 所謂小尾巴是指在複製文本的最後會多一個作者和出處信息,如下: ···(複製 ...
  • ​1. 什麼是邊框圖片 為了實現豐富多彩的邊框效果,在css3中,新增了 border-image屬性,這個新增屬性允許指定一副圖像作為元素的邊框。​ 2. 邊框圖片的使用場景 盒子大小不一,但是邊框樣式相同,此時就需要邊框圖片來完成,不是背景圖片,而是用邊框圖片來實現。 3. 邊框圖片的切圖原理 ...
  • 1 簡介 jsx(JavaScript XML),即可拓展的JavaScript,是react定義的一種類似於XML的js擴展語法:JS+XML。它本質上是React.createElement(type,config,...children)的語法糖,主要用於創建React元素,生成虛擬DOM 2 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...