国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發(fā) > HTML5 > 正文

HTML高亮關(guān)鍵字的實(shí)現(xiàn)代碼

2024-09-05 07:19:17
字體:
供稿:網(wǎng)友

本以為一個 innerHTML replace 就能實(shí)現(xiàn)的簡單操作,卻遇到了許多的問題。本文就記錄這些問題和最終的完美解決辦法, 希望能對有同樣遭遇的小伙伴有所幫助。只對結(jié)果感興趣的,忽略過程,直接跳過看結(jié)果吧~

常用做法:正則替換

思路:要想高亮元素,那么需要將關(guān)鍵字提取出來用標(biāo)簽包裹,然后對標(biāo)簽進(jìn)行樣式調(diào)整。使用 innerHTML,或 outHTML, 而不能使用 innerText,outText。

const regex = new RegExp(keyword,"g")element.innerHTML = element.innerHTML.replace(regex,"<b class="a">"+keyword+"</b>")element.classList.add("highlight")

這樣做存在的隱患有如下:

  • keyword 如果是 ()/ 等正則對象的關(guān)鍵字將會構(gòu)建正則對象失敗。(可以通過轉(zhuǎn)義解決)
  • keyword 如果是一些 HTML 標(biāo)簽如 div 將會對 innerHTML 進(jìn)行錯誤的替換
  • keyword 如果和一些DOM屬性名、值相同,也會導(dǎo)致異常替換。如下當(dāng) keyword 為 test 時,會將 class 名也錯誤的替換掉:
<div id="parent">    <div class="test">test</div>  </div>

關(guān)鍵字父節(jié)點(diǎn) element 通過 class 來進(jìn)行背景染色處理,對原始DOM有一定程度污染,可能對 element 再次定位造成影響。(作為插件希望盡可能少改變原始DOM)

正則優(yōu)化一:僅處理位于標(biāo)簽內(nèi)的元素

var formatKeyword = text.replace(/[-////^$*+?.()|[/]{}]/g, '//$&') // 轉(zhuǎn)義處理keyword包含的特殊字符,如 /.var finder = new RegExp(">.*?"++".*?<") // 提取位于標(biāo)簽內(nèi)的文本,避免誤操作 class、id 等element.innerHTML = element.innerHTML.replace(finder,function(matched){        return matched.replace(text,"<br>"+text+</br>)})// 對提取的標(biāo)簽內(nèi)文本進(jìn)行關(guān)鍵字替換

以能解決大多數(shù)問題,但依舊存在的問題是,只要標(biāo)簽屬性存在類似 < 符號,將會打破匹配規(guī)則導(dǎo)致正則提取內(nèi)容錯誤, HTML5 dataset 可以自定義任意內(nèi)容,故這些特殊字符是無法避免的。

<div dataset="p>d">替換</div>

正則優(yōu)化二:清除可能影響的標(biāo)簽

<div id="keyword">keyword</div>  =》將閉合標(biāo)簽用變量替換  [replaced1]keyword[replaced2]//閉合標(biāo)簽內(nèi) id="keyword" 不會被處理  =》  [replaced1]<b>keyword</b>[replaced2]  =》將暫存變量 replaced 替換為原先標(biāo)簽  <div id="keyword"><b>keyword</b></div>

這種思路及源碼從這里來, 但存在問題是:

  • 如果 [replaced1] 包含 keyword, 那么替換時將發(fā)生異常
  • 最重要的,當(dāng)標(biāo)簽值中包含 <> 符號時,此方法也不能正確的提取標(biāo)簽

總之在經(jīng)過了N多嘗試之后,通過正則都沒能有效的處理各種情況。然后換了個思路,不通過字符串的方式,通過節(jié)點(diǎn)處理。element.childNodes 可以最有效的清理標(biāo)簽內(nèi)的干擾信息。

[完美解決方案]通過 DOM 節(jié)點(diǎn)處理

<div id="parent">    keyword 1  <span id="child">    keyword 2  </span> </div>

通過 parent.childNodes 得到所有子節(jié)點(diǎn)。child 節(jié)點(diǎn)可以通過 innerText.replce(keyword,result) 的方式替換得到想要的高亮效果,如下: <span id="child"><b>keyword</b> 2</span> (遞歸處理:當(dāng)child節(jié)點(diǎn)不含子節(jié)點(diǎn)時進(jìn)行replace操作)。

但是 keyword 1 是屬于文本節(jié)點(diǎn),只能修改文本內(nèi)容,無法增加 HTML,更無法單獨(dú)控制其樣式。而文本節(jié)點(diǎn)也不能轉(zhuǎn)換為普通節(jié)點(diǎn),這也是最苦惱的事情。

最后~,本文的重點(diǎn)來了,因?yàn)檫@個功能,讓我第一次認(rèn)真接觸到了文本節(jié)點(diǎn)這個東西。從這里發(fā)現(xiàn)了Text,使用切割文本節(jié)點(diǎn)并替換的方式實(shí)現(xiàn)高亮。

源碼以及還原高亮見源碼

const reg = new RegExp(keyword.replace(/[-////^$*+?.()|[/]{}]/g, '//$&'))highlight = function (node,reg){    if (node.nodeType == 3) {  //只處理文本節(jié)點(diǎn)        const match = node.data.match(new RegExp(reg));        if (match) {          const highlightEl = document.createElement("b");          highlightEl.dataset.highlight="y"          const wordNode = node.splitText(match.index)          wordNode.splitText(match[0].length); // 切割成前 關(guān)鍵詞 后三個Text 節(jié)點(diǎn)          const wordNew = document.createTextNode(wordNode.data);          highlightEl.appendChild(wordNew);//highlight 節(jié)點(diǎn)構(gòu)建成功          wordNode.parentNode.replaceChild(highlightEl, wordNode);// 替換該文本節(jié)點(diǎn)        }    } else if (node.nodeType == 1 && node.dataset.highlight!="y"    ) {        for (var i = 0; i < node.childNodes.length; i++) {            highlight(node.childNodes[i], reg);            i++        }    }  }

最后,留個彩蛋,以上方法也是存在一個小 bug 的,有興趣可以去發(fā)現(xiàn)一下。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持武林網(wǎng)。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 麻城市| 鲁甸县| 钦州市| 四会市| 南阳市| 台中市| 柏乡县| 大石桥市| 临夏县| 分宜县| 凤庆县| 潼南县| 万荣县| 开原市| 乌海市| 门源| 绥阳县| 武陟县| 萍乡市| 达日县| 富蕴县| 平谷区| 德格县| 庐江县| 汕头市| 象州县| 老河口市| 通道| 南川市| 东明县| 德昌县| 上高县| 阳曲县| 青铜峡市| 新巴尔虎左旗| 磴口县| 临漳县| 德格县| 伊金霍洛旗| 丹阳市| 丹棱县|