iview tree树形控件多选,自定义内容

news2025/4/6 11:13:40

项目中需要一个iview框架的树形控件,需要里面包含以下功能

    • 1、控件宽度可展开,可缩小
    • 2、树形控件可搜索,并且定位到搜索的节点
    • 3、控件可以一键勾选,一键取消
    • 4、控件图标自定义
    • 5、 点击最后一个节点时可以进入到二级节点,点击上一节点可返回
    • 完整代码:
    • listToTree文件

效果图:
请添加图片描述

具体实现

1、控件宽度可展开,可缩小

这个比较简单,就是控制控件的宽度

    // 放大
    expansion() {
      this.isZhankai = true;
      this.width = "422";
      this.$store.commit("compration/saveLeftWidth", this.isZhankai);
    },
    // 缩小
    packUp() {
      this.isZhankai = false;
      this.width = "230";
      this.$store.commit("compration/saveLeftWidth", this.isZhankai);
    },

2、树形控件可搜索,并且定位到搜索的节点

    //输入节点名称后按回车健搜索
    inputChange() {
      let exist = false;
      if (this.isShowTwo) {
        // 二级树形控件搜索
        let data = JSON.parse(JSON.stringify(this.treeDataRowTwo));
        for (var i = 0; i < data.length; i++) {
          if (data[i].title.indexOf(this.value) !== -1) {
            //在树中存在
            exist = true;
            this.currentId = data[i].id;

            /*需要重新获取treeDataRowTwo,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/
            data = JSON.parse(JSON.stringify(this.treeDataRowTwo));
            this.treeDataTwo = listToTree(data);
            /*需要重新获取treeDataRowTwo,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/

            this.treeDataTwo = this.fineCurrentIdRecursive(this.treeDataTwo);
            break;
          } else if (!exist && i === data.length - 1) {
            //在树中不存在
            this.$Message.error("无搜索结果");
          }
        }
      } else {
        // 一级树形控件搜索
        for (var i = 0; i < this.treeRawData.length; i++) {
          if (this.treeRawData[i].title.indexOf(this.value) !== -1) {
            //在树中存在
            exist = true;
            this.currentId = this.treeRawData[i].id;

            /*需要重新获取treeRawData,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/
            this.treeRawData = JSON.parse(JSON.stringify(this.authTreeData));
            this.treeData = listToTree(this.treeRawData);
            /*需要重新获取treeRawData,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/

            this.treeData = this.fineCurrentIdRecursive(this.treeData);
            break;
          } else if (!exist && i === this.treeRawData.length - 1) {
            //在树中不存在
            this.$Message.error("无搜索结果");
          }
        }
      }
    },
    //通过节点id选中树中节点并展开它的父节点-递归方式
    fineCurrentIdRecursive(list) {
      for (var i = 0; i < list.length; i++) {
        if (list[i].id === this.currentId) {
          list[i].selected = true; //如果节点id等于currentId,则选中该节点
          break;
        } else {
          if (list[i].children && list[i].children.length > 0) {
            list[i].children = this.fineCurrentIdRecursive(list[i].children); //找不到想要的节点则继续找孩子的(递归)
            for (var j = 0; j < list[i].children.length; j++) {
              if (list[i].children[j].selected || list[i].children[j].expand) {
                list[i].expand = true; //如果子节点(末端节点)选中或者子节点(非末端节点)展开,则展开该子节点的父节点
                break;
              }
            }
          }
        }
      }
      return list;
    },

3、控件可以一键勾选,一键取消

  // 全选按钮
    handleCheckAll(val) {
      this.indeterminate = false;
      if (val) {
        let arr = this.treeData;
        const addKey = arr =>
          arr.map(item => ({
            ...item,
            checked: true,
            children: item.children ? addKey(item.children) : [] // 这里要判断原数据有没有子级如果没有判断会报错
          }));

        this.treeData = addKey(arr);
      } else {
        let arr = this.treeData;
        const addKey = arr =>
          arr.map(item => ({
            ...item,
            checked: false,
            children: item.children ? addKey(item.children) : [] // 这里要判断原数据有没有子级如果没有判断会报错
          }));

        this.treeData = addKey(arr);
      }
      // 全选的ids
      if (this.checkAll) {
        this.ids = [];
        this.authTreeData.forEach(k => {
          this.ids.push(k.id);
        });
        this.removeDuplicate(this.ids);
      } else {
        this.ids = [];
      }
      this.$emit("changePower", this.ids);
    },



  // 树形控件勾选多选框
    checkChange(data) {
      // 半选
      if (data.length === this.authTreeData.length) {
        this.indeterminate = false;
        this.checkAll = true;
      } else if (data.length > 0) {
        this.indeterminate = true;
        this.checkAll = false;
      } else {
        this.indeterminate = false;
        this.checkAll = false;
      }
      this.ids = [];
      data.forEach(k => {
        this.ids.push(k.id);
      });
      this.removeDuplicate(this.ids);
      this.$emit("changePower", this.ids);
    },

4、控件图标自定义

    // 自定义树结构图标
    renderContent(h, { root, node, data }) {
      return h("span", [
        h(
          "Tooltip",
          {
            props: {
              placement: "top",
              transfer: true
            }
          },
          [
            h(
              "span",
              {
                slot: "content",
                style: {
                  whiteSpace: "normal"
                }
              },
              data.title
            ),
            h("img", {
              attrs: {
                src:
                  data.parentId === 0
                    ? this.yiji
                    : data.children === undefined || data.children.length === 0
                    ? this.sanji
                    : this.erji
              },
              style: {
                marginRight: "8px",
                width: "16px",
                height: "16px"
              }
            }),
            h(
              "span",
              {
                style: {
                  fontSize: "14px",
                  width: this.width === "230" ? "89px" : "",
                  overflow: this.width === "230" ? "hidden" : "inherit",
                  textOverflow: this.width === "230" ? "ellipsis" : "inherit"
                }
              },
              data.title
            )
          ]
        )
      ]);
    },
    renderContentTwo(h, { root, node, data }) {
      return h("span", [
        h(
          "Tooltip",
          {
            props: {
              placement: "top",
              transfer: true
            }
          },
          [
            h(
              "span",
              {
                slot: "content",
                style: {
                  whiteSpace: "normal"
                }
              },
              data.title
            ),
            h("img", {
              attrs: {
                src:
                  data.parentId === this.powerId
                    ? this.one
                    : data.children === undefined || data.children.length === 0
                    ? this.three
                    : this.two
              },
              style: {
                marginRight: "8px",
                width: "16px",
                height: "16px",
                verticalAlign: "text-top"
              }
            }),
            h(
              "span",
              {
                style: {
                  fontSize: "14px",
                  width: this.width === "230" ? "89px" : "",
                  overflow: this.width === "230" ? "hidden" : "inherit",
                  textOverflow: this.width === "230" ? "ellipsis" : "inherit"
                }
              },
              data.title
            )
          ]
        )
      ]);
    },

5、 点击最后一个节点时可以进入到二级节点,点击上一节点可返回

    // 点击节点展开收缩
    selectChange(data, selectedNode) {
      this.$set(selectedNode, "expand", !selectedNode.expand);
        // 二级树形的标题
      this.twoTitle = data[0].title;
      // 获取选择节点的id
      this.powerId = data[0].id;
      this.$emit("domainId", this.powerId, this.twoTitle);
      if (this.isTwo) {
        if (data[0].children === undefined || data[0].children.length === 0) {
          this.isSearch = false;
          this.checkAll = false;
          this.value = "";
          this.isShowTwo = true;
          // 二级树形结构处理
          this.treeDataTwo = this.transitionToTree(
            JSON.parse(JSON.stringify(this.treeDataRowTwo)),
            1
          );
        } else {
          this.isShowTwo = false;
        }
      }
    },
    selectChangeTwo(data, selectedNode) {
      this.$set(selectedNode, "expand", !selectedNode.expand);
    },

完整代码:

父组件

      <leftTree
        @changePower="changePower"
        @domainId="domainId"
        :showCheckBox="true"
        :isCheckAll="true"
        :isTwo="true"
        :isExpand="true"
      ></leftTree>

子组件

<template>
  <div class="leftTree" :style="'width:' + width + 'px'">
    <div class="powerList">
      <div class="initPower" v-if="!isSearch">
        <div v-if="isShowTwo" class="twoDianZhan">
          <div style="margin-right:5px; cursor: pointer;" @click="back">
            &lt;
          </div>
          <div>{{ twoTitle }}</div>
        </div>
        <div v-if="!isShowTwo" class="dianzhanliebiao">肖战影视作品</div>
        <div style="display: flex;justify-content: center;align-items: center;">
          <Tooltip content="搜索">
            <img
              class="img"
              @click="search"
              src="../../assets/images/overview/sousuo.svg"
              alt=""
            />
          </Tooltip>
          <Tooltip content="放大">
            <img
              class="img"
              @click="expansion"
              v-show="!isZhankai"
              src="../../assets/images/overview/zhankai.svg"
              alt=""
            />
          </Tooltip>
          <Tooltip content="缩小">
            <img
              class="img"
              @click="packUp"
              v-show="isZhankai"
              src="../../assets/images/overview/shouqi.svg"
              alt=""
            />
          </Tooltip>
          <Tooltip content="全选" v-if="isShowCheckAll">
            <Checkbox
              @on-change="handleCheckAll"
              v-model="checkAll"
              :indeterminate="indeterminate"
              style="margin-left: 10px;margin-right: 0;margin-bottom: 0;"
            ></Checkbox>
          </Tooltip>
        </div>
      </div>
      <div v-if="isSearch" class="searchPower">
        <Input
          ref="power"
          v-model="value"
          placeholder="请输入电站"
          style="width: 180px"
          @on-blur="inputBlur"
          @on-clear="inputClear"
          @on-enter="inputChange"
        />
      </div>
    </div>

    <div class="powerTree">
      <Tree
        v-if="isShowTwo"
        :data="treeDataTwo"
        :show-checkbox="true"
        :render="renderContentTwo"
        @on-select-change="selectChangeTwo"
        @on-check-change="checkChangeTwo"
      ></Tree>
      <Tree
        v-if="!isShowTwo"
        :data="treeData"
        :show-checkbox="showCheck"
        :render="renderContent"
        @on-select-change="selectChange"
        @on-check-change="checkChange"
      ></Tree>
    </div>
  </div>
</template>
<script>
import { listToTree } from "./../../utils/tree";
export default {
  name: "leftTree",
  components: {},
  props: {
    // 是否显示树形多选框
    showCheckBox: {
      type: Boolean,
      default: true
    },
    // 是否全选
    isCheckAll: {
      type: Boolean,
      default: true
    },
    // 集控中心是否展开
    isExpand: {
      type: Boolean,
      default: false
    },
    // 是否显示二级树形控件
    isTwo: {
      type: Boolean,
      default: false
    }
  },
  data() {
    return {
      // 是否全选
      isShowCheckAll: this.isCheckAll,
      // 是否显示树形多选框
      showCheck: this.showCheckBox,
      // 全选半选
      indeterminate: false,
      // 全选绑定数据
      checkAll: false,
      // 树形控件原始数据
      authTreeData: [
        {
          id: 1,
          title: "肖战",
          parentId: 0
        },
        {
          id: 2,
          title: "现代剧",
          parentId: 1
        },
        {
          id: 3,
          title: "古装剧",
          parentId: 1
        },
        {
          id: 4,
          title: "《余生,请多指教》",
          parentId: 2
        },
        {
          id: 5,
          title: "《梦中那片海》",
          parentId: 2
        },
        {
          id: 6,
          title: "《王牌部队》",
          parentId: 2
        },
        {
          id: 7,
          title: "《超星星学园》",
          parentId: 2
        },
        {
          id: 8,
          title: "《陈情令》",
          parentId: 3
        },
        {
          id: 9,
          title: "《狼殿下》",
          parentId: 3
        },

        {
          id: 10,
          title: "《玉骨遥》",
          parentId: 3
        },
        {
          id: 11,
          title: "《哦,我的皇帝陛下》",
          parentId: 3
        },
        {
          id: 12,
          title: "《斗罗大陆》",
          parentId: 3
        }
      ],
      // 电站搜索绑定数据
      value: "",
      // 展开缩小宽度
      width: "230",
      // 展开
      isZhankai: false,
      // 搜索
      isSearch: false,
      // 勾选的节点数据
      ids: [],
      currentId: "", //需要选中的节点id
      treeData: [], //前端处理后的数据
      // 树形控件过渡数据
      treeRawData: [],
      // 树形控件自定义图标
      yiji: require("../../assets/images/overview/yijitubiao.svg"),
      erji: require("../../assets/images/overview/erjitubiao.svg"),
      sanji: require("../../assets/images/overview/sanjitubiao.svg"),
      // 树形控件自定义图标

      // 二级树形控件数据
      isShowTwo: false,
      twoTitle: "",
      // 二级树形控件数据
      treeDataTwo: [],
      // 二级树形控件处理数据
      treeDataRowTwo: [
        {
          id: 2,
          title: "音乐",
          parentId: 1
        },
        {
          id: 3,
          title: "广告",
          parentId: 1
        },
        {
          id: 4,
          title: "光点",
          parentId: 2
        },
        {
          id: 5,
          title: "最幸运的幸运",
          parentId: 2
        },
        {
          id: 6,
          title: "余年",
          parentId: 2
        },
        {
          id: 7,
          title: "满足",
          parentId: 2
        },
        {
          id: 8,
          title: "余生,请多指教",
          parentId: 2
        },
        {
          id: 9,
          title: "问少年",
          parentId: 2
        },

        {
          id: 10,
          title: "等等",
          parentId: 2
        },
        {
          id: 11,
          title: "TOD‘S",
          parentId: 3
        },
        {
          id: 12,
          title: "GUCCI",
          parentId: 3
        },

        {
          id: 13,
          title: "真力时",
          parentId: 3
        },
        {
          id: 14,
          title: "石头科技",
          parentId: 3
        },
        {
          id: 15,
          title: "倍轻松",
          parentId: 3
        },
        {
          id: 16,
          title: "开小灶",
          parentId: 3
        },
        {
          id: 17,
          title: "等等",
          parentId: 3
        }
      ],
      one: require("../../assets/images/overview/one.svg"),
      two: require("../../assets/images/overview/two.svg"),
      three: require("../../assets/images/overview/three.svg"),
      powerId: ""
      // 二级树形控件数据
    };
  },
  created() {
    this.getData();
  },
  mounted() {
    this.$store.state.compration.isZhankai = this.isZhankai;
  },
  methods: {
    // 全选按钮
    handleCheckAll(val) {
      this.indeterminate = false;
      if (val) {
        let arr = this.treeData;
        const addKey = arr =>
          arr.map(item => ({
            ...item,
            checked: true,
            children: item.children ? addKey(item.children) : [] // 这里要判断原数据有没有子级如果没有判断会报错
          }));

        this.treeData = addKey(arr);
      } else {
        let arr = this.treeData;
        const addKey = arr =>
          arr.map(item => ({
            ...item,
            checked: false,
            children: item.children ? addKey(item.children) : [] // 这里要判断原数据有没有子级如果没有判断会报错
          }));

        this.treeData = addKey(arr);
      }
      // 全选的ids
      if (this.checkAll) {
        this.ids = [];
        this.authTreeData.forEach(k => {
          this.ids.push(k.id);
        });
        this.removeDuplicate(this.ids);
      } else {
        this.ids = [];
      }
      this.$emit("changePower", this.ids);
    },
    // tree树结构转换
    transitionToTree(data, rootValue) {
      // data是转换树形结构的数据源
      // rootValue是根节点的特征
      const parents = data.filter(item => item.parentId === rootValue); // 把所有顶级节点拆分出来
      const children = data.filter(item => item.parentId !== rootValue); // 把所有子级节点(无论多少级,二三四五六级..)拆分出来
      dataToTree(parents, children);
      function dataToTree(parents, children) {
        parents.map(p => {
          children.map((c, index) => {
            // 判断当前循环的子节点是否是当前循环父节点的子级
            if (c.parentId === p.id) {
              let newChildren = JSON.parse(JSON.stringify(children));
              newChildren.splice(index, 1);
              dataToTree([c], newChildren);
              // 如果是当前循环父节点的子级,那就看父节点有没有children属性,
              // 如果有就push到children属性的数组,
              // 如果没有就给父节点一个children属性并把当前循环的子级放进去
              if (p.children) {
                p.children.push(c);
              } else {
                p.children = [c];
              }
            }
          });
        });
      }
      return parents;
    },
    // 自定义树结构图标
    renderContent(h, { root, node, data }) {
      return h("span", [
        h(
          "Tooltip",
          {
            props: {
              placement: "top",
              transfer: true
            }
          },
          [
            h(
              "span",
              {
                slot: "content",
                style: {
                  whiteSpace: "normal"
                }
              },
              data.title
            ),
            h("img", {
              attrs: {
                src:
                  data.parentId === 0
                    ? this.yiji
                    : data.children === undefined || data.children.length === 0
                    ? this.sanji
                    : this.erji
              },
              style: {
                marginRight: "8px",
                width: "16px",
                height: "16px"
              }
            }),
            h(
              "span",
              {
                style: {
                  fontSize: "14px",
                  width: this.width === "230" ? "89px" : "",
                  overflow: this.width === "230" ? "hidden" : "inherit",
                  textOverflow: this.width === "230" ? "ellipsis" : "inherit"
                }
              },
              data.title
            )
          ]
        )
      ]);
    },
    renderContentTwo(h, { root, node, data }) {
      return h("span", [
        h(
          "Tooltip",
          {
            props: {
              placement: "top",
              transfer: true
            }
          },
          [
            h(
              "span",
              {
                slot: "content",
                style: {
                  whiteSpace: "normal"
                }
              },
              data.title
            ),
            h("img", {
              attrs: {
                src:
                  data.parentId === this.powerId
                    ? this.one
                    : data.children === undefined || data.children.length === 0
                    ? this.three
                    : this.two
              },
              style: {
                marginRight: "8px",
                width: "16px",
                height: "16px",
                verticalAlign: "text-top"
              }
            }),
            h(
              "span",
              {
                style: {
                  fontSize: "14px",
                  width: this.width === "230" ? "89px" : "",
                  overflow: this.width === "230" ? "hidden" : "inherit",
                  textOverflow: this.width === "230" ? "ellipsis" : "inherit"
                }
              },
              data.title
            )
          ]
        )
      ]);
    },
    back() {
      this.isShowTwo = false;
    },
    // 点击节点展开收缩
    selectChange(data, selectedNode) {
      this.$set(selectedNode, "expand", !selectedNode.expand);
      // 二级树形的标题
      this.twoTitle = data[0].title;
      // 获取选择节点的id
      this.powerId = data[0].id;
      this.$emit("domainId", this.powerId, this.twoTitle);
      if (this.isTwo) {
        if (data[0].children === undefined || data[0].children.length === 0) {
          //
          this.isSearch = false;
          this.checkAll = false;
          this.value = "";
          this.isShowTwo = true;
          // 二级树形结构处理
          this.treeDataTwo = this.transitionToTree(
            JSON.parse(JSON.stringify(this.treeDataRowTwo)),
            1
          );
        } else {
          this.isShowTwo = false;
        }
      }
    },
    selectChangeTwo(data, selectedNode) {
      this.$set(selectedNode, "expand", !selectedNode.expand);
    },
    checkChangeTwo(data) {
      this.ids = [];
      data.forEach(k => {
        if (k.name) {
          this.ids.push({
            unit: k.unit,
            name: k.name,
            title: k.title
          });
        }
      });
      this.$emit("changePower", this.ids);
    },
    // 树形控件勾选多选框
    checkChange(data) {
      // 半选
      if (data.length === this.authTreeData.length) {
        this.indeterminate = false;
        this.checkAll = true;
      } else if (data.length > 0) {
        this.indeterminate = true;
        this.checkAll = false;
      } else {
        this.indeterminate = false;
        this.checkAll = false;
      }
      this.ids = [];
      data.forEach(k => {
        this.ids.push(k.id);
      });
      this.removeDuplicate(this.ids);
      this.$emit("changePower", this.ids);
    },
    //输入节点名称后按回车健搜索
    inputChange() {
      let exist = false;

      if (this.isShowTwo) {
        // 二级树形控件搜索
        let data = JSON.parse(JSON.stringify(this.treeDataRowTwo));
        for (var i = 0; i < data.length; i++) {
          if (data[i].title.indexOf(this.value) !== -1) {
            //在树中存在
            exist = true;
            this.currentId = data[i].id;

            /*需要重新获取treeDataRowTwo,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/
            data = JSON.parse(JSON.stringify(this.treeDataRowTwo));
            this.treeDataTwo = listToTree(data);
            /*需要重新获取treeDataRowTwo,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/

            this.treeDataTwo = this.fineCurrentIdRecursive(this.treeDataTwo);
            break;
          } else if (!exist && i === data.length - 1) {
            //在树中不存在
            this.$Message.error("无搜索结果");
          }
        }
      } else {
        // 一级树形控件搜索
        for (var i = 0; i < this.treeRawData.length; i++) {
          if (this.treeRawData[i].title.indexOf(this.value) !== -1) {
            //在树中存在
            exist = true;
            this.currentId = this.treeRawData[i].id;

            /*需要重新获取treeRawData,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/
            this.treeRawData = JSON.parse(JSON.stringify(this.authTreeData));
            this.treeData = listToTree(this.treeRawData);
            /*需要重新获取treeRawData,不然会出现这种情况:数据是对的,但是视图渲染不出来,导致没有搜索结果*/

            this.treeData = this.fineCurrentIdRecursive(this.treeData);
            break;
          } else if (!exist && i === this.treeRawData.length - 1) {
            //在树中不存在
            this.$Message.error("无搜索结果");
          }
        }
      }
    },
    //通过节点id选中树中节点并展开它的父节点-递归方式
    fineCurrentIdRecursive(list) {
      for (var i = 0; i < list.length; i++) {
        if (list[i].id === this.currentId) {
          list[i].selected = true; //如果节点id等于currentId,则选中该节点
          break;
        } else {
          if (list[i].children && list[i].children.length > 0) {
            list[i].children = this.fineCurrentIdRecursive(list[i].children); //找不到想要的节点则继续找孩子的(递归)
            for (var j = 0; j < list[i].children.length; j++) {
              if (list[i].children[j].selected || list[i].children[j].expand) {
                list[i].expand = true; //如果子节点(末端节点)选中或者子节点(非末端节点)展开,则展开该子节点的父节点
                break;
              }
            }
          }
        }
      }
      return list;
    },
    // 电站获取数据
    getData() {
      this.treeRawData = JSON.parse(JSON.stringify(this.authTreeData));
      this.treeData = this.transitionToTree(this.treeRawData, 0);
      if (this.isExpand) {
        this.treeData.forEach(k => {
          if (k.title === "肖战") {
            k.expand = true;
          }
        });
      }
    },
    // 去重
    removeDuplicate(arr) {
      let len = arr.length;
      for (let i = 0; i < len; i++) {
        for (let j = i + 1; j < len; j++) {
          if (arr[i] === arr[j]) {
            arr.splice(j, 1);
            len--; // 减少循环次数提高性能
            j--; // 保证j的值自加后不变
          }
        }
      }
      return arr;
    },
    // 放大
    expansion() {
      this.isZhankai = true;
      this.width = "422";
      this.$store.commit("compration/saveLeftWidth", this.isZhankai);
    },
    // 缩小
    packUp() {
      this.isZhankai = false;
      this.width = "230";
      this.$store.commit("compration/saveLeftWidth", this.isZhankai);
    },
    // 列表搜索
    search() {
      this.isSearch = true;
      this.$nextTick(() => {
        this.$refs.power.focus();
      });
    },
    // 失去焦点显示
    inputBlur() {
      if (this.value === "") {
        this.isSearch = false;
      }
    },
    // 清除搜索框
    inputClear() {
      this.isSearch = false;
      this.value = "";
    }
  }
};
</script>
<style lang="scss" scoped>
::v-deep .ivu-input {
  width: 100% !important;
}
.leftTree {
  height: 100%;
  .powerList {
    height: 52px;
    background-color: #fff;
    margin-bottom: 1px;
    padding: 0 16px;
    display: flex;
    align-items: center;
    .initPower {
      width: 100%;
      height: 100%;
      display: flex;
      align-items: center;
      justify-content: space-between;
    }
    .img {
      cursor: pointer;
      margin-left: 10px;
    }
    .twoDianZhan {
      display: flex;
      font-family: PingFangSC-Medium;
      font-size: 14px;
      color: #3d3d3d;
    }
    .dianzhanliebiao {
      font-family: PingFangSC-Medium;
      font-size: 14px;
      color: #3d3d3d;
    }
  }
  .powerTree {
    height: calc(100% - 53px);
    background-color: #fff;
  }
}
::v-deep .ivu-tooltip-rel {
  display: flex;
  align-items: center;
}
::v-deep .ivu-tree {
  height: 100%;
  overflow-y: auto;
}

::v-deep .ivu-tree-children .ivu-checkbox-wrapper {
  position: absolute;
  right: 30px;
  top: 5px;
}
::v-deep .ivu-tree-children li {
  &::before {
    content: none !important;
  }
  &::after {
    content: none !important;
  }
}
::v-deep .ivu-tree .ivu-tree-children li .ivu-tree-arrow .ivu-icon {
  &::before {
    content: "\f11f" !important;
    border: none;
    top: 2px;
    transition: all 0.2s ease-in-out;
    -webkit-transition: all 0.2s ease-in-out;
  }
}
::v-deep .ivu-tree .ivu-tree-children li .ivu-tree-arrow-open .ivu-icon {
  &::before {
    -webkit-transform: rotate(90deg);
    transform: rotate(90deg);
    transition: all 0.2s ease-in-out;
    -webkit-transition: all 0.2s ease-in-out;
  }
}
</style>

listToTree文件

//将后端返回的list数据转化为树结构
export const listToTree = list => {
  var arr = [];
  let items = {};
  var idsStr = "";
  let array = [];
  // 获取每个节点的直属子节点(是直属,不是所有子节点)
  for (let i = 0; i < list.length; i++) {
    let key = list[i].parentId;
    if (items[key]) {
      items[key].push(list[i]);
    } else {
      items[key] = [];
      items[key].push(list[i]);
    }
    array.push(list[i].id);
    // idsStr += idsStr === "" ? list[i].id : "," + list[i].id;
  }
  for (var key in items) {
    if (array.indexOf(Number(key)) === -1) {
      //找到最大的父节点key
      arr = formatTree(items, Number(key));
    }
  }
  delete arr[0].parentId;
  return arr;
};
function formatTree(items, parentId) {
  let result = [];
  if (!items[parentId]) {
    return result;
  }
  for (let t of items[parentId]) {
    t.children = formatTree(items, t.id); //递归获取children
    result.push(t);
  }
  return result;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/701310.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

标准模版库 知识点总结 C++程序设计与算法笔记总结(八) 北京大学 郭炜

标准模版库 https://blog.csdn.net/shaozheng0503/article/details/129101932?ops_request_misc%257B%2522request%255Fid%2522%253A%2522168802585416800211563089%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fblog.%2522%257D&request_id1688025854168…

前端判断img图片未能正确加载时显示一个默认图片。img标签onerror事件设置了本地默认图片还是不显示的问题

前端img标签中引用后端交给的图片地址时&#xff0c;有可能这个路径下图片被删除或者损坏了&#xff0c;那么页面在加载img图片时就显示为&#xff0c;太丑了。 img标签有个onerror事件&#xff0c;就是当图片未被正确加载的时候调用此事件&#xff0c;写的时候在onerror事件里…

安装国产系统Kylin-Desktop实战

前言 随着国产化系统的呼声越来越高&#xff0c;再加上今年6月红帽公司发表声明&#xff0c;只有付费才能拿到代码的通告&#xff0c;学习国产化势在必行。 在众多的国产化系统中&#xff0c;我选择了排名比较高的系统&#xff0c;今天我们就一起来安装下麒麟的操作系统 &…

习题-集合框架

目录 1.打印信息 2.ArrayList的使用 User类 测试类 3.Set的使用 汽车类 测试类 4.ArrayList使用 5.新闻 新闻类 测试类 6.返回索引 7.删除重复QQ 8.Map集合操作 9.成绩排序 学生类 测试类 10.Map基础题 学员类 测试类 11.遍历Map综合-国家名称 12.Collec…

1. Linux手写简单的线程池

目录 一、线程池的概念二、线程池的核心组件三、数据结构设计1、任务队列2、线程池 四、接口设计1、创建线程池2、销毁线程池3、抛出任务的接口 五、实现一个线程池及测试1、测试单生成者——多消费者2、测试多生产者——多消费者3、thrd_pool.h4、thrd_pool.c5、main.c6、thrd…

<C语言> 操作符

1.算术操作符 加法&#xff08;&#xff09;&#xff1a;用于将两个操作数相加。减法&#xff08;-&#xff09;&#xff1a;用于将第一个操作数减去第二个操作数。乘法&#xff08;*&#xff09;&#xff1a;用于将两个操作数相乘。除法&#xff08;/&#xff09;&#xff1a;…

使用JavaScript获取随机数序列

使用Javascript 生成随机数 要在 Javascript 中生成随机数&#xff0c;可以使用 Math 对象的 random() 方法。该方法返回一个大于等于 0 小于 1 的伪随机浮点数。 Javascript中的 Math.random() 函数是一个用于生成随机数的内置函数。 MDN 官方解释 Math.random() 函数返回…

idea常用快捷方式,保姆级!图文并茂【建议收藏】

大家好&#xff0c;我是三叔&#xff0c;很高兴这期又和大家见面了&#xff0c;一个奋斗在互联网的打工人。 给大家分享一下idea在开发过程中使用的快捷方式把&#xff0c;可以极大的提升生产力&#xff0c;提高自己的开发速度&#xff0c;需要在开发中不断地使用&#xff0c;…

《Linux操作系统编程》 第十章 线程与线程控制: 线程的创建、终止和取消,detach以及线程属性

&#x1f337;&#x1f341; 博主 libin9iOak带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——libin9iOak的博客&#x1f390; &#x1f433; 《面试题大全》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33…

arcgis-elasticsearch矢量数据导入及索引设计工具

插件说明 插件支持单图层导入和多图层同时导入&#xff0c;依赖elasticsearch包和urlLib包&#xff0c;使用之前请用pip安装&#xff0c;具体的依赖包的requirements.txt文件放在压缩包里面了。 pip install -r requirements.txt插件下载地址&#xff1a;https://download.cs…

DocuSign:在全球电子签名市场具有巨大上涨潜力的SaaS股

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 总结 &#xff08;1&#xff09;DocuSign的核心电子签名业务还在持续增长&#xff0c;尽管在疫情后增速有所放缓&#xff0c;但第一季度的收入已经达到了6.61亿美元&#xff0c;增长率为12%。 &#xff08;2&#xff09;Do…

Linux:通过wget下载安装mysql数据库(5.7版本)

目前&#xff0c;主要使用的MySQL有5.7和8.0两个版本&#xff0c;在安装上&#xff0c;5.7和8.0版本基本一致&#xff0c;区别只在于配置root密码和远程登陆上不同。本次将以5.7版本作为对象&#xff0c;进行后续安装。 1.wget下载MySQL安装文件 下载完成&#xff0c;得到mysq…

PySpark如何输入数据到Spark中?【RDD对象】

PySpark支持多种数据的输入&#xff0c;在输入完成后&#xff0c;都会得到一个&#xff1a;RDD类的对象RDD全称为弹性分布式数据集(Resilient Distributed Datasets)&#xff0c;PySpark针对数据的处理&#xff0c;都是以RDD对象作为载体&#xff0c;即&#xff1a; •数据存储…

ansible实训-Day3(playbook的原理、结构及其基本使用)

一、前言 该篇是对ansible实训第三天内容的归纳总结&#xff0c;主要包括playbook组件的原理、结构及其基本使用方式。 二、Playbook 原理 Playbook是Ansible的核心组件之一&#xff0c;它是用于定义任务和配置的自动化脚本。 Ansible Playbook使用YAML语法编写&#xff0c;可…

Linux 学习记录42(C++篇)

Linux 学习记录42(C篇) 本文目录 Linux 学习记录42(C篇)一、class 类1. 类中的this指针(1. this指针的格式(2. 使用this指针 2. 类中特殊的成员函数(1. 构造函数>1 格式/定义>2 调用构造函数的时机>3 构造函数的初始化列表 (2. 析构函数>1 功能/格式>2 析构函数…

Redis的数据复制到另一台Redis

Redis的数据复制到另一台Redis 最近用到一个问题&#xff0c;需要把Redis的数据复制到另一台Redis&#xff0c;现在总结下解决问题的方法 解决方法一&#xff1a; redis-dump导出 [root ~]# redis-dump -u :password172.20.0.1:6379 > 172.20.0.1.jsonredis-load导入 [ro…

快速打造属于你的接口自动化测试框架

目录 1 接口测试 2 框架选型​​​​​​​ 3 环境搭建 4 需求 5 整体实现架构 6 RF用例实现​​​​​​​ 7 集成到CICD流程 总结&#xff1a; 1 接口测试 接口测试是对系统或组件之间的接口进行测试&#xff0c;主要是校验数据的交换&#xff0c;传递和控制管理过程…

Redis 高可用 RDB AOF

---------------------- Redis 高可用 ---------------------------------------- 在web服务器中&#xff0c;高可用是指服务器可以正常访问的时间&#xff0c;衡量的标准是在多长时间内可以提供正常服务&#xff08;99.9%、99.99%、99.999%等等&#xff09;。 但是在Redis语境…

基于Java人力资源管理系统设计实现(源码+lw+部署文档+讲解等)

博主介绍&#xff1a;✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专…

webassembly简单Demo——hello world

参考官网 Emscripten Tutorial 一、创建C/C文件 hello.c #include <stdio.h>int main() {printf("hello, world!\n");return 0; } 二、编译成html 命令行切到hello.c目录下&#xff0c;执行如下命令(注意需要em的环境变量&#xff0c;参考&#xff1a;emsr…