mirror of
https://gitee.com/hhyykk/ipms-sjy.git
synced 2025-07-06 15:15:07 +08:00
uni-app引入使用uView
This commit is contained in:
193
yudao-ui-app/uni_modules/uview-ui/components/u-slider/nvue.js
Normal file
193
yudao-ui-app/uni_modules/uview-ui/components/u-slider/nvue.js
Normal file
@ -0,0 +1,193 @@
|
||||
/**
|
||||
* 使用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
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user