【组件】前端ElementUi 下拉Tree树形组件 带模糊搜索自动展开高亮功能

【组件】前端ElementUi 下拉Tree树形组件 带模糊搜索自动展开高亮功能

https://live.csdn.net/v/435737

javascript 复制代码
<template>
  <div>
    <el-popover
      style="overflow-y: auto; "
      placement="bottom"
      trigger="click">
      <el-input
        style="margin-bottom: 10px"
        v-model="搜索内容"
        clearable
        @clear="searchHandleIptClear"
      >
        <el-button slot="append" icon="el-icon-search" @click="search" size="small" >搜索</el-button>
      </el-input>
<!--      树结构-->
      <el-tree
        style="width:auto"
        ref="tree"
        :props="props"
        :data="data"
        :check-strictly="false"
        @check-change="handleCheckChange"
        show-checkbox
        node-key="id"
        :default-expanded-keys="需要展开的节点keys"
        :default-checked-keys="[]"
        @node-click="handleNodeClick"
      >
        <span class="custom-tree-node" slot-scope="{ node, data }">
<!--          高亮的部分-->
          <span v-if="需要高亮节点数组.indexOf(data.id) !== -1" style="background-color: yellow">{{ node.label }}</span>
<!--          不需要高亮的部分-->
          <span v-else>{{ node.label }}</span>
        </span>
      </el-tree>
<!--      输入框 用来显示选中的节点内容-->
      <el-input slot="reference"
                style="width:380px"
                v-model="value.label"
                placeholder="节点"
                clearable
                readonly
                @clear="handleIptClear">
      </el-input>
    </el-popover>
  </div>
</template>
<script>
export default {
  data() {
    return {
      data: [{
        id: 1,
        label: '一级 1 A',
        children: [{
          id: 4,
          label: '二级 1-1 张三B',
          children: [{
            id: 9,
            label: '三级 1-1-1 李四C'
          }, {
            id: 10,
            label: '三级 1-1-2 王五a'
          }]
        }]
      }, {
        id: 2,
        label: '一级 2 赵六c',
        children: [{
          id: 5,
          label: '二级 2-1 张三b'
        }, {
          id: 6,
          label: '二级 2-2 李四a'
        }]
      }, {
        id: 3,
        label: '一级 3 王五D',
        children: [{
          id: 7,
          label: '二级 3-1 赵六d'
        }, {
          id: 8,
          label: '二级 3-2 钱七a'
        }]
      }],
      props: {
        label: 'label',
        children: 'children'
      },
      count: 1,
      value:{id:'', label: ''},
      需要高亮节点数组:[],
      需要展开的节点keys:[],
      搜索内容:''
    };
  },
  methods: {
    searchHandleIptClear(){
      this.需要高亮节点数组 = []
    },
    高亮模糊查询的节点递归(arr, 父节点){
      if(!arr || arr.length <= 0){
        return
      }
      let that = this
      arr.forEach(item=>{
        //不区分大小写匹配
        if(item.label.toUpperCase().indexOf(that.搜索内容.toUpperCase()) !== -1){
          console.log('搜索内容匹配', that.搜索内容, item, 父节点)
          that.需要高亮节点数组.push(item.id);
          if(父节点){
            that.需要展开的节点keys.push(父节点.id)
          }
        }
        if(item.children){
          that.高亮模糊查询的节点递归(item.children, item)
        }
      })
    },
    /**
     * 点击搜索
     */
    search(){
      let that = this
      //1 过滤所有内容
      that.需要高亮节点数组 = []
      that.需要展开的节点keys = []
      if(that.搜索内容){
        this.高亮模糊查询的节点递归(this.data, null)
      }
      //3 展开匹配到的节点内容
    },
    // 清空输入框内容
    handleIptClear(){
      console.log('清空输入框内容')
      //清空选中内容
      this.$refs.tree.setCheckedNodes([])
      this.value.id = ''
      this.value.label = ''
    },
    /**
     * 更新被选中的值
     */
    updateCheck(){
      const seltedNodes = this.$refs.tree.getCheckedNodes()
      console.log(seltedNodes)
      const ids = seltedNodes.map(n => n.id)
      const labels = seltedNodes.map(n => n.label)
      this.value.id = ids + ''
      this.value.label = labels + ''
    },
    // checkbox被选中或取消选中
    handleCheckChange(arg1, arg2, arg3){
      console.log(arg1, arg2, arg3);
      this.updateCheck()
    },
    // 节点被点击
    handleNodeClick(arg1, arg2, arg3){
      console.log('nodes:', arg1, arg2, arg3)
      this.updateCheck()
    },
  }
};
</script>
相关推荐
一个很帅的帅哥1 小时前
Vue keep-alive
前端·javascript·vue.js·keep-alive
lbh1 小时前
Chrome DevTools 详解(一):Elements 面板
前端·javascript·浏览器
明里人1 小时前
React 状态库:Zustand 和 Jotai 怎么选?
前端·javascript·react.js
儒雅的烤地瓜2 小时前
JS | 如何把一个伪数组转换成一个真正的数组?
javascript·from方法·数组转换·扩展运算符·slice方法·push方法
β添砖java4 小时前
交互动效设计
前端·javascript·交互
一条有腹肌的咸鱼4 小时前
vue3+vite+element-plus封装npm插件遇到的问题,求大神搭救
vue.js
用户56170657111474 小时前
scratch二次开发--如何在舞台区开启网络摄像头(Turbowarp版)
javascript
JamesGosling6665 小时前
详解 Vue 3.6 Vapor Mode:从原理到问题,看透 VDOM 逐步退场的底层逻辑
前端·vue.js
一个很帅的帅哥5 小时前
Vue中的hash模式和history模式
前端·vue.js·history模式·hash模式
进阶的鱼5 小时前
React+ts+vite脚手架搭建(三)【状态管理篇】
前端·javascript·react.js