用vue2+elementUI封装手机端选择器picker组件,支持单选、多选、远程搜索多选
单选注意点:
-
@touchmove.prevent: 在
touchmove事件上添加.prevent修饰符,以阻止默认的滚动行为。 -
handleTouchStart: 记录触摸开始的 Y 坐标和当前的
translateY值。 -
handleTouchMove: 计算触摸移动的距离,并更新
translateY值。 -
handleTouchEnd: 根据
translateY计算当前选中的索引,并更新translateY值。 -
handleCancel: 触发取消事件。
-
handleConfirm: 触发确认事件,并传递当前选中的选项。
-
clampTranslateY: 确保
translateY值在合理范围内。
多选注意点:
-
clickItem: 为选中的选项改变样式
-
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>
相关文章:
用vue2+elementUI封装手机端选择器picker组件,支持单选、多选、远程搜索多选
单选注意点: touchmove.prevent: 在 touchmove 事件上添加 .prevent 修饰符,以阻止默认的滚动行为。 handleTouchStart: 记录触摸开始的 Y 坐标和当前的 translateY 值。 handleTouchMove: 计算触摸移动的距离,并更新 translateY 值。 han…...
『古籍自有答案』古风H5案例赏析
「古籍自有答案」,一部由新京报与字节跳动公益联合打造的古风H5,以诗意盎然的开篇引领用户穿梭于千年文脉。 part1. 创意定位 "人生有惑问先贤,先贤答案存古籍",在这里,每一个灵魂的探问,都能在…...
Laravel模型事件完全指南:触发应用程序的动态行为
标题:Laravel模型事件完全指南:触发应用程序的动态行为 在Laravel框架中,模型事件提供了一种优雅的方式来处理Eloquent模型生命周期中的各种关键时刻。通过监听和响应这些事件,开发者可以自动化许多常见的任务,如日志…...
hot100 |八、二叉树
1-leetcode94. 二叉树的中序遍历 注意:√ 递归方法已经很熟练了,两种不同的递归方式迭代法需要注意,zrm就遇到了要求迭代实现,前序遍历和后续遍历其实不难,中序遍历用的少,注意看一看 // 1.递归方法1Lis…...
Matlab协方差矩阵分解法生成随机场
Matlab协方差矩阵分解法生成随机场 相关系数矩阵 % function outcohesion(x,y,mu,theta) % end % xyload(F:\Research-OUC\基于机器许学习模型的海底斜坡可靠度研究\基于comsol的斜坡稳定性分析\comsol网格操作\grid_operate-matlab.mphtxt); % xxy(:,1); % yxy(:,2); Xlinspac…...
android 在清单文件中配置receiver,系统是何时会注册此广播接收者的?
在 Android 中,通过清单文件(AndroidManifest.xml)配置的广播接收器(BroadcastReceiver),系统会在特定的时机自动注册这些广播接收器。以下是详细的说明: 静态注册的广播接收器 静态注册的广播…...
嵌入式硬件电路常用设计软件
目录 1. Cadence Allegro 2. PADS 3. Altium Designer 4. Multisim 5. Protues 1. Cadence Allegro 功能: Cadence Allegro是Cadence公司推出的先进PCB(Printed Circuit Board,印刷电路板)设计布线工具,也是目前…...
c#的List<T>的SelectMany 和Select
在C#中,List<T>(以及任何实现了IEnumerable<T>的集合)的Select和SelectMany扩展方法都是LINQ(Language Integrated Query)的一部分,用于对集合中的元素进行查询和转换。 尽管它们的作用有些相…...
69.WEB渗透测试-信息收集- WAF、框架组件识别(9)
免责声明:内容仅供学习参考,请合法利用知识,禁止进行违法犯罪活动! 内容参考于: 易锦网校会员专享课 上一个内容:68.WEB渗透测试-信息收集- WAF、框架组件识别(8) 有无waf存在&am…...
ASCII码对照表(Matplotlib颜色对照表)
文章目录 1、简介1.1 颜色代码 2、Matplotlib库简介2.1 简介2.2 安装2.3 后端2.4 入门例子 3、Matplotlib库颜色3.1 概述3.2 颜色图的分类3.3 颜色格式表示3.4 内置颜色映射3.5 xkcd 颜色映射3.6 颜色命名表 4、Colorcet库5、颜色对照表结语 1、简介 1.1 颜色代码 颜色代码是…...
Mysql-常用函数及其用法总结
1、字符串函数 测试用例如下: 1.1 CONCAT() 将多个字符串连接成一个字符串。 SELECT CONCAT(first_name, , last_name) AS full_name FROM users; -- 期望结果:John Doe, Jane Smith, Michael Johnson 1.2 SUBSTRING() 提取子字符串 SELECT SUBSTR…...
【c++刷题笔记-数组】day29:452. 用最少数量的箭引爆气球、 435. 无重叠区间 、 763.划分字母区间
452. 用最少数量的箭引爆气球 - 力扣(LeetCode) 思路:先按照左边界排序,当前的左边界大于前一个的右边界的时候,表示没有覆盖所以需要一根箭,反之则要更新为最小的右边界 重点:是区间覆盖问题…...
【数据结构】链表带环问题分析及顺序表链表对比分析
【C语言】链表带环问题分析及顺序表链表对比分析 🔥个人主页:大白的编程日记 🔥专栏:C语言学习之路 文章目录 【C语言】链表带环问题分析及顺序表链表对比分析前言一.顺序表和链表对比1.1顺序表和链表的区别1.2缓存利用率&#…...
快速解决找不到krpt.dll,无法继续执行代码问题
对于那些遇到计算机开机出现由于无法找到krpt.dll,进而无法继续执行代码问题的用户。 krpt.dll是计算机系统中与DirectX紧密相关的重要文件,如果它出现问题,可能会对一些特定的软件或游戏的运行产生影响。实际上,我们有多种解决该…...
C# List、LinkedList、Dictionary性能对比
数据结构性能对比 List、LinkedList、Dictionary 1. ArrayList (List:前传) ArrayList 是一个特殊数组, 通过添加和删除元素就可以动态改变数组的长度。 ArrayList集合相对于数组的优点: 支持…...
【Spring Cloud】微服务的简单搭建
文章目录 🍃前言🎄开发环境安装🌳服务拆分的原则🚩单一职责原则🚩服务自治🚩单向依赖 🍀搭建案例介绍🌴数据准备🎋工程搭建🚩构建父子工程🎈创建父…...
全球首款商用,AI为视频自动配音配乐产品上线
近日,海外推出了一款名为Resona V2A的产品,这是全球首款商用视频转音频 (V2A) 技术产品。这项突破性技术利用AI,仅凭视频数据即可自动生成高质量、与上下文相关的音频,包括声音设计、音效、拟音和环境音,为电影制作人、…...
Git管理源代码、git简介,工作区、暂存区和仓库区,git远程仓库github,创建远程仓库、配置SSH,克隆项目
学习目标 能够说出git的作用和管理源代码的特点能够如何创建git仓库并添加忽略文件能够使用add、commit、push、pull等命令实现源代码管理能够使用github远程仓库托管源代码能够说出代码冲突原因和解决办法能够说出 git 标签的作用能够使用使用git实现分支创建,合并…...
【机器学习】机器学习与时间序列分析的融合应用与性能优化新探索
文章目录 引言第一章:机器学习在时间序列分析中的应用1.1 数据预处理1.1.1 数据清洗1.1.2 数据归一化1.1.3 数据增强 1.2 模型选择1.2.1 自回归模型1.2.2 移动平均模型1.2.3 长短期记忆网络1.2.4 卷积神经网络 1.3 模型训练1.3.1 梯度下降1.3.2 随机梯度下降1.3.3 A…...
执行力不足是因为选择模糊
选择模糊:执行力不足的根源 选择模糊是指在面对多个选项时,缺乏明确的目标和方向。这种模糊感会导致犹豫不决,进而影响我们的执行力。 选择模糊的表现: 目标不明确,不知道应该做什么。优先级混乱,不清楚…...
基于算法竞赛的c++编程(28)结构体的进阶应用
结构体的嵌套与复杂数据组织 在C中,结构体可以嵌套使用,形成更复杂的数据结构。例如,可以通过嵌套结构体描述多层级数据关系: struct Address {string city;string street;int zipCode; };struct Employee {string name;int id;…...
工业安全零事故的智能守护者:一体化AI智能安防平台
前言: 通过AI视觉技术,为船厂提供全面的安全监控解决方案,涵盖交通违规检测、起重机轨道安全、非法入侵检测、盗窃防范、安全规范执行监控等多个方面,能够实现对应负责人反馈机制,并最终实现数据的统计报表。提升船厂…...
【力扣数据库知识手册笔记】索引
索引 索引的优缺点 优点1. 通过创建唯一性索引,可以保证数据库表中每一行数据的唯一性。2. 可以加快数据的检索速度(创建索引的主要原因)。3. 可以加速表和表之间的连接,实现数据的参考完整性。4. 可以在查询过程中,…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院查看报告小程序
一、开发环境准备 工具安装: 下载安装DevEco Studio 4.0(支持HarmonyOS 5)配置HarmonyOS SDK 5.0确保Node.js版本≥14 项目初始化: ohpm init harmony/hospital-report-app 二、核心功能模块实现 1. 报告列表…...
HarmonyOS运动开发:如何用mpchart绘制运动配速图表
##鸿蒙核心技术##运动开发##Sensor Service Kit(传感器服务)# 前言 在运动类应用中,运动数据的可视化是提升用户体验的重要环节。通过直观的图表展示运动过程中的关键数据,如配速、距离、卡路里消耗等,用户可以更清晰…...
C++.OpenGL (14/64)多光源(Multiple Lights)
多光源(Multiple Lights) 多光源渲染技术概览 #mermaid-svg-3L5e5gGn76TNh7Lq {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-3L5e5gGn76TNh7Lq .error-icon{fill:#552222;}#mermaid-svg-3L5e5gGn76TNh7Lq .erro…...
【C++进阶篇】智能指针
C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...
LabVIEW双光子成像系统技术
双光子成像技术的核心特性 双光子成像通过双低能量光子协同激发机制,展现出显著的技术优势: 深层组织穿透能力:适用于活体组织深度成像 高分辨率观测性能:满足微观结构的精细研究需求 低光毒性特点:减少对样本的损伤…...
Caliper 负载(Workload)详细解析
Caliper 负载(Workload)详细解析 负载(Workload)是 Caliper 性能测试的核心部分,它定义了测试期间要执行的具体合约调用行为和交易模式。下面我将全面深入地讲解负载的各个方面。 一、负载模块基本结构 一个典型的负载模块(如 workload.js)包含以下基本结构: use strict;/…...
第7篇:中间件全链路监控与 SQL 性能分析实践
7.1 章节导读 在构建数据库中间件的过程中,可观测性 和 性能分析 是保障系统稳定性与可维护性的核心能力。 特别是在复杂分布式场景中,必须做到: 🔍 追踪每一条 SQL 的生命周期(从入口到数据库执行)&#…...
