vue+el-select 多数据分页搜索组件

project.Vue

//子页面1
<template>
	<div>
		<el-select  :value="defaultValue" :loading="loading" :multiple="multiple" :placeholder="placeholder"
			:allow-create="allowCreate"   :isConcatShowText="isConcatShowText" filterable remote clearable default-first-option :remote-method="remoteMethod"
			style="width: 100%;" @input="handleInput" @visible-change="visibleChange" @change="change"
			@clear="clearChange">
			<el-option v-for="(item, index) in optionsList" :key="'s' + index" :label="isConcatShowText==true?concatenatedLabel2(item):concatenatedLabel(item)"
				:value="item[valueString]">
				{{ concatenatedLabel(item) }}
			</el-option>
			<el-pagination class="select-pagination" @size-change="handleSizeChange"
				@current-change="handleCurrentChange" :current-page.sync="localCurrentPage" :page-size="pageSize"
				:total="total" layout="prev, pager, next, total"></el-pagination>
		</el-select>
	</div>
</template>

<script>
	export default {
		name: 'CustomSelect',
		props: {
			defaultValue: {
				type: [Number, String, Array],
				default: null
			},
			loading: {
				type: Boolean,
				default: false
			},
			拼接后 被选择后输入框是否显示拼接的字符串
			isConcatShowText:{
				type: Boolean,
				default: false
			},
			multiple: {
				type: Boolean,
				default: false
			},
			placeholder: {
				type: String,
				default: 'Please select'
			},
			allowCreate: {
				type: Boolean,
				default: false
			},
			remoteMethod: {
				type: Function,
				default: null
			},
			optionsList: {
				type: Array,
				default: () => []
			},
			label: {
				type: String,
				default: 'label'
			},
			labelTwo: {
				type: String,
				default: ''
			},
			labelThree: {
				type: String,
				default: ''
			},
			labelFour: {
				type: String,
				default: ''
			},
			valueString: {
				type: String,
				default: 'value'
			},
			currentPage: {
				type: Number,
				default: 1
			},
			pageSize: {
				type: Number,
				default: 10
			},
			total: {
				type: Number,
				default: 0
			},
		},
		watch: {
			// 监听 prop 的变化,并更新本地数据属性
			currentPage(newValue) {
				this.localCurrentPage = newValue;
			}
		},
		data() {
			return {
				localCurrentPage: 1
			}
		},
		methods: {
			handleInput(value) {
				this.$emit('input', value);
			},
			visibleChange(visible) {
				this.$emit('visible-change', visible);
			},
			change(value) {
				this.$emit('change', value);
			},
			clearChange() {
				this.$emit('clear');
			},
			handleSizeChange(size) {
				this.$emit('size-change', size);
			},
			handleCurrentChange(page) {
				this.$emit('current-change', page);
			},
			concatenatedLabel(item) {
				return [item[this.label], item[this.labelTwo], item[this.labelThree], item[this.labelFour]].filter(Boolean)
					.join(' || ');
			},
			concatenatedLabel2(item) {
				return item[this.label]
			}
		}
	};
</script>

<style scoped>
	.select-pagination {
		margin-top: 10px;
	}
</style>

projectParent.Vue

<template>
	<!-- 所有的项目 也可以单独用这个页面 -->
	<div>
		<!-- is-concat 是否拼接字符串 concat-symbol拼接字符  allowCreate是否允许创建条目 默认显示多少条 是否多选  -->
		<projectSelect :defaultValue="selectedValue" :loading="isLoading" :multiple="isMultiple"
			:isConcatShowText="isConcatShowText" :placeholder="placeholder" :allowCreate="allowCreation"
			:remoteMethod="remoteFetch" :optionsList="options" label="projectCode" labelTwo="name" labelThree=""
			labelFour="" :valueString="valueString" :currentPage="currentPages" :pageSize="pageSize" :total="totalItems"
			@input="handleInput" @visible-change="handleVisibleChange" @change="handleChange" @clear="handleClear"
			@size-change="handleSizeChange" @current-change="handleCurrentChange" />
	</div>
</template>

<script>
	import projectSelect from './project.vue'; 
	//管理员查看往来单位
	import {
		listInfo
	} from "@/api/project/index.js"; //客户
	export default {
		name: 'ParentComponent',
		components: {
			projectSelect
		},
		props: {
			index: {
				type: Number,
				default: 0
			},
			placeholder: {
				type: String,
				default: '请选择'
			},
			valueString: {
				type: String,
				default: 'id'
			},
			selectedVal: {
				default: null
			}
		},
		watch: {
			selectedVal: {
				handler(val, oldVal) {
					this.selectedValue = val
				},
				immediate: true,//为要监视的prop添加immediate属性并设置为true,可以使得watch函数在组件创建时立即执行一次
				deep: true
			}
		},
		data() {
			return {
				querySearch:null,
				isConcatShowText: true, //拼接后 被选择后输入框是否显示拼接的字符串
				selectedValue: [], // 如果是多选,则为数组;否则为单个值
				isLoading: false,
				isMultiple: false, // 是否允许多选
				allowCreation: false, // 是否允许用户创建新选项
				options: [], // 选项列表,应从服务器或本地获取
				currentPages: 1, // 当前页码
				pageSize: 6, // 每页显示的数量
				totalItems: 0, // 总项目数,用于分页
			};
		},
		mounted() {
			this.remoteFetch()
		},
		methods: {
			remoteFetch(query) {
				if(query){
					this.querySearch=query
				}
				let that = this
				// 远程获取数据的函数,根据 query 参数进行搜索
				this.isLoading = true;
				// 假设 fetchData 是一个从服务器获取数据的函数
				listInfo({
					isStop:0,//0否 1是 是否停用
					search: this.querySearch,
					pageSize: that.pageSize,
					pageNum: that.currentPages
				}).then(response => {
					//每次进入后  数据都是重新填充
					if (response.rows.length < 1) {
						this.selectedValue = []; // 清空已选值
					}
					that.options = response.rows;
					that.totalItems = response.total;
					this.isLoading = false;
				});
			},
			handleInput(value) {
				// 处理输入事件,更新 selectedValue
				this.selectedValue = value;
				this.$emit('handleInput', value);
			},
			handleVisibleChange(visible) {
				// 处理下拉菜单的可见性变化
				this.currentPages = 1;
				this.remoteFetch(); // 可能需要重新获取当前页的数据
			},
			handleChange(value) {
				this.querySearch=null
				// 处理选项变化事件
				this.$emit('handlePageChange', value);
			},
			handleClear() {
				this.querySearch=null
				// 处理清除事件
				this.selectedValue = []; // 清空已选值
				this.$emit('clear', this.index);
				// this.$parent.productClear(this.index)
			},
			handleSizeChange(size) {
				// 处理每页显示数量变化事件
				this.pageSize = size;
				this.remoteFetch(this.querySearch); // 可能需要重新获取当前页的数据
			},
			handleCurrentChange(page) {
				// 处理当前页码变化事件
				this.currentPages = page;
				this.remoteFetch(this.querySearch); // 获取当前页的数据
			}
		},
		// 其他逻辑和生命周期钩子...
	};
</script>

<style scoped>
	/* 父组件的样式 */
</style>

最后在页面引用(因为我很多页面用到了,所以用了projectSelect 组件

<el-form>
   <el-form-item label="项目" prop="projectName">
			<project-select class="width100bfb" valueString="id" :selectedVal="form.projectName"   @clear='projectClear()' @handlePageChange="projectChange($event)" placeholder="请选择项目" />
	</el-form-item>
</el-form>

<script>
import projectSelect from '@/views/components/elSelect/projectParent'
export default {
components: {
			projectSelect
		},
methods: {
/**项目列表change*/
			projectChange(val) {
				if (val) {
				//根据自己情况去调用接口
					//getProjectInfo(val).then(res => {
					//	this.form.projectName = res.data.name
						
					//})
				}
			},
		/*清除项目**/
			projectClear() {
			//根据自己情况去设置
				//this.form.projectName = null
			},
 }
}
</script>
### 关于 `el-select` 分页搜索样式的配置 对于带有分页功能的 `el-select` 组件,在 Element Plus 或者旧版的 Element UI 中,可以通过自定义 CSS 来调整其外观。下面提供了一个具体的例子来展示如何设置样式。 #### 自定义 `el-select` 样式 为了修改 `el-select` 的样式,可以利用 Vue 单文件组件中的 `<style scoped>` 部分或者全局样式表来进行覆盖。这里给出一个简单的案例: ```html <template> <div style="width: 300px;"> <!-- 使用 el-select 并绑定远程搜索 --> <el-select v-model="value" filterable remote placeholder="请输入关键词" :remote-method="remoteMethod"> <el-option v-for="(item, index) in options" :key="index" :label="item.label" :value="item.value"></el-option> </el-select> </div> </template> <script setup lang="ts"> import { ref } from 'vue'; const value = ref(''); const options = ref([]); function remoteMethod(query) { if (query !== '') { // 这里模拟异步获取数据的过程 setTimeout(() => { const result = [ { label: query + '-选项一', value: query }, { label: query + '-选项二', value: query } ]; options.value = result; }, 200); } else { options.value = []; } } </script> <style scoped> /* 修改下拉框宽度 */ .el-select-dropdown__wrap { max-height: 200px !important; /* 设置最大高度以便滚动条生效 */ } /* 调整输入框内部间距 */ .el-input__inner { padding-left: 15px; padding-right: 30px; } /* 更改箭头图标颜色 */ .el-icon-arrow-up:before, .el-icon-arrow-down:before { color: #409eff; } /* 当鼠标悬停时改变背景色 */ .el-select .el-input__suffix-inner:hover { background-color: rgba(64, 158, 255, 0.1); } /* 对弹出菜单项应用特定样式 */ .el-select-dropdown li { font-size: 14px; line-height: normal; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; margin: 0; padding: 0 20px; position: relative; list-style: none; cursor: pointer; transition: all .3s cubic-bezier(.645,.045,.355,1); &:hover { background-color: #f5f7fa; } &.is-disabled { color: #a6a9ad; cursor: not-allowed; } } </style> ``` 上述代码片段展示了如何创建一个具有分页特性的 `el-select` 输入控件,并对其进行了基本的样式定制[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值