摘要:有這么一個功能在網(wǎng)頁中高亮關(guān)鍵字。如下當為時,會將名也錯誤的替換掉關(guān)鍵字父節(jié)點通過來進行背景染色處理,對原始有一定程度污染,可能對再次定位造成影響。
有這么一個功能:在網(wǎng)頁中高亮關(guān)鍵字。
本以為一個 innerHTML replace 就能實現(xiàn)的簡單操作,卻遇到了許多的問題。本文就記錄這些問題和最終的完美解決辦法,
希望能對有同樣遭遇的小伙伴有所幫助。只對結(jié)果感興趣的,忽略過程,直接跳過看結(jié)果吧~
思路:要想高亮元素,那么需要將關(guān)鍵字提取出來用標簽包裹,然后對標簽進行樣式調(diào)整。使用 innerHTML,或 outHTML, 而不能使用 innerText,outText。
const regex = new RegExp(keyword,"g") element.innerHTML = element.innerHTML.replace(regex,""+keyword+"") element.classList.add("highlight")
這樣做存在的隱患有如下:
keyword 如果是 () 等正則對象的關(guān)鍵字將會構(gòu)建正則對象失敗。(可以通過轉(zhuǎn)義解決)
keyword 如果是一些 HTML 標簽如 div 將會對 innerHTML 進行錯誤的替換
keyword 如果和一些DOM屬性名、值相同,也會導(dǎo)致異常替換。如下當 keyword 為 test 時,會將 class 名也錯誤的替換掉:
test
關(guān)鍵字父節(jié)點 element 通過 class 來進行背景染色處理,對原始DOM有一定程度污染,可能對 element 再次定位造成影響。(作為插件希望盡可能少改變原始DOM)
正則優(yōu)化一:僅處理位于標簽內(nèi)的元素var formatKeyword = text.replace(/[-/^$*+?.()|[]{}]/g, "$&") // 轉(zhuǎn)義處理keyword包含的特殊字符,如 /. var finder = new RegExp(">.*?"++".*?<") // 提取位于標簽內(nèi)的文本,避免誤操作 class、id 等 element.innerHTML = element.innerHTML.replace(finder,function(matched){ return matched.replace(text,"
"+text+) })// 對提取的標簽內(nèi)文本進行關(guān)鍵字替換
以能解決大多數(shù)問題,但依舊存在的問題是,只要標簽屬性存在類似 < 符號,將會打破匹配規(guī)則導(dǎo)致正則提取內(nèi)容錯誤, HTML5 dataset 可以自定義任意內(nèi)容,故這些特殊字符是無法避免的。
正則優(yōu)化二:清除可能影響的標簽替換
keyword=》將閉合標簽用變量替換 [replaced1]keyword[replaced2]//閉合標簽內(nèi) id="keyword" 不會被處理 =》 [replaced1]keyword[replaced2] =》將暫存變量 replaced 替換為原先標簽keyword
這種思路及源碼從這里來,
但存在問題是:
如果 [replaced1] 包含 keyword, 那么替換時將發(fā)生異常
最重要的,當標簽值中包含 <> 符號時,此方法也不能正確的提取標簽
總之在經(jīng)過了N多嘗試之后,通過正則都沒能有效的處理各種情況。然后換了個思路,不通過字符串的方式,通過節(jié)點處理。element.childNodes 可以最有效的清理標簽內(nèi)的干擾信息。[完美解決方案]通過 DOM 節(jié)點處理
keyword 1 keyword 2
通過 parent.childNodes 得到所有子節(jié)點。child 節(jié)點可以通過 innerText.replce(keyword,result) 的方式替換得到想要的高亮效果,如下:
keyword 2
(遞歸處理:當child節(jié)點不含子節(jié)點時進行replace操作)。
但是 keyword 1 是屬于文本節(jié)點,只能修改文本內(nèi)容,無法增加 HTML,更無法多帶帶控制其樣式。而文本節(jié)點也不能轉(zhuǎn)換為普通節(jié)點,這也是最苦惱的事情。
最后~,本文的重點來了,因為這個功能,讓我第一次認真接觸到了文本節(jié)點這個東西。從這里發(fā)現(xiàn)了Text對象,使用切割文本節(jié)點并替換的方式實現(xiàn)高亮。
源碼以及還原高亮見源碼
const reg = new RegExp(keyword.replace(/[-/^$*+?.()|[]{}]/g, "$&")) highlight = function (node,reg){ if (node.nodeType == 3) { //只處理文本節(jié)點 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é)點 const wordNew = document.createTextNode(wordNode.data); highlightEl.appendChild(wordNew);//highlight 節(jié)點構(gòu)建成功 wordNode.parentNode.replaceChild(highlightEl, wordNode);// 替換該文本節(jié)點 } } 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)一下。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/53115.html
摘要:有這么一個功能在網(wǎng)頁中高亮關(guān)鍵字。如下當為時,會將名也錯誤的替換掉關(guān)鍵字父節(jié)點通過來進行背景染色處理,對原始有一定程度污染,可能對再次定位造成影響。 有這么一個功能:在網(wǎng)頁中高亮關(guān)鍵字。 本以為一個 innerHTML replace 就能實現(xiàn)的簡單操作,卻遇到了許多的問題。本文就記錄這些問題和最終的完美解決辦法,希望能對有同樣遭遇的小伙伴有所幫助。只對結(jié)果感興趣的,忽略過程,直接跳過...
摘要:高亮顯示通過實現(xiàn)對關(guān)鍵詞的替換,通過添加實現(xiàn)關(guān)鍵詞高亮顯示匹配關(guān)鍵字正則高亮替換值在查詢到結(jié)果后執(zhí)行方法將查詢出來的數(shù)據(jù)傳遞過來通過來將關(guān)鍵詞替換成標簽,同時用中的進行綁定。 最近在做移動real-time-search于實時搜索和關(guān)鍵詞高亮顯示的功能,通過博客的方式總結(jié)一下,同時希望能夠幫助到別人~~~ 如果不喜歡看文字的朋友我寫了一個demo方便已經(jīng)上傳到了github上,可以cl...
摘要:高亮顯示通過實現(xiàn)對關(guān)鍵詞的替換,通過添加實現(xiàn)關(guān)鍵詞高亮顯示匹配關(guān)鍵字正則高亮替換值在查詢到結(jié)果后執(zhí)行方法將查詢出來的數(shù)據(jù)傳遞過來通過來將關(guān)鍵詞替換成標簽,同時用中的進行綁定。 最近在做移動real-time-search于實時搜索和關(guān)鍵詞高亮顯示的功能,通過博客的方式總結(jié)一下,同時希望能夠幫助到別人~~~ 如果不喜歡看文字的朋友我寫了一個demo方便已經(jīng)上傳到了github上,可以cl...
摘要:除官方外的參考文章微信小程序?qū)嵗齽?chuàng)建下發(fā)模板消息實例手把手教你開發(fā)微信小程序之模版消息開發(fā)教你突破小程序模板消息的推送限制獲取用戶信息接口的廢棄問題接口是獲取用戶信息昵稱,頭像等的接口,在官方文檔上寫是即將廢棄。 ----------------更新-------------- 2018年10月10日官網(wǎng)3個接口廢棄的通知: 1、分享監(jiān)聽接口分享消息給好友時,開發(fā)者將無法從callba...
閱讀 2742·2021-09-02 15:11
閱讀 913·2019-08-26 18:18
閱讀 1872·2019-08-26 11:57
閱讀 3325·2019-08-23 16:59
閱讀 2002·2019-08-23 16:51
閱讀 2311·2019-08-23 16:11
閱讀 3131·2019-08-23 14:58
閱讀 1113·2019-08-23 11:34