下拉刷新,上拉加载 的基础款(基本实现)

前言

现在网上 下拉刷新,上拉加载 插件一搜一大堆,如果你想用在生产环境,那你可以直接网上搜一个靠谱的,我所做的就是不依赖任何插件,一步一步把这个插件的过程写一下,各位同学可以在此基础上定制,没有写过插件的,可以了解下插件怎么写的,整个过程定位入门级,看不懂?那么百度插件直接用就好了,修炼一段时间在考虑怎么写插件吧。废话不多说,上效果图

原理

下拉刷新的原理就是,添加一个 div,这个 div 里面标识加载的过程,根据拉下的距离改变div的高度,从而显示 下拉加载 的文字,松手后根据下拉的距离,判断是否请求数据

构建过程

定义默认的一些配置项

var defaults = {
    threshold: 100,   // 滑动触发下拉刷新的距离
    stop: 40,         // 下拉刷新时停留的位置距离屏幕顶部的距离
    dis: 20           // 距离屏幕底端触发 上拉加载 的距离
}

定义构造函数

function JrRefresh (el, options) {
    this.options = Object.assign({}, defaults, options)  // 合并参数
    this.el = typeof el === ‘string‘ ? document.querySelector(el) : el  // 定义要操作对象
    this.progress = null       // 下拉刷新显示的 dom
    this.loadMore = null       // 上拉加载显示的 dom
    this.progressHeight = 0    // 下拉刷新的 dom 的高度
    this.rotate = null         // 下拉刷新 转圈 的角度
    this.touchstartY = 0       // 触摸到屏幕的坐标起始 Y 值
    this.currentY = 0          // 移动时实时记录的坐标 Y 值
    this.isAnimation = false   // 是否在自动回滚
    this.isRefresh = false     // 是否正在刷新数据
    this.isLoadMore = false    // 是否正在加载数据
    this.hasMore = true        // 是否有更多数据, 下拉加载会用
    this.rotateTimer = null    // 控制 下拉刷新 转圈 的时间计时器
    this.event()
    this.init()
}

初始化,添加一些需要用到的 dom 元素

JrRefresh.prototype.init = function () {
    // 增加下拉刷新的显示
    var refreshHtml = `<div class="jrscroll-downwarp jrscroll-downwarp-reset" style="height: 0px;"><div class="downwarp-content"><p class="downwarp-progress" style="transform: rotate(0deg);"></p ><p class="downwarp-tip">下拉刷新</p ></div></div>`
    var divm = document.createElement(‘div‘)
    divm.innerHTML = refreshHtml
    this.progress = divm.children[0]
    this.el.prepend(this.progress)
    // 增加上拉加载的显示
    var loadMoreHtml = `<div class="jrscroll-upwarp" style="visibility: hidden;"><p class="upwarp-progress"></p><p class="upwarp-tip">加载中...</p></div>`
    var div = document.createElement(‘div‘)
    div.innerHTML = loadMoreHtml
    this.loadMore = div.children[0]
    this.el.appendChild(this.loadMore)
}

定义事件,解释一下这里用到的 bind 吧,这里的 JrRefresh.prototype.pullDown、JrRefresh.prototype.touchend 等函数里的 this 绑定到 JrRefresh 的构造函数上, 如果不用的话,pullDown 函数是由 this.el 调用的,this 会指向 this.el。(假装你们都懂了?)这是写插件常用到一种绑定 this 的方法之一

JrRefresh.prototype.event = function () {
    this.el.addEventListener(‘touchstart‘, this.handleTouchStart.bind(this))
    this.el.addEventListener(‘touchmove‘, this.pullDown.bind(this))
    this.el.addEventListener(‘touchend‘, this.touchend.bind(this))
    window.addEventListener(‘scroll‘, this.handleScroll.bind(this))
}

手指触摸时记录Y值坐标,用来判断是向上滑还是向下

JrRefresh.prototype.handleTouchStart = function (e) {
    // 记录手指触摸屏幕的 Y 值坐标
    this.touchstartY = e.changedTouches[0].clientY
}

手指开始滑动时,触发 pullDown 事件,用 this.currentY 来记录实时的 Y 值坐标,当 this.currentY - this.touchstartY > 0 时,说明是手指向下滑动,这里面 e.preventDefault() 是为了防止微信还有 ios 的浏览器下拉时顶部出现出现默认的黑色空白,或者触发 uc 等浏览器自带的下拉刷新, 由 this.moveDown 专门来控制 下拉刷新显示的dom(this.progress) 的高度

JrRefresh.prototype.pullDown = function (e) {
    var scrollTop = document.documentElement.scrollTop ||
                    window.pageYOffset ||
                    document.body.scrollTop
    this.currentY = e.targetTouches[0].clientY
    if (this.currentY - this.touchstartY >= 0 && scrollTop <= 0) {
        e.preventDefault()
        if (!this.isAnimation && !this.isRefresh) {
            this.moveDown(this.currentY - this.touchstartY)
        }
    }
}

moveDown 函数是用来专门控制 this.progress 的高度,rotateProgress 函数用来专门控制圈圈的旋转角度 changeProgressState 函数用来专门控制this.progress 显示内容

JrRefresh.prototype.moveDown = function (dis) {
    if (dis < this.options.threshold) {
        this.progress.style.height = this.progressHeight + dis + ‘px‘
        this.rotateProgress(dis*2)
        this.changeProgressState(‘下拉刷新‘)
    } else {
        // 当滑动距离超过 threshold 时,放慢下拉速度
        var aftDis = this.options.threshold + (dis - this.options.threshold) / 3
        var aftAngle = this.options.threshold * 2 + (dis - this.options.threshold) / 1.7
        this.progress.style.height = this.progressHeight + aftDis + ‘px‘
        this.rotateProgress(aftAngle)
        this.changeProgressState(‘释放刷新‘)
    }
}

圈圈只有两种情况,一种随手指移动,超过距离转圈速度变慢,一种是自己不停转,前一种情况,手指下滑距离跟旋转角度的比例并不一定按我这个写法,主要有一点,就是转动变慢时候,圈圈旋转角度不要出现突变,这也是这个插件的难点之一,不懂的同学多看多想哈

JrRefresh.prototype.rotateProgress = function (rotate) {
    var rotateDom = this.progress.querySelector(‘.downwarp-progress‘)
    if (rotate != undefined) {
        rotateDom.style.transform = ‘rotate(‘ + rotate + ‘deg)‘
        this.rotate = rotate
    } else {
        var t = 0;
        this.rotateTimer = setInterval(() => {
            t++
            var angle = (this.rotate + t*15) % 360
            rotateDom.style.transform = ‘rotate(‘ + angle + ‘deg)‘
            rotateDom.style.WebkitTransform = ‘rotate(‘ + angle + ‘deg)‘
        }, 16)
    }
}

changeProgressState 这个函数没什么好说的了

JrRefresh.prototype.changeProgressState = function (name) {
    this.progress.querySelector(‘.downwarp-tip‘).innerHTML = name
}

至此,下滑效果有点样子了,下面是手指松开时候的逻辑

JrRefresh.prototype.touchend = function () {
    var scrollTop = document.documentElement.scrollTop ||
                    window.pageYOffset ||
                    document.body.scrollTop
    if (scrollTop > 0 || this.isRefresh|| this.isAnimation) return     //只有 1.在屏幕顶部 2.已完成请求数据 3.不在回滚 三条都满足才进行处理
    if ((this.currentY - this.touchstartY) > this.options.threshold) {
        this.options.downCallback()  // 触发参数穿过来的请求数据
        this.isRefresh = true
        this.moveBack(this.options.stop)            // 下拉刷新时停留的位置距离屏幕顶部的距离

    } else  {
        this.moveBack()
    }

}

moveBack 函数专门把进度返回到对应位置

JrRefresh.prototype.moveBack = function (dis) {
    var dis = dis || 0;
    this.isAnimation = true   // 正在回退
    var currentHeight = this.progress.offsetHeight
    var t = 0,                        // 进行的步数
        b = 10,                       // 总步数
        c = (currentHeight - dis)/b   // 每一步的距离
    var timer = setInterval(() => {
        t++;
        this.progress.style.height = currentHeight - c * t + ‘px‘
        if (t == b) {
            if (dis === 0) {
                this.changeProgressState(‘下拉刷新‘)
                this.progressHeight = 0
            } else {
                this.changeProgressState(‘正在刷新‘)
                this.progressHeight = this.options.stop
                this.rotateProgress()
            }
            this.touchstartY = ‘‘
            this.isAnimation = false     // 回退完成
            clearInterval(timer)
        }
    }, 16)
}

当请求数据完成,要回滚到原始位置,参数是 boolean 类型,表明有没有更多数据

JrRefresh.prototype.endSuccess = function (bool) {
    if (this.isRefresh) {      //  如果是正在刷新数据
        this.changeProgressState(‘刷新成功‘)
        if (bool) {
            setTimeout(() => {     //延迟 500ms 回滚
                this.moveBack(0)
                this.isRefresh = false
                clearInterval(this.rotateTimer)
            },500)
        } else {
            this.toggleLoadingText(true)
        }
    }
    if (this.isLoadMore) {     //  如果是正在加载数据
        this.isLoadMore = false
        this.loadMore.style.visibility = ‘hidden‘
        this.toggleLoadingText(bool)
    }
}
JrRefresh.prototype.toggleLoadingText = function (hasMore) {
    if (hasMore) {
        this.loadMore.querySelector(‘.upwarp-tip‘).innerHTML = ‘加载中...‘
        this.loadMore.querySelector(‘.upwarp-progress‘).style.display = ‘inline-block‘
    } else {
        this.loadMore.style.visibility = ‘visible‘
        this.loadMore.querySelector(‘.upwarp-tip‘).innerHTML = ‘没有更多数据了‘
        this.loadMore.querySelector(‘.upwarp-progress‘).style.display = ‘none‘
    }
}

至此,下拉刷新的逻辑终于完成了,下面开始上拉加载,比较麻烦的是获取页面高度,数据过来的同时请求页面高度,页面还没有渲染完成,页面高度可能会不准,所以我处理方法是延迟100ms 获取高度,

JrRefresh.prototype.handleScroll = function () {
    var top = this.loadMore.getBoundingClientRect().top;    // 获取最底部标签距离屏幕顶部的距离
    if (top + 10 < window.innerHeight && !this.isLoadMore && this.hasMore) {
        this.isLoadMore = true
        this.loadMore.style.visibility = ‘visible‘
        this.options.up.callback()
    }
}

用法

<div class="jrscroll">
//列表内容,如:<ul>列表数据</ul> ..
</div>
<script>
    var scroll = new JrRefresh(scrollWrap, {
        downCallback: pullDownRefresh,
        upCallback: pullUpLoadMore
    })
    function pullDownRefresh() {
        setTimeout(() => {
            console.log(‘刷新成功‘)
            // 处理数据
            scroll.endSuccess(true)
        }, 1000)
    }
    function pullUpLoadMore() {
        setTimeout(() => {
            console.log(‘请求成功‘)
            // 处理数据
            scroll.endSuccess(true)
        }, 2000)

    }
</script>

好了,大致就介绍这么多,想看源码的同学,移步这里,后续的完善和更新也会在 github 上,有兴趣的同学 star 或者 fork 哦

原文地址:https://www.cnblogs.com/baimeishaoxia/p/12035274.html

时间: 2024-10-27 06:14:57

下拉刷新,上拉加载 的基础款(基本实现)的相关文章

最新Android ListView 下拉刷新 上滑加载

开发项目过程中基本都会用到listView的下拉刷新和上滑加载更多,之前大家最常用的应该是pull to refresh或它的变种版吧,google官方在最新的android.support.v4包中增加了一个新类SwipeRefreshLayout,地址 这个类的作用就是提供官方的下拉刷新,并且效果相当不错,而上拉加载更多则用我们自定义的listview,也是相当简单. 下拉刷新 简单的介绍下: 首先它是一个viewgroup,但是它只允许有一个子控件,子控件能是任何view,使用的时候,所在

十分钟实现ListView下拉刷新上滑加载更多

说到ListView下拉刷新几乎每个APP都会用到,所以ListView下拉刷新是很重要的,就像ListView优化一样是你必会的东西. ListView实现下拉刷新如果我们开发人员自己编写相对来说比较费事的,当我们使用第三方库之后我们再来开发这个功能就会省事很多.相比与自己实现可以少编写不少代码,Android-PullToRefresh库可以轻松实现ListView的下拉刷新功能. 要使用Android—PullToRefesh库对ListView实现下拉刷新要经过以下几个步骤: 1.下载A

Android 下拉刷新上啦加载SmartRefreshLayout + RecyclerView

在弄android刷新的时候,可算是耗费了一番功夫,最后发觉有现成的控件,并且非常好用,这里记录一下. 原文是 https://blog.csdn.net/huangxin112/article/details/78781682 ,这里是看了之后,结合自己实际遇到的问题写的. 首先引入包. //下拉框 implementation 'com.android.support:recyclerview-v7:28.0.0-beta01' implementation 'com.scwang.smar

android 安卓 listview 支持下拉刷新 上拉加载更多

[1]重写listView import java.text.SimpleDateFormat; import java.util.Date; import android.content.Context; import android.util.AttributeSet; import android.util.Log; import android.view.MotionEvent; import android.view.View; import android.view.ViewGrou

带你实现开发者头条APP(五)--RecyclerView下拉刷新上拉加载

转载请注明出处:http://blog.csdn.net/lowprofile_coding/article/details/51321896 一 .前言 最近实在太忙,一个多礼拜没有更新文章了,于是今晚加班加点把demo写出来,现在都12点了才开始写文章. 1.我们的目标 把RecyclerView下拉刷新上拉加载更多加入到我们的开发者头条APP中. 2.效果图 3.实现步骤 找一个带上拉刷新下载加载更多的RecyclerView开源库,我们要站在巨人的肩膀上 下载下来自己先运行下demo,然

Android 下拉刷新上拉加载 多种应用场景 超级大放送(上)

转载请标明原文地址:http://blog.csdn.net/yalinfendou/article/details/47707017 关于Android下拉刷新上拉加载,网上的Demo太多太多了,这里不是介绍怎么去实现下拉刷新上拉加载,而是针对下拉刷新上拉加载常用的一些应用场景就行了一些总结,包含了下拉刷新上拉加载过程中遇到的一些手势冲突问题的解决方法(只能算是抛砖引玉). 去年9月的时候,那时自己正在独立做Android项目.记得刚刚写完那个ListView列表页面(木有下拉刷新,上拉加载)

vue10行代码实现上拉翻页加载更多数据,纯手写js实现下拉刷新上拉翻页不引用任何第三方插件

vue10行代码实现上拉翻页加载更多数据,纯手写js实现下拉刷新上拉翻页不引用任何第三方插件/库 一提到移动端的下拉刷新上拉翻页,你可能就会想到iScroll插件,没错iScroll是一个高性能,资源占用少,无依赖,多平台的javascript滚动插件.iScroll不仅仅是 滚动.它可以处理任何需要与用户进行移动交互的元素.在你的项目中包含仅仅4kb大小的iScroll,你的项目便拥有了滚动,缩放,平移,无限滚动,视差滚动,旋转功能.iScroll的强大毋庸置疑,本人也非常欢迎大家使用iScr

android 下拉刷新上拉加载更多,高仿ios左滑动删除item,解决了众多手势问题

一.前言 老规矩,别的不说,这demo是找了很相关知识集合而成的,可以说对我这种小白来说是绞尽脑汁!程序员讲的是无图无真相!现在大家一睹为快! 二.比较关键的还是scroller这个类的 package com.icq.slideview.view; import android.content.Context; import android.util.AttributeSet; import android.util.Log; import android.util.TypedValue; i

Android 下拉刷新上拉加载效果功能,使用开源项目android-pulltorefresh实现

应用场景: 在App开发中,对于信息的获取与演示,不可能全部将其获取与演示,为了在用户使用中,给予用户以友好.方便的用户体验,以滑动.下拉的效果动态加载数据的要求就会出现.为此,该效果功能就需要应用到所需要的展示页面中. 知识点介绍: 本文主要根据开源项目android-pulltorefresh展开介绍. android-pulltorefresh [一个强大的拉动刷新开源项目,支持各种控件下拉刷新 ListView.ViewPager.WevView.ExpandableListView.G

使用MJRefresh自定义下拉刷新,上拉加载动画

有时候我们需要自己设置下拉刷新,上拉加载动画的实现,这里主要是记录下使用MJRefresh自定义下拉刷新,上拉加载动画..... 下拉刷新我们只需要继承MJRefreshGifHeader即可: 实现代码如下: - (void)prepare{ [super prepare]; self.stateLabel.hidden = NO; self.lastUpdatedTimeLabel.hidden = YES; [self setImages:@[[UIImage imageNamed:@"v