基于elementui的远程搜索下拉选择分页组件

在开发一个练手项目的时候,需要一个远程搜索的下拉选择组件;

elementui自带的el-select支持远程搜索;但如果一次性查询的数据过多;会导致卡顿。故自己实现一个可分页的远程下拉选择组件

效果:

代码:

javascript 复制代码
<template>
  <el-select
    v-model="selectedValue"
    filterable
    remote
    reserve-keyword
    placeholder="请输入关键词"
    :remote-method="remoteSearch"
    :loading="loading"
    @change="handleChange"
    @focus="handleFocus"
    @clear="handleClear"
    clearable
  >
    <el-option
      v-for="item in options"
      :key="item[keyField]"
      :label="item[labelField]"
      :value="item[valueField]"
    />
    <div class="select-footer" v-if="total > 0">
      <div class="total-count">共 {{ total }} 条</div>
      <div class="select-pagination" v-if="total > pageSize">
        <el-pagination
          small
          layout="prev, pager, next"
          :total="total"
          :page-size="pageSize"
          :current-page.sync="currentPage"
          @current-change="handlePageChange"
        />
      </div>
    </div>

  </el-select>
</template>

<script>
import request from '@/utils/request'

export default {
  name: 'RemoteSearchSelect',
  props: {
    value: {
      type: [String, Number],
      default: ''
    },
    apiUrl: {
      type: String,
      required: true
    },
    // 自定义字段映射
    labelField: {
      type: String,
      default: 'label'
    },
    valueField: {
      type: String,
      default: 'value'
    },
    keyField: {
      type: String,
      default: 'id'
    },
    // 请求参数自定义
    searchParamName: {
      type: String,
      default: 'keyword'
    },
    pageParamName: {
      type: String,
      default: 'pageNum'
    },
    pageSizeParamName: {
      type: String,
      default: 'pageSize'
    },
    // 响应数据结构配置
    responseListField: {
      type: String,
      default: 'list'
    },
    responseTotalField: {
      type: String,
      default: 'total'
    },
    // 额外的请求参数
    extraParams: {
      type: Object,
      default: () => ({})
    }
  },
  data() {
    return {
      selectedValue: this.value,
      options: [],
      loading: false,
      currentPage: 1,
      pageSize: 10,
      total: 0,
      keyword: '',
      isFirstFocus: true,
      selectedOption: null
    }
  },
  created() {
    this.remoteSearch('');
  },
  methods: {
    async remoteSearch(query) {
      this.loading = true
      this.keyword = query
      try {
        const params = {
          [this.searchParamName]: query,
          [this.pageParamName]: this.currentPage,
          [this.pageSizeParamName]: this.pageSize,
          ...this.extraParams
        }
        // 使用配置的响应数据字段
        const responseData = await request({
          url: this.apiUrl,
          method: 'post',
          data: params
        })
        this.options = responseData[this.responseListField] || []
        this.total = responseData[this.responseTotalField] || 0
      } catch (error) {
        console.error('搜索出错:', error)
        this.$message.error('搜索失败,请重试')
      } finally {
        this.loading = false
      }
    },
    handleChange(value) {
      this.selectedOption = this.options.find(item => item[this.valueField] === value)
      this.$emit('input', value)
      this.$emit('change', value)
      this.$emit('select', this.selectedOption)
    },
    handleFocus() {
      // 如果是首次点击或当前没有选项,则执行搜索
        this.remoteSearch(this.keyword)
        this.isFirstFocus = false
    },
    handleClear() {
      this.selectedValue = ''
      this.keyword = ''
      this.selectedOption = null
      this.$emit('input', '')
      this.$emit('change', '')
      this.$emit('select', null)
      this.remoteSearch('')
    },
    async handlePageChange(page) {
      this.currentPage = page
      if (this.keyword) {
        await this.remoteSearch(this.keyword)
      }
    }
  },
  watch: {
    value: {
      handler(newVal) {
        this.selectedValue = newVal
      },
      immediate: true
    }
  }
}
</script>

<style scoped>
.select-footer {
  padding: 8px;
  border-top: 1px solid #EBEEF5;
  display: flex;
  align-items: center;
  justify-content: space-between;
}

.total-count {
  color: #909399;
  font-size: 12px;
  padding-left: 8px;
}

:deep(.el-pagination) {
  padding: 0;
  margin-left: 8px;
}
</style>

用法

javascript 复制代码
<remote-search-select
            v-model="assignForm.ownerId"
            api-url="remoteUrl"
            label-field="label"
            value-field="value"
            key-field="id"
            search-param-name="childName"
            response-list-field="rows"
            response-total-field="total"
            @select="handleProxyerSelectChange"
          />

参数介绍:

参数 说明
v-model 绑定的data数据
api-url 远程查询的数据接口路径,方法名,请求方式为post
label-field 需要展示的名称属性名
value-field 下拉值的属性名
key-field 主键的属性名
search-param-name 模糊搜索的参数的字段名
response-list-field 返回list的属性名
response-total-field 返回list的总数的属性名
@select funtion,接收选中的值

组件可根据自己的具体需求做源码修改。

相关推荐
光影少年2 分钟前
AI前端开发需要会哪些及未来发展?
前端·人工智能·前端框架
Vincent_Vang13 分钟前
多态 、抽象类、抽象类和具体类的区别、抽象方法和具体方法的区别 以及 重载和重写的相同和不同之处
java·开发语言·前端·ide
菩提小狗15 分钟前
小迪安全_第4天:基础入门-30余种加密编码进制&Web&数据库&系统&代码&参数值|小迪安全笔记|网络安全|
前端·网络·数据库·笔记·安全·web安全
闲蛋小超人笑嘻嘻17 分钟前
非父子通信: provide和inject
前端·javascript·vue.js
周亚鑫18 分钟前
vue3 js代码混淆
开发语言·javascript·ecmascript
止观止32 分钟前
不止解构:深入掌握 ES6+ 对象与函数的高级语法糖
前端·javascript·es6
C_心欲无痕35 分钟前
react - useTransition标记低优先级更新
前端·react.js·前端框架
捻tua馔...38 分钟前
antd3的表单实现(HOC解决方案)
前端·javascript·react.js
支付宝体验科技40 分钟前
支付宝 KJS Compose 动态化方案与架构设计
前端·客户端
AllinLin1 小时前
JS中的call apply bind全面解析
前端·javascript·vue.js