【vue-treeselect+el-table】数据量大的时候懒加载,数据回显,输入框绑值,末级节点不要前面的箭头等问题详解

vue-treeselect官网,戳我
最近做了个功能,el-table可编辑表格,有一列是treeselect下拉树,接口数据900多kb,请求花了400毫秒左右,按道理说数据不算很大,但是渲染很慢,一刷新或者刚进页面,有1.89秒的卡顿,并且点击下拉树或者同行其他的el-select下拉,卡顿非常明显。

这个列表需要填写上百行数据,不做优化肯定不行的。

优化的思路就是懒加载,思路是:
调两次树接口,第一次什么都不传或者传0,看后端怎么要求,最重要的是拿到的是根节点。如下:
在这里插入图片描述
子节点children都为空,只有根节点的数据。第二次还是调这个接口,传父id,拿到父id下的子节点,然后每次点击都是拿自己下面的子节点,
如下:
在这里插入图片描述

思路有了,下面就是具体的实现:
在这里插入图片描述
这是官网对延迟加载的说明,也许很多人都不会去看文档,但是知道怎么回事,下次遇到了才会不慌。
实现懒加载少不了loadOptions以及children为null
下面开始贴代码:

1、:ref=“‘treeselect’ + scope.$index” 是因为此列为必填项,有校验,和懒加载无关

2、:options=“deptOptions” 就是绑树数据
3、 :normalizer=“normalizer” 把后端返的格式换成树自己的格式
4、:load-options="loadOptions"懒加载方法
5、 @select=“node => treeHandleSelect(scope.$index, node)” 点击树节点把id和code传给后端
6、option-label插槽是显示下拉的内容,用treeselect自己的,通过node赋值及回显
7、value-label插槽是显示文本框的内容,要用el-table的scope插槽赋值及回显
8、要引入LOAD_CHILDREN_OPTIONS,不然会一直报找不到的错,即使你已经log打印出来了
import { LOAD_CHILDREN_OPTIONS } from ‘@riophae/vue-treeselect’

<TreeSelect
                  :ref="'treeselect' + scope.$index"
                  v-model="scope.row.assetTypeCode"
                  :options="deptOptions"
                  :normalizer="normalizer"
                  :load-options="loadOptions"
                  clearable
                  no-options-text="暂无可用选项"
                  noChildrenText="暂无数据"
                  noResultsText="暂无匹配项"
                  loadingText="数据加载中"
                  :matchKeys="['name', 'code']"
                  no-results-text="没有匹配的搜索结构"
                  placeholder="请选择资产分类名称"
                  style="width: 100%"
                  :appendToBody="true"
                  
                  @select="node => treeHandleSelect(scope.$index, node)"
                >
                  <div slot="option-label" slot-scope="{ node }" :style="{ marginLeft: !node.raw.children ? '16px' : '0' }">
                    [{{ node.raw.code }}]{{ node.raw.name }}
                  </div>
                  <div slot="value-label" slot-scope="{ node }">
                    {{ scope.row.assetTypeCode ? `[${scope.row.assetTypeCode}]` : '' }}{{ scope.row.assetTypeName }}
                  </div>
                </TreeSelect>

格式化

 normalizer(node) {
      if (node.children && !node.children.length) {
        delete node.children
      }
      return {
        id: node.code,
        label: node.name,
        children: node.children,
      }
    },

初始化的时候,先调树接口,拿到根节点

// 初始化数据
    initData() {
     //什么都不传,拿到根节点     
      lazyTreeselect().then(res => {      
        for (let index of res.data) {
          let modeInfo = {}
          modeInfo.id = index.id
          modeInfo.code = index.code
          modeInfo.label = index.name
          modeInfo.name = index.name
          modeInfo.children = null
          this.deptOptions.push(modeInfo)
        }      
      })     
    },

点击的时候拿到子节点,这时要用到loadOptions方法


  // 树懒加载
    loadOptions({ action, parentNode, callback }) {    
      if (action === LOAD_CHILDREN_OPTIONS) {
        //加载点击节点的子节点,parentNode.id即父节点,
        //还是调树接口,传id进去
        lazyTreeselect({ id: parentNode.id }).then(response => {
          let arr = []
          for (let index of response.data) {
            let chiledModeInfo = {}
            chiledModeInfo.id = index.id
            chiledModeInfo.code = index.code
            chiledModeInfo.label = index.name
            chiledModeInfo.name = index.name
            chiledModeInfo.children = null
            //后端返的接口里正常都会有个类似于isLeaf/hasLaef字段,末级为1或者true
            //如果没有,直接和后端要就行了
             if (index.hasLeaf == '1') {
              delete chiledModeInfo.children//为末级的时候删掉children,就不会出现末级也有小箭头可以点
            }
            arr.push(chiledModeInfo)
          }
          parentNode.children = arr
        })
        callback()
      }
    },

点击树节点的时候,把code和id都传给后端

 treeHandleSelect(index, node) {     
      this.standTableList[index].assetTypeName = node.label
      this.standTableList[index].assetTypeName = node.name
      this.standTableList[index].assetTypeid = node.id
      this.standTableList[index].assetTypeCode = node.code
    },

选完节点并回显后,调保存接口,把数据传给后端,然后就完成了,我们调详情接口回显也是可以正常回显的,因为上面文本框的插槽绑的已经是scope.row.字段的值,相当于自己自定义的值后端已经保存了。

在写代码的过程中遇到报错,节点id和code重复的报错,是因为loadOption里还没开始写,记录下,下次遇到就不慌了

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值