<template> <view> <view class="maskbox" v-show="active" @tap="maskClick"></view> <view class="uni-combox" style="width: 100%;"> <view v-if="label" class="uni-combox__label" :style="labelStyle"> <text>{{label}}</text> </view> <view class="u-dropdown"> <view class="u-dropdown__menu" :style="{height:height+'rpx'}"> <view class="u-dropdown__menu__item"> <view class="u-flex " style="width: 100%;"> <view class="u-close-wrap" v-if="isSearch"> <u-icon class="u-clear-icon" :size="30" :name="searchIcon" :color="searchIconColor ? searchIconColor : color"></u-icon> </view> <input confirm-type="search" class="uni-combox__input" type="text" v-model="inputVal" @confirm="confirm()" :placeholder="placeholder" /> <icon class="uni-combox__icon" type="clear" size="16" @tap="clearInputValue" v-if="inputVal!=''" style="height: 100%;align-items: center;justify-content: center;" /> <!-- <image size="16" src="/static/icons/close-circle2.svg" style="height: 35%;width: 10%;align-items: center;justify-content: center;background-color: aqua;display: flex;"> --> <!-- </image> --> <view :style="{width:height + 'rpx'}" class="u-dropdown__menu__item__arrow" :class="{ 'u-dropdown__menu__item__arrow--rotate': active }" @tap.stop="menuClick()"> <u-icon :custom-style="{display: 'flex'}" :name="menuIcon" :size="$u.addUnit(menuIconSize)" :color="active ? activeColor : '#c0c4cc'"></u-icon> </view> </view> </view> </view> <view class="u-dropdown__content" :style="[contentStyle, { transition: `opacity ${duration / 1000}s linear`, top: $u.addUnit(height), height: contentHeight + 'px'}]" @tap="maskClick" @touchmove.stop.prevent> <view class="u-dropdown__content__popup" :style="[popupStyle]" @touchmove.stop.prevent="() => {}" @tap.stop.prevent="() => {}"> <view class="uni-combox__selector-empty" v-if="optionsLength === 0" @click="empty()"> <text>{{emptyTips}}</text> </view> <view v-for="(item, index) in filterOptions" style="width: 100%;" @click="onSelectorClick(index)"> <view class="uni-combox-item"> <text class="uni-combox-item-text" :title-style="{ color: inputVal === item ? activeColor : inactiveColor }">{{item}}</text> <u-icon v-if="inputVal === item" name="checkbox-mark" :color="activeColor" size="32"> </u-icon> </view> <u-line class="line_color"></u-line> </view> </view> <!-- <view class="u-dropdown__content__mask"></view> --> </view> </view> </view> </view> </template> <script> /** * dropdown 下拉菜单 * @description 该组件一般用于向下展开菜单,同时可切换多个选项卡的场景 * @tutorial http://uviewui.com/components/dropdown.html * @property {String} active-color 标题和选项卡选中的颜色(默认#2979ff) * @property {String} inactive-color 标题和选项卡未选中的颜色(默认#606266) * @property {Boolean} close-on-click-mask 点击遮罩是否关闭菜单(默认true) * @property {Boolean} close-on-click-self 点击当前激活项标题是否关闭菜单(默认true) * @property {String | Number} duration 选项卡展开和收起的过渡时间,单位ms(默认300) * @property {String | Number} height 标题菜单的高度,单位任意(默认80) * @property {String | Number} border-radius 菜单展开内容下方的圆角值,单位任意(默认0) * @property {Boolean} border-bottom 标题菜单是否显示下边框(默认false) * @property {String | Number} title-size 标题的字体大小,单位任意,数值默认为rpx单位(默认28) * @event {Function} open 下拉菜单被打开时触发 * @event {Function} close 下拉菜单被关闭时触发 * @example <pulldown></pulldown> */ export default { name: 'pulldown', emits: ["open", "close", "update:modelValue", "input", "change", "confirm"], props: { // 菜单标题 label: { type: String, default: '' }, // 菜单标题长度 labelWidth: { type: String, default: 'auto' }, // 菜单标题和选项的激活态颜色 activeColor: { type: String, default: '#2979ff' }, // 菜单标题和选项的未激活态颜色 inactiveColor: { type: String, default: '#606266' }, // 点击遮罩是否关闭菜单 closeOnClickMask: { type: Boolean, default: true }, // 点击当前激活项标题是否关闭菜单 closeOnClickSelf: { type: Boolean, default: true }, // 过渡时间 duration: { type: [Number, String], default: 300 }, // 标题菜单的高度,单位任意,数值默认为rpx单位 height: { type: [Number, String], default: 80 }, // 是否显示下边框 borderBottom: { type: Boolean, default: false }, // 标题的字体大小 titleSize: { type: [Number, String], default: 28 }, // 下拉出来的内容部分的圆角值 borderRadius: { type: [Number, String], default: 0 }, // 菜单右侧的icon图标 menuIcon: { type: String, default: 'arrow-down' }, // 菜单右侧图标的大小 menuIconSize: { type: [Number, String], default: 26 }, emptyTips: { type: String, default: '无匹配项' }, // 是否开启搜索模式 isSearch: { type: Boolean, default: false }, // 占位提示文字 hint: { type: String, default: '' }, // 搜索图标的颜色,默认同输入框字体颜色 searchIconColor: { type: String, default: '' }, // 输入框字体颜色 color: { type: String, default: '#606266' }, // 左边输入框的图标,可以为uView图标名称或图片路径 searchIcon: { type: String, default: 'search' }, // 选项数据,如果传入了默认slot,此参数无效 options: { type: Array, default () { return []; } }, }, data() { return { inputVal: '', showDropdown: true, // 是否打开下来菜单, menuList: [], // 显示的菜单 active: false, // 下拉菜单的状态 // 当前是第几个菜单处于激活状态,小程序中此处不能写成false或者"",否则后续将current赋值为0, // 无能的TX没有使用===而是使用==判断,导致程序认为前后二者没有变化,从而不会触发视图更新 current: 99999, // 外层内容的样式,初始时处于底层,且透明 contentStyle: { // zIndex: 11, zIndex: -1, opacity: 0 }, // 让某个菜单保持高亮的状态 highlightIndex: 99999, contentHeight: 0 } }, computed: { // 下拉出来部分的样式 popupStyle() { let style = {}; // 进行Y轴位移,展开状态时,恢复原位。收齐状态时,往上位移100%,进行隐藏 style.transform = `translateY(${this.active ? 0 : '-100%'})` style['transition-duration'] = this.duration / 1000 + 's'; style.borderRadius = `0 0 ${this.$u.addUnit(this.borderRadius)} ${this.$u.addUnit(this.borderRadius)}`; return style; }, labelStyle() { if (this.labelWidth === 'auto') { return {} } return { width: this.labelWidth } }, optionsLength() { return this.options.length }, filterOptions() { if (this.isSearch) { return this.options.filter((item) => { return item.indexOf(this.inputVal) > -1 }) } else { return this.options; } }, placeholder() { if (this.hint) { return this.hint; } else { if (this.isSearch) { return '请输入关键字'; } else { return '请选择'; } } }, }, created() { // 引用所有子组件(u-dropdown-item)的this,不能在data中声明变量,否则在微信小程序会造成循环引用而报错 this.children = []; }, mounted() { this.getContentHeight(); }, methods: { //清空数据 clearInputValue() { this.inputVal = ""; }, //回车监听 confirm() { this.$emit("confirm", this.inputVal); }, init() { // 当某个子组件内容变化时,触发父组件的init,父组件再让每一个子组件重新初始化一遍 // 以保证数据的正确性 this.menuList = []; this.children.map(child => { child.init(); }) }, // 点击菜单 menuClick() { // 判断是否被禁用 // if (this.menuList[index].disabled) return; // 如果点击时的索引和当前激活项索引相同,意味着点击了激活项,需要收起下拉菜单 if (this.active) { this.close(); // 等动画结束后,再移除下拉菜单中的内容,否则直接移除,也就没有下拉菜单收起的效果了 // setTimeout(() => { // this.active = false; // }, this.duration) return; } this.open(); }, onSelectorClick(index) { // console.log(index); this.inputVal = this.options[index]; this.close(); // 修改通过v-model绑定的值 this.$emit("input", this.inputVal); this.$emit("update:modelValue", this.inputVal); // 发出事件,抛出当前勾选项的value this.$emit("change", index, this.inputVal); }, // 打开下拉菜单 open() { console.log(this.options[0]); console.log(this.options.length); // console.log(this.filterOptions()); // 重置高亮索引,否则会造成多个菜单同时高亮 // this.highlightIndex = 9999; // 展开时,设置下拉内容的样式 this.contentStyle = { zIndex: 11, } this.$forceUpdate(); // 标记展开状态以及当前展开项的索引 this.active = true; // this.current = index; // 历遍所有的子元素,将索引匹配的项标记为激活状态,因为子元素是通过v-if控制切换的 // 之所以不是因display: none,是因为nvue没有display这个属性 this.$emit('open', this.current); }, // 设置下拉菜单处于收起状态 close() { this.$emit('close', this.current); // 设置为收起状态,同时current归位,设置为空字符串 this.active = false; this.current = 99999; // 下拉内容的样式进行调整,不透明度设置为0 this.contentStyle = { zIndex: -1, opacity: 0 } }, //点击无匹配选项关闭下拉菜单 empty() { // 设置为收起状态,同时current归位,设置为空字符串 this.active = false; this.current = 99999; // 下拉内容的样式进行调整,不透明度设置为0 this.contentStyle = { zIndex: -1, opacity: 0 } }, // 点击遮罩 maskClick() { // 如果不允许点击遮罩,直接返回 if (!this.closeOnClickMask) return; this.close(); }, // 外部手动设置某个菜单高亮 highlight(index = undefined) { this.highlightIndex = index !== undefined ? index : 99999; }, // 获取下拉菜单内容的高度 getContentHeight() { // 这里的原理为,因为dropdown组件是相对定位的,它的下拉出来的内容,必须给定一个高度 // 才能让遮罩占满菜单一下,直到屏幕底部的高度 // this.$u.sys()为uView封装的获取设备信息的方法 let windowHeight = this.$u.sys().windowHeight; this.$uGetRect('.u-dropdown__menu').then(res => { // 这里获取的是dropdown的尺寸,在H5上,uniapp获取尺寸是有bug的(以前提出修复过,后来又出现了此bug,目前hx2.8.11版本) // H5端bug表现为元素尺寸的top值为导航栏底部到到元素的上边沿的距离,但是元素的bottom值确是导航栏顶部到元素底部的距离 // 二者是互相矛盾的,本质原因是H5端导航栏非原生,uni的开发者大意造成 // 这里取菜单栏的botton值合理的,不能用res.top,否则页面会造成滚动 this.contentHeight = windowHeight - res.bottom; // console.log(this.contentHeight); // console.log(res.bottom); }) } } } </script> <style scoped lang="scss"> @import "../../uni_modules/vk-uview-ui/libs/css/style.components.scss"; .u-clear-icon { @include vue-flex; align-items: center; } .u-close-wrap { width: 30px; height: 100%; @include vue-flex; align-items: center; justify-content: center; border-radius: 50%; } .maskbox { position: fixed; top: 0; left: 0; width: 100vw; height: 100vh; z-index: 4; } .uni-combox { /* #ifndef APP-NVUE */ display: flex; /* #endif */ min-height: 40px; flex-direction: row; align-items: center; position: relative; background-color: #FFFFFF; // padding-right: 5px; // padding-left: 5px; // z-index: 3; } .uni-combox__label { font-size: 16px; line-height: 22px; padding-right: 10px; } .uni-combox__selector-empty, .uni-combox__selector-item { /* #ifdef APP-NVUE */ display: flex; /* #endif */ line-height: 36px; font-size: 14px; text-align: center; border-bottom: solid 1px #DDDDDD; /* margin: 0px 10px; */ } .uni-combox__icon { display: flex; height: 100%; align-items: center; justify-content: center; background-color: #FFFFFF; z-index: 4; } .uni-combox__input { // flex: 1; font-size: 16px; height: 100%; line-height: 100%; background-color: #FFFFFF; width: 100%; z-index: 5; } .uni-combox-item { display: flex; flex-direction: row; font-size: 16px; height: 100%; line-height: 100%; background-color: #FFFFFF; width: 100%; } .uni-combox-item-text { font-size: 16px; height: 100%; line-height: 100%; background-color: #FFFFFF; width: 100%; padding: 20rpx; display: flex; } .uni-combox__input-box { /* #ifndef APP-NVUE */ display: flex; /* #endif */ flex: 1; flex-direction: column; } .uni-combox__selector { box-sizing: border-box; position: absolute; top: 42px; left: 0; width: 100%; background-color: #FFFFFF; border-radius: 6px; box-shadow: #DDDDDD 4px 4px 8px, #DDDDDD -4px -4px 8px; z-index: 2; } .u-dropdown { flex: 1; width: 100%; position: relative; background-color: #FFFFFF; margin-right: 2rpx; margin-left: 2rpx; &__menu { @include vue-flex; // position: relative; z-index: 11; width: 100%; &__item { // flex: 1; @include vue-flex; justify-content: center; // align-items: center; border: 2rpx solid $uni-border-color; width: 100%; margin-top: 2rpx; // margin-right: 2rpx; // margin-left: 2rpx; &__text { font-size: 28rpx; color: $u-content-color; } &__arrow { transition: transform .3s; align-items: center; justify-content: center; @include vue-flex; &--rotate { transform: rotate(180deg); } } } } &__content { position: absolute; z-index: 8; width: 100%; left: 0px; bottom: 0; overflow: hidden; // z-index: 10; // box-sizing: border-box; // position: absolute; // top: 42px; // left: 0; // width: 100%; // // background-color: #FFFFFF; // border-radius: 6px; // box-shadow: #DDDDDD 4px 4px 8px, #DDDDDD -4px -4px 8px; // z-index: 10; &__mask { position: relative; z-index: 9; background: rgba(0, 0, 0, .3); width: 100%; height: 100%; left: 0; top: 0; bottom: 0; } &__popup { position: relative; z-index: 11; transition: all 0.3s; transform: translate3D(0, -100%, 0); overflow: hidden; width: 100%; background-color: aliceblue; border: 2rpx solid $uni-border-color; // z-index: 10; } } } </style>