Vue2中使用自定义指令实现el-table虚拟列表的代码示例

 更新时间:2025年01月15日 09:29:48   作者:程序员张张  
在实际开发中,我们可能会面临其他需求,例如在 el-table 中无法使用分页技术的情况下展示海量数据,这种情况下,页面可能会出现卡顿,严重时甚至可能引发浏览器崩溃,所以针对这个问题本文给大家介绍了vue2中使用自定义指令实现el-table虚拟列表,需要的朋友可以参考下

引言

上一篇文章我们讲了虚拟列表的实现原理,以及在vue3中的使用实例,感兴趣的友友们可以查看相关内容 传送门。然而,在实际开发中,我们可能会面临其他需求,例如在 el-table 中无法使用分页技术的情况下展示海量数据。这种情况下,页面可能会出现卡顿,严重时甚至可能引发浏览器崩溃。针对这一问题,我们需要寻找其他的解决方案。

创建virtualTable组件

首先新建一个virtualTable.vue文件,代码如下:

<template>
  <div style="width: 100%">
    <el-table v-virtual-list="virtual" width="100%" height="600" :data="tableData.slice(start, over)">
      <el-table-column align="center" prop="date" label="日期" :height="virtual.itemHeight" min-width="180"></el-table-column>
      <el-table-column align="center" prop="name" label="姓名" :height="virtual.itemHeight" min-width="180"></el-table-column>
      <el-table-column align="center" prop="address" label="地址" :height="virtual.itemHeight" min-width="180"></el-table-column>
    </el-table>
  </div>
</template>

<script>
let timeout = null
export default {
  name: 'virtualTable',
  data() {
    return {
      tableData: [],
      virtual: {
        itemHeight: 40, // 每一项高度
        scrollClass: '.el-table__body-wrapper', // 有滚动条元素
        scrollTop: 0, // 滚动条距离顶部距离
        tableHeight: 0 // 可视区高度
      }
    }
  },
  computed: {
    // 起始下标
    start() {
      return Math.max(Math.ceil(this.virtual.scrollTop / this.virtual.itemHeight - 5), 0)
    },
    // 结束下标
    over() {
      return Math.min(
        Math.ceil((this.virtual.scrollTop + this.virtual.tableHeight + 1) / this.virtual.itemHeight + 5),
        this.tableData.length
      )
    },
    // 为了保持列表高度完整且滚动条能正常滚
    paddingAttr() {
      let bottom = (this.tableData.length - this.over) * this.virtual.itemHeight
      let top = this.start * this.virtual.itemHeight
      return `${top}px 0 ${bottom}px 0`
    }
  },
  mounted() {
    this.init()
  },
  methods: {
    init() {
      let _arr = []
      for (let i = 1; i <= 100000; i++) {
        _arr.push({
          date: '2016-05-02',
          name: '王小虎' + i,
          address: `上海市普陀区金沙江路 ${i} 弄`
        })
      }
      this.tableData = _arr
      timeout = setTimeout(() => {
        const target = document.querySelector(this.virtual.scrollClass)
        this.virtual.tableHeight = target.clientHeight ?? 0
        this.setPaddingAttr(target)
      })
    },
    // 更新padding属性
    setPaddingAttr(target) {
      let _table = target.querySelector('table')
      _table.style.padding = this.paddingAttr
    }
  },
  beforeDestroy() {
    clearTimeout(timeout)
  }
}
</script>

<style scoped>
::v-deep th,
::v-deep td {
  height: 40px;
  line-height: 40px;
  padding: 0;
}
</style>

代码介绍

这边重点说一下里面的关键代码,具体原理可以参考我上一篇文章 传送门

  • virtual对象

virtual对象虚拟列表相关的属性,放在一个对象中方便管理,需要传入到自定义指令中。

  • timeout = setTimeout(() => {})

为了解决组件初始化时nextTick中无法获取到正确的clientHeight

  • this.virtual.tableHeight = target.clientHeight ?? 0

组件初始化时获取容器的可视区高度。

  • this.setPaddingAttr(target)

设置paddingAttr方法,重新计算padding,为了确保虚拟列表的渲染效果正确。

tips

当然这里我们可以将其封装成一个公共的组件,如果有相同的需求直接使用当前组件。这里我就不去封装了,感兴趣的友友们可以去尝试一下。

添加自定义指令

// 虚拟列表自定义指令
Vue.directive('virtual-list', {
  bind(el, bind, vNode) {
    const that = vNode.context
    let virtual = bind.value
    if (virtual) {
      let target = el.querySelector(virtual.scrollClass)
      target.addEventListener('scroll', () => {
        // 更新滚动条位置
        that.virtual.scrollTop = target.scrollTop
        // 重新计算padding
        that.setPaddingAttr(target)
      })
    }
  }
})

关键代码介绍

  • const taht = vNode.context

获取当前指令所在组件的上下文this,以便在后续的代码中访问组件的方法和数据。

  • let virtual = bind.value

获取组件中传入的值,内部包含虚拟列表的相关信息。

  • target.addEventListener('scroll', () => {...})

给容器添加滚动监听事件,当用户滚动时触发该事件。

  • that.virtual.scrollTop = target.scrollTop

更新组件中的 virtual.scrollTop 属性,记录当前滚动条的位置。

  • that.setPaddingAttr(target)

该方法由当前指令所在组件内部提供。重新计算padding,为了确保虚拟列表的渲染效果正确。

最终效果如下:

tips

该自定义指令不仅可以在 el-table 中实现虚拟列表,还可以在 el-select 等组件中使用。在实际开发中,只要了解其原理,就可以有效地应用于实际操作中。

文章小尾巴

以上就是Vue2中使用自定义指令实现el-table虚拟列表的代码示例的详细内容,更多关于Vue2 el-table虚拟列表的资料请关注脚本之家其它相关文章!

相关文章

  • vue 移动端适配方案详解

    vue 移动端适配方案详解

    这篇文章主要介绍了vue 移动端适配方案详解,详细的介绍2种方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-11-11
  • 解决vue-seamless-scroll滚动加点赞衔接处数据不同步问题

    解决vue-seamless-scroll滚动加点赞衔接处数据不同步问题

    这篇文章主要介绍了解决vue-seamless-scroll滚动加点赞衔接处数据不同步问题,初步判断可能是因为下方悬接vue-seamless-scroll是静态的,没同步DOM,本文给大家分享解决方法,感兴趣的朋友一起看看吧
    2021-11-11
  • 详解vuex commit保存数据技巧

    详解vuex commit保存数据技巧

    这篇文章主要介绍了详解vuex commit保存数据技巧,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • element-ui el-upload实现上传文件及简单的上传文件格式验证功能

    element-ui el-upload实现上传文件及简单的上传文件格式验证功能

    前端上传文件后,后端接受文件进行处理后直接返回处理后的文件,前端直接再将文件下载下来,下面这篇文章主要给大家介绍了关于element-ui el-upload实现上传文件及简单的上传文件格式验证功能的相关资料,需要的朋友可以参考下
    2022-11-11
  • 使用Element的InfiniteScroll 无限滚动组件报错的解决

    使用Element的InfiniteScroll 无限滚动组件报错的解决

    这篇文章主要介绍了使用Element的InfiniteScroll 无限滚动组件报错的解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-07-07
  • 基于vue实现网站前台的权限管理(前后端分离实践)

    基于vue实现网站前台的权限管理(前后端分离实践)

    这篇文章主要介绍了基于vue实现网站前台的权限管理(前后端分离实践),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-01-01
  • VUE前端工程报错监控问题及解决

    VUE前端工程报错监控问题及解决

    文章介绍了在Vue3项目中实现错误捕获和邮件报错上报的方法,首先使用了`errorHandler`全局错误捕获,接着通过Nodemailer发送邮件实现错误上报功能,过程中遇到一些坑,如需本地调试需用node启动服务,同时总结了两种方案的,包括直接存在的问题和限制
    2026-04-04
  • vue组件中的数据传递方法

    vue组件中的数据传递方法

    这篇文章主要介绍了vue组件中的数据传递方法,非常不错,具有一定的参考借鉴价值,需要的朋友参考下吧
    2018-05-05
  • 在 vue-cli v3.0 中使用 SCSS/SASS的方法

    在 vue-cli v3.0 中使用 SCSS/SASS的方法

    关于如何在 vue-cli v3.0 中使用 SCSS/SASS,这里提供三种方案。感兴趣的朋友通过本文一起学习吧
    2018-06-06
  • vue3使用socket.io的踩坑实战记录

    vue3使用socket.io的踩坑实战记录

    Socket.io将Websocket和轮询机制以及其它的实时通信方式封装成了通用的接口,并且在服务端实现了这些实时机制的相应代码,下面这篇文章主要给大家介绍了关于vue3使用socket.io踩坑的相关资料,需要的朋友可以参考下
    2023-03-03

最新评论