最近把老板辞了,得空写写随笔;开发中遇到的问题实在太多,记不住。这不最近用到了scrollViewDelegate,结果以前用到过但不常用的方法又忘了。写写加深印象,欢迎大家指正。
- 此文并不是要详细解释代理方法
- 总结一些平时不常用的(至少我不常用。。)
- 所有scrollViewDelegate方法执行时scrollView属性值都是默认的;
scrollViewDelegate方法主要分3类,滚动类,zoom(缩放)类,还有个回到顶部;
一、scrollViewDelegate调用顺序
首先从scrollViewDelegate方法调用顺序入手;
1.初始化完毕
- 初始化完毕只会调用下边这一个方法,并且滚动和缩放时都会频繁调用,这里要注意避免此方法返回的View或者其子类重复创建
- (nullable UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView{}
2.滚动时
- 各方法执行顺序如下
///将要开始拖动
- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{}
///实时滚动
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{}
///返回需要缩放的View
- (nullable UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView{}
///将要结束拖动
- (void)scrollViewWillEndDragging:(UIScrollView *)scrollView withVelocity:(CGPoint)velocity targetContentOffset:(inout CGPoint *)targetContentOffset {}
///已经结束拖动
- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate{}
///将要开始减速
- (void)scrollViewWillBeginDecelerating:(UIScrollView *)scrollView{}
///减速完毕 降为0
- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView{}
- 逐个方法依次说明
不实现缩放功能时一般不写
viewForZoomingInScrollView
方法,而且此方法在滚动时几乎与scrollViewDidScroll
同时穿插着调用,在此说明中就忽略了;
①scrollViewWillBeginDragging
方法,在手指将要实现拖拽时触发,只执行一次;
②scrollViewDidScroll
方法,只要界面在滚动,不管是手指拖动,减速滑动,还是缩放都会触发。多次执行;
③在多次执行scrollViewDidScroll
后就会依次连续执行后面scrollViewWillEndDragging
scrollViewDidEndDragging
scrollViewWillBeginDecelerating
三个方法,并且只执行一次;
④在滑动速度减为0前扔会多次执行scrollViewDidScroll
方法,最后执行scrollViewDidEndDecelerating
方法,滚动结束;
3.zoom(缩放)时
- 各方法执行顺序如下
///返回需要缩放的View
- (nullable UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView{}
///将要开始缩放
- (void)scrollViewWillBeginZooming:(UIScrollView *)scrollView withView:(nullable UIView *)view {}
///实时滚动
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{}
///实时缩放
- (void)scrollViewDidZoom:(UIScrollView *)scrollView{}
///缩放结束
- (void)scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(nullable UIView *)view atScale:(CGFloat)scale{}
- 逐个方法依次说明
有趣的是
viewForZoomingInScrollView
方法,在缩放时并不执行 ,只在缩放开始与结束前执行了几次而已;
缩放时频繁调用的方法是scrollViewDidZoom
scrollViewDidScroll
这两个,由此可见只要scrollView发生矢量变化scrollViewDidScroll
方法都会执行;
进行缩放时,需要设置scrollView的
maximumZoomScale
与minimumZoomScale
两个最大伸缩比例与最小伸缩比例,如果不设置默认值为1.0原始大小,捏合时,代理方法不触发;
进行缩放时,
contentSize
会发生变化;
①viewForZoomingInScrollView
方法返回要缩放的View,多次执行,具体触发诱因还不清楚,欢迎补充;
②scrollViewWillBeginZooming
方法,在手指将要捏合时触发,只执行一次;
③缩放时,在多次执行scrollViewDidScroll
scrollViewDidZoom
两个方法;
④在手指离开屏幕后,执行scrollViewDidEndZooming
方法一次,缩放结束;
3.触发返回顶部时
- 各方法执行顺序如下
///确定回到顶部?
- (BOOL)scrollViewShouldScrollToTop:(UIScrollView *)scrollView{}
///实时滚动
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{}
///已经回到顶部
- (void)scrollViewDidScrollToTop:(UIScrollView *)scrollView{}
- 逐个方法依次说明
①用户点击状态栏时触发scrollViewShouldScrollToTop
方法,需要返回BOOL
。若返回NO
,则响应失效,不回到顶部;
②在回到顶部滚动过程中触发scrollViewDidScroll
方法,执行多次;
③回到顶部后触发scrollViewDidScrollToTop
;
4.小结
- 只要scrollView发生矢量变化
scrollViewDidScroll
方法都会执行,所以此方法使用时要注意; - zoom相关代理方法最好单独使用;
二、手指滚动与代码滚动scrollView
手指滚动其实上一节已经说的差不多了;代码滚动多是通过改变contentOffset
的值来进行;
1. 通过手指滚动
- 通过手指在屏幕上拖拽而产生滚动,所触发的方法如 第一节中的 2.;
2.通过contentOffset滚动
- 滚动过程中触发
scrollViewDidScroll
,执行一次;
3.通过setContentOffset:animated:YES 滚动
- 滚动过程中仍然是多次触发
scrollViewDidScroll
;滚动结束后会触发scrollViewDidEndScrollingAnimation
,执行一次;
3.通过scrollView的子类的属性滚动
- 例如tableView的
- (void)scrollToRowAtIndexPath:(NSIndexPath *)indexPath atScrollPosition:(UITableViewScrollPosition)scrollPosition animated:(BOOL)animated;
和 collectionView 的
- (void)scrollToItemAtIndexPath:(NSIndexPath *)indexPath atScrollPosition:(UICollectionViewScrollPosition)scrollPosition animated:(BOOL)animated;
等等,诸如此类的方法都和- (void)setContentOffset:(CGPoint)contentOffset animated:(BOOL)animated;
触发效果相同;
三、手指缩放与代码缩放scrollView
手指缩放仍然与第一节说的差不多;代码缩放多是通过改变zoomScale
的值来进行;
1. 通过手指缩放
- 通过手指在屏幕上捏合而产生滚动,所触发的方法如 第一节中的 3.;
2.通过zoomScale缩放
- 缩放一次触发
scrollViewDidZoom
执行一次;
3.通过setZoomScale:animated:YES 滚动
- 缩放过程中与手势缩放执行发方法顺序相同;
写在最后
对于这次想要小记一下,是因为做上下scrollView联动这个页面时想要优化,所以就挨个了解每个代理方法的用处;并且我还尝试了上collectionView下scrollView+tableViewController 、上collectionView下collectionView+tableViewController两种不同的构建方法;至于这两种的优缺嘛,个人总结了一下两点,有兴趣的同学欢迎补充:
- 若tableViewController内容布局大不相同 用前者的组合;但tableViewController的增多会使内存增加些许;
- 若tableViewController内容布局相似,用后者的组合可以减少内存的分配,因为collectionView会把tableViewController.view循环重用;但同时也会增加些许CPU计算量;
告一段落,欢迎指正,第一次写,不好的地方请大家见谅~