123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113 |
- /**
- * 使用普通的js方案实现slider
- */
- export default {
- watch: {
- value(n) {
- // 只有在非滑动状态时,才可以通过value更新滑块值,这里监听,是为了让用户触发
- if (this.status === 'end') {
- this.updateSliderPlacement(n, true)
- }
- }
- },
- mounted() {
- this.init()
- },
- methods: {
- init() {
- this.getSliderRect()
- },
- // 获取slider尺寸
- getSliderRect() {
- // 获取滑块条的尺寸信息
- setTimeout(() => {
- this.$uGetRect('.u-slider').then((rect) => {
- this.sliderRect = rect
- this.updateSliderPlacement(this.value, true)
- })
- }, 10)
- },
- // 是否可以操作
- canNotDo() {
- return this.disabled
- },
- // 获取当前手势点的X轴位移值
- getTouchX(e) {
- return e.touches[0].clientX
- },
- formatStep(value) {
- // 移动点占总长度的百分比
- return Math.round(Math.max(this.min, Math.min(value, this.max)) / this.step) * this.step
- },
- // 发出事件
- emitEvent(event, value) {
- this.$emit(event, value || this.value)
- },
- // 标记当前手势的状态
- setTouchStatus(status) {
- this.status = status
- },
- onTouchStart(e) {
- if (this.canNotDo()) {
- return
- }
- // 标示当前的状态为开始触摸滑动
- this.emitEvent('start')
- this.setTouchStatus('start')
- },
- onTouchMove(e) {
- if (this.canNotDo()) {
- return
- }
- // 滑块的左边不一定跟屏幕左边接壤,所以需要减去最外层父元素的左边值
- const x = this.getTouchX(e)
- const { left, width } = this.sliderRect
- const distanceX = x - left
- // 获得移动距离对整个滑块的百分比值,此为带有多位小数的值,不能用此更新视图
- // 否则造成通信阻塞,需要每改变一个step值时修改一次视图
- const percent = (distanceX / width) * 100
- this.setTouchStatus('moving')
- this.updateSliderPlacement(percent, true, 'moving')
- },
- onTouchEnd() {
- if (this.canNotDo()) {
- return
- }
- this.emitEvent('end')
- this.setTouchStatus('end')
- },
- // 设置滑点的位置
- updateSliderPlacement(value, drag, event) {
- // 去掉小数部分,同时也是对step步进的处理
- const { width } = this.sliderRect
- const percent = this.formatStep(value)
- // 设置移动的值
- const barStyle = {
- width: `${percent / 100 * width}px`
- }
- // 移动期间无需过渡动画
- if (drag === true) {
- barStyle.transition = 'none'
- } else {
- // 非移动期间,删掉对过渡为空的声明,让css中的声明起效
- delete barStyle.transition
- }
- // 修改value值
- this.$emit('input', percent)
- // 事件的名称
- if (event) {
- this.emitEvent(event, percent)
- }
- this.barStyle = barStyle
- },
- onClick(e) {
- if (this.canNotDo()) {
- return
- }
- // 直接点击滑块的情况,计算方式与onTouchMove方法相同
- const { left, width } = this.sliderRect
- const value = ((e.detail.x - left) / width) * 100
- this.updateSliderPlacement(value, false, 'click')
- }
- }
- }
|