基础 (十四) : UIScrollView

掌握

UIScrollView的常见属性

UIScrollView的常用代理方法

UIScrollView的缩放

UIScrollView和UIPageControl的分页

NSTimer的使用

什么是UIScrollView

移动设备的屏幕大小是极其有限的,因此直接展示在用户眼前的内容也相当有限当展示的内容较多,超出一个屏幕时,用户可通过滚动手势来查看屏幕以外的内容,普通的UIView不具备滚动功能,不能显示过多的内容,UIScrollView是一个能够滚动的视图控件,可以用来展示大量的内容,并且可以通过滚动查看所有的内容

举例:手机上的“设置”、其他示例程序

UIScrollView的基本使用

UIScrollView的用法很简单

将需要展示的内容添加到UIScrollView中

设置UIScrollView的contentSize属性,告诉UIScrollView所有内容的尺寸,也就是告诉它滚动的范围(能滚多远,滚到哪里是尽头)

未命名图片.png

UIScrollView的常见属性

@property(nonatomic) CGSize contentSize; //这个属性用来表示UIScrollView内容的尺寸,滚动范围(能滚多远)

@property(nonatomic) CGPoint contentOffset;

(又叫内容偏移量:内容左上角减去scoreView左上角差的值,用户只要稍微动一下,图片就会再次弹回原始的位置)

这个属性用来表示UIScrollView滚动到的位置

@property(nonatomic) UIEdgeInsets contentInset;

这个属性能够在UIScrollView的4周增加额外的滚动区域(也可以把这个理解成周围的厚度)

未命名图片1.png

UIScrollView的其他属性

@property(nonatomic) BOOL bounces; //设置UIScrollView是否需要弹簧效果

@property(nonatomic,getter=isScrollEnabled) BOOL scrollEnabled; //设置UIScrollView是否能滚动

@property(nonatomic) BOOL showsHorizontalScrollIndicator; // 是否显示水平滚动条

3

@property(nonatomic) BOOL showsVerticalScrollIndicator; // 是否显示垂直滚动条

tracking // 当touch 后还没有拖动的时候值是YES,否则NO

zoomBouncing //当内容放大到最大或者最小的时候值是
YES,否则 NO

zooming // 当正在缩放的时候值是
YES,否则 NO

decelerating // 当滚动后,手指放开但是还在继续滚动中。这个时候是
YES,其它时候是 NO

decelerationRate // 设置手指放开后的减速率

maximumZoomScale // 一个浮点数,表示能放最大的倍数

minimumZoomScale // 一个浮点数,表示能缩最小的倍数

pagingEnabled // 当值是YES 会自动滚动到 subview 的边界。默认是NO

scrollEnabled // 决定是否可以滚动

如果scrollView不是通过storyboard、xib创建,那么一开始的subviews里面没有子控件这就是和手动创建的最大的区别

在scoreView里面在水平滚动条和垂直滚动条没有设置为NO的情况下,它也算做ScrollView的一部分(在scoreView里面删除子控件的时候要格外的注意,但是可以把这两个滚动条给禁止设置为NO)

意思是需要动画 但是时间是苹果自己定义的第一个参数是你要滚动到的位置 第二个参数是是否要动画效果!

    [self.scrollView

setContentOffset:CGPointMake(0, self.scrollView.contentOffset.y) animated:YES];

    是否有弹簧效果

    self.scrollView.bounces= NO;

 不管有没有contentSize,都能拥有弹簧效果(一般适用于下拉刷新功能,上拉刷新)

    self.scrollView.alwaysBounceVertical= YES;(弹簧效果)



    self.scrollView.alwaysBounceHorizontal= YES;(水平效果可以滚来滚去)



    是否要显示滚动条

    self.scrollView.showsHorizontalScrollIndicator= NO;

    self.scrollView.showsVerticalScrollIndicator= NO;



    self.scrollView.indicatorStyle= UIScrollViewIndicatorStyleWhite;



    NSLog(@"%@", self.scrollView.subviews);



    [[self.scrollView.subviews firstObject] removeFromSuperview];

/**

UIControlStateNormal;

UIControlStateHighlighted;

 UIControlStateDisabled; 禁止状态//

只能通过enabled=NO达到这个状态

设置userInteractionEnabled=NO,可以禁止一些事件,比如点击和拉拽(意思是禁止和用户交互)

*/

未命名图片2.png

UIScrollView的代理(delegate)

代理三部曲:作用

1.代理一般都是控制器对象

2.代理一般都是id类型 (注意代理是弱指针,如果是强指针会内存泄露)

3.代理协议的格式:控件类名Delegata

4.代理方法:方法名一般都是以控件名开头,比如UIScrollView的代理方法就是ScrollViewDelegata

5.如何监听控件的行为

通过addTarget(只有继承自UIControl的控件,才有这个功能)

通过Delegate(只有拥有delagate的属性的控件,才有这个功能,有的控件两个功能都有)

(比如让控制器监听ScoreView的行为)

注意:给其设置背景色的时候只有运行的时候才能看到

1.设置scorollView的deledate(代理)为控制器对象

scorollView.deledata = 控制器

2.控制器要遵守协议 <UIScrollViewDelegata>

3.控制器要实现UIscoreView协议里面的代理方法

很多时候,我们想在UIScrollView正在滚动 或
滚动到某个位置 或者 停止滚动 时做一些特定的操作

要想完成上述功能,前提条件就是能够监听到UIScrollView的整个滚动过程

当UIScrollView发生一系列的滚动操作时, 会自动通知它的代理(delegate)对象,给它的代理发送相应的消息,让代理得知它的滚动情况

也就是说,要想监听UIScrollView的滚动过程,就必须先给UIScrollView设置一个代理对象,然后通过代理得知UIScrollView的滚动过程

未命名图片4.png

仔细看这个方法

UIImageView*imageView = [[UIImageView

alloc] initWithImage:[UIImage imageNamed:@"minion"]];

这个方法的意思是通过一张图片来初始化这个UIImageView,并且UIImageView的尺寸就是图片的尺寸!

[scrollView addSubview:imageView];

scrollView.contentSize = imageView.frame.size;

 设置代理(监听scrollView的各种行为)

scrollView.delegate =self;

pragma mark - <UIScrollViewDelegate> 代理方法

只要scrollView在滚动,就会调用这个方法(监听scrollView的滚动)

  • (void)scrollViewDidScroll:(UIScrollView
    *)scrollView

{

NSLog(@"scrollViewDidScroll");

}

用户即将开始拖拽scrollView,就会调用这个方法

  • (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView

{

NSLog(@"scrollViewWillBeginDragging");

}

用户即将停止拖拽scrollView,就会调用这个方法

  • (void)scrollViewWillEndDragging:(UIScrollView *)scrollView withVelocity:(CGPoint)velocity targetContentOffset:(inout CGPoint
    *)targetContentOffset

{

NSLog(@"scrollViewWillEndDragging");

}

scrollView已经停止减速,就会调用这个方法(停止滚动)

  • (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView

{

NSLog(@"scrollView减速完毕,停止滚动,完全静止");

}

用户已经停止拖拽scrollView,就会调用这个方法

  • (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate

{

if (decelerate == NO)
{

    NSLog(@"scrollView停止滚动,完全静止");

}else {

    NSLog(@"用户停止拖拽,但是scrollView由于惯性,会继续滚动,并且减速");

}

}

textField的代理方法

/**

  • 当textField的输入文字发生改变时,就会调用这个方法

  • @param string 用户当时输入的文字
    (NSRange暂且不要管)

  • @return YES:允许用户输入;NO:禁止用户输入

*/

  • (BOOL)textField:(UITextField
    *)textField shouldChangeCharactersInRange:(NSRange)range
    replacementString:(NSString *)string

{

if ([string isEqualToString:@"0"]) return
NO;

return YES;

}
UIScrollView和delegate的通信

未命名图片5.png
未命名图片6.png

成为delegate的条件

UIScrollView将delegate需要实现的方法都定义在了UIScrollViewDelegate协议中,因此要想成为UIScrollView的delegate,必须遵守UIScrollViewDelegate协议,然后实现协议中相应的方法,就可以监听UIScrollView的滚动过程了

未命名图片7.png
未命名图片8.png

UIScrollView和控制器
一般情况下,就设置UIScrollView所在的控制器 为 UIScrollView的delegate设置控制器为UIScrollView的delegate有2种方法:

通过代码(self就是控制器)

self.Edison.delegate = self;

通过storyboard拖线(右击UIScrollView)

1.png

scrollView的代理方法

     scrollView 正在滚动的时候调用  偏移量一直变化
  • (void)scrollViewDidScroll:(UIScrollView
    *)scrollView;

     scrollView正在缩放
    
  • (void)scrollViewDidZoom:(UIScrollView
    *)scrollView

      开始拖拽的时候调用
    
  • (void)scrollViewWillBeginDragging:(UIScrollView
    *)scrollView;

即将停止拖拽的时候调用

  • (void)scrollViewWillEndDragging:(UIScrollView
    *)scrollView withVelocity:(CGPoint)velocity targetContentOffset:(inout CGPoint *)targetContentOffset
    NS_AVAILABLE_IOS(5_0);

已经停止拖拽调用

  • (void)scrollViewDidEndDragging:(UIScrollView
    *)scrollView willDecelerate:(BOOL)decelerate;

     scrollView即将开始减速
    
  • (void)scrollViewWillBeginDecelerating:(UIScrollView
    *)scrollView;

     scrollView 停止减速  停止
    
  • (void)scrollViewDidEndDecelerating:(UIScrollView
    *)scrollView;

     scrollView 停止滚的的动画
    
  • (void)scrollViewDidEndScrollingAnimation:(UIScrollView
    *)scrollView;

在scrollView中 哪些控件是需要缩放的

  • (UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView;

     scrollView即将开始缩放
    
  • (void)scrollViewWillBeginZooming:(UIScrollView
    *)scrollView withView:(UIView *)view NS_AVAILABLE_IOS(3_2);

     scrollView结束缩放
    
  • (void)scrollViewDidEndZooming:(UIScrollView
    *)scrollView withView:(UIView *)view atScale:(CGFloat)scale;

     scrollView 即将滚动到顶部
    
  • (BOOL)scrollViewShouldScrollToTop:(UIScrollView
    *)scrollView;

     scrollView 已经滚动到顶部
    
  • (void)scrollViewDidScrollToTop:(UIScrollView
    *)scrollView;

UIScrollView和控制器

2.png
3.png
4.png

缩放实现步骤

跟缩放相关的其他代理方法

准备开始缩放的时候调用

  • (void)scrollViewWillBeginZooming:(UIScrollView *)scrollView withView:(UIView *)view

正在缩放的时候调用

  • (void)scrollViewDidZoom:(UIScrollView
    *)scrollView

核心代码:设置内容大小

 contentSize.height == 0 设置为0仅仅代表竖直方向上不能滚动

self.scrollView.contentSize= CGSizeMake(count * w, 0);



 分页

 每一页的尺寸都是跟scrollView的frame.size一样的

self.scrollView.pagingEnabled= YES;
  • (void)scrollViewDidScroll:(UIScrollView
    *)scrollView

{

 --四舍五入为整数的方法-- 

(int)(小数+ 0.5)

 0.3  -> (int)(0.3 + 0.5) == 0

 0.4 -> (int)(0.4 + 0.5) == 0

 1.5 -> (int)(1.5 + 0.5) == 2

 0.7 -> (int)(0.7 + 0.5) == 1

 小数加上0.5再转换成整数,这样页码显示的就是正确的页数(仔细想)

int page = (int)(scrollView.contentOffset.x
/ scrollView.frame.size.width + 0.5);

self.pageControl.currentPage= page;

}

  • (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView

{

int page = scrollView.contentOffset.x / scrollView.frame.size.width;

self.pageControl.currentPage= page;

}

分页

只要将UIScrollView的pageEnabled属性设置为YES,UIScrollView会被分割成多个独立页面,里面的内容就能进行分页展示

一般会配合UIPageControl增强分页效果,UIPageControl常用属性如下

一共有多少页

@property(nonatomic)
NSInteger numberOfPages;

当前显示第几页

@property(nonatomic) NSInteger
currentPage;

只有一页时,是否需要隐藏页码指示器

@property(nonatomic)
BOOL hidesForSinglePage;

其他页码指示器的颜色

@property(nonatomic,retain) UIColor
*pageIndicatorTintColor;

当前页码指示器的颜色

@property(nonatomic,retain) UIColor
*currentPageIndicatorTintColor;

NSTimer

注意:(它现在是弱指针,NSTimer的内部会有一个强指针引用着它)

NSTimer叫做“定时器”,它的作用如下

在指定的时间执行指定的任务

每隔一段时间执行指定的任务

调用下面的方法就会开启一个定时任务

(NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget
selector:(SEL)aSelector userInfo:(id)userInfo repeats:(BOOL)yesOrNo;

每隔ti秒,调用一次aTarget的aSelector方法,yesOr No决定了是否重复执行这个任务

通过invalidate方法可以停止定时器的工作,一旦定时器被停止了,就不能再次执行任务。只能再创建一个新的定时器才能执行新的任务

停止定时器:

  • (void)invalidate;

当用户即将开始拖拽scrollView时,停止定时器

  • (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView

{

[self stopTimer];  这个stopTimer方法是自己定义的,并不是系统的!

}

当用户已经结束拖拽scrollView时,开启定时器

  • (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate

{

[self startTimer]; 这个startTimer方法是自己定义的,并不是系统的!

}

返回一个自动开始执行任务的定时器

self.timer= [NSTimer

scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(nextPage:) userInfo:@"123" repeats:YES];

 修改NSTimer在NSRunLoop中的模式:NSRunLoopCommonModes

 主线程不管在处理什么操作,都会抽时间处理NSTimer

[[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];

*******************************************************************笔记**************************

  1. 常见属性

    设置只能在一个方向上滚动

    self.scrollView.directionalLockEnabled =YES;

    内容在scrollView四周的弹簧效果

    self.scrollView.bounces = NO;

    实现下拉刷新

    不管有没有contentSize 都可以水平 垂直方向的拉

    self.scrollView.alwaysBounceVertical =YES;

    self.scrollView.alwaysBounceHorizontal =YES;

    分页效果

    self.scrollView.pagingEnabled =YES;

    UIButton
    *btn ;

    btn.enabled
    = NO;

    btn.userInteractionEnabled
    = NO;

    设置scrollView能不能滚动

    self.scrollView.scrollEnabled =NO;

    任何继承自UIView的子类都能使用以下属性设置与用户是否能交互

    self.scrollView.userInteractionEnabled =NO;

    滚动条在scrollView的subview中

    NSLog(@"------subviews = %@",self.scrollView.subviews);

    设置水平方向的滚动条

    self.scrollView.showsHorizontalScrollIndicator =NO;

    self.scrollView.showsVerticalScrollIndicator =NO;

    subviews中没有滚动条

    NSLog(@"----2--subviews = %@",self.scrollView.subviews);

    滚动条的样式

    self.scrollView.indicatorStyle =
    UIScrollViewIndicatorStyleBlack;

    惯性停下来的速度 靠自己的手验证

    self.scrollView.decelerationRate = 0.1f;

  1. 重点常见属性

    设置内容尺寸

    self.scrollView.contentSize =
    imageView.image.size;

    设置内容的偏移量

    self.scrollView.contentOffset = CGPointMake(200,
    200);

    内边距

    self.scrollView.contentInset =
    UIEdgeInsetsMake(20, 30, 40, 50);

3.delegate
代理(assign 弱引用)

面试题:

代理的好处:可以实现监听的思想,假如让一个对象A监听另一个对象B的状态,这个时候可以用代理

通知的思想:一个对象B的状态发生了改变(做了某些事情),想通知另一个对象A,这就是通知

  1. 遵守协议
  1. 成为代理
  1. 实现代理方法

一般:

  1. 代理的名称一般是控件的名称+ Delegate

UITextFieldDelegate,UITableViewDelegate,UIScrollViewDelegate,UIAlertViewDelegate

  2. 代理需要实现的的方法一般都是代理名称- UI - delegate

 
  - (BOOL)textField:(UITextField

*)textField shouldChangeCharactersInRange:(NSRange)range
replacementString:(NSString *)string;

  - (void)tableView:(UITableView

*)tableView didDeselectRowAtIndexPath:(NSIndexPath *)indexPath;

  - (void)alertView:(UIAlertView

*)alertView clickedButtonAtIndex:(NSInteger)buttonIndex;

  3. 成为代理


      一般需要在A类中写代理方法,那么就让A类成为代理


          遵守协议 - 写在.h文件中

(如果是需要当前控制器,那么就.m中遵守协议就可以了)

          实现(你需要和必须实现的)代理方法


          成为代理 

scrollView.delegate(scrollView 的属性) = self

4.图片的缩放

  1. 遵守协议
  1. 成为代理
  1. 实现代理方法
  1. 在- (UIView *)viewForZoomingInScrollView:(UIScrollView
    *)scrollView 返回一个需要缩放的控件
  1. 设置scrollView的最大和最小缩放比例
  1. 用到的代理方法
      以下方法中  view 就是正在缩放的控件


                scale 缩放比例

 
  - (void)scrollViewWillBeginZooming:(UIScrollView

*)scrollView withView:(UIView *)view

  - (void)scrollViewWillBeginZooming:(UIScrollView

*)scrollView withView:(UIView *)view

缩放前:

5.png
6.png

5.UITextField 的代理方法

  1. 常见设置
   数字键盘

textField.keyboardType = UIKeyboardTypeNumberPad;

改变键盘的右下角return 按钮的名称

textField.returnKeyType = UIReturnKeyDone;

  1. 代理方法

    textField : A输入框发送改变 就把A输入框传入代理方法中
    
    range : 其实就是光标的位置  长度就是代表需要替换哪些字符串
    
    string : 输入的是什么就是什么
    
  • (BOOL)textField:(UITextField
    *)textField shouldChangeCharactersInRange:(NSRange)range
    replacementString:(NSString *)string
  1. 图片轮播器
  1. 往scrollView 中添加imageView (5个),imageView的尺寸保持根scrollView的尺寸一致
  1. 设置分页pagingEnabled = YES
  1. pageControl 在storyboard中
    根 scrollView 同级
  1. 设置pageControl的一些属性 numberOfPages currentPage...
  1. 在scrollViewDidScroll代理方法中设置currentPage
  使用x偏移量/ scrollView的宽度+ 0.5  "记得打印偏移量"

6.

nextPage

  1. 获取pageControl的当前页+ 1 (跳到下一个) = index


  2. 判断index 是不是为5  (如果是  index = 0)(循环切换)


  3. 设置偏移量(使用偏移量切换到下一个imageView)

 
  index *width(scrollView)
  1. 定时器
  1. performSector
  1. NSTimer
  self.timer = [NSTimer

scheduledTimerWithTimeInterval:2.0f target:self
selector:@selector(nextPage:) userInfo:@"ceshi" repeats:YES];

      timer是一次性, 用了就不能再用了


  3. 添加timer到主线程

 
  [[NSRunLoop mainRunLoop] addTimer:self.timer

forMode:NSRunLoopCommonModes];

效果展示:

8.png

7.png
  1. 封装图片轮播器
  1. 类方法加载xib
  1. 先把awakeFromNib
  1. pageControl的一次性操作
  1. 重写图片数组set方法
      将控制器中viewDidLoad里面所有的代码除了pageControl的一次性操作
  1. copy控制器中所有的方法
  1. 将报错的地方补上
  1. 使用scrollView做购物车
  1. 设置contentSize
  拿出scrollView中的最后一个控件


  算出这个空间的bottom = (控件的高+ 控件的y)


  bottom 就是contentSize 的height
  1. 设置contentOffset
  首先判断最后一个控件的bottom 是不是比scrollView的高度大


  如果大设置偏移量height 就 = bottom
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容