前端页面如何让关键字凸显

今天分享下”前端页面如何让关键字凸显“这篇文章,文中根据实例编码详细介绍,或许对大家的编程之路有着一定的参考空间与使用价值,需要的朋友接下来跟着云南仟龙Mark一起学习一下吧。 近期做新项目碰到这种的一个作用:在页面中高亮关键字。

本认为一个 innerHTML replace 就能完成的简便实际操作,却碰到了众多的难题。文中就纪录这种情况和最后的极致解决方案, 期待能对有一样遇到的朋友有一定的协助。只对結果有兴趣的,忽视全过程,立即绕过看效果吧~

常见作法:正则替换

思路:要想高亮原素,那麼必须将关键字获取出去用标识包囊,随后对标贴开展款式调节。应用 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”)​​

这样做存在的隐患有如下:

1

2

3

4

5

​​()\​​

​​div​​

​​<div id=​​​​"parent"​​​​>​​

​​<div class=​​​​"test"​​​​>test​​

​​​​

关键字父节点 element 通过 class 来进行背景染色处理,对原始DOM有一定程度污染,可能对 element 再次定位造成影响。(作为插件希望尽可能少改变原始DOM)

正则优化一:仅处理位于标签内的元素

1

2

3

4

5

6

​​var formatKeyword = text.replace(/[-/\^$*+?.()|[]{}]/g, ‘\$&’) // 转义处理keyword包含的特殊字符,如 /.​​

​​var finder = new RegExp(">.?"++".?<") // 提取位于标签内的文本,避免误操作 class、id 等​​

​​element.innerHTML = element.innerHTML.replace(finder,function(matched){​​

​​return matched.replace(text,"<​​​​br​​​​>"+text+</​​​​br​​​​>)​​

​​})// 对提取的标签内文本进行关键字替换​​

以能解决大多数问题,但依旧存在的问题是,只要标签属性存在类似 < 符号,将会打破匹配规则导致正则提取内容错误, HTML5 dataset 可以自定义任意内容,故这些特殊字符是无法避免的。

1

​​<​​​​div​​​ ​​dataset=“p>d”>替换</​​​​div​​​​>​​

正则优化二:清除可能影响的标签

1

2

3

4

5

6

7

​​<​​​​div​​​ ​​id​​​​=​​​​"keyword"​​​​>keyword</​​​​div​​​​>​​

​​=》将闭合标签用变量替换​​

​​[replaced1]keyword[replaced2]//闭合标签内 id=“keyword” 不会被处理​​

​​= ​​http://www.qlyl1688.com/​​》​​

​​[replaced1]<​​​​b​​​​>keyword</​​​​b​​​​>[replaced2]​​

​​=》将暂存变量 replaced 替换为原先标签​​

​​<​​​​div​​​ ​​id​​​​=​​​​"keyword"​​​​><​​​​b​​​​>keyword</​​​​b​​​​></​​​​div​​​​>​​

这种思路及源码从这里来, 但存在问题是:
如果 [replaced1] 包含 keyword, 那么替换时将发生异常
最重要的,当标签值中包含 <> 符号时,此方法也不能正确的提取标签

总之在经过了N多尝试之后,通过正则都没能有效的处理各种情况。然后换了个思路,不通过字符串的方式,通过节点处理。element.childNodes 可以最有效的清理标签内的干扰信息。

[完美解决方案]通过 DOM 节点处理

1

2

3

4

5

6

​​<​​​​div​​​ ​​id​​​​=​​​​"parent"​​​​>​​

​​keyword 1​​

​​<​​​​span​​​ ​​id​​​​=​​​​"child"​​​​>​​

​​keyword 2​​

​​</​​​​span​​​​>​​

​​</​​​​div​​​​>​​

通过 parent.childNodes 得到所有子节点。child 节点可以通过 ​​innerText.replce(keyword,result) ​​​的方式替换得到想要的高亮效果,如下: ​​keyword 2 ​​(递归处理:当child节点不含子节点时进行replace操作)。

但是 keyword 1 是属于文本节点,只能修改文本内容,无法增加 HTML,更无法单独控制其样式。而文本节点也不能转换为普通节点,这也是最苦恼的事情。

最后~,本文的重点来了,因为这个功能,让我第一次认真接触到了文本节点这个东西。从这里发现了Text,使用切割文本节点并替换的方式实现高亮。

​ ​源码以及还原高亮见源码​​

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

​​const reg = new RegExp(keyword.replace(/[-/\^$*+?.()|[]{}]/g, ‘\$&’))​​

​​highlight = function (node,reg){​​

​​if (node.nodeType == 3) { //只处理文本节点​​

​​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); // 切割成前 关键词 后三个Text 节点​​

​​const wordNew = document.createTextNode(wordNode.data);​​

​​highlightEl.appendChild(wordNew);//highlight 节点构建成功​​

​​wordNode.parentNode.replaceChild(highlightEl, wordNode);// 替换该文本节点​​

​​}​​

​​} else if (node.nodeType == 1 && node.dataset.highlight!="y"​​

​​) {​​

​​for (var i = 0; i < node.childNodes.length; i++) {​​

​​highlight(node.childNodes[i], reg);​​

​​i++​​

​​}​​

​​} ​​

​​}​​

今天分享下”“这篇文章,文中根据实例编码详细介绍,或许对大家的编程之路有着一定的参考空间与使用价值,需要的朋友接下来跟着云南仟龙Mark一起学习一下吧。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值