ios监听ScrollView/TableView滚动的正确姿势
主要介绍
- 监测tableView垂直滚动的舒畅姿势
- 监测scrollView/collectionView横向滚动的正确姿势
1.监测tableView垂直滚动的舒畅姿势
通常我们用KVO或者在scrollViewDidScroll代理方法中监听ScrollView/TableView的contentOffset,比如监听TableView的contentOffset来设置导航栏的透明度或者拉伸顶部的图片。
常见的姿势是在scrollViewDidScroll的代理方法中监听scrollView.contentOffset.y,会发现有导航栏时scrollView.contentOffset.y初始值可能会等于-64,如果再手动设置tableView.contentInset这个值又会改变,这个时候就需要计算出初始偏移量,然后再算偏移的差值,要是过几天再改下需求......重新计算
那有没有更舒畅的姿势?
有
首先定义2个成员属性,一个是监测范围的临界值,另一个用来记录scrollView.contentInset.top(重点)
// 监测范围的临界点,>0代表向上滑动多少距离,<0则是向下滑动多少距离
@property (nonatomic, assign)CGFloat threshold;
// 记录scrollView.contentInset.top
@property (nonatomic, assign)CGFloat marginTop;
// 这里设值-80,即向下滑动80,-80到0就是咱们重点监测的范围
self.threshold = -80;
然后在KVO回调或者scrollViewDidScroll代理方法中加上下面的代码。
需要理解的就是contentInset,四个值代表tableView的contentView上下左右距离边界的值,界面有导航栏时,系统会自动将tableView的contentView到顶部的距离设为64,即contentInset.top=64,如果导航栏透明就可以看到有空白区域。没有导航栏或者我们调用self.automaticallyAdjustsScrollViewInsets = NO时,tableView的contentInset.top就会变为0。通过下面的算法,我们就可以不用去刻意计算初始的偏移量,只要设置好临界值就行,newoffsetY 就是我们实际要监测的偏移。PS:手动设置tableView.contenInset需要在viewDidAppear中进行。
// 实时监测scrollView.contentInset.top, 系统优化以及手动设置contentInset都会影响contentInset.top。
if (self.marginTop != self.scrollView.contentInset.top) {
self.marginTop = self.scrollView.contentInset.top;
}
//CGFloat offsetY = [change[@"new"] CGPointValue].y;
CGFloat offsetY = scrollView.contentOffset.y;
// newoffsetY 便是我们想监测的偏移offset.y,初始值为0
// 向下滑动时<0,向上滑动时>0;
CGFloat newoffsetY = offsetY + self.marginTop;
if (newoffsetY >= self.threshold && newoffsetY <= 0) {
CGFloat progress = newoffsetY/self.threshold;
}
是骡子是马,拉出来溜溜,再看看效果图。
第一个页面(首页),上划变透明
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{
if (self.marginTop != scrollView.contentInset.top) {
self.marginTop = scrollView.contentInset.top;
}
CGFloat offsetY = scrollView.contentOffset.y;
CGFloat newoffsetY = offsetY + self.marginTop;
// 临界值150,向上拖动时变透明
if (newoffsetY >= 0 && newoffsetY <= 150) {
[self.navigationController.navigationBar jp_setNavigationBarBackgroundAlpha:1- newoffsetY/150];
}else if (newoffsetY > 150){
[self.navigationController.navigationBar jp_setNavigationBarBackgroundAlpha:0];
}else{
[self.navigationController.navigationBar jp_setNavigationBarBackgroundAlpha:1];
}
}
第二个界面仿知乎日报的效果,一生X命在他的文章仿写知乎日报 - 主页面补遗(Part 2)用另外的方法实现过。
简单说下我的思路
- tableView的y坐标为20,然后组头高度44,加起来刚好就是导航栏的高度。因为组头悬停在tableView的最上方,所以y坐标为20。
- 在navigationBar上面添加一个高度为20的view,放在状态栏下面,跟组头一样的颜色。navigationBar透明后view看上去会和组头连接起来。
- 在viewDidAppear方法里面计算第一个组头的frame,取y坐标。(最好在viewDidAppear方法里面,不然加了tableHeaderView可能会产生偏差)
- 然后在scrollViewDidScroll方法实现导航栏的透明以及切换titleView
- (void)viewDidAppear:(BOOL)animated{
[super viewDidAppear:animated];
HeaderFrame = [self.tableView rectForHeaderInSection:1];
}
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{
if (self.marginTop != scrollView.contentInset.top) {
self.marginTop = scrollView.contentInset.top;
}
CGFloat offsetY = scrollView.contentOffset.y;
CGFloat newoffsetY = offsetY + self.marginTop;
if (newoffsetY >= 0 && newoffsetY <= 150) {
[self.navigationController.navigationBar jp_setStatusBarBackgroundViewAlpha:newoffsetY/150];
}else if (newoffsetY > 150){
[self.navigationController.navigationBar jp_setStatusBarBackgroundViewAlpha:1];
}
if (newoffsetY >= HeaderFrame.origin.y) {
self.refrshView.hidden = YES;
[self.navigationController.navigationBar setBackgroundColor:[[UIColor purpleColor] colorWithAlphaComponent:0]];
}else if (newoffsetY < HeaderFrame.origin.y){
[self.refrshView resetNavigationItemTitle:@"首页"];
self.refrshView.hidden = NO;
}
}
2.监测scrollView/collectionView横向滚动的正确姿势
先看效果图,下面的scrollView翻页后才移动上面的导航条,你会怎么实现?
首先想到的肯定是结束减速的代理方法:scrollViewDidEndDecelerating,但是滑动过快的时候是不会调用scrollViewDidEndDecelerating代理方法的,如果做过用3个界面+scrollView实现循环滚动展示图片,那么基本上都会碰到这么问题。如何准确的监听翻页?我的解决的思路如下
- 把原来减速后需要处理的代码整合到一个方法中,并且需传入scrollView的contentOffset,来确定当前的page/index。后面简称【方法A】
- 整个过程中主要留意三个代理方法,拖动后 开始减速-->结束减速-->开始拖动
- 正常滑动速度的时候,先调用scrollViewWillBeginDecelerating再调用scrollViewDidEndDecelerating,然后调用方法A
- 快速滑动的时候,先调用scrollViewWillBeginDecelerating,再调用scrollViewWillBeginDragging,不会调用scrollViewWillBeginDragging。咱们可以加个flag,来判断是否减速。没减速再调用方法A
// 开始减速的时候开始self.didEndDecelerating = NO;结束减速就会置为YES,如果滑动很快就还是NO。
- (void)scrollViewWillBeginDecelerating:(UIScrollView *)scrollView{
self.didEndDecelerating = NO;
}
- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView{
self.didEndDecelerating = YES;
// 调用方法A,传scrollView.contentOffset
}
// 再次拖拽的时候,判断有没有因为滑动太快而没有调用结束减速的方法。
// 如果没有,四舍五入手动确定位置。这样就可以解决滑动过快的问题
- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{
if (!self.didEndDecelerating) {
// 先计算当期的page/index
CGFloat index = scrollView.contentOffset.x/self.screenWidth;
//再四舍五入推算本该减速时的scrollView的contentOffset。即:roundf(index)*self.screenWidth]
}
}
原文链接:http://www.jianshu.com/p/2172cca95cdc
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
ios监听ScrollView/TableView滚动的正确姿势的更多相关文章
- Android中监听ScrollView滑动停止和滑动到底部
1.监听ScrollView滑动停止: /********************监听ScrollView滑动停止*****************************/ scrollView.s ...
- Android监听ScrollView滑动到顶端和底部
Android监听ScrollView滑动到顶端和底部 package cn.testscrollview; import android.os.Bundle; import android. ...
- Android 监听 ScrollView 滑动到最底部。
做产品时,有一个需求,需要监听ScrollView滑动到最底部.在网上找了些方法,都有这样或那样的问题,要不就是监听不精确, 要不就是重复监听,那些代码没有产品化,很不可靠. 经过自己试验,终于找到了 ...
- vue中监听页面滚动和监听某元素滚动
①监听页面滚动 在生命周期mounted中进行监听滚动: mounted () { window.addEventListener('scroll', this.scrollToTop) }, 在方法 ...
- jq监听页面的滚动事件,
jQuery监听页面的滚动状态,实现代码: $(document).scroll(function() { var scroH = $(document).scrollTop(); //滚 ...
- iOS监听tableView组头切换事件
- (void)tableView:(UITableView *)tableView willDisplayHeaderView:(UIView *)view forSection:(NSIntege ...
- iOS scrollView/tableView滚动到底部
//项目要求tableView滚动到底部就自动加载下一页,UITableView继承自UIScrollView 所以可以在//scrollViewDidEndDecelerating这个方法中进行判断 ...
- iOS监听模式系列之键值编码KVC、键值监听KVO的简单介绍和应用
键值编码KVC 我们知道在C#中可以通过反射读写一个对象的属性,有时候这种方式特别方便,因为你可以利用字符串的方式去动态控制一个对象.其实由于ObjC的语言特性,你根部不必进行任何操作就可以进行属性的 ...
- iOS监听模式系列之关于delegate(代理,委托)的学习
首先,大家应该都明白的是委托是协议的一种,顾名思义,就是委托他人帮自己去做什么事.也就是当自己做什么事情不方便的时候,就可以建立一个委托,这样就可以委托他人帮自己去实现什么方法. 其次,我简单的总结了 ...
随机推荐
- D8
=-=昨天被老师拉去吃点心了就没有发题解...忧伤..昨天的T2貌似都没有调完嗯 今天脑洞是大啊.. T1模拟写挂..呵呵我一面 T2数学题..刚开始只会求素数表的那种方法暴力..不过后面他们都知道一 ...
- poj1001求幂
这道题目是实质上就是高精度的乘法,虽然是带小数点的数多少次幂,但是开始我们需要将它变为整数进行求幂,然后再加上小数点,然后要考虑前导0,有效数位问题,做的时候要十分的小心 #include<io ...
- 企业架构研究总结(35)——TOGAF架构内容框架之构建块(Building Blocks)
之前忙于搬家移居,无暇顾及博客,今天终于得闲继续我的“政治课”了,希望之后至少能够补完TOGAF方面的内容.从前面文章可以看出,笔者并无太多能力和机会对TOGAF进行理论和实际的联系,仅可对标准的文本 ...
- Weka开发[4]-特征选择
特征选择,我对这一部分也不熟,大概讲一下,用AttributeSelection进行特征选择,它需要设置3个方面,第一:对属性评价的类(自己到Weka软件里看一下,英文Attribute Evalua ...
- Oracle实现主键自增长
-- 主键设置:xx_id number(24) primary key 1 create sequence XX_seq --序列名称 increment by 1 -- 每次加几个 start - ...
- 取SQL分组中的某几行数据
取SQL分组中的某几行数据 对表中数据分组,有时只需要某列的聚合值:有时却需要返回整行数据,常用的方法有:子查询.ROW_NUMBER.APPLY,总体感觉还是ROW_NUMBER比较直观.测试数据: ...
- SQL Server中的高可用性1
SQL Server中的高可用性(1)----高可用性概览 自从SQL Server 2005以来,微软已经提供了多种高可用性技术来减少宕机时间和增加对业务数据的保护,而随着SQL Server ...
- Mongodb 集群搭建以及常见错误
Mongodb 集群搭建以及常见错误 1 关于Replica Sets +Sharding(主从复制加分片)搭建,不这详细去说,网上有很多,大部分的例子就三台服务器之间做主从复制,分2个shard,架 ...
- the selected server is enabled,but is not configured properly.Deployment to it will not be permitted
用Tomcat添加部署项目的时候报错: the selected server is enabled,but is not configured properly.Deployment to it w ...
- ColorMatrixFilter色彩矩阵滤镜(as3)
matrix是一个长度为4*5=20的数组,其构成如下所示: R ,G, B, A, offset [1, 0, 0, 0, 0]); // red [0, 1, 0, 0, 0 ...