Vue中用element-ui封装的el-tree和el-select实现下拉树状多选单选,灵活自定义模糊查询、单选多选,默认选中值。

一.实现效果

1.多选样式

在这里插入图片描述

2.单选样式

![在这里插入图片描述](https://img-blog.csdnimg.cn/20210428161723110.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2ltYXBpZ18=,size_16,color_FFFFFF,t_70

二.组件代码

文件路径:pages/publicComp/selectTree

<template>
  <div>
    <el-popover
      popper-class="selectTree"
      v-clickoutside="replyOutside"
      placement="bottom-start"
      :width="popoverWidth"
      trigger="manual"
      v-model="isShowSelect"
      @hide="popoverHide"
    >
      <div class="pdlr_1">
        <el-input
          class="mb_5"
          v-if="filterable"
          v-model="filterText"
          :placeholder="$t('search')"
        ></el-input>
      </div>

      <el-tree
        class="common-tree"
        :indent="multiple ? 46 : 24"
        :width="width"
        ref="tree"
        :data="treeData"
        :props="obj"
        :filter-node-method="filterNode"
        :show-checkbox="multiple"
        :node-key="obj.id"
        :check-strictly="checkStrictly"
        :default-expanded-keys="defaultKey"
        :default-expand-all="defaulexpand"
        :expand-on-click-node="multiple && expandClickNode"
        :check-on-click-node="checkClickNode"
        :highlight-current="true"
        @check-change="nodeClick"
        @node-click="nodeClick"
      ></el-tree>

      <el-select
        slot="reference"
        ref="select"
        :size="size"
        v-model="returnDataKeys"
        :multiple="multiple"
        :clearable="clearable"
        :collapse-tags="collapseTags"
        @click.native="selectClick"
        @remove-tag="removeTag"
        @clear="clean"
        class="tree-select"
        :placeholder="placeholder"
      >
        <el-option
          v-for="item in options"
          :key="item.value"
          :label="item.label"
          :value="item.value"
        ></el-option>
      </el-select>
      <div class="mt_5" v-if="selectAll">
        <el-checkbox
          v-model="checkedAll"
          :indeterminate="returnDataKeys.length == 0 ? false : !checkedAll"
          @change="selectedAll"
          >全选</el-checkbox
        >
      </div>
    </el-popover>
  </div>
</template>
 
<script>
//点击外部关闭下拉框
import Clickoutside from "element-ui/src/utils/clickoutside";
import $ from "jquery";
import util from "@/assets/utils/util";
export default {
  directives: { Clickoutside },
  props: {
    //全选
    selectAll: {
      type: Boolean,
      default() {
        return false;
      },
    },
    // 树结构数据
    data: {
      type: Array,
      default() {
        return [];
      },
    },
    obj: {
      type: Object,
      required: false,
      default: () => {
        return {
          id: "", // id
          pid: "", //父级id
          label: "", // 显示名称
          children: "", //子级字段名
        };
      },
    },
    //配置是否可以搜索
    filterable: {
      type: Boolean,
      default() {
        return false;
      },
    },
    //输入框占位值
    placeholder: {
      type: String,
      default() {
        return "请选择";
      },
    },
    //配置是否可多选
    multiple: {
      type: Boolean,
      default() {
        return false;
      },
    },
    // 配置是否可清空选择
    clearable: {
      type: Boolean,
      default() {
        return false;
      },
    },
    // 配置多选时是否将选中值按文字的形式展示
    collapseTags: {
      type: Boolean,
      default() {
        return false;
      },
    },
    // 显示复选框情况下,是否严格遵循父子不互相关联
    checkStrictly: {
      type: Boolean,
      default() {
        return false;
      },
    },
    radioStrictly: {
      type: Boolean,
      default() {
        return false;
      },
    },
    //多选是设置点击节点是否可以选中
    checkClickNode: {
      type: Boolean,
      default() {
        return true;
      },
    },
    //多选时:点击节点展开还是点三角标
    expandClickNode: {
      type: Boolean,
      default() {
        return false;
      },
    },
    //默认展开
    defaulexpand: {
      type: Boolean,
      default() {
        return true;
      },
    },
    // 默认选中的节点key
    defaultKey: {
      type: [Number, String, Array, Object],
      default() {
        return [];
      },
    },
    size: {
      type: String,
      default() {
        return "small";
      },
    },
    width: {
      type: String,
      default() {
        return "100%";
      },
    },
    height: {
      type: String,
      default() {
        return "300px";
      },
    },
    curValue: {
      default() {
        return "";
      },
    },
  },

  //上面是父组件可传入参数
  data() {
    return {
      popoverWidth: "0px", //下拉框大小
      isShowSelect: false, // 是否显示树状选择器
      options: [], //select option选项
      returnDatas: [], //返回给父组件数组对象
      returnDataKeys: [], //返回父组件数组主键值
      filterText: "", //筛选绑定值
      checkedAll: false, //是否有全选按钮(只有在多选下才能配置)
      treeArrId: [],
    };
  },

  mounted() {
    // var that = this; //this的指向问题
    // document.addEventListener("click", function (e) {
    //   console.log(e);
    //   that.isShowSelect = false; //这里that代表组件,this代表document
    // });
    this.$nextTick(() => {
      if (this.curValue) {
        if (
          typeof this.curValue != "string" &&
          typeof this.curValue != "number"
        ) {
          this.setKeys(this.curValue);
        } else {
          this.setKey(this.curValue);
        }
      }
    });
  },

  computed: {
    treeData() {
      // 若非树状结构,则转化为树状结构数据
      if (this.obj.children != "") {
        // return JSON.stringify(this.data).indexOf(this.obj.children) !== -1
        //   ? this.data
        //   : this.switchTree();
        return this.data;
      } else {
        return this.data;
      }
    },
  },
  watch: {
    returnDataKeys(newValue, oldValue) {
      if (newValue.length === this.treeArrId.length) {
        this.checkedAll = true;
      } else {
        this.checkedAll = false;
      }
    },
    curValue(val) {
      this.$nextTick(() => {
        if (val) {
          if (typeof val != "string" && typeof val != "number") {
            this.setKeys(val);
          } else {
            this.setKey(val);
          }
        }
      });
    },

    filterText(val) {
      this.$refs.tree.filter(val);
    },
    // eslint-disable-next-line no-unused-vars
    isShowSelect(val) {
      // 隐藏select自带的下拉框
      this.$refs.select.blur();
    },
    treeData() {
      //监听tree数据
      this.$nextTick(() => {
        this.init();
      });
    },
  },
  created() {
    this.treeArrId = [];
    this.originTree(this.treeData);
  },
  methods: {
    originTree(treeData) {
      treeData.forEach((item) => {
        this.treeArrId.push(item[this.obj.id]);
        if (item.children && item.children.length > 0) {
          this.originTree(item.children);
        }
      });
    },
    // 点击外部关闭
    replyOutside() {
      let e = e || window.event;
      let elem = e.target;
      if ($(elem).is(".selectTree") || $(elem).is(".selectTree *")) {
        return;
      } else if (this.isShowSelect) {
        this.isShowSelect = false;
      }
    },
    filterNode(value, data) {
      if (!value) return true;
      return data[this.obj.label].indexOf(value) !== -1;
    },

    init() {
      // debugger
      if (this.defaultKey != undefined && this.defaultKey.length > 0) {
        if (this.multiple) {
          // 多选
          if (
            Object.prototype.toString.call(this.defaultKey).indexOf("Array") !=
            -1
          ) {
            if (
              Object.prototype.toString
                .call(this.defaultKey[0])
                .indexOf("Object") != -1
            ) {
              //对象
              this.setDatas(this.defaultKey);
            } else if (
              Object.prototype.toString
                .call(this.defaultKey[0])
                .indexOf("Number") != -1 ||
              Object.prototype.toString
                .call(this.defaultKey[0])
                .indexOf("String") != -1
            ) {
              this.setKeys(this.defaultKey);
            } else {
              console.log("多选:传入参数类型不匹配");
              return;
            }
          } else {
            console.log("多选:传入参数类型不匹配");
            return;
          }
        } else {
          // 单选
          if (
            Object.prototype.toString.call(this.defaultKey).indexOf("Number") !=
              -1 ||
            Object.prototype.toString.call(this.defaultKey).indexOf("String") !=
              -1 ||
            Object.prototype.toString.call(this.defaultKey).indexOf("Object") !=
              -1
          ) {
            this.setKey(this.defaultKey);
          } else {
            console.log("单选:传入参数类型不匹配");
            return;
          }
        }
      }
    },
    //下拉框select点击[入口]
    selectClick() {
      this.$nextTick(function () {
        //设置下拉框自适应宽度
        this.popoverWidth = this.$refs.select.$el.clientWidth;
      });
      //显示下拉框
      return (this.isShowSelect = !this.isShowSelect);
    },
    //单选: 树点击方法
    nodeClick(data, node) {
      if (!this.multiple) {
        this.isShowSelect = false;
        //单选父子节点不关联
        if (this.radioStrictly) {
          this.setKey(node.key);
        } else {
          if (node.level != 1) {
            this.setKey(node.key);
          } else if (node.level == 1 && this.obj.children == "") {
            this.setKey(node.key);
          }
        }
      } else {
        //多选
        var checkedKeys = this.$refs.tree.getCheckedKeys(); // 所有被选中的节点的 key 所组成的数组数据

        checkedKeys = checkedKeys.filter(function (s) {
          if (typeof s == "String") {
            return s && s.trim();
          } else {
            return s;
          }
        });
        var t = [];
        checkedKeys = util.unique(checkedKeys);
        this.options = checkedKeys.map((item) => {
          if (item !== undefined) {
            var node = this.$refs.tree.getNode(item); // 所有被选中的节点对应的node
            t.push(node.data);
          }
          //设置option选项
          return { label: node.label, value: node.key };
        });

        this.returnDataKeys = this.options.map((item) => {
          return item.value;
        });
        this.returnDatas = t;
        this.popoverHide();
      }
    },
    //单选:清空选中
    clean() {
      this.$refs.tree.setCurrentKey(null); //清除树选中key
      this.returnDatas = null;
      this.returnDataKeys = "";
      this.popoverHide();
    },
    //单选:设置、初始化值 key
    setKey(thisKey) {
      this.$refs.tree.setCurrentKey(thisKey);
      var node = this.$refs.tree.getNode(thisKey);
      if (node) {
        this.setData(node.data);
      }
    },
    //单选:设置、初始化对象
    setData(data) {
      this.options = [];
      this.options.push({
        label: data[this.obj.label],
        value: data[this.obj.id],
      });
      this.returnDatas = data;
      this.returnDataKeys = data[this.obj.id];
    },
    //多选:设置、初始化值 keys
    setKeys(thisKeys) {
      this.$refs.tree.setCheckedKeys(thisKeys);
      this.returnDataKeys = thisKeys;
      var t = [];
      thisKeys.map((item) => {
        //设置option选项
        var node = this.$refs.tree.getNode(item); // 所有被选中的节点对应的node
        t.push(node.data);
        return { label: node.label, value: node.key };
      });
      this.returnDatas = t;
      this.popoverHide();
    },
    //多选:设置、初始化对象
    setDatas(data) {
      this.$refs.tree.setCheckedNodes(data);
      this.returnDatas = data;
      var t = [];
      data.map((item) => {
        //设置option选项
        t.push(item[this.obj.id]);
      });
      this.returnDataKeys = t;
      this.popoverHide();
    },
    // 多选,删除任一select选项的回调
    removeTag(val) {
      this.$refs.tree.setChecked(val, false); //设置为未选中
      var node = this.$refs.tree.getNode(val); //获取节点
      if (!this.checkStrictly && node.childNodes.length > 0) {
        this.treeToList(node).map((item) => {
          if (item.childNodes.length <= 0) {
            this.$refs.tree.setChecked(item, false);
          }
        });
      }
      this.nodeClick();
      this.popoverHide();
    },
    //下拉框关闭执行
    popoverHide() {
      this.$emit("getValue", this.returnDataKeys, this.returnDatas);
    },
    // 多选,清空所有勾选
    clearSelectedNodes() {
      var checkedKeys = this.$refs.tree.getCheckedKeys(); // 所有被选中的节点的 key 所组成的数组数据
      for (let i = 0; i < checkedKeys.length; i++) {
        this.$refs.tree.setChecked(checkedKeys[i], false);
      }
    },
    //多选  全选
    selectedAll(val) {
      if (val) {
        this.setKeys(this.treeArrId);
      } else {
        this.clearSelectedNodes();
      }
    },
    //树形转为集合
    treeToList(tree) {
      var queen = [];
      var out = [];
      queen = queen.concat(tree);
      while (queen.length) {
        var first = queen.shift();
        if (first.childNodes) {
          queen = queen.concat(first.childNodes);
        }
        out.push(first);
      }
      return out;
    },
    switchTree() {
      return this.buildTree(this.data, this.defaultValue);
    },
    // 将一维的扁平数组转换为多层级对象
    buildTree(data, id) {
      const fa = (id) => {
        const temp = [];
        for (let i = 0; i < data.length; i++) {
          const n = data[i];
          if (n[this.obj.pid] === id) {
            n[this.obj.children] = fa(n[this.obj.id]);
            temp.push(n);
          }
        }
        return temp;
      };
      return fa(id);
    },
  },
};
</script>
 
<style lang="scss" scoped>
.el-select {
  width: 100%;
}
.common-tree {
  overflow: auto;
  max-height: 300px;
  height: auto;
}
.common-tree /deep/ .el-tree-node__expand-icon.is-leaf {
  display: none !important;
}
.common-tree /deep/ .el-tree-node__content > .el-tree-node__expand-icon {
  padding: 0;
  margin-left: 6px;
  margin-right: 0px;
}
.el-checkbox {
  margin-left: 10px !important;
}
.el-row {
  padding-top: 0px !important;
}
</style>

三.引入方法

直接上代码

<template>
	<div>
	//调用组件  参数根据自己的字段更改
	 <selectTree
	       ref="selectTree"
	       @getValue="checkChange"
	       :curValue="defaulData"
	       :data="data"     
	       :obj="{                  
	         id: 'id',
	         label: 'lable',
	         children: 'children',
	       }"
	       checkStrictly   //父子组件严格不关联
	       clearable	//是否显示清除数据icon
	       filterable	//是否可以搜索
	       multiple		//多选或者单选
	       collapseTags	 //是否折叠选项(多选时生效)
	     ></selectTree>
	</div>
</template>
<script>
//引入组件
import selectTree from "@/pages/publicComp/selectTree";
export default {
	components: {
   		 selectTree //注册组件
  	},
  	methods:{
  	//用data的key赋值给你需要的数据
	  	checkChange(key, data) {
	      this.key = key;
	    },
  	}
  }
</script>

如有不理解的欢迎下方留言

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值