| <template> | 
|     <view class="u-dropdown"> | 
|         <view class="u-dropdown__menu" :style="{ | 
|             height: $u.addUnit(height) | 
|         }" :class="{ | 
|             'u-border-bottom': borderBottom | 
|         }"> | 
|             <view class="u-dropdown__menu__item" v-for="(item, index) in menuList" :key="index" @tap.stop="menuClick(index)"> | 
|                 <view class="u-flex"> | 
|                     <text class="u-dropdown__menu__item__text" :style="{ | 
|                         color: item.disabled ? '#c0c4cc' : (index === current || highlightIndex == index) ? activeColor : inactiveColor, | 
|                         fontSize: $u.addUnit(titleSize) | 
|                     }">{{item.title}}</text> | 
|                     <view class="u-dropdown__menu__item__arrow" :class="{ | 
|                         'u-dropdown__menu__item__arrow--rotate': index === current | 
|                     }"> | 
|                         <u-icon :custom-style="{display: 'flex'}" :name="menuIcon" :size="$u.addUnit(menuIconSize)" :color="index === current || highlightIndex == index ? 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 @tap.stop.prevent class="u-dropdown__content__popup" :style="[popupStyle]"> | 
|                 <slot></slot> | 
|             </view> | 
|             <view class="u-dropdown__content__mask"></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 <u-dropdown></u-dropdown> | 
|      */ | 
|     export default { | 
|         name: 'u-dropdown', | 
|         props: { | 
|             // 菜单标题和选项的激活态颜色 | 
|             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 | 
|             } | 
|         }, | 
|         data() { | 
|             return { | 
|                 showDropdown: true, // 是否打开下来菜单, | 
|                 menuList: [], // 显示的菜单 | 
|                 active: false, // 下拉菜单的状态 | 
|                 // 当前是第几个菜单处于激活状态,小程序中此处不能写成false或者"",否则后续将current赋值为0, | 
|                 // 无能的TX没有使用===而是使用==判断,导致程序认为前后二者没有变化,从而不会触发视图更新 | 
|                 current: 99999, | 
|                 // 外层内容的样式,初始时处于底层,且透明 | 
|                 contentStyle: { | 
|                     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; | 
|             } | 
|         }, | 
|         created() { | 
|             // 引用所有子组件(u-dropdown-item)的this,不能在data中声明变量,否则在微信小程序会造成循环引用而报错 | 
|             this.children = []; | 
|         }, | 
|         mounted() { | 
|             this.getContentHeight(); | 
|         }, | 
|         methods: { | 
|             init() { | 
|                 // 当某个子组件内容变化时,触发父组件的init,父组件再让每一个子组件重新初始化一遍 | 
|                 // 以保证数据的正确性 | 
|                 this.menuList = []; | 
|                 this.children.map(child => { | 
|                     child.init(); | 
|                 }) | 
|             }, | 
|             // 点击菜单 | 
|             menuClick(index) { | 
|                 // 判断是否被禁用 | 
|                 if (this.menuList[index].disabled) return; | 
|                 // 如果点击时的索引和当前激活项索引相同,意味着点击了激活项,需要收起下拉菜单 | 
|                 if (index === this.current && this.closeOnClickSelf) { | 
|                     this.close(); | 
|                     // 等动画结束后,再移除下拉菜单中的内容,否则直接移除,也就没有下拉菜单收起的效果了 | 
|                     setTimeout(() => { | 
|                         this.children[index].active = false; | 
|                     }, this.duration) | 
|                     return; | 
|                 } | 
|                 this.open(index); | 
|             }, | 
|             // 打开下拉菜单 | 
|             open(index) { | 
|                 // 重置高亮索引,否则会造成多个菜单同时高亮 | 
|                 // this.highlightIndex = 9999; | 
|                 // 展开时,设置下拉内容的样式 | 
|                 this.contentStyle = { | 
|                     zIndex: 11, | 
|                 } | 
|                 // 标记展开状态以及当前展开项的索引 | 
|                 this.active = true; | 
|                 this.current = index; | 
|                 // 历遍所有的子元素,将索引匹配的项标记为激活状态,因为子元素是通过v-if控制切换的 | 
|                 // 之所以不是因display: none,是因为nvue没有display这个属性 | 
|                 this.children.map((val, idx) => { | 
|                     val.active = index == idx ? true : false; | 
|                 }) | 
|                 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 | 
|                 } | 
|             }, | 
|             // 点击遮罩 | 
|             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; | 
|                 }) | 
|             } | 
|         } | 
|     } | 
| </script> | 
|   | 
| <style scoped lang="scss"> | 
|     @import "../../libs/css/style.components.scss"; | 
|   | 
|     .u-dropdown { | 
|         flex: 1; | 
|         width: 100%; | 
|         position: relative; | 
|   | 
|         &__menu { | 
|             @include vue-flex; | 
|             position: relative; | 
|             z-index: 11; | 
|             height: 80rpx; | 
|   | 
|             &__item { | 
|                 flex: 1; | 
|                 @include vue-flex; | 
|                 justify-content: center; | 
|                 align-items: center; | 
|   | 
|                 &__text { | 
|                     font-size: 28rpx; | 
|                     color: $u-content-color; | 
|                 } | 
|   | 
|                 &__arrow { | 
|                     margin-left: 6rpx; | 
|                     transition: transform .3s; | 
|                     align-items: center; | 
|                     @include vue-flex; | 
|   | 
|                     &--rotate { | 
|                         transform: rotate(180deg); | 
|                     } | 
|                 } | 
|             } | 
|         } | 
|   | 
|         &__content { | 
|             position: absolute; | 
|             z-index: 8; | 
|             width: 100%; | 
|             left: 0px; | 
|             bottom: 0; | 
|             overflow: hidden; | 
|              | 
|   | 
|             &__mask { | 
|                 position: absolute; | 
|                 z-index: 9; | 
|                 background: rgba(0, 0, 0, .3); | 
|                 width: 100%; | 
|                 left: 0; | 
|                 top: 0; | 
|                 bottom: 0; | 
|             } | 
|   | 
|             &__popup { | 
|                 position: relative; | 
|                 z-index: 10; | 
|                 transition: all 0.3s; | 
|                 transform: translate3D(0, -100%, 0); | 
|                 overflow: hidden; | 
|             } | 
|         } | 
|   | 
|     } | 
| </style> |