|
|
/** * 使用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 } } }
|