先看一下效果圖javascript
下拉效果的樣子參考的新浪微博,滾動加載是ydui的滾動加載組件php
由於滾動加載使用的ydui的組件,我這裏便再也不累述css
在線體驗點這裏html
1.頁面滾動到頂部時,用戶手指向下拖動
2.頁面總體開始隨着手指向下移動,同時出現下拉的動畫
3.用戶拖動超過指定長度以後鬆開手指,頁面開始回彈而且執行加載中的動畫
4.加載完成以後執行結束的動畫
複製代碼
1、touchstart事件中
1.判斷是否是滾動到了頂部,若是不是則什麼也不用作
2.判斷上次的下拉刷新是否是結束了,若是沒有則阻止瀏覽器默認行爲
3.若是滾動到了頂部 且 沒有進行中的下拉刷新 則記錄觸摸的位置event.touches[0].clientY
2、touchmove事件中
1.判斷是否是滾動到了頂部,若是不是則什麼也不用作
2.判斷上次的下拉刷新是否是結束了,若是沒有則阻止瀏覽器默認行爲
3.若是滾動到了頂部 且 沒有進行中的下拉刷新
1.判斷手指是向上滑仍是向下滑,向上則正常滾動頁面,向下則執行下拉刷新
2.若是手指向下拉,則判斷滑動的距離是否超過了指定的距離,超過了則改變更畫效果(由下拉刷新-> 釋放刷新)
3、touchend事件中
與touchmove中判斷同,惟一不一樣的是滑動的距離是否超過了指定的距離觸發回調而且執行刷新中的動畫
複製代碼
獲取頁面滾動的位置(或者是div內部滾動位置)vue
/** * 傳入一個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
}
}
複製代碼
事件處理java
//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 //手指滑動的距離0
return
}
//正在執行下拉刷新則返回
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 = false
this.moveOffset = 0
return
}
const currentY = event.changedTouches[0].clientY
//說明此時手指已經在觸摸位置上方了
if (
currentY - touches.startClientY < 0 ||
this.$refs.dragBox.getBoundingClientRect().top < this.offsetTop
) {
this.touches.isDraging = false
event.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組件 使用方法node
//引入
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 關閉提示, 默認true
2.更新成功請調用this.$refs.infinitescrollDemo.$emit('ydui.pullrefresh.finishLoad.fail',true) 參數 true 開始提示, false 關閉提示, 默認true
3.傳入pullTipBgColor參數來修改下拉刷新成功狀態的背景色,默認藍色(#171dca)
複製代碼
組件源碼,直接copy源碼保存爲InfiniteScroll.vue便可開始使用微信
<template>
<div> <!-- 下拉刷新 --> <div class="dragBox" ref="dragBox" :style="{ transform: `translateY(${moveOffset / double}px)` }" > <!-- 下拉刷新動畫效果 --> <div class="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` }" > <img v-if="touches.loading" src="" /> <img v-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> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100" preserveAspectRatio="xMidYMid" class="lds-ellipsis" > <circle cx="84" cy="50" r="5.04711" fill="#f3b72e"> <animate attributeName="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> <animate attributeName="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"> <animate attributeName="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> <animate attributeName="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"> <animate attributeName="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> <animate attributeName="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"> <animate attributeName="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> <animate attributeName="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"> <animate attributeName="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> <animate attributeName="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: 'yd-infinitescroll', 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: '#171dca' } }, 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 = false this.isDone = true }) this.$on('ydui.infinitescroll.finishLoad', () => { this.isLoading = false }) this.$on('ydui.infinitescroll.reInit', () => { this.isLoading = false this.isDone = false }) }, pullinit() { const dragBox = this.$refs.dragBox dragBox.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) }) }, scrollHandler() { if (this.isLoading || this.isDone) return const scrollview = this.scrollview const contentHeight = document.body.offsetHeight const isWindow = scrollview === window const offsetTop = isWindow ? 0 : scrollview.getBoundingClientRect().top const scrollviewHeight = isWindow ? contentHeight : scrollview.offsetHeight if (!scrollview) { // eslint-disable-next-line console.warn("Can't find the scrollview!") return } if (!this.$refs.tag) { // eslint-disable-next-line console.warn("Can't find the refs.tag!") return } const tagOffsetTop = Math.floor(this.$refs.tag.getBoundingClientRect().top) - 1 const 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 = true this.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 = el while ( currentNode && currentNode.tagName !== 'HTML' && currentNode.tagName !== 'BODY' && currentNode.nodeType === 1 ) { let overflowY = document.defaultView.getComputedStyle( currentNode ).overflowY if (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 = false this.touches.startClientY = 0 this.touches.isDraging = false this.touches.statusText = '下拉刷新' this.moveOffset = 0 //記錄觸摸位置 // this.touches.startClientX = event.touches[0].clientX this.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 = false this.moveOffset = 0 return } //正在執行下拉刷新則返回 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 = false this.moveOffset = 0 return } 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 = false event.preventDefault() return } //下拉刷新阻止瀏覽器默認行爲 if (this.getScrollTop(this.scrollview) === 0) { event.preventDefault() } //手指滑動的距離 let deltaSlide = currentY - touches.startClientY //若是超過了指定的距離 if (deltaSlide >= touches.distance * this.double) { //進行更新的動畫 this.touches.statusText = '加載中' // alert('下拉刷新') this.touches.startClientY = 0 this.touches.isDraging = false this.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 = offsetTop clearInterval(timer) if (offsetTop !== 0) { this.touches.loading = true this.pullcallback && this.pullcallback() } else { //重置 this.touches.loading = false this.touches.startClientY = 0 this.touches.isDraging = false this.touches.statusText = '下拉刷新' //執行加載中動畫 if (tip) { //執行更新成功或者失敗動畫 this.pullupdateStatus = true setTimeout(() => { this.pullupdateStatus = false }, 1000) } } } }, 1000 / 60) } }, mounted() { 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%; } } .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: 30px; line-height: 30px; font-size: 0.24rem; 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> 複製代碼