近日开发微信小程序时,需要一个下拉框的组件,使用和PC端下拉框一样的方式会显得很丑,于是发现微信官方API有个wx.showActionSheet
,但是有诸多限制,例如:最大长度只能是6项,多选等等。
由于业务需要,下面用到下拉框(actionSheet)支持自定义 高度, 选项,单选,多选,以及遮罩层的透明度等参数。
看完效果直接上代码
组件代码
index.js
Component({ properties: { //数据集 options: { type: JSON, value: '' }, //内容层高度 topsHeight: { type: String, value: '40%' }, //遮罩层透明度 opacity: { type: String, value: '0.5' }, //是否单选 IsSingle: { type: Boolean, value: true } }, data: { isShow: false, contentAnimate: null, //内容动画 overlayAnimate: null, //遮罩层动画 }, methods: { /** * 关闭 */ _close: function () { var that = this; this.contentAnimate.bottom("-40%").step(); this.overlayAnimate.opacity(0).step(); this.setData({ contentAnimate: this.contentAnimate.export(), overlayAnimate: this.overlayAnimate.export(), }); setTimeout(function () { that.setData({ isShow: false, }) }, 200) }, /** * 显示 */ _show: function () { var that = this; that.setData({ isShow: true, }) // 容器上弹 var contentAnimate = wx.createAnimation({ duration: 200, }) contentAnimate.bottom(0).step(); //遮罩层 var overlayAnimate = wx.createAnimation({ duration: 250, }) overlayAnimate.opacity(`${this.properties.opacity}`).step(); this.contentAnimate = contentAnimate; this.overlayAnimate = overlayAnimate; this.setData({ contentAnimate: contentAnimate.export(), overlayAnimate: overlayAnimate.export(), }) }, /** * 设置选中 */ _setSelect: function (even) { let data = this.data.options; let index = even.currentTarget.dataset.index; if (this.properties.IsSingle) { data.forEach((item, i) => { if (i == index) { item.Selected = !item.Selected; } else { item.Selected = false; } }); } else { data[index].Selected = !data[index].Selected; } this.setData({ options: data }); }, /** * 确定 * */ _submit: function () { this.triggerEvent('OnSelectFinish', this.properties.options); this._close(); }, /** * 外部方法调用,显示组件 */ showPopup: function () { this._show(); } } })
index.json
{ "usingComponents": {}, "component": true }
index.wxml
<view class="selectComponent" style="height:{{topsHeight}};bottom: -{{topsHeight}};" wx:if="{{isShow}}" animation="{{contentAnimate}}"> <image src="../../images/icon_close_gray.png" class="close" catchtap="_close"></image> <view class="select-ul"> <view class="select-li" wx:for="{{options}}"> {{item.Text}} <block wx:if="{{IsSingle}}"> <view class="checkbox {{item.Selected ? 'singlechecked' : 'singleunchecked'}}" data-index="{{index}}" bindtap="_setSelect"></view> </block> <block wx:else> <view data-single="{{IsSingle}}" class="checkbox {{item.Selected ? 'checked' : 'unchecked'}}" data-index="{{index}}" bindtap="_setSelect"></view> </block> </view> </view> <view class="SubmitBtn" bindtap="_submit"> 确 定 </view> </view> <!-- 遮罩层 --> <view class="overlay" catchtap ="_close" wx:if="{{isShow}}" animation="{{overlayAnimate}}"></view>
index.wxss
.overlay{ position: fixed; top: 0; left: 0; z-index: 1; width: 100%; height: 100%; background-color: rgba(0, 0, 0); } .selectComponent{ background-color: white; position: fixed; /* bottom: -40%; */ left: 0; width: calc( 100% - 100rpx); /* height: 40%; */ z-index: 10000; padding:20rpx 50rpx; border-top-right-radius: 20rpx; border-top-left-radius: 20rpx; } .SubmitBtn{ border-top: 10px #F7F8FA solid; height: 120rpx; text-align: center; line-height: 90rpx; position: absolute; bottom: 0; left: 0; width: 100%; background-color: #4876F9; color: white; } .select-ul{ height: calc( 100% - 200rpx); overflow: auto; margin-top: 50rpx; } .select-li{ text-align: left; line-height: 80rpx; height: 80rpx; position: relative; font-size: 14px; } .checkbox{ top:24rpx !important; right: 30rpx !important; height: 34rpx; } .close{ width: 14px; height: 14px; position: absolute; right: 30rpx; top: 30rpx; } .checkbox{ width: 30rpx; height: 30rpx; top: 58rpx; right: 0rpx; position: absolute; background-repeat: no-repeat; background-size: 30rpx; } .singlechecked{ background-image: url(""); } .singleunchecked{ background-image: url(""); } .checked{ background-image: url(""); } .unchecked{ background-image: url(""); }
调用页面代码
index.wxml
<!-- 出库类型下拉框 --> <select id="select" options="{{options}}" topsHeight="{{topsHeight}}" opacity="{{opacity}}" IsSingle="{{IsSingle}}" bind:OnSelectFinish="SelectFinish"></select>
json文件需要引用组件
{ "usingComponents": { "select":"../../../../components/select/index" } }
js 文件
const app = getApp(); Page({ /** * 页面的初始数据 */ data: { options: [], //数据集 topsHeight: "30%", //高度 opacity: "0.5", //透明度 IsSingle: true //是否单选 }, /** * 生命周期函数--监听页面加载 */ onl oad: function (options) { //初始化出库类型 this.initExitType(); }, /** * 初始化出库类型 */ initExitType() { /* 这里是系统封装的函数,下面需要换成自己的访问代码*/ var that = this; HTTP.GetExitTypeList().then((res) => { if (res.ErrorCode === 0) { var newArray = []; res.Data.forEach(item => { newArray.push({ "Text": item.Text, "Value": item.Value }); }); that.setData({ options: res.Data }); } }) }, /** * 下拉框选择完成事件 */ SelectFinish: function (even) { console.log(even.detail) }, /** * 显示出库类型下拉框 */ selectStockOutType: function () { this.selectComponent("#select").showPopup(); } })