React实现模糊搜索和关键字高亮

模糊搜索与关键词点亮

背景

公司需要一个可视化页面,作为Redis查询展示的工具,方便同事进行快速检索。

要求将后端返回的报文,完整展现在识图中,并且可以提供关键词检索和关键词点亮的功能,以便快速定位关键词。

整体效果预览:

主要功能

  1. 点击搜索:返回的大字符串形式报文被组装成JSON的格式,点击搜索匹配相应字段, 并高亮显示
  2. 点击区分大小写:模糊匹配 OR 精确匹配
  3. 查找下一个:被选中的字段,往下移动一个,下一个的背景颜色切换为另一高亮色

解决

  1. 匹配到的文本,动态插入样式
  2. 使用正则解决匹配问题
  3. 点击下一个时,动态切换类名和背景色

代码实现

仅仅贴出主要代码

DOM

jsx 复制代码
...
<Input 
    value={this.state.txt}
    placeholder="请输入检索内容"
    onChange={e => this.setState({txt: e.target.value})}
></Input>
<Button onClick={this.dispatchHighLightText}>
    搜索
</Button>
...
<CheckBos 
    checked={this.state.isCheck}
    onChange={e => this.setState({isCheck: e.target.checked})}
>
	区分大小写
</CheckBos>
...
<Button
    onClick={this.searchToNext}
>
    查找下一个
</Button>

...
<pre
    dangerouslySetInnerHTML={{__html: this.state.information}}
    contentEditable
>
    {/*动态插入html的容器*/}
</pre>

CSS

less 复制代码
// 每个匹配到的都有的类名
.highLight{
    background-color: #a9a9a9;
}
// 点击下一个时的类名,默认为第一个
.current{
    background-color: #32aaf8;
}

JS

  • 点击搜索时
jsx 复制代码
colorIndex = 0; // 初始化高亮标记
_information = ''; // 文本备份
dispatchHighLightText = () => {
    const {txt} = this.state; // 检索内容
    if(!txt){
        message.warning("请先输入查询条件"); 
        return;
    }
    // 重置高亮标记
    colorIndex = 0;
    const regex = new RegExp(txt, this.state.isCheck ? 'g' : 'gi'); // 不勾选 => 模糊匹配
    // 由于每次插入样式都会污染文本,所以每次都需要从备份文本中重新渲染
    const newInformation = _information;
    const hightLightTxt = newInformation.replace(regex, (match,index) => `<span class="highLight"> ${match} </span>`);
    this.setState({infomation: hightLightTxt}, () => {
        // 视图更新渲染然后,获取到dom
        let highLightEle = document.querySelectorAll('.highLight');
        this.updateHeight(highLightEle)
    }); // render
}

// 点亮目标关键字
updateHeight = (highLightEle) => {
    highLightEle.forEach((element, index) => {
        if(index === colorIndex){
            element.classList.add('current');
            element.scrollIntoView({ behavior: 'smooth', block: 'center'});
        }else{
            element.classList.remove('current');
        }
    })
}
  • 点击 "下一个"
jsx 复制代码
searchToNext = () => {
    let highLightEle = document.querySelectorAll('.highLight');
    if(heighLightEle.length){
        colrIndex = (colorIndex + 1 + highLightEle.length) % highLightEle.length;
        this.updateHeight(highLightEle)
    }
}

反思

  1. 本来想用偏移量window.getSelection 和 document.createRange 以及相关api模拟富文本实现光标跳转定位的,可是发现偏移量不好确定(小菜鸡)
  2. 每次渲染的时候需要拿新的备份的文本进行替换,不要用脏数据
  3. pre标签可以很好的保留文本原来的样式,不能用textarea
  4. dangerouslySetInnerHTML接收的是一个对象

仅以本文作为工作记录

相关推荐
Mr Xu_6 分钟前
从后端数据到前端图表:深入解析 reduce 与 flatMap 的数据整形实战
前端·javascript
玖玖passion24 分钟前
Windows 上部署 Hermes Agent 完整指南 - 让你的 AI 助手在 WSL2 中跑起来
前端·后端·github
AC赳赳老秦1 小时前
OpenClaw多平台部署:Windows+Linux跨系统协同,实现全场景覆盖
linux·服务器·前端·网络·windows·deepseek·openclaw
喜欢吃鱿鱼1 小时前
DES加解密(附带解决转义问题)-VUE
开发语言·前端·javascript
腹黑天蝎座2 小时前
前端性能优化实战指南:从原理到落地的全方位解决方案
前端·性能优化·监控
忆往wu前2 小时前
一文通透 Vue动态组件体系:插槽|数据监听|组件通信|动态切换|缓存—闭环
前端·面试
奇奇怪怪的问题2 小时前
问题总结:关于封装axios问题,导致外部使用接口报错,无法进入error回调
前端·axios
Jenlybein2 小时前
速学 VS Code 插件开发入门,客制化你的开发体验
前端·javascript·visual studio code
qq_437100662 小时前
SSE 流式响应(Server-Sent Events)
前端·sse