vue3+elPlus 选择框select 下拉数据过千条,页面卡顿,分页解决

vue3+elPlus 选择框select 下拉数据过千条,页面卡顿,分页解决

注意:我这里是提前将下拉数据请求到,保存到本地,来分页

也可分页请求接口来获取,性质一样

最重要的是,监听滑动到底部的vue指令

v-loadmore="selectListPageHandle"

在文件中directive---loadmore---index.tx

自定义指令

javascript 复制代码
import { Directive, DirectiveBinding } from 'vue'
/**
 * 操作权限处理
 */
export const loadmore: Directive = {
  beforeMount(el: HTMLElement, binding: DirectiveBinding) {
    let classDirect = el.className.split(" ")[1]
    const selectDom = document.querySelector(`.event-select-poper${classDirect} .el-scrollbar__wrap`)
    let loadMores = function () {
      const isBase = this.scrollHeight - this.scrollTop <= this.clientHeight + 20
      if (isBase) {
        // 增加防抖
        binding.value && binding.value()
      }
    }
    // 将获取到的dom和函数挂载到el-select上,实例销毁时好处理
    el.selectDomInfo = selectDom
    el.selectLoadMore = loadMores
    // 监听滚动事件
    selectDom?.addEventListener('scroll', loadMores.bind(selectDom))
  },
  // 实例销毁
  beforeUnmount(el) {
    if (el.selectLoadMore) {
      el.selectDomInfo.removeEventListener('scroll', el.selectLoadMore)
      delete el.selectDomInfo
      delete el.selectLoadMore
    }
  }
}

导入全局Index.ts

javascript 复制代码
import { loadmore } from './loadmore'
import { App } from 'vue';
export default (app: App) => {

  app.directive('loadmore', loadmore);

};

// 自定义指令

main.ts中引入(百度怎么引入全局指令,这里不详细说了)

javascript 复制代码
import directive from '@/directive';
// 自定义指令
directive(app)

直接贴代码

javascript 复制代码
// 封装的组件
<template>
  <el-combo-box
    :popper-class="'event-select-poper'+directCustom"
    :class="directCustom"
    v-loadmore="selectListPageHandle"
    style="width: 100%"
    v-model="valueName"
    filterable
    remote
    :remote-method="remoteMethod"
    :loading="loading"
    :dropdown-visible-always="true"
    :show-option-tooltip="true"
    @focus="focusChange"
    @change="selectChange"
  >
    <el-option
      v-for="dict in selectList"
      :key="dict.code"
      :label="dict.label"
      :value="dict.code"
    >
    </el-option>
  </el-combo-box>
</template>

<script setup lang="ts">
import { ref, watch, computed } from 'vue'
const props = defineProps({
  modelValue: [String, Object, Array],
  selectAllList: {
    type: Array,
    default: []
  },
  disabled: {
    type: Boolean,
    default: false
  },
  directCustom: {
    type: String,
    default: ''
  }
})
const emit = defineEmits(['update:modelValue', 'changeCallback'])
watch(
  () => props.modelValue,
  async (val) => {
    if (val) {
      // 回显默认值
      valueName.value = val
    } else {
      valueName.value = []
    }
  }
)

const selectList = ref<any>([])
const valueName = ref('')
const loading = ref(false)

// 本地分页方法
let pageSize = 10
let pageNumber = 1
const resetselectListInit = () => {
  pageNumber = 1 // 重置页数
  selectList.value = [] // 重置选项
  selectListPageHandle()
}
const selectListPageHandle = ($event) => {
  if (loading.value) {
    return false
  }
  // 数据加载完之后,不需要再获取
  if (selectList.value.length >= props.selectAllList.length) {
    return false
  }
  loading.value = true
  let resData = props.selectAllList.slice((pageNumber - 1) * pageSize, pageNumber * pageSize)
  selectList.value = selectList.value.concat(resData)
  pageNumber += 1
  //加载完成
  loading.value = false
}
// 本地过滤
const remoteMethod = (query: string) => {
  if (query) {
    // 手动触发下拉框回滚至顶部,避免触发v-selectLazyLoad指令
    // document.querySelector(`.el-select-dropdown__wrap`).scrollTop = 0;
    if (loading.value) {
      return false
    }
    loading.value = true
    selectList.value = props.selectAllList.filter((item) => {
      return item.codeAndName.toLowerCase().includes(query.toLowerCase())
    })
    loading.value = false
  } else {
    // 为空时,显示第一页数据
    resetselectListInit()
  }
}
const focusChange = () => {
  // 重置(显示第一页数据)
  resetselectListInit()
}
const selectChange = (data) => {
  emit('update:modelValue', data)
  emit('changeCallback', data)
}
</script>

<style scoped lang="scss"></style>

引用组件

import elComboxNew from '@/components/elComboxNew/index.vue'

javascript 复制代码
<el-row>
   <el-col :span="12" class="left-form-gen">
       <el-form-item label="合同文本管理组织" prop="archivesDepartmentCode">
         <elComboxNew
           directCustom="archivesDepartmentCode"
           v-model="dataForm.archivesDepartmentCode"
           :selectAllList="archivesDepartmentCodeList"
         ></elComboxNew>
       </el-form-item>
     </el-col>
   </el-row>
// 请求所有数据
onMounted(() => {
  getArchivesDepartmentCodeList()
})

const archivesDepartmentCodeList = ref<any>([])
const getArchivesDepartmentCodeList = () => {
  return new Promise((resolve, reject) => {
    AJX_archivesDepartmentCodeList().then((res: any) => {
      let archivesDepartmentCodeListStore = res.data ?? []
      archivesDepartmentCodeList.value = archivesDepartmentCodeListStore.map((item) => {
        return {
          code: item.archivesDepartmentCode,
          label: item.archivesDepartmentCode + item.archivesDepartmentName
        }
      })
      resolve(true)
    })
  })
}
相关推荐
장숙혜8 分钟前
JavaScript正则表达式解析:模式、方法与实战案例
开发语言·javascript·正则表达式
工业互联网专业25 分钟前
基于springboot+vue的高校社团管理系统的设计与实现
java·vue.js·spring boot·毕业设计·源码·课程设计
白宇横流学长2 小时前
基于SpringBoot+Vue的旅游管理系统【源码+文档+部署讲解】
vue.js·spring boot·旅游
努力搬砖的程序媛儿2 小时前
uniapp广告飘窗
前端·javascript·uni-app
张人玉3 小时前
小白误入(需要一定的vue基础 )使用node建立服务器——vue前端登录注册页面连接到数据库
服务器·前端·vue.js
大大。3 小时前
element el-table合并单元格
前端·javascript·vue.js
一纸忘忧3 小时前
Bun 1.2 版本重磅更新,带来全方位升级体验
前端·javascript·node.js
杨.某某3 小时前
若依 v-hasPermi 自定义指令失效场景
前端·javascript·vue.js
不爱学英文的码字机器4 小时前
[操作系统] 环境变量详解
开发语言·javascript·ecmascript
Lysun0014 小时前
vue2的$el.querySelector在vue3中怎么写
前端·javascript·vue.js