194 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
		
		
			
		
	
	
			194 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
|  | /** | |||
|  |  * 使用bindingx方案实现slider | |||
|  |  * 只能使用于nvue下 | |||
|  |  */ | |||
|  | // 引入bindingx,此库类似于微信小程序wxs,目的是让js运行在视图层,减少视图层和逻辑层的通信折损
 | |||
|  | const BindingX = uni.requireNativePlugin('bindingx') | |||
|  | // nvue操作dom的库,用于获取dom的尺寸信息
 | |||
|  | const dom = uni.requireNativePlugin('dom') | |||
|  | // nvue中用于操作元素动画的库,类似于uni.animation,只不过uni.animation不能用于nvue
 | |||
|  | const animation = uni.requireNativePlugin('animation') | |||
|  | 
 | |||
|  | export default { | |||
|  |     data() { | |||
|  |         return { | |||
|  |             // 位移的偏移量
 | |||
|  |             x: 0, | |||
|  |             // 是否正在触摸过程中,用于标记动画类是否添加或移除
 | |||
|  |             touching: false, | |||
|  |             changeFromInside: false | |||
|  |         } | |||
|  |     }, | |||
|  |     watch: { | |||
|  |         // 监听vlaue的变化,此变化可能是由于内部修改v-model的值,或者外部
 | |||
|  |         // 从服务端获取一个值后,赋值给slider的v-model而导致的
 | |||
|  |         value(n) { | |||
|  |             if (!this.changeFromInside) { | |||
|  |                 this.initX() | |||
|  |             } else { | |||
|  |                 this.changeFromInside = false | |||
|  |             } | |||
|  |         } | |||
|  |     }, | |||
|  |     mounted() { | |||
|  |         this.init() | |||
|  |     }, | |||
|  |     methods: { | |||
|  |         init() { | |||
|  |             // 更新滑块尺寸信息
 | |||
|  |             this.getSliderRect().then((size) => { | |||
|  |                 this.sliderRect = size | |||
|  |                 this.initX() | |||
|  |             }) | |||
|  |         }, | |||
|  |         // 获取节点信息
 | |||
|  |         // 获取slider尺寸
 | |||
|  |         getSliderRect() { | |||
|  |             // 获取滑块条的尺寸信息
 | |||
|  |             // 通过nvue的dom模块,查询节点信息
 | |||
|  |             return new Promise((resolve) => { | |||
|  |                 this.$nextTick(() => { | |||
|  |                     dom.getComponentRect(this.$refs.slider, (res) => { | |||
|  |                         resolve(res.size) | |||
|  |                     }) | |||
|  |                 }) | |||
|  |             }) | |||
|  |         }, | |||
|  |         // 初始化按钮位置
 | |||
|  |         initButtonStyle({ | |||
|  |             barStyle, | |||
|  |             buttonWrapperStyle | |||
|  |         }) { | |||
|  |             this.barStyle = barStyle | |||
|  |             this.buttonWrapperStyle = buttonWrapperStyle | |||
|  |         }, | |||
|  |         emitEvent(event, value) { | |||
|  |             this.$emit(event, value || this.value) | |||
|  |         }, | |||
|  |         // 滑动开始
 | |||
|  |         async onTouchStart(e) { | |||
|  |             // if (this.disabled) return
 | |||
|  |             // // 阻止页面滚动,可以保证在滑动过程中,不让页面可以上下滚动,造成不好的体验
 | |||
|  |             // e.stopPropagation && e.stopPropagation()
 | |||
|  |             // e.preventDefault && e.preventDefault()
 | |||
|  |             // // 更新滑块的尺寸信息
 | |||
|  |             // this.sliderRect = await this.getSliderRect()
 | |||
|  |             // // 标记滑动过程中触摸点的信息
 | |||
|  |             // this.touchStart(e)
 | |||
|  |             // this.startValue = this.format(this.value)
 | |||
|  |             // this.dragStatus = 'start'
 | |||
|  | 
 | |||
|  |             // 标记滑动过程中触摸点的信息
 | |||
|  |             // this.touchStart(e)
 | |||
|  |         }, | |||
|  |         // 开始滑动
 | |||
|  |         onTouchMove(e) { | |||
|  |             // if (this.disabled) return;
 | |||
|  |             // if (this.dragStatus === 'start') {
 | |||
|  |             // 	this.$emit('drag-start')
 | |||
|  |             // }
 | |||
|  |             // // 标记当前滑动过程中的触点信息,此方法在touch mixin中
 | |||
|  |             // this.touchMove(e)
 | |||
|  |             // this.dragStatus = 'draging'
 | |||
|  |             // const {
 | |||
|  |             // 	width: sliderWidth
 | |||
|  |             // } = this.sliderRect
 | |||
|  |             // const diff = (this.deltaX / sliderWidth) * this.getRange()
 | |||
|  |             // this.newValue = this.startValue + diff
 | |||
|  |             // this.updateValue(this.newValue, false, true)
 | |||
|  |             // 获取元素ref
 | |||
|  |             // const button = this.$refs['nvue-button'].ref
 | |||
|  |             // const gap = this.$refs['nvue-gap'].ref
 | |||
|  | 
 | |||
|  |             //          animation.transition(gap, {
 | |||
|  |             // 	styles: {
 | |||
|  |             //                  width: `${this.startX + this.deltaX}px`
 | |||
|  |             // 	}
 | |||
|  |             // })
 | |||
|  |             // // console.log(this.startX + this.deltaX);
 | |||
|  |             // animation.transition(button, {
 | |||
|  |             // 	styles: {
 | |||
|  |             //         transform: `translateX(${this.startX + this.deltaX}px)`
 | |||
|  |             // 	}
 | |||
|  |             // })
 | |||
|  |             // this.barStyle = {
 | |||
|  |             // 	width: `${this.startX + this.deltaX}px`
 | |||
|  |             // }
 | |||
|  |             const { | |||
|  |                 x | |||
|  |             } = this.getTouchPoint(e) | |||
|  |             this.buttonWrapperStyle = { | |||
|  |                 transform: `translateX(${x}px)` | |||
|  |             } | |||
|  |             // this.buttonWrapperStyle = {
 | |||
|  |             // 	transform: `translateX(${this.format(this.startX + this.deltaX)}px)`
 | |||
|  |             // }
 | |||
|  |         }, | |||
|  |         // onTouchEnd() {
 | |||
|  |         // 	if (this.disabled) return;
 | |||
|  |         // 	if (this.dragStatus === 'draging') {
 | |||
|  |         // 		this.updateValue(this.newValue, true)
 | |||
|  |         // 		this.$emit('drag-end');
 | |||
|  |         // 	}
 | |||
|  |         // },
 | |||
|  |         updateValue(value, end, drag) { | |||
|  |             value = this.format(value) | |||
|  |             const { | |||
|  |                 width: sliderWidth | |||
|  |             } = this.sliderRect | |||
|  |             const width = `${((value - this.min) * sliderWidth) / this.getRange()}` | |||
|  |             this.value = value | |||
|  |             this.barStyle = { | |||
|  |                 width: `${width}px` | |||
|  |             } | |||
|  |             // console.log('width', width);
 | |||
|  |             if (drag) { | |||
|  |                 this.$emit('drag', { | |||
|  |                     value | |||
|  |                 }) | |||
|  |             } | |||
|  |             if (end) { | |||
|  |                 this.$emit('change', value) | |||
|  |             } | |||
|  |             if ((drag || end)) { | |||
|  |                 this.changeFromInside = true | |||
|  |                 this.$emit('update', value) | |||
|  |             } | |||
|  |         }, | |||
|  |         // 从value的变化,倒推得出x的值该为多少
 | |||
|  |         initX() { | |||
|  |             const { | |||
|  |                 left, | |||
|  |                 width | |||
|  |             } = this.sliderRect | |||
|  |             // 得出x的初始偏移值,之所以需要这么做,是因为在bindingX中,触摸滑动时,只能的值本次移动的偏移值
 | |||
|  |             // 而无法的值准确的前后移动的两个点的坐标值,weex纯粹为阿里巴巴的KPI(部门业绩考核)产物,也就这样了
 | |||
|  |             this.x = this.value / 100 * width | |||
|  |             // 设置移动的值
 | |||
|  |             const barStyle = { | |||
|  |                 width: `${this.x}px` | |||
|  |             } | |||
|  |             // 按钮的初始值
 | |||
|  |             const buttonWrapperStyle = { | |||
|  |                 transform: `translateX(${this.x - this.blockHeight / 2}px)` | |||
|  |             } | |||
|  |             this.initButtonStyle({ | |||
|  |                 barStyle, | |||
|  |                 buttonWrapperStyle | |||
|  |             }) | |||
|  |         }, | |||
|  |         // 移动点占总长度的百分比,此处需要先除以step,是为了保证step大于1时,比如10,那么在滑动11,12px这样的
 | |||
|  |         // 距离时,实际上滑块是不会滑动的,到了16,17px,经过四舍五入后,就变成了20px,进行了下一个跳变
 | |||
|  |         format(value) { | |||
|  |             return Math.round(uni.$u.range(this.min, this.max, value) / this.step) * this.step | |||
|  |         }, | |||
|  |         getRange() { | |||
|  |             const { | |||
|  |                 max, | |||
|  |                 min | |||
|  |             } = this | |||
|  |             return max - min | |||
|  |         } | |||
|  |     } | |||
|  | } |