264 lines
6.5 KiB
Vue
264 lines
6.5 KiB
Vue
|
<template>
|
|||
|
<view
|
|||
|
class="tn-slider-class tn-slider"
|
|||
|
:class="{'tn-slider--disabled': disabled}"
|
|||
|
:style="{
|
|||
|
backgroundColor: inactiveColor
|
|||
|
}"
|
|||
|
@tap="click"
|
|||
|
>
|
|||
|
<!-- slider滑动线 -->
|
|||
|
<view
|
|||
|
class="tn-slider__gap"
|
|||
|
:style="[
|
|||
|
barStyle,
|
|||
|
{
|
|||
|
height: this.$tn.string.getLengthUnitValue(lineHeight),
|
|||
|
backgroundColor: activeColor
|
|||
|
}
|
|||
|
]"
|
|||
|
>
|
|||
|
<!-- slider滑块 -->
|
|||
|
<view
|
|||
|
class="tn-slider__button-wrap"
|
|||
|
@touchstart="touchStart"
|
|||
|
@touchmove="touchMove"
|
|||
|
@touchend="touchEnd"
|
|||
|
@touchcancel="touchEnd"
|
|||
|
>
|
|||
|
<view v-if="$slots.default || $slots.$default">
|
|||
|
<slot></slot>
|
|||
|
</view>
|
|||
|
<view
|
|||
|
v-else
|
|||
|
class="tn-slider__button"
|
|||
|
:style="[blockStyle, {
|
|||
|
height: this.$tn.string.getLengthUnitValue(blockWidth),
|
|||
|
width: this.$tn.string.getLengthUnitValue(blockWidth),
|
|||
|
backgroundColor: blockColor
|
|||
|
}]"
|
|||
|
></view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</template>
|
|||
|
|
|||
|
<script>
|
|||
|
export default {
|
|||
|
name: 'tn-slider',
|
|||
|
props: {
|
|||
|
// 进度值
|
|||
|
value: {
|
|||
|
type: [Number, String],
|
|||
|
default: 0
|
|||
|
},
|
|||
|
// 最小值
|
|||
|
min: {
|
|||
|
type: Number,
|
|||
|
default: 0
|
|||
|
},
|
|||
|
// 最大值
|
|||
|
max: {
|
|||
|
type: Number,
|
|||
|
default: 100
|
|||
|
},
|
|||
|
// 步进值
|
|||
|
step: {
|
|||
|
type: Number,
|
|||
|
default: 1
|
|||
|
},
|
|||
|
// 禁用
|
|||
|
disabled: {
|
|||
|
type: Boolean,
|
|||
|
default: false
|
|||
|
},
|
|||
|
// 滑块宽度
|
|||
|
blockWidth: {
|
|||
|
type: Number,
|
|||
|
default: 30
|
|||
|
},
|
|||
|
// 滑动条高度
|
|||
|
lineHeight: {
|
|||
|
type: Number,
|
|||
|
default: 8
|
|||
|
},
|
|||
|
// 滑动条激活的颜色
|
|||
|
activeColor: {
|
|||
|
type: String,
|
|||
|
default: '#01BEFF'
|
|||
|
},
|
|||
|
// 滑动条未被激活的颜色
|
|||
|
inactiveColor: {
|
|||
|
type: String,
|
|||
|
default: '#E6E6E6'
|
|||
|
},
|
|||
|
// 滑块的颜色
|
|||
|
blockColor: {
|
|||
|
type: String,
|
|||
|
default: '#FFFFFF'
|
|||
|
},
|
|||
|
// 自定义滑块的样式
|
|||
|
blockStyle: {
|
|||
|
type: Object,
|
|||
|
default() {
|
|||
|
return {}
|
|||
|
}
|
|||
|
}
|
|||
|
},
|
|||
|
data() {
|
|||
|
return {
|
|||
|
startX: 0,
|
|||
|
status: 'end',
|
|||
|
newValue: 0,
|
|||
|
distanceX: 0,
|
|||
|
startValue: 0,
|
|||
|
barStyle: {},
|
|||
|
sliderRect: {
|
|||
|
left: 0,
|
|||
|
width: 0
|
|||
|
}
|
|||
|
}
|
|||
|
},
|
|||
|
watch: {
|
|||
|
value(val) {
|
|||
|
// 只有在非滑动状态时,才可以通过value更新滑块值,这里监听,是为了让用户触发
|
|||
|
if (this.status === 'end') this.updateValue(val, false)
|
|||
|
}
|
|||
|
},
|
|||
|
created() {
|
|||
|
this.updateValue(this.value, false)
|
|||
|
},
|
|||
|
mounted() {
|
|||
|
this._tGetRect('.tn-slider').then(res => {
|
|||
|
this.sliderRect = res
|
|||
|
})
|
|||
|
},
|
|||
|
methods: {
|
|||
|
// 开始滑动
|
|||
|
touchStart(event) {
|
|||
|
if (this.disabled) return
|
|||
|
if (!event.changedTouches[0]) return
|
|||
|
|
|||
|
this.startX = 0
|
|||
|
// 触摸点
|
|||
|
this.startX = event.changedTouches[0].pageX
|
|||
|
this.startValue = this.format(this.value)
|
|||
|
|
|||
|
// 标识当前开始触摸
|
|||
|
this.status = 'start'
|
|||
|
},
|
|||
|
// 滑动移动中
|
|||
|
touchMove(event) {
|
|||
|
this._tGetRect('.tn-slider').then(res => {
|
|||
|
this.sliderRect = res
|
|||
|
});
|
|||
|
if (this.disabled) return
|
|||
|
if (!event.changedTouches[0]) return
|
|||
|
|
|||
|
// 连续触摸的过程会一直触发本方法,但只有手指触发且移动了才被认为是拖动了,才发出事件
|
|||
|
// 触摸后第一次移动已经将status设置为moving状态,故触摸第二次移动不会触发本事件
|
|||
|
if (this.status === 'start') this.$emit('start')
|
|||
|
let movePageX = event.changedTouches[0].pageX
|
|||
|
// 滑块的左边不一定跟屏幕左边接壤,所以需要减去最外层父元素的左边值
|
|||
|
let marginLeft = this.sliderRect.left;
|
|||
|
marginLeft = marginLeft < 0 ? 0 : marginLeft;
|
|||
|
this.distanceX = movePageX - marginLeft;
|
|||
|
// 获得移动距离对整个滑块的百分比值,此为带有多位小数的值,不能用此更新视图
|
|||
|
// 否则造成通信阻塞,需要每改变一个step值时修改一次视图
|
|||
|
this.newValue = ((this.distanceX / this.sliderRect.width) * (this.max - this.min)) + this.min
|
|||
|
this.status = 'moving'
|
|||
|
this.$emit('moving')
|
|||
|
|
|||
|
this.updateValue(this.newValue, true)
|
|||
|
},
|
|||
|
// 滑动结束
|
|||
|
touchEnd() {
|
|||
|
if(this.disabled) return
|
|||
|
if (this.status === 'moving') {
|
|||
|
this.updateValue(this.newValue, false)
|
|||
|
this.$emit('end')
|
|||
|
}
|
|||
|
this.status = 'end'
|
|||
|
},
|
|||
|
// 更新数值
|
|||
|
updateValue(value, drag) {
|
|||
|
// 去掉小数部分,对step进行步进处理
|
|||
|
value = this.format(value)
|
|||
|
const width = Math.round((value - this.min) / (this.max - this.min) * 100)
|
|||
|
// 不允许滑动的距离小于0和超过100
|
|||
|
if (width < 0 || width > 100) return
|
|||
|
// 设置移动的百分比
|
|||
|
let barStyle = {
|
|||
|
width: width + '%'
|
|||
|
}
|
|||
|
// 移动期间取消动画
|
|||
|
if (drag === true) {
|
|||
|
barStyle.transition = 'none'
|
|||
|
} else {
|
|||
|
// 非移动期间,删掉对过渡为空的声明,让css中的声明起效
|
|||
|
delete barStyle.transition
|
|||
|
}
|
|||
|
|
|||
|
// 修改value值
|
|||
|
this.$emit('input', value)
|
|||
|
this.barStyle = barStyle
|
|||
|
},
|
|||
|
// 点击事件
|
|||
|
click(event) {
|
|||
|
this._tGetRect('.tn-slider').then(res => {
|
|||
|
this.sliderRect = res
|
|||
|
})
|
|||
|
if (this.disabled) return
|
|||
|
// 直接点击的情况,计算方式和touchMove方法一致
|
|||
|
const value = (((event.detail.x - this.sliderRect.left) / this.sliderRect.width) * (this.max - this.min)) + this.min
|
|||
|
this.updateValue(value, false)
|
|||
|
},
|
|||
|
// 格式化滑动的值
|
|||
|
format(value) {
|
|||
|
return Math.round(Math.max(this.min, Math.min(value, this.max)) / this.step) * this.step
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
</script>
|
|||
|
|
|||
|
<style lang="scss" scoped>
|
|||
|
.tn-slider {
|
|||
|
width: 100%;
|
|||
|
position: relative;
|
|||
|
border-radius: 1000rpx;
|
|||
|
// 增加点击的范围
|
|||
|
border-width: 20rpx;
|
|||
|
border-style: solid;
|
|||
|
border-color: transparent;
|
|||
|
background-color: $tn-font-holder-color;
|
|||
|
background-clip: content-box;
|
|||
|
|
|||
|
&__gap {
|
|||
|
position: relative;
|
|||
|
border-radius: inherit;
|
|||
|
transition: width 0.2s;
|
|||
|
background-color: #01BEFF;
|
|||
|
}
|
|||
|
|
|||
|
&__button {
|
|||
|
width: 30rpx;
|
|||
|
height: 30rpx;
|
|||
|
border-radius: 50%;
|
|||
|
box-shadow: 0 2rpx 4rpx rgba(0, 0, 0, 0.6);
|
|||
|
background-color: #FFFFFF;
|
|||
|
cursor: pointer;
|
|||
|
|
|||
|
&-wrap {
|
|||
|
position: absolute;
|
|||
|
top: 50%;
|
|||
|
right: 0;
|
|||
|
transform: translate3d(50%, -50%, 0);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
&--disabled {
|
|||
|
opacity: 0.6;
|
|||
|
}
|
|||
|
}
|
|||
|
</style>
|