181 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			181 lines
		
	
	
		
			5.3 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 {
 | 
						||
			// bindingx的回调值,用于取消绑定
 | 
						||
			panEvent: null,
 | 
						||
			// 标记是否移动状态
 | 
						||
			moving: false,
 | 
						||
			// 位移的偏移量
 | 
						||
			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()
 | 
						||
		},
 | 
						||
		// 获取节点信息
 | 
						||
		// 获取slider尺寸
 | 
						||
		getSliderRect() {
 | 
						||
			// 获取滑块条的尺寸信息
 | 
						||
			// 通过nvue的dom模块,查询节点信息
 | 
						||
			setTimeout(() => {
 | 
						||
				dom.getComponentRect(this.$refs['slider'], res => {
 | 
						||
					this.sliderRect = res.size
 | 
						||
					this.initX()
 | 
						||
				})
 | 
						||
			}, 10)
 | 
						||
		},
 | 
						||
		// 初始化按钮位置
 | 
						||
		initButtonStyle({
 | 
						||
			barStyle,
 | 
						||
			buttonWrapperStyle
 | 
						||
		}) {
 | 
						||
			this.barStyle = barStyle
 | 
						||
			this.buttonWrapperStyle = buttonWrapperStyle
 | 
						||
		},
 | 
						||
		emitEvent(event, value) {
 | 
						||
			this.$emit(event, value ? value : this.value)
 | 
						||
		},
 | 
						||
		formatStep(value) {
 | 
						||
			// 移动点占总长度的百分比
 | 
						||
			return Math.round(Math.max(this.min, Math.min(value, this.max)) / this.step) * this.step
 | 
						||
		},
 | 
						||
		// 滑动开始
 | 
						||
		onTouchStart(e) {
 | 
						||
			// 阻止页面滚动,可以保证在滑动过程中,不让页面可以上下滚动,造成不好的体验
 | 
						||
			e.stopPropagation && e.stopPropagation()
 | 
						||
			e.preventDefault && e.preventDefault()
 | 
						||
			if (this.moving || this.disabled) {
 | 
						||
				// 释放上一次的资源
 | 
						||
				if (this.panEvent?.token != 0) {
 | 
						||
					BindingX.unbind({
 | 
						||
						token: this.panEvent.token,
 | 
						||
						// pan为手势事件
 | 
						||
						eventType: 'pan'
 | 
						||
					})
 | 
						||
					this.gesToken = 0
 | 
						||
				}
 | 
						||
				return
 | 
						||
			}
 | 
						||
 | 
						||
			this.moving = true
 | 
						||
			this.touching = true
 | 
						||
 | 
						||
			// 获取元素ref
 | 
						||
			const button = this.$refs['nvue-button'].ref
 | 
						||
			const gap = this.$refs['nvue-gap'].ref
 | 
						||
 | 
						||
			const {
 | 
						||
				min,
 | 
						||
				max,
 | 
						||
				step
 | 
						||
			} = this
 | 
						||
			const {
 | 
						||
				left,
 | 
						||
				width
 | 
						||
			} = this.sliderRect
 | 
						||
 | 
						||
			// 初始值为本次偏移量x,加上次停止滑动时的结束值
 | 
						||
			let exporession = `(${this.x} + x)`
 | 
						||
			// 将偏移的x值,转为总位移的百分比值,为了和min和max进行判断
 | 
						||
			exporession = `(${exporession} / ${width}) * 100`
 | 
						||
			if (step > 1) {
 | 
						||
				// 如果step步进大于1,需要跳步,所以需要使用Math.round进行取整
 | 
						||
				exporession = `round(max(${min}, min(${exporession}, ${max})) / ${step}) * ${step}`
 | 
						||
			} else {
 | 
						||
				// 当step=1时,无需跳步,充分利用bindingx性能,滑块实时跟随手势,达到丝滑的效果
 | 
						||
				exporession = `max(${min}, min(${exporession}, ${max}))`
 | 
						||
			}
 | 
						||
			// 将百分比最后转化为对应的px值
 | 
						||
			exporession = `${exporession} / 100 * ${width}`
 | 
						||
			// 最大值不允许超过轨迹的宽度
 | 
						||
			const {
 | 
						||
				sliderWidth
 | 
						||
			} = this.sliderRect
 | 
						||
			exporession = `min(${sliderWidth}, ${exporession})`
 | 
						||
			// 滑块点总是需要一个左偏移的值,为自身宽度的一半
 | 
						||
			const buttonExpression = `${exporession} - ${this.blockHeight / 2}`
 | 
						||
			// 阿里为了KPI而开源的BindingX
 | 
						||
			this.panEvent = BindingX.bind({
 | 
						||
				anchor: button,
 | 
						||
				eventType: 'pan',
 | 
						||
				props: [{
 | 
						||
					element: gap,
 | 
						||
					// 绑定width属性,设置其宽度值
 | 
						||
					property: 'width',
 | 
						||
					expression
 | 
						||
				}, {
 | 
						||
					element: button,
 | 
						||
					// 绑定width属性,设置其宽度值
 | 
						||
					property: 'transform.translateX',
 | 
						||
					expression: buttonExpression
 | 
						||
				}]
 | 
						||
			}, (e) => {
 | 
						||
				if (e.state === 'end' || e.state === 'exit') {
 | 
						||
					// 
 | 
						||
					this.x = uni.$u.range(0, left + width, e.deltaX + this.x)
 | 
						||
					// 根据偏移值,得出移动的百分比,进而修改双向绑定的v-model的值
 | 
						||
					const value = (this.x / width) * 100
 | 
						||
					const percent = this.formatStep(value)
 | 
						||
					// 修改value值
 | 
						||
					this.$emit('input', percent)
 | 
						||
					// 标记下一次触发value的watch时,这个值的变化,是由内部改变的
 | 
						||
					this.changeFromInside = true
 | 
						||
					this.moving = false
 | 
						||
					this.touching = false
 | 
						||
				}
 | 
						||
			})
 | 
						||
		},
 | 
						||
		// 从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
 | 
						||
			})
 | 
						||
		}
 | 
						||
	}
 | 
						||
}
 |