200字范文,内容丰富有趣,生活中的好帮手!
200字范文 > 移动端下拉刷新 兼容ios Android及微信浏览器

移动端下拉刷新 兼容ios Android及微信浏览器

时间:2019-03-17 07:23:34

相关推荐

移动端下拉刷新 兼容ios Android及微信浏览器

先看一下效果图

下拉效果的样子参考的新浪微博,滚动加载是ydui的滚动加载组件

因为滚动加载使用的ydui的组件,我这里便不再累述

在线体验点这里

首先分析下拉刷新是怎么实现的

1.页面滚动到顶部时,用户手指向下拖动2.页面整体开始随着手指向下移动,同时出现下拉的动画3.用户拖动超过指定长度之后松开手指,页面开始回弹并且执行加载中的动画4.加载完成之后执行结束的动画

实现原理

一、touchstart事件中1.判断是不是滚动到了顶部,如果不是则什么也不用做2.判断上次的下拉刷新是不是结束了,如果没有则阻止浏览器默认行为3.如果滚动到了顶部 且 没有进行中的下拉刷新 则记录触摸的位置event.touches[0].clientY二、touchmove事件中1.判断是不是滚动到了顶部,如果不是则什么也不用做2.判断上次的下拉刷新是不是结束了,如果没有则阻止浏览器默认行为3.如果滚动到了顶部 且 没有进行中的下拉刷新1.判断手指是向上滑还是向下滑,向上则正常滚动页面,向下则执行下拉刷新2.如果手指向下拉,则判断滑动的距离是否超过了指定的距离,超过了则改变动画效果(由下拉刷新-> 释放刷新)三、touchend事件中与touchmove中判断同,唯一不同的是滑动的距离是否超过了指定的距离触发回调并且执行刷新中的动画

获取页面滚动的位置(或者是div内部滚动位置)

/*** 传入一个dom对象,返回获取滚动的位置* @method getScrollTop* @param {dom} dom节点* @return {Number} 滚动的位置*/function getScrollTop(element) {if (element === window) {return Math.max(window.pageYOffset || 0,document.documentElement.scrollTop)} else {return element.scrollTop}}

事件处理

//touchstart事件function touchStartHandler(event) {//正在执行下拉刷新则返回if (this.touches.loading) {event.preventDefault()return}//当向下滚动了则直接返回//this.getScrollTop(this.scrollview)获取元素滚动的位置,实现方法见源码//this.$refs.dragBox.getBoundingClientRect().top为元素距离窗口顶部的距离//this.offsetTop为页面初始化时 this.$refs.dragBox.getBoundingClientRect().top的值if (this.getScrollTop(this.scrollview) > 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {return}//数据初始化this.touches.loading = false //是否在下拉刷新回调中this.touches.startClientY = 0 //触摸初始位置this.touches.isDraging = false //是否开始下拉刷新this.touches.statusText = '下拉刷新' //下拉刷新动画中的描述文字this.moveOffset = 0 //手指滑动的距离//记录触摸位置this.touches.startClientY = event.touches[0].clientY}//touchmove事件function touchMoveHandler(event) {const touches = this.touches//记录当前触摸位置,为了和下一个触摸位置作比较,判断是向上还是向下移动touches.currentClientY = event.touches[0].clientY//当向下滚动了则直接返回if (this.getScrollTop(this.scrollview) > 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {this.touches.isDraging = false //没有开始下拉刷新this.moveOffset = 0 //手指滑动的距离0return}//正在执行下拉刷新则返回if (this.touches.loading) {event.preventDefault()return}//当前触摸的位置const currentY = event.touches[0].clientY//防止手指直接下滑造成页面不能正常的滚动if (!touches.isDraging && currentY < touches.startClientY) {return}//手指先先下拉,再向上滑,说明此时手指已经在触摸位置上方了if (touches.isDraging &&(currentY - touches.startClientY < 0 ||this.$refs.dragBox.getBoundingClientRect().top <this.offsetTop)) {// this.isDragToUp = true;event.preventDefault()return}//手指向下滑if (touches.isDraging && this.getScrollTop(this.scrollview) === 0) {event.preventDefault()}// //开始下拉刷新this.touches.isDraging = true//手指滑动的距离let deltaSlide = currentY - touches.startClientY//如果超过了指定的距离, 达到了释放更新的条件//touches.distance为顶部加载中动画的高度//this.double为手指移动距离和页面实际移动距离的倍数if (deltaSlide >= touches.distance * this.double) {this.touches.statusText = '释放更新'} else {this.touches.statusText = '下拉刷新'}//记录滑动的距离this.moveOffset = deltaSlide}//touchend事件function touchEndHandler(event) {const touches = this.touches//正在执行下拉刷新则返回if (this.touches.loading) {event.preventDefault()return}//当向下滚动了则直接返回if (this.getScrollTop(this.scrollview) > 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {this.touches.isDraging = falsethis.moveOffset = 0return}const currentY = event.changedTouches[0].clientY//说明此时手指已经在触摸位置上方了if (currentY - touches.startClientY < 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {this.touches.isDraging = falseevent.preventDefault()return}//下拉刷新阻止浏览器默认行为if (this.getScrollTop(this.scrollview) === 0) {event.preventDefault()}//手指滑动的距离let deltaSlide = currentY - touches.startClientY//如果超过了指定的距离if (deltaSlide >= touches.distance * this.double) {//进行更新的回调及动画的改变,这部分见源码,耐心的看源码,还是能很容易看明白的some code...} else {this.touches.isDraging = false//距离不够则不刷新this.Retract(0, false)}}

上方只展示了主要部分,完整源码见文末

然后我把下拉刷新和上拉加载封装为一个vue组件

使用方法

//引入import YdInfinitescroll from './components/InfiniteScroll.vue'//注册components: {YdInfinitescroll }//使用<yd-infinitescroll:pullcallback="pullcallback":callback="callback"ref="infinitescrollDemo"><div slot='list'>...这里放你的内容</div></yd-infinitescroll>

说明:

1.传入callback参数表示开启滚动加载功能1.this.$refs.infinitescrollDemo.$emit('ydui.infinitescroll.finishLoad')表示单次数据请求完毕2.this.$refs.infinitescrollDemo.$emit('ydui.infinitescroll.loadedDone')表示所有数据请求完毕3.this.$refs.infinitescrollDemo.$emit('ydui.infinitescroll.reInit')表示重新初始化2.传入pullcallback参数表示开启下拉刷新功能1.更新成功请调用this.$refs.infinitescrollDemo.$emit('ydui.pullrefresh.finishLoad.success',true) 参数 true 开始提示, false 关闭提示, 默认true2.更新成功请调用this.$refs.infinitescrollDemo.$emit('ydui.pullrefresh.finishLoad.fail',true) 参数 true 开始提示, false 关闭提示, 默认true3.传入pullTipBgColor参数来修改下拉刷新成功状态的背景色,默认蓝色(#171dca)

组件源码,直接copy源码保存为InfiniteScroll.vue即可开始使用

<template><div><!-- 下拉刷新 --><divclass="dragBox"ref="dragBox":style="moveOffset? {transform: `translateY(${moveOffset / double}px)` }: ''"><!-- 下拉刷新动画效果 --><divclass="yd-pullTip":style="{height: `${moveOffset / double}px`,top: `-${moveOffset / double}px`,paddingBottom:moveOffset / double > (touches.distance - 20) / 2? (touches.distance - 20) / 2 + 'px': `${moveOffset / double}px`}"><imgv-if="touches.loading"src=""/><imgv-else:class="{ rotate: moveOffset >= touches.distance * double }"src=""/>{{touches.statusText }}</div><!-- 下拉刷新提示效果 --><div class="yd-Tip" v-if="pullupdateStatus">{{pullupdateText }}<span:style="pullupdateText === '更新成功'? `backgroundColor: ${pullTipBgColor};`: 'backgroundColor: #aeaeae;'"></span></div><slot name="list"></slot></div><div ref="tag" style="height: 0;"></div><div class="yd-list-loading" v-if="!isDone"><div v-show="isLoading"><slot name="loadingTip"><template><svgxmlns="/2000/svg"viewBox="0 0 100 100"preserveAspectRatio="xMidYMid"class="lds-ellipsis"><circle cx="84" cy="50" r="5.04711" fill="#f3b72e"><animateattributeName="r"values="10;0;0;0;0"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="0s"></animate><animateattributeName="cx"values="84;84;84;84;84"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="0s"></animate></circle><circle cx="66.8398" cy="50" r="10" fill="#E8574E"><animateattributeName="r"values="0;10;10;10;0"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="-0.85s"></animate><animateattributeName="cx"values="16;16;50;84;84"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="-0.85s"></animate></circle><circle cx="32.8398" cy="50" r="10" fill="#43A976"><animateattributeName="r"values="0;10;10;10;0"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="-0.425s"></animate><animateattributeName="cx"values="16;16;50;84;84"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="-0.425s"></animate></circle><circle cx="16" cy="50" r="4.95289" fill="#304153"><animateattributeName="r"values="0;10;10;10;0"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="0s"></animate><animateattributeName="cx"values="16;16;50;84;84"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="0s"></animate></circle><circle cx="16" cy="50" r="0" fill="#f3b72e"><animateattributeName="r"values="0;0;10;10;10"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="0s"></animate><animateattributeName="cx"values="16;16;16;50;84"keyTimes="0;0.25;0.5;0.75;1"keySplines="0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1;0 0.5 0.5 1"calcMode="spline"dur="1.7s"repeatCount="indefinite"begin="0s"></animate></circle></svg></template></slot></div></div><div class="yd-list-donetip" v-show="!isLoading && isDone"><slot name="doneTip">没有更多数据了</slot></div></div></template><script type="text/babel">export default {name: 'BaseInfinitescroll',data() {return {isLoading: false,isDone: false,num: 1,touches: {loading: false, //是否在下拉刷新回调中distance: 60, //滑动距离大于100时可释放刷新startClientY: 0,currentClientY: Math.pow(2, 32), //当前触摸位置isDraging: false, //是否开始下拉刷新statusText: '下拉刷新' //此时的状态描述},moveOffset: 0, //手指下拉的长度滚动的double: 3, //手滑动距离与下拉距离的倍数// step: 20 //松开手指界面向上滑动的速度time: 100,pullupdateStatus: false, //是否展示下拉刷新更新后的提示pullupdateText: '更新成功' //展示下拉刷新更新后的提示}},props: {callback: {type: Function},pullcallback: {type: Function},distance: {default: 0,validator(val) {return /^\d*$/.test(val)}},scrollTop: {type: Boolean,default: true},pullTipBgColor: {type: String,default: '#1989fa'}},methods: {init() {if (this.scrollTop) {if (this.scrollview === window) {window.scrollTo(0, 0)} else {this.scrollview.scrollTop = 0}}this.scrollview.addEventListener('scroll',this.throttledCheck,false)this.$on('ydui.infinitescroll.loadedDone', () => {this.isLoading = falsethis.isDone = true})this.$on('ydui.infinitescroll.finishLoad', () => {this.isLoading = false})this.$on('ydui.infinitescroll.reInit', () => {this.isLoading = falsethis.isDone = false})},pullinit() {const dragBox = this.$refs.dragBoxdragBox.addEventListener('touchstart', this.touchStartHandler)dragBox.addEventListener('touchmove', this.touchMoveHandler)dragBox.addEventListener('touchend', this.touchEndHandler)//防止微信浏览器下拉出现域名document.body.addEventListener('touchmove', this.stopDragEvent, {passive: false //调用阻止默认行为})//容器距离顶部的距离this.offsetTop = this.$refs.dragBox.getBoundingClientRect().top//上拉加载完成this.$on('ydui.pullrefresh.finishLoad.success', (tip = true) => {this.pullupdateText = '更新成功'this.Retract(0, tip)})this.$on('ydui.pullrefresh.finishLoad.fail', (tip = true) => {this.pullupdateText = '更新失败'this.Retract(0, tip)})// eslint-disable-next-line no-unused-varsthis.$on('ydui.pullrefresh.finishLoad.load', (tip = true) => {this.touches.statusText = '加载中'this.touches.loading = truethis.moveOffset = this.double * this.touches.distancethis.pullupdateStatus = false})},scrollHandler() {if (this.isLoading || this.isDone) returnconst scrollview = this.scrollviewconst contentHeight = document.body.offsetHeightconst isWindow = scrollview === windowconst offsetTop = isWindow? 0: scrollview.getBoundingClientRect().topconst scrollviewHeight = isWindow? contentHeight: scrollview.offsetHeightif (!scrollview) {// eslint-disable-next-lineconsole.warn("Can't find the scrollview!")return}if (!this.$refs.tag) {// eslint-disable-next-lineconsole.warn("Can't find the refs.tag!")return}const tagOffsetTop =Math.floor(this.$refs.tag.getBoundingClientRect().top) - 1const distance =!!this.distance && this.distance > 0? ~~this.distance: Math.floor(contentHeight / 10)if (tagOffsetTop > offsetTop &&tagOffsetTop - (distance + offsetTop) * this.num <=contentHeight &&this.$el.offsetHeight > scrollviewHeight) {this.isLoading = truethis.callback && this.callback()this.num++}},throttle(method, context) {clearTimeout(method.tId)method.tId = setTimeout(() => {method.call(context)}, 30)},throttledCheck() {this.throttle(this.scrollHandler)},getScrollview(el) {let currentNode = elwhile (currentNode &&currentNode.tagName !== 'HTML' &&currentNode.tagName !== 'BODY' &&currentNode.nodeType === 1) {let overflowY = document.defaultView.getComputedStyle(currentNode).overflowYif (overflowY === 'scroll' || overflowY === 'auto') {return currentNode}currentNode = currentNode.parentNode}return window},/*** 获取滚动的位置* @method getScrollTop* @return {Number} 滚动的位置*/getScrollTop(element) {if (element === window) {return Math.max(window.pageYOffset || 0,document.documentElement.scrollTop)} else {return element.scrollTop}},touchStartHandler(event) {//正在执行下拉刷新则返回if (this.touches.loading) {event.preventDefault()return}//当向下滚动了则直接返回if (this.getScrollTop(this.scrollview) > 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {return}//数据初始化this.touches.loading = falsethis.touches.startClientY = 0this.touches.isDraging = falsethis.touches.statusText = '下拉刷新'this.moveOffset = 0//记录触摸位置// this.touches.startClientX = event.touches[0].clientXthis.touches.startClientY = event.touches[0].clientY},touchMoveHandler(event) {const touches = this.touches//记录当前触摸位置touches.currentClientY = event.touches[0].clientY//当向下滚动了则直接返回if (this.getScrollTop(this.scrollview) > 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {// this.dragTip.translate = 0;// this.resetParams();this.touches.isDraging = falsethis.moveOffset = 0return}//正在执行下拉刷新则返回if (this.touches.loading) {event.preventDefault()return}// console.log(this.getScrollTop(this.scrollview))// console.log('执行了')const currentY = event.touches[0].clientY// const currentX = event.touches[0].clientX//防止手指直接下滑造成页面不能正常的滚动if (!touches.isDraging && currentY < touches.startClientY) {return}//手指先先下拉,再向上滑,说明此时手指已经在触摸位置上方了if (touches.isDraging &&(currentY - touches.startClientY < 0 ||this.$refs.dragBox.getBoundingClientRect().top <this.offsetTop)) {// this.isDragToUp = true;event.preventDefault()return}//手指向下滑if (touches.isDraging && this.getScrollTop(this.scrollview) === 0) {event.preventDefault()}// //开始下拉刷新this.touches.isDraging = true// const touchAngle =//(Math.atan2(// Math.abs(currentY - touches.startClientY),// Math.abs(currentX - touches.startClientX)//) *// 180) ///Math.PI// if (90 - touchAngle > 45) return//手指滑动的距离let deltaSlide = currentY - touches.startClientY//如果超过了指定的距离, 达到了释放更新的条件if (deltaSlide >= touches.distance * this.double) {this.touches.statusText = '释放更新'} else {this.touches.statusText = '下拉刷新'}//记录滑动的位置this.moveOffset = deltaSlide// console.log(this.moveOffset)},touchEndHandler(event) {const touches = this.touches// console.log(this.touches.isDraging)//正在执行下拉刷新则返回if (this.touches.loading) {event.preventDefault()return}//当向下滚动了则直接返回if (this.getScrollTop(this.scrollview) > 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {this.touches.isDraging = falsethis.moveOffset = 0return}const currentY = event.changedTouches[0].clientY// const currentX = event.changedTouches[0].clientX//说明此时手指已经在触摸位置上方了if (currentY - touches.startClientY < 0 ||this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop) {this.touches.isDraging = falseevent.preventDefault()return}//下拉刷新阻止浏览器默认行为if (this.getScrollTop(this.scrollview) === 0 &&currentY !== this.touches.startClientY) {event.preventDefault()}//手指滑动的距离let deltaSlide = currentY - touches.startClientY//如果超过了指定的距离if (deltaSlide >= touches.distance * this.double) {//进行更新的动画this.touches.statusText = '加载中'// alert('下拉刷新')this.touches.startClientY = 0this.touches.isDraging = falsethis.Retract(touches.distance * this.double)return} else {this.touches.isDraging = false//距离不够则不刷新this.Retract(0, false)}},stopDragEvent(event) {this.touches.isDraging && event.preventDefault()},Retract(offsetTop, tip) {let timer = setInterval(() => {//根据时间计算出每次运动的距离// 总时间 / 每次运动时间 = 运动次数// 总长度 / 运动次数 = 每次运动距离let step = ((this.touches.distance * this.double) /((this.time * 60) / 1000).toFixed(2)).toFixed(2)if (this.moveOffset - step > offsetTop) {this.moveOffset -= step} else {this.moveOffset = offsetTopclearInterval(timer)if (offsetTop !== 0) {this.touches.loading = truethis.pullcallback && this.pullcallback()} else {//重置this.touches.loading = falsethis.touches.startClientY = 0this.touches.isDraging = falsethis.touches.statusText = '下拉刷新'//执行加载中动画if (tip) {//执行更新成功或者失败动画this.pullupdateStatus = truesetTimeout(() => {this.pullupdateStatus = false}, 1000)}}}}, 1000 / 60)}},mounted() {// console.log(this.distance, this.touches)this.scrollview = this.getScrollview(this.$el)if (this.callback) {this.init()}if (this.pullcallback) {this.pullinit()}},beforeDestroy() {this.scrollview.removeEventListener('scroll', this.throttledCheck)this.$refs.dragBox.removeEventListener('touchstart',this.touchStartHandler)this.$refs.dragBox.removeEventListener('touchmove',this.touchMoveHandler)this.$refs.dragBox.removeEventListener('touchend', this.touchEndHandler)document.body.removeEventListener('touchmove', this.stopDragEvent)}}</script><style lang="scss" scoped>@keyframes intact {0% {border-radius: 50%;}100% {border-radius: 0%;}}.dragBox {position: relative;}.yd {&-list-loading {padding: 0.1rem 0;text-align: center;font-size: 0.26rem;color: #999;height: 0.66rem;box-sizing: content-box;&-box {height: 0.66rem;overflow: hidden;line-height: 0.66rem;}img {height: 0.66rem;display: inline-block;}svg {width: 0.66rem;height: 0.66rem;}}&-list-donetip {font-size: 0.24rem;text-align: center;padding: 0.25rem 0;color: #777;}&-pullTip {text-align: center;font-size: 0.24rem;position: absolute;left: 0;right: 0;background: #eeeeee;color: #a5a5a5;overflow: hidden;display: flex;align-items: flex-end;justify-content: center;img {height: 20px;margin-right: 12px;margin-bottom: -1px;transition: transform 0.1s linear;transform: rotate(180deg);}img.rotate {transform: rotate(0deg);}}&-Tip {z-index: 99999999;position: absolute;left: 0;right: 0;top: 0;height: 36px;line-height: 36px;font-size: 0.26rem;overflow: hidden;text-align: center;color: #fff;span {position: absolute;top: 0;left: 0;z-index: -1;display: block;width: 100%;padding-top: 100%;border-radius: 50%;animation: intact 0.1s linear forwards;}}}</style>

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。