用vue2+elementUI封装手机端选择器picker组件,支持单选、多选、远程搜索多选

单选注意点:

  1. @touchmove.prevent: 在 touchmove 事件上添加 .prevent 修饰符,以阻止默认的滚动行为。

  2. handleTouchStart: 记录触摸开始的 Y 坐标和当前的 translateY 值。

  3. handleTouchMove: 计算触摸移动的距离,并更新 translateY 值。

  4. handleTouchEnd: 根据 translateY 计算当前选中的索引,并更新 translateY 值。

  5. handleCancel: 触发取消事件。

  6. handleConfirm: 触发确认事件,并传递当前选中的选项。

  7. clampTranslateY: 确保 translateY 值在合理范围内。

多选注意点:

  1. clickItem: 为选中的选项改变样式

  2. handleConfirm: 触发确认事件,并传递当前选中的选项。

远程搜索多选注意点:

     1. 单独给el-popper设置样式,发现无效,原因是el-popper和<div id="app">...</div>组件处于同一层级,解决方法是使用popper-class属性给el-popper定义一个class,另外在style中去掉scoped。

效果如下:

        

单选/多选picker.vue组件:

<template><div><div class="picker-mask"></div><divclass="picker"@touchstart="handleTouchStart"@touchmove="handleTouchMove"@touchmove.prevent="handleTouchMove"@touchend="handleTouchEnd"><div class="picker-actions"><el-button type="text" @click="handleCancel">取消</el-button><el-button type="text" @click="handleConfirm">确认</el-button></div><div class="picker-box" v-if="chooseOptions.length"><divclass="picker-content":style="{ transform: `translateY(${translateY}px)` }"><divv-for="(item, index) in chooseOptions":key="index":class="item.chooseFlag ? 'choose-item' : 'picker-item'"@click="multiple ? clickItem(index) : null">{{ labelKey ? item[labelKey] : item }}</div></div></div><div class="empty" v-else>暂无数据</div><divv-if="chooseOptions.length && !multiple"class="picker-highlight"></div></div></div>
</template><script>
export default {props: {options: {type: Array,default: () => []},labelKey: {type: String,default: ''},selectedOption: {type: [Object, Array],default: () => []},multiple: {type: Boolean,default: false}},data() {return {startY: 0,translateY: 40,currentIndex: 0,startTranslateY: 40,chooseOptions: this.multiple? this.options?.map(v => ({ ...v, chooseFlag: false })) || []: this.options || []};},mounted() {if (this.multiple && this.selectedOption?.length) {console.log('selectedOption', this.selectedOption);console.log(this.options);this.chooseOptions =this.options?.map(v => ({...v,chooseFlag: this.selectedOption?.some(item => item[this.labelKey] === v[this.labelKey])})) || [];}// 根据选项列表和当前选中项,设置当前索引和滚动位置if (!this.multiple && this.options.indexOf(this.selectedOption) !== -1) {this.currentIndex = this.options.indexOf(this.selectedOption);this.translateY = -40 * this.currentIndex + 40;}},methods: {// 记录触摸开始的 Y 坐标和当前的 translateY 值handleTouchStart(event) {this.startY = event.touches[0].clientY;this.startTranslateY = this.translateY ?? 0;},// 计算触摸移动的距离,并更新 translateY 值。handleTouchMove(event) {const deltaY = event.touches[0].clientY - this.startY;this.translateY = this.startTranslateY + deltaY;this.clampTranslateY();},// 根据 translateY 计算当前选中的索引,并更新 translateY 值handleTouchEnd() {const index = Math.round(this.translateY / 40);this.translateY = index * 40;this.currentIndex = -Math.round((this.translateY - 40) / 40);},// 确保 translateY 值在合理范围内clampTranslateY() {const itemHeight = 40;const maxTranslateY = 40;const minTranslateY =maxTranslateY - (this.options.length - 1) * itemHeight;this.translateY = Math.max(minTranslateY,Math.min(maxTranslateY, this.translateY));},clickItem(index) {this.chooseOptions[index].chooseFlag =!this.chooseOptions[index].chooseFlag;},handleCancel() {console.log('cancel');this.$emit('cancel');},handleConfirm() {const result = this.multiple? this.chooseOptions?.filter(v => v.chooseFlag): this.chooseOptions?.[this.currentIndex];this.$emit('confirm', result);}}
};
</script><style scoped>
.picker-mask {z-index: 2014;position: fixed;top: 0;left: 0;width: 100%;height: 100%;background: rgba(0, 0, 0, 0.7);border-radius: 8px;
}
.picker {position: fixed;bottom: 0;left: 0;width: 100%;height: 200px;overflow: hidden;background-color: #fff;z-index: 2014;color: #323233;font-size: 16px;border-radius: 8px;
}
.picker-box {flex: 1;overflow: hidden;
}.picker-content {display: flex;flex-direction: column;align-items: center;transition: transform 0.3s ease;
}.picker-item {height: 40px;line-height: 40px;text-align: center;width: 100%;
}.picker-highlight {position: absolute;top: 60%;left: 0;width: 100%;height: 40px;transform: translateY(-50%);background-color: rgba(255, 255, 255, 0.7);border-top: 1px solid #ccc;border-bottom: 1px solid #ccc;
}.picker-actions {width: 100%;display: flex;justify-content: space-between;padding: 10px 0;z-index: 2024;background-color: #fff;.el-button {margin: 0 8px;}
}.choose-item {height: 40px;line-height: 40px;text-align: center;color: #2c68ff;width: 100%;
}.choose-item::after {position: absolute;right: 20px;font-family: 'element-icons';content: '';font-size: 12px;font-weight: bold;-webkit-font-smoothing: antialiased;
}.empty {text-align: center;
}
</style>

远程搜索多选picker组件,可以增加远程搜索属性,自己改造使用:

<template><div><div class="picker-mask"></div><div class="picker"><div class="picker-actions"><el-button type="text" @click="handleCancel">取消</el-button><el-button type="text" @click="handleConfirm">确认</el-button></div><div class="picker-box"><el-selectcollapse-tagsfilterablemultiplevalue-key="id"default-first-option:clearable="true"v-model="selectList"placeholder="请输入"popper-class="select-popper"@change="handleChange"><el-optionv-for="item in userSuccessorNowList":key="item.id":value="item":label="item.vname"/></el-select></div></div></div>
</template><script>
export default {props: {selectedOption: {type: Array,default: () => []},labelKey: {type: String,default: ''}},data() {return {selectList: [],userSuccessorNowList: [{ id: 1, vname: 'aaa' },{ id: 2, vname: 'bbb' },{ id: 3, vname: 'ccc' },{ id: 4, vname: 'aaa' },{ id: 5, vname: 'bbb' },{ id: 6, vname: 'ccc' }]};},mounted() {if (this.selectedOption?.length) {console.log('selectedOption', this.selectedOption);this.selectList = this.selectedOption;}},methods: {handleChange(val) {this.selectList = val;console.log('selectList', this.selectList);},handleCancel() {this.$emit('cancel');},handleConfirm() {this.$emit('confirm', this.selectList);}}
};
</script>
<style>
.select-popper {width: 100vw !important;z-index: 2014 !important;left: 0 !important;box-shadow: none;height: 120px !important;overflow: auto;.popper__arrow {display: none !important;}.el-scrollbar__view {text-align: center;}
}
</style><style scoped>
.picker-mask {z-index: 1014;position: fixed;top: 0;left: 0;width: 100%;height: 100%;background: rgba(0, 0, 0, 0.7);border-radius: 8px;
}
.picker {position: fixed;bottom: 0;left: 0;width: 100%;height: 250px;overflow: hidden;background-color: #fff;z-index: 1014;color: #323233;font-size: 16px;border-radius: 8px;.el-select {width: 100%;.el-input__suffix {display: none;}}
}.picker-actions {width: 100%;display: flex;justify-content: space-between;padding: 10px 0;z-index: 2024;background-color: #fff;.el-button {margin: 0 8px;}
}.picker-box {flex: 1;overflow: hidden;
}
</style>

选择器父组件:

<template><div><div class="select-box" @click="togglePicker"><div class="select-content"><div class="select-label"><span class="label">{{ label }}<span v-if="required" style="color: rgba(253, 75, 76, 1)"> * </span></span></div><spanclass="select-value":style="{ color: `${!selectedOption ? 'rgba(0,0,0,0.25)' : ''}` }">{{ selectedOption ? showResults(selectedOption) : placeholder }}</span></div><i class="el-icon-arrow-right"></i></div><div v-if="showPicker"><SearchPickerv-if="pickerType === 'search'":options="options":labelKey="labelKey"@confirm="handleConfirm"@cancel="handleCancel":selectedOption="selectedOption"/><Pickerv-else:options="options":labelKey="labelKey"@confirm="handleConfirm"@cancel="handleCancel":selectedOption="selectedOption":multiple="multiple"/></div></div>
</template><script>
import Picker from '../Picker';
import SearchPicker from '../SearchPicker';export default {components: {Picker,SearchPicker},props: {label: {type: String,default: ''},pickerType: {type: String,default: ''},required: {type: Boolean,default: false},options: {type: Array,default: () => []},labelKey: {type: String,default: ''},placeholder: {type: String,default: '请选择'},multiple: {type: Boolean,default: false}},data() {return {selectedOption: null,showPicker: false};},mounted() {this.getResults();},methods: {getResults() {if (this.multiple) {this.selectedOption = this.selectedOption || null;} else {this.selectedOption = this.selectedOption?.[this.labelKey];}},showResults(selectedOption) {if (this.multiple) {return selectedOption.map(v => v[this.labelKey]).join(',');} else {return selectedOption;}},togglePicker() {this.showPicker = !this.showPicker;},handleConfirm(selectedOption) {if (this.multiple) {this.selectedOption = selectedOption?.length ? selectedOption : null;} else {this.selectedOption = selectedOption;}this.showPicker = false;this.getResults();},handleCancel() {this.showPicker = false;}}
};
</script><style lang="scss" scoped>
.select-box {width: 100%;height: 48px;line-height: 40px;text-align: left;background-color: #fff;cursor: pointer;display: flex;justify-content: space-between;align-items: center;.select-content {display: flex;align-items: center;.select-label {width: 90px;}.select-value {@include textElipsis(1);width: calc(100% - 90px);}}span {font-family: PingFangSC, PingFang SC;font-weight: 400;font-size: 16px;color: rgba(0, 0, 0, 0.85);line-height: 24px;text-align: left;font-style: normal;text-transform: none;}i {color: rgba(0, 0, 0, 0.45);}
}
</style>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/371178.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android 开发中 C++ 和Java 日志调试

在 C 中添加堆栈日志 先在 Android.bp 中 添加 ‘libutilscallstack’ shared_libs:["liblog"," libutilscallstack"]在想要打印堆栈的代码中添加 #include <utils/CallStack.h> using android::CallStack;// 在函数中添加 int VisualizerLib_Crea…

基于支持向量机、孤立森林和LSTM自编码器的机械状态异常检测(MATLAB R2021B)

异常检测通常是根据已有的观测数据建立正常行为模型&#xff0c;从而将不同机制下产生的远离正常行为的数据划分为异常类&#xff0c;进而实现对异常状态的检测。常用的异常检测方法主要有&#xff1a;统计方法、信息度量方法、谱映射方法、聚类方法、近邻方法和分类方法等。 …

26_嵌入式系统网络接口

以太网接口基本原理 IEEE802标准 局域网标准协议工作在物理层和数据链路层&#xff0c;其将数据链路层又划分为两层&#xff0c;从下到上分别为介质访问控制子层(不同的MAC子层&#xff0c;与具体接入的传输介质相关),逻辑链路控制子层(统一的LLC子层&#xff0c;为上层提供统…

MySQL之备份与恢复(八)

备份与恢复 还原逻辑备份 如果还原的是逻辑备份而不是物理备份&#xff0c;则与使用操作系统简单地复制文件到适当位置的方式不同&#xff0c;需要使用MySQL服务器本身来加载数据到表中。在加载导出文件之前&#xff0c;应该先花一点时间考虑文件有多大&#xff0c;需要多久加…

240707_昇思学习打卡-Day19-基于MindSpore通过GPT实现情感分类

240707_昇思学习打卡-Day19-基于MindSpore通过GPT实现情感分类 今天基于GPT实现一个情感分类的功能&#xff0c;假设已经安装好了MindSpore环境。 # 该案例在 mindnlp 0.3.1 版本完成适配&#xff0c;如果发现案例跑不通&#xff0c;可以指定mindnlp版本&#xff0c;执行!pip…

14-20 Vision Transformer用AI的画笔描绘新世界

概述 毫无疑问,目前最受关注且不断发展的最重要的主题之一是使用人工智能生成图像、视频和文本。大型语言模型 (LLM) 已展示出其在文本生成方面的卓越能力。它们在文本生成方面的许多问题已得到解决。然而,LLM 面临的一个主要挑战是它们有时会产生幻觉反应。 最近推出的新模…

【Unity小技巧】Unity字典序列化

字典序列化 在 Unity 中&#xff0c;标准的 C# 字典&#xff08;Dictionary<TKey, TValue>&#xff09;是不能直接序列化的&#xff0c;因为 Unity 的序列化系统不支持非 Unity 序列化的集合类型。可以通过手写字典实现 效果&#xff1a; 实现步骤&#xff1a; 继承ISe…

【TB作品】51单片机 Proteus仿真 MAX7219点阵驱动数码管驱动

1、8乘8点阵模块&#xff08;爱心&#xff09; 数码管测试程序与仿真 实验报告: MAX7219 数码管驱动测试 一、实验目的 通过对 MAX7219 芯片的编程与控制&#xff0c;了解如何使用单片机驱动数码管显示数字&#xff0c;并掌握 SPI 通信协议的基本应用。 二、实验器材 51…

AI中药处方模型构建与案例

在中医领域,人工智能(AI)可以生成各种指令来辅助诊断、治疗和研究。 1. 诊断辅助指令: 根据患者的症状和体征,自动分析并生成可能的中医证候诊断建议。利用中医望闻问切四诊信息,智能识别关键症状,提供对应的中医辨证思路。2. 治疗建议指令: 根据辨证结果,自动推荐相应…

JVM专题之垃圾收集算法

标记清除算法 第一步:标记 (找出内存中需要回收的对象,并且把它们标记出来) 第二步:清除 (清除掉被标记需要回收的对象,释放出对应的内存空间) 缺点: 标记清除之后会产生大量不连续的内存碎片,空间碎片太多可能会导致以后在程序运行过程中需 要分配较大对象时,无法找到…

Python代码设置Excel工作表背景色或背景图

Excel是工作中数据处理和分析数据的重要工具。面对海量的数据和复杂的表格&#xff0c;如何提高工作效率、减少视觉疲劳并提升数据的可读性是不容忽视的问题。而给工作表设置合适的背景是表格优化的一个有效方式。为Excel工作表设置背景色或背景图不仅能够美化工作表&#xff0…

chrome 谷歌浏览器插件打包

1、找到id对应的字符串去搜索 C:\Users\<你的用户名>\AppData\Local\Google\Chrome\User Data\Default\Extensions2、选择根目录 直接加载下面的路径扩展可用&#xff1a;

AI绘画Stable Diffusion【图生图教程】:图片高清修复的三种方案详解,你一定能用上!(附资料)

大家好&#xff0c;我是画画的小强 今天给大家分享一下用AI绘画Stable Diffusion 进行 高清修复&#xff08;Hi-Res Fix&#xff09;&#xff0c;这是用于提升图像分辨率和细节的技术。在生成图像时&#xff0c;初始的低分辨率图像会通过放大算法和细节增强技术被转换为高分辨…

qt 如何添加子项目

首先我们正常流程创建一个项目文件&#xff1a; 这是我已经创建好的&#xff0c;请无视红线 然后找到该项目的文件夹&#xff0c;在文件夹下创建一个文件夹&#xff0c;再到创建好的文件夹下面创建一个 .pri 文件&#xff1a; &#xff08;创建文件夹&#xff09; &#xff08…

中国石油大学(华东)24计算机考研数据速览,计科学硕复试线288分!

中国石油大学&#xff08;华东&#xff09;计算机与通信工程学院是中国石油大学(华东)十三个教学院部之一&#xff0c;其前身是创建于1984年的计算机科学系&#xff0c;2001年撤系建院。伴随着学校50多年的风雨历程&#xff0c;计算机与通信工程学院也已经有了20多年的发展历史…

Python【打包exe文件两步到位】

Python打包Exe 安装 pyinstaller&#xff08;pip install pyinstaller&#xff09; 执行打包命令&#xff08;pyinstaller demo.py&#xff09; 打完包会生成 dist 文件夹&#xff0c;如下如

04.ffmpeg打印音视频媒体信息

目录 1、相关头文件 2、相关结构体 3、相关函数 4、函数详解 5、源码附上 1、相关头文件 #include <libavformat/avformat.h> 包含格式相关的函数和数据结构 #include <libavutil/avutil.h> 包含一些通用实用函数 2、相关结构体 AV…

【代码管理的必备工具:Git的基本概念与操作详解】

一、Git 初识 1.提出问题 不知道你工作或学习时&#xff0c;有没有遇到这样的情况&#xff1a;我们在编写各种⽂档时&#xff0c;为了防止⽂档丢失&#xff0c;更改失误&#xff0c;失误后能恢复到原来的版本&#xff0c;不得不复制出⼀个副本&#xff0c;比如&#xff1a; “…

多元微分学中可微、连续、存在问题

一、偏导存在 与一元证明相同&#xff0c;利用偏导定义式&#xff0c;证明偏导数左右极限存在且相同。 二、偏导连续 与一元证明相同&#xff0c;证明 三、极限存在 1、找一条路径&#xff0c;一般地找 y kx 2、代入f(x,y)&#xff0c;得f(x,kx) 3、证明f(x,kx)极限存在 注意&…

基于SpringBoot的休闲娱乐代理售票系统

本系统主要包括管理员和用户两个角色组成&#xff1b;主要包括&#xff1a;首页、个人中心、用户管理、折扣票管理、分类管理、订单信息管理、退票信息管理、出票信息管理、系统管理等功能的管理系统。 &#x1f495;&#x1f495;作者&#xff1a;Weirdo &#x1f495;&#x…