什么是无感知上拉加载更多
什么是无感知,这个这样理解:在网络情况正常的情况下,用户对列表进行连续的上拉时,该列表可以无卡顿不停出现新的数据。
如果要体验话,Web 端很多已经做到了,比如掘金的首页,还有比如掘金 iOS 的 App,列表都是无感知上拉加载更多。
说来惭愧,写了这久的代码,还真的没有认真思考这个功能怎么实现。
我在看见这位网友留言的时候,就开始思考了。
在我看来,有下面几个着手点:
UIScrollView
的代理回调中去找和 scrollView
的位置(contentOffset
)大小(contentSize
)关系密切的回调。public protocol UIScrollViewDelegate : NSObjectProtocol {
@available(iOS 2.0, *)
optional func scrollViewDidScroll(_ scrollView: UIScrollView) // any offset changes
@available(iOS 3.2, *)
optional func scrollViewDidZoom(_ scrollView: UIScrollView) // any zoom scale changes
.
.
.
.
.
.
/// 代码很多,这里就不放上来,给大家压力了。
}
直接上结论吧:看了一圈,反正没有和 contentSize
或者位置相关的回调代理。scrollViewDidScroll
这个回调里面虽然可以回参 scrollView
,但是对于我们需要的信息还不够具体。
思考:既然
UIScrollViewDelegate
的代理没有现成的代理回调,自己使用KVO
去监听试试?
就在我思考的同时,我也在网络上需求实现这个功能的答案,让后看到这样一个思路:
实现方法很简单,需要用到
tableView
的一个代理方法,就可轻松实现。- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath
就是这个方法,自定义显示cell
。这个方法不太常用。但是这个方法可在每个cell
将要第一次出现的时候触发。然后我们可设置当前页面第几个cell
将要出现时,触发请求加载更多数据。
我看了之后,心想着,多写一个 TableView
的代理,总比写 KVO
的代码少,先试试再说,于是代码撸起:
extension SwiftCoinRankListController: UITableViewDelegate {
func tableView(_ tableView: UITableView, willDisplay cell: UITableViewCell, forRowAt indexPath: IndexPath) {
let row = indexPath.row
let distance = dataSource.count - 25
print("row: \(row), distance:\(distance) ")
if row == distance {
loadMore()
}
}
}
本代码可以在开源项目中的 SwiftCoinRankListController.swift
文件查看具体的逻辑,其主要就是通过 cell
显示的个数去提前请求加载数据,然后我们看看效果:
Gif可能看起来还好,我说我调试的感受:
虽然做到了上拉无感知,但是当手滑的速度比较快的时候,到底了新的数据没有回来,就会在底部等一段时间。
功能达到了,但是感受却不理想,果然还是监听的细腻程度不够。
然后在继续的搜索中,我看到了另外一个方案:
很多时候我们上拉刷新需要提前加载新数据,这时候利用
MJRefreshAutoFooter
的属性triggerAutomaticallyRefreshPercent
就可以实现,该属性triggerAutomaticallyRefreshPercent
默认值为 1,然后改成 0 的话划到底部就会自动刷新,改成 -1 的话,在快划到底部 44px 的时候就会自动刷新。
MJRefresh
?使用 MJRefreshAutoFooter
,这个简单,我直接把基类的 footer
给替换掉就可以了,本代码可以在开源项目中的 BaseTableViewController.swift
文件查看:
/// 设置尾部刷新控件,更新为无感知加载更多
let footer = MJRefreshAutoFooter()
footer.triggerAutomaticallyRefreshPercent = -1
tableView.mj_footer = footer
再来看看效果:
直接说感受:
代码改动性少,编写简单,达到预期效果,爽歪歪。比方案一更丝滑,体验好。
到此,功能就实现,难道就完了?
当然,不会,我们去看看源码吧。
首先我们看看 MJRefreshAutoFooter.h
文件:
这里有个专门的属性 triggerAutomaticallyRefreshPercent
去做自动刷新,那么我们去 MJRefreshAutoFooter.m
中去看看吧:
注意看喔,这个 .m
文件有一个 - (void)scrollViewContentOffsetDidChange:(NSDictionary *)change
方法,并且还调用了 super
,从这个方法名中我们可以明显的得到当 scrollView
的 contentOffset
变化的时候进行回调的监听。,我们顺藤摸瓜,看看 super
是什么,会不会有新的发现:
稍微跟着一下源代码,MJRefreshAutoFooter
的继承关系如下:
MJRefreshAutoFooter => MJRefreshFooter => MJRefreshComponent
所以这个 super
的调用我们就去 MJRefreshComponent.m
里面去看看吧:
通过上面的截图我们可以得到下面的一些信息与结论:
MJRefreshComponent
是通过 KVO
去监听 scrollView
的 contentOffset
变化,思路上我们对齐一致了。MJRefreshComponent.h
的注释可以得到:MJRefreshComponent
从本质上更像虚基类。如果不是掘友提出这个问题,我可能都不会太仔细的去研究这个功能,也许继续普普通通的使用一般的上拉加载更多就够了。
这次的实践,其实是从思路到寻找方法,最后再到源码阅读。
思路也许不困难,但是真正一点点实现并完善功能,每一步都并不容易,这次我也仅仅是继续使用了 MJRefresh
这个轮子。
iOS 关于列表上拉(平滑加载数据)自动加载数据的问题[1]
MJRefresh小技巧(上拉提前刷新)[2]
[1]
iOS 关于列表上拉(平滑加载数据)自动加载数据的问题: https://www.jianshu.com/p/55c0f5b5670f
[2]
MJRefresh小技巧(上拉提前刷新): https://www.jianshu.com/p/227976796819
-End-
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有