vue3 el-select懒加载以及自定义指令

<template>

<el-select

v-model="state.input"

:loading="loading"

loading-text="数据加载中..."

filterable

clearable

popper-class="event-select-poper"

v-el-select-loadmore="loadmore"

placeholder="请输入属性值"

style="width: 100%"

:filter-method="filterOptions"

@change="handleEventsChange"

@visible-change="handleVisibleChange">

<el-option

v-for="item in dataList"

:key="item.id"

:label="item.name"

:value="item.name">

</el-option>

<div style="margin:10 auto; text-align: center; color:#ccc">----已到底----</div>

</el-select>

</template>

<script setup>

import { ref, nextTick } from 'vue';

import axios from 'axios';

const emits = defineEmits(['change'])

const dataList = ref([]); //渲染数据数组

let loading = ref(false)

const state = ref({input: '', val: ''});

const pageIndex = ref(1);

const scrollpSize = ref(10);

const scrollPages = ref(1);

// 自定义v-el-select-loadmore指令

const vElSelectLoadmore = {

beforeMount(el, binding) {

/**

* vue2时:

* el.querySelector('.el-select-dropdown .el-select-dropdown__wrap');

* vue3时:

* 在el-select给一个参数popper-class="event-select-poper"

* element-plus中el-select的选项是使用的popper.js生成的,无法直接获取

*/

const selectDom = document.querySelector('.event-select-poper .el-select-dropdown__wrap')

let loadMores = function() {

/**

* scrollHeight 获取元素内容高度(只读)

* scrollTop 获取或者设置元素的偏移值,常用于, 计算滚动条的位置, 当一个元素的容器没有产生垂直方向的滚动条, 那它的scrollTop的值默认为0.

* clientHeight 读取元素的可见高度(只读)

* 如果元素滚动到底, 下面等式返回true, 没有则返回false:

* ele.scrollHeight - ele.scrollTop === ele.clientHeight;

*/

// 判断是否到底

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

}

}

}

// onBeforeMount(async () => {

// await loadEvents()

// })

loadEvents();

// 模拟懒加载

const loadmore = () => {

// 数据加载完成之后,不需要再执行懒加载

if (pageIndex.value >= scrollPages.value) { return }

pageIndex.value++

nextTick(() => {

loading.value = true

getList();

loading.value = false

})

}

function getList() {

axios({

method: 'post',

url: 'https://xxx.xx.x.xxx/dataSource/list',

data: { name : state.value.val, pageIndex: pageIndex.value, pageSize: scrollpSize.value },

headers: {'Content-Type': 'application/json;charset=UTF-8'}

}).then(res => {

let resdata = res.data && res.data.data || res.data;

let respage = res.page || (res.data && res.data.page);

for (let index = 0; index < resdata.length; index++) {

dataList.value.push({ name:resdata[index].name, id:resdata[index].id });

}

scrollPages.value = respage.totalPage;

pageIndex.value = respage.pageIndex;

});

}

// 自定义过滤函数

const filterOptions = (query = '') => {

pageIndex.value = 1

nextTick(async () => {

dataList.value = [];

state.value.val = query;

getList();

})

}

// 下拉框隐藏时,重置分页已经过滤得到的列表

const handleVisibleChange = (visible) => {

if(!visible){

pageIndex.value = 1;

}

}

// 获取事件列表

function loadEvents () {

loading.value = true;

getList();

// 调用接口

loading.value = false;

}

const handleEventsChange = (val) => {

console.log(val);

emits('change', val)

}

</script>

相关推荐
Mr Xu_1 天前
告别冗长 switch-case:Vue 项目中基于映射表的优雅路由数据匹配方案
前端·javascript·vue.js
前端摸鱼匠1 天前
Vue 3 的toRefs保持响应性:讲解toRefs在解构响应式对象时的作用
前端·javascript·vue.js·前端框架·ecmascript
sleeppingfrog1 天前
zebra通过zpl语言实现中文打印(二)
javascript
未来之窗软件服务1 天前
未来之窗昭和仙君(六十五)Vue与跨地区多部门开发—东方仙盟练气
前端·javascript·vue.js·仙盟创梦ide·东方仙盟·昭和仙君
baidu_247438611 天前
Android ViewModel定时任务
android·开发语言·javascript
VT.馒头1 天前
【力扣】2721. 并行执行异步函数
前端·javascript·算法·leetcode·typescript
有位神秘人1 天前
Android中Notification的使用详解
android·java·javascript
phltxy1 天前
Vue 核心特性实战指南:指令、样式绑定、计算属性与侦听器
前端·javascript·vue.js
Byron07071 天前
Vue 中使用 Tiptap 富文本编辑器的完整指南
前端·javascript·vue.js
Mr Xu_1 天前
告别硬编码:前端项目中配置驱动的实战优化指南
前端·javascript·数据结构