vue3如何利用自定义指令实现下拉框分页懒加载

 更新时间:2024年07月29日 11:38:46   作者:前端小怪兽zmy  
下拉框一开始请求第一页的内容,滚动到最后的时候,请求第二页的内容,如此反复,直到所有数据加载完成,这篇文章主要介绍了vue3如何利用自定义指令实现下拉框分页懒加载,需要的朋友可以参考下

需求:下拉框一开始请求第一页的内容,滚动到最后的时候,请求第二页的内容,如此反复,直到所有数据加载完成。

selectLoadMore.ts

//自定义指令:实现下拉框下拉到末尾时,加载下一页的内容
// 使用时传两个参数,一个是下拉框的class,一个是下拉框滚动到末尾时触发的函数,比如:
// v-el-select-loadmore="{
//   selector: '.myOption .el-select-dropdown .el-select-dropdown__wrap',
//   loadFunction: loadMore
// }"
export default {
  mounted(el, binding) {
    //解构传来的值
    const {
      value: { selector, loadFunction }
    } = binding
    const SELECTWRAP_DOM = document.querySelector(selector)
    if (SELECTWRAP_DOM) {
      // 监听事件的处理函数,把函数单独写出来,方便销毁
      const scrollHandler = () => {
        const condition = SELECTWRAP_DOM.scrollTop + SELECTWRAP_DOM.clientHeight >= SELECTWRAP_DOM.scrollHeight - 1
        if (condition) {
          loadFunction()
        }
      }
      //赋值,为了方便销毁,这里很重要,不然销毁的时候找不到dom和对应的回调函数!!!
      el.dom = SELECTWRAP_DOM
      el.event = scrollHandler
      //监听滚动事件
      SELECTWRAP_DOM.addEventListener('scroll', scrollHandler)
    }
  },
//销毁,会在关闭弹窗时触发(这里的el-select写在弹窗里)
  beforeUnmount(el) {
    if (el.dom) {
      el.dom.removeEventListener('scroll', el.event)
    }
  }
}

记得在main.ts里面注册成全局组件!!!! 

main.ts

import vElSelectLoadmore from '@/utils/tools/selectLoadMore'
app.directive('el-select-loadmore', vElSelectLoadmore) //全局自定义指令
app.mount('#app')

使用的vue文件 

//使用时在指令里传值,这里有个坑!
//在el-select给一个参数popper-class="myOption",因为element-plus中ei-select的选项是使用的popper.js生成的,无法直接获取,直接获取下拉框的dom获取不到
          <el-select
            v-el-select-loadmore="{
              selector: '.myOption .el-select-dropdown .el-select-dropdown__wrap',
              loadFunction: loadMore
            }"
            v-model="userForm.accountName"
            placeholder="请选择用户"
            popper-class="myOption"
          >
            <el-option
              v-for="(item, index) in userData"
              v-loading="optionLoading(index)"
              :key="item.id"
              :label="item.username"
              :value="item.id"
            />
          </el-select>
//总共的数据条数
let totalCount: number = 0
//当前滚动页
let page: number
/**
 * 自定义指令触发的回调
 */
function loadMore() {
  //计算总页数
  let maxPagSize: number = Math.max(Math.ceil(totalCount / 10), 1)
  //不超过总页数才发请求
  if (page < maxPagSize) {
    page += 1
    //发请求,获得接口数据
    getUserListWrap(page, 10)
  }
}
/**
 * 控制下拉框loding是否出现,isOptionLoading是在getUserListWrap请求函数里面的,userData是请求函数获得是下拉框数据,这样可以使下拉到最后一个option的时候,出现loding效果,更加完善美观
 * @param index 下拉框循环的index
 */
function optionLoading(index: number): boolean {
  if (isOptionLoading.value && index == userData.length - 1) {
    return true
  } else {
    return false
  }
}

总结:

1、 在el-select给一个参数popper-class="myOption",因为element-plus中el-select的选项是使用的popper.js生成的,无法直接获取,直接获取下拉框的dom获取不到

2、在自定义指令里面销毁事件的时候,在mounted必须把事件存在el上,不然不好销毁,一开始是以下这么写的:发现这样不好销毁,因为在 beforeUnmount拿不到function,this只能按以下这么写才行,如果单独把funtion拎出来,this就报错找不到

export default {
  mounted(el, binding) {
    const {
      value: { selector, loadFunction }
    } = binding
    const SELECTWRAP_DOM = document.querySelector(selector)
    if (SELECTWRAP_DOM) {
      SELECTWRAP_DOM.addEventListener('scroll', function () {
        const condition = this.scrollTop + this.clientHeight >= this.scrollHeight - 1
        if (condition) {
          loadFunction()
        }
      })
    }
  }
}

 vue3的demo代码如下:

<template>
  <el-form-item label="用户名称:">
        <el-select 
        popper-class="myOption"
          v-model="accountName" 
          placeholder="请输入或选择用户"  
           v-el-select-loadmore="loadMore">
            <el-option v-for="item in options" 
            :key="item.value" 
            :label="item.label" 
            :value="item.value" />
        </el-select>
        </el-form-item>
</template>
<script setup lang="ts">
import {ref,onMounted} from 'vue'
let page = ref(1)
let pageSize = ref(10)
let total = 100
let totalCount=6
let stopLoading = false
let accountName = ref('')
const options = ref([])
onMounted(() => {
  options.value.length=0
  loadOptions(1)
})
function loadMore() {
  //这里设置接口的最大页数totalCount为6,超过6页就没数据了
        if (page.value <= totalCount) {
            page.value += 1;
        //获得接口数据
          loadOptions(page.value);
        }
    }
function loadOptions(page:number) {
      // 模拟接口发送数据,异步加载数据
      setTimeout(() => {
        for (let i = (page-1)*10+1; i <= page*10; i++) {
        options.value.push({
        value: `Option${i}`,
        label: `Option${i}`
        });
        }
      }, 500);
}
//在自定义指令
const vElSelectLoadmore = {
  mounted(el: any, binding: any) {
    // 坑:
    const SELECTWRAP_DOM:Element|null = document.querySelector(
                '.myOption .el-select-dropdown .el-select-dropdown__wrap'
    );
    if (SELECTWRAP_DOM) {
      SELECTWRAP_DOM.addEventListener('scroll', function () {
                const condition = this.scrollTop+this.clientHeight >= this.scrollHeight-1;
        // 当滚动条滚动到最底下的时候执行接口加载下一页
        if (condition) {
          binding.value && binding.value()
        }
            });
    }
  },
}
</script>

到此这篇关于vue3如何利用自定义指令实现下拉框分页懒加载的文章就介绍到这了,更多相关vue3自定义指令分页懒加载内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • vue项目中axios配置方式(代理配置)

    vue项目中axios配置方式(代理配置)

    这篇文章主要介绍了vue项目中axios配置方式(代理配置),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-07-07
  • van-dialog 组件调用报错的解决

    van-dialog 组件调用报错的解决

    这篇文章主要介绍了van-dialog 组件调用报错的解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-05-05
  • 基于Vue渲染与插件的加载顺序的问题详解

    基于Vue渲染与插件的加载顺序的问题详解

    下面小编就为大家分享一篇基于Vue渲染与插件的加载顺序的问题详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-03-03
  • vue3中不支持.sync语法糖的解决方法

    vue3中不支持.sync语法糖的解决方法

    在 Vue 3 中,.sync 修饰符已经被移除,在 Vue 2 中,.sync 修饰符是一个语法糖,用于简化子组件和父组件之间的双向数据绑定,那么本文将给大家介绍一下vue3中不支持.sync语法糖的解决方法,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-01-01
  • VUE v-bind 数据绑定的示例详解

    VUE v-bind 数据绑定的示例详解

    这篇文章主要介绍了VUE v-bind 数据绑定,简单点来说就是对 HTML 中的元素,我们可以使用 v-bind 来进行绑定和动态的数据输出,本文结合示例代码给大家介绍的非常详细,需要的朋友可以参考下
    2023-05-05
  • Vuex实现计数器以及列表展示效果

    Vuex实现计数器以及列表展示效果

    这篇文章主要为大家详细介绍了Vuex实现计数器以及列表展示效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-03-03
  • vue如何使用router.meta.keepAlive对页面进行缓存

    vue如何使用router.meta.keepAlive对页面进行缓存

    这篇文章主要介绍了vue如何使用router.meta.keepAlive对页面进行缓存问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-05-05
  • Vue源码学习defineProperty响应式数据原理实现

    Vue源码学习defineProperty响应式数据原理实现

    这篇文章主要为大家介绍了Vue源码学习defineProperty响应式数据原理实现,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • 浅谈Vue3中key的作用和工作原理

    浅谈Vue3中key的作用和工作原理

    本文主要介绍了Vue3中key的作用和工作原理,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • 解决ElementUI组件中el-upload上传图片不显示问题

    解决ElementUI组件中el-upload上传图片不显示问题

    这篇文章主要介绍了解决ElementUI组件中el-upload上传图片不显示问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-10-10

最新评论