You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

395 lines
9.5 KiB

<template>
<view class="wSelect" :class="show ? 'active' : ''" :style="{
width: width,
height: height,
backgroundColor: bgColor,
borderRadius: round ? '6rpx' : 'none'
}">
<view @click="openSelect" class="pickerSelect">
<view :style="{ height: height }" v-if="multiple" class="multipleChoice">
<view class="defaultValue" :style="{ color: color }" v-if="!inputData && multiSelectList.length < 1">
{{ defaultValue }}
</view>
<view class="option" v-if="multiSelectList.length > 0">
<view class="">{{ multiSelectList[0][valueName] }}</view>
<image class="img" @click.stop="multiDelete" :src="refreshUrl" mode=""></image>
</view>
<view v-if="multiSelectList.length > 1" class="more">{{ multiLength }}</view>
<input :disabled="!filterable"
:style="{ height: height, paddingRight: '30rpx', fontSize: '28rpx', color: color }"
@input="inputChange" type="text" v-model="inputData" />
</view>
<input v-else :disabled="!filterable"
:style="{ height: height, paddingRight: '30rpx', fontSize: '30rpx', color:'#000' }" @input="inputChange"
:placeholder="defaultValue" :placeholder-style="placeholderColor" type="text" v-model="inputData" />
<view v-if="!inputData || !showClose" :class="show ? 'arrow-up' : 'arrow'"></view>
<view class="showClose" v-if="showClose && inputData">
<image @click.stop="refreshValue" :src="refreshUrl" mode=""></image>
</view>
<view v-show="show" :class="isShow ? 'animation-top' : 'animation-bottom'" class="tips"
:style="{ top: tipsPositon }"></view>
</view>
<!-- 单选时的下拉框 -->
<view v-if="!multiple" :style="{ width: '75%', color: '#737478' }" v-show="show"
:class="isShow ? 'animation-top' : 'animation-bottom'" class="content">
<view class="item" :class="value == item[valueName] ? 'choose' : ''" v-for="(item, index) in listData"
:key="index" @click="clickSelect(item)">
{{ item[valueName] }}
</view>
<view class="item" v-if="listData.length < 1">无匹配项</view>
</view>
<!-- 多选的下拉框 -->
<view v-else :style="{ width: width }" v-show="show" :class="isShow ? 'animation-top' : 'animation-bottom'"
class="content">
<view class="item" :class="multiSelectList.find(res => res[valueName] == item[valueName]) ? 'choose' : ''"
v-for="(item, index) in listData" :key="index" @click="multiSelect(item)">
{{ item[valueName] }}
</view>
<view class="item" v-if="listData.length < 1">无匹配项</view>
</view>
<view v-if="show" @click="closeContentSelect" class="contentMask"></view>
</view>
</template>
<script>
export default {
name: 'wSelect',
props: {
//是否多选
multiple: {
type: Boolean,
default: false
},
//是否可搜索
filterable: {
type: Boolean,
default: false
},
//是否显示关闭按钮
showClose: {
type: Boolean,
default: false
},
width: {
type: String,
default: '300rpx'
},
color: {
type: String,
default: '#606266'
},
placeholderColor: {
type: String,
default: 'color:#bcbec4'
},
bgColor: {
type: String,
default: '#fff'
},
height: {
type: String,
default: '60rpx'
},
//默认显示的内容
defaultValue: {
type: String,
default: '请选择'
},
//显示的内容
valueName: {
type: String,
required: true
},
list: {
type: Array,
default: () => [],
required: true
},
//双向绑定的值
value: {
type: String,
default: ''
},
round: {
type: Boolean,
default: true
}
},
watch: {
value: {
immediate: true,
handler(newValueData) {
if (!this.inputData) {
this.inputData = newValueData
}
}
}
},
data() {
return {
show: false,
isShow: false,
multiSelectList: [],
inputData: '',
listData: this.list,
refreshUrl: ''
}
},
computed: {
multiLength() {
const length = this.multiSelectList.length - 1
return '+' + length
},
tipsPositon() {
let num = this.height.replace('rpx', '')
let numAdd = Number(num) + Number(10)
return numAdd + 'rpx'
}
},
methods: {
openSelect() {
this.listData = this.list
console.log("listData>>>", this.listData)
if (this.show === true) {
this.isShow = false
} else {
this.isShow = true
}
setTimeout(() => {
this.show = !this.show
}, 200)
},
clickSelect(item) {
this.$emit('input', item[this.valueName])
this.inputData = item[this.valueName]
this.$emit('change', item)
this.show = false
},
inputChange(e) {
console.log(e.detail.value)
let value = e.detail.value
this.$emit('input', value)
if (value) {
this.listData = this.listData.filter(item => item[this.valueName].includes(value))
} else {
this.listData = this.list
}
},
multiSelect(item) {
let index = this.multiSelectList.findIndex(res => res[this.valueName] == item[this.valueName])
if (index > -1) {
this.multiSelectList.splice(index, 1)
} else {
this.multiSelectList.push(item)
}
this.inputData = ''
this.listData = this.list
this.$emit('input', this.multiSelectList)
this.$emit('change', item)
},
refreshValue() {
this.$emit('input', '')
this.inputData = ''
this.listData = this.list
this.$emit('change', '')
this.show = false
},
multiDelete() {
this.multiSelectList.splice(0, 1)
this.$emit('input', this.multiSelectList)
if (this.multiSelectList.length < 1) {
this.show = false
}
},
closeContentSelect() {
this.isShow = false
setTimeout(() => {
this.show = false
}, 200)
}
}
}
</script>
<style lang="scss" scoped>
.wSelect {
// border: 2rpx solid #dcdfe6;
transition: all 0.5s;
.pickerSelect {
border-radius: 6rpx;
padding: 0 20rpx;
position: relative;
transition: all 0.6s;
display: flex;
align-items: center;
input {
flex: 1;
}
.tips {
position: absolute;
margin-left: 30rpx;
width: 0;
height: 0;
border-bottom: 12rpx solid #fff;
border-left: 12rpx solid transparent;
border-right: 12rpx solid transparent;
z-index: 1000;
}
.showClose {
width: 30rpx;
height: 30rpx;
position: absolute;
right: 14rpx;
top: 50%;
transform: translateY(-50%);
image {
width: 100%;
height: 100%;
}
}
.arrow {
transition: all 0.3s;
position: absolute;
top: 44%;
right: 20rpx;
border-left: 2rpx solid #999999;
border-bottom: 2rpx solid #999999;
height: 16rpx;
width: 16rpx;
transform: translateY(-50%) rotate(-45deg);
-webkit-transform: translateY(-50%) rotate(-45deg);
border-right: 2rpx solid transparent;
border-top: 2rpx solid transparent;
display: inline-block;
}
.arrow-up {
@extend .arrow;
transform: rotate(-235deg);
}
.multipleChoice {
display: flex;
align-items: center;
input {
flex: 1;
}
.defaultValue {
position: absolute;
left: 20rpx;
}
.option {
display: flex;
align-items: center;
background-color: #f4f4f5;
border-radius: 10rpx;
font-size: 24rpx;
color: #aa93b1;
padding: 6rpx 12rpx;
.img {
margin-left: 4rpx;
width: 30rpx;
height: 30rpx;
}
}
.more {
background-color: #f4f4f5;
border-radius: 10rpx;
font-size: 24rpx;
color: #aa93b1;
margin-left: 10rpx;
padding: 6rpx 12rpx;
}
}
}
.content {
position: absolute;
margin-top: 20rpx;
padding: 20rpx;
border-radius: 4px;
background-color: #fff;
box-shadow: rgba(0, 0, 0, 0.5) 0px 3px 8px;
z-index: 999;
max-height: 300rpx;
overflow-y: auto;
.item {
padding: 10rpx;
height: 60rpx;
line-height: 40rpx;
margin-bottom: 10rpx;
font-size: 28px;
color: #000;
}
.choose {
background-color: #f5f7fa;
color: #409eff;
font-weight: 700;
}
}
.animation-top {
animation-name: slide-top;
animation-duration: 0.5s;
animation-timing-function: ease-out;
animation-fill-mode: both;
}
.animation-bottom {
animation-name: slide-bottom;
animation-duration: 0.2s;
animation-timing-function: ease-out;
animation-fill-mode: both;
}
@keyframes slide-top {
0% {
opacity: 0;
transform: translateY(-20%);
}
100% {
opacity: 1;
transform: translateY(0);
}
}
@keyframes slide-bottom {
0% {
opacity: 1;
transform: translateY(0);
}
100% {
opacity: 0;
transform: translateY(-20%);
}
}
}
.active {
// border: 2rpx solid #409eff;
}
.contentMask {
position: fixed;
left: 0;
top: 0;
bottom: 0;
right: 0;
width: 100%;
height: 100%;
z-index: 998;
}
</style>