首页 > 技术文章 > UIScrollview 技巧

YungMing 2014-04-09 11:48 原文

设置UIScrollView的contentSize


如果使用自动布局,那么它会自动帮你基于这个scrollview的子视图的约束来计算这个内容大小。在非自动布局情况下,如果app旋转导致scrollview 的bounds改变,不会影响到scrollview的contentSize,而如果重新设置contentSize,也不会影响scrollview的子视图,这个contentSize仅仅是决定了滚动的范围。

下面我们用代码创建一个UIScrollView,UILabel在y轴上顺序排列:

UIScrollView* sv = [UIScrollView new];
sv.backgroundColor = [UIColor whiteColor];
sv.translatesAutoresizingMaskIntoConstraints = NO;
[self.view addSubview:sv];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[sv]|"
                                         options:0 metrics:nil
                                           views:@{@"sv":sv}]];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[sv]|"
                                         options:0 metrics:nil
                                           views:@{@"sv":sv}]];
UILabel* previousLab = nil;
for (int i=0; i<30; i++) {
    UILabel* lab = [UILabel new];
    lab.translatesAutoresizingMaskIntoConstraints = NO;
    lab.text = [NSString stringWithFormat:@"This is label %d", i+1];
    [sv addSubview:lab];
    [sv addConstraints:
    [NSLayoutConstraint constraintsWithVisualFormat:@"H:|-(10)-[lab]"
                                             options:0 metrics:nil
                                               views:@{@"lab":lab}]];
    if (!previousLab) { // first one, pin to top
        [sv addConstraints:
        [NSLayoutConstraint constraintsWithVisualFormat:@"V:|-(10)-[lab]"
                                                 options:0 metrics:nil
                                                   views:@{@"lab":lab}]];
    } else { // all others, pin to previous
        [sv addConstraints:
        [NSLayoutConstraint
          constraintsWithVisualFormat:@"V:[prev]-(10)-[lab]"
          options:0 metrics:nil
          views:@{@"lab":lab, @"prev":previousLab}]];
    }
    previousLab = lab;
}

运行代码,你会发现label 都放置在正确的位置,但是scrollView 却不能滚动。而且,即使你手动设置了contentSize也无法滚动。原因就是在页面布局时,scrollview的contentSize会自动根据它和子视图之间的约束来计算得出。解决方法就是,再添加多一个约束,告诉scroll view 它的contentSize的高度应该是多少:

// 给底部最后一个label设置一个y轴方向的约束,这样高度就可以确定了
[sv addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:[lab]-(10)-|"
                                         options:0 metrics:nil
                                           views:@{@"lab":previousLab}]];

可以看到,我们可以在垂直方向上滚动scrollview了,在水平方向上仍然不能滚动(contentSize的宽度默认就是0,正好是我们需要的)。

使用一个Content View

我们一般不会把scrollview 的子视图直接添加到scrollview上,而是添加到UIView上,然后把这个视图再添加到scrollview上,这个视图就是这个Content View。这样可以更加方便的组织和使用。

在自动布局下,我们可以有两种设置scroll view的contentSize 方法:

  • 设置content view 的translatesAutoresizingMaskIntoConstraints为YES,然后手动设置这个scroll view的 contentSize为这个content view的大小。

  • 设置content view 的translatesAutoresizingMaskIntoConstraints为NO,然后给这个content view 设置宽度和高度约束。

设置content view的大小或者约束这种方式是与它的子视图怎么定位(设置frame,还是实用约束)无关。有四种可能的组合,这四种组合开头的代码都是一样的:

UIScrollView* sv = [UIScrollView new];
sv.backgroundColor = [UIColor whiteColor];
sv.translatesAutoresizingMaskIntoConstraints = NO;
[self.view addSubview:sv];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[sv]|"
                                        options:0 metrics:nil
                                          views:@{@"sv":sv}]];
[self.view addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[sv]|"
                                     options:0 metrics:nil
                                       views:@{@"sv":sv}]];
UIView* v = [UIView new]; // content view
[sv addSubview: v];

第一种组合不使用约束:

CGFloat y = 10;
for (int i=0; i<30; i++) {
    UILabel* lab = [UILabel new];
    lab.text = [NSString stringWithFormat:@"This is label %d", i+1];
    [lab sizeToFit];
    CGRect f = lab.frame;
    f.origin = CGPointMake(10,y);
    lab.frame = f;
    [v addSubview:lab]; // add to content view, not scroll view
    y += lab.bounds.size.height + 10;
}
// set content view frame and content size explicitly
v.frame = CGRectMake(0,0,0,y);
sv.contentSize = v.frame.size;

第二种组合,content view使用约束,但它的子视图不使用:

CGFloat y = 10;
for (int i=0; i<30; i++) {
    // ... same as before, create labels, keep incrementing y
}
// configure the content view using constraints
v.translatesAutoresizingMaskIntoConstraints = NO;
[sv addConstraints:
     [NSLayoutConstraint constraintsWithVisualFormat:@"V:|[v(y)]|"
                options:0 metrics:@{@"y":@(y)} views:@{@"v":v}]];
[sv addConstraints:
      [NSLayoutConstraint constraintsWithVisualFormat:@"H:|[v(0)]|"
                   options:0 metrics:nil views:@{@"v":v}]];

第三种组合,content view 和它的子视图都使用约束:

UILabel* previousLab = nil;
for (int i=0; i<30; i++) {
    UILabel* lab = [UILabel new];
    lab.translatesAutoresizingMaskIntoConstraints = NO;
    lab.text = [NSString stringWithFormat:@"This is label %d", i+1];
    [v addSubview:lab];
    [v addConstraints:
          [NSLayoutConstraint constraintsWithVisualFormat:@"H:|-(10)-[lab]"
                                                  options:0 metrics:nil
                                                    views:@{@"lab":lab}]];
    if (!previousLab) { // first one, pin to top
        [v addConstraints:
            [NSLayoutConstraint constraintsWithVisualFormat:@"V:|-(10)-[lab]"
                                                    options:0 metrics:nil
                                                      views:@{@"lab":lab}]];
    } else { // all others, pin to previous
        [v addConstraints:
            [NSLayoutConstraint
                 constraintsWithVisualFormat:@"V:[prev]-(10)-[lab]"
                                     options:0 metrics:nil
                                       views:@{@"lab":lab, @"prev":previousLab}]];
}
    previousLab = lab;
}
// last one, pin to bottom, this dictates content size height
[v addConstraints:
[NSLayoutConstraint constraintsWithVisualFormat:@"V:[lab]-(10)-|"
                                        options:0 metrics:nil
                                          views:@{@"lab":previousLab}]];
// configure the content view using constraints
v.translatesAutoresizingMaskIntoConstraints = NO;
[sv addConstraints:
    [NSLayoutConstraint constraintsWithVisualFormat:@"V:|[v]|"
                        options:0 metrics:nil views:@{@"v":v}]];
[sv addConstraints:
    [NSLayoutConstraint constraintsWithVisualFormat:@"H:|[v(0)]|"
                        options:0 metrics:nil views:@{@"v":v}]];

第四种组合是很奇怪的组合,content view的子视图使用约束布局,但是content view 和 scrollview 不使用:

UILabel* previousLab = nil;
// ... same as before, add subviews and constraints to content view
// autolayout helps us learn the consequences of those constraints
CGSize minsz = [v systemLayoutSizeFittingSize:UILayoutFittingCompressedSize];
// set content view frame and content size explicitly
v.frame = CGRectMake(0,0,0,minsz.height);
sv.contentSize = v.frame.size;

滚动


在iOS 7 下,由于app都是全屏的,status bar,navigation bar 都是半透明的,所以scroll view 也会在 status bar 下面,那么有可能导致scrollview 的内容被status bar遮盖,我们可以设置:

sv.contentInset = UIEdgeInsetsMake(20, 0, 0, 0);

当我们设置了contentInset,我们通常也会设置scrollIndicatorInsets,让滚动条也适应:

sv.contentInset = UIEdgeInsetsMake(20, 0, 0, 0);
sv.scrollIndicatorInsets = sv.contentInset;

这样,即可以在status bar 看到半透明的scrollview 内容,又不会滚动不下导致的显示不全,但是我们硬编码这个20的值好像不是很优雅,我们可以:

- (void) viewWillLayoutSubviews {
    if (self.sv) {
        CGFloat top = self.topLayoutGuide.length;
        CGFloat bot = self.bottomLayoutGuide.length;
        self.sv.contentInset = UIEdgeInsetsMake(top, 0, bot, 0);
        self.sv.scrollIndicatorInsets = self.sv.contentInset;
    }
}

这些如果在nib中,如果我们设置view controller的automaticallyAdjustsScrollViewInsets为YES,那么iOS 7 运行时会帮我们自动适应这个scrollview 的contentInset 和 scrollIndicatorInsets,而不需要上面的代码。这个仅仅在nib上有用,一旦我们手动创建scrollview,即使设置这个automaticallyAdjustsScrollViewInsets为YES也没有用,还是需要上面的代码。

Tiling 平铺

假如我们有一个非常大的内容需要显示在scroll view上,这么大的图片,用户可以滚动来查看所有的内容细节。在内存中存放整张图片是不现实的,也是不可能的。

Tiling是一个种解决方法。我们把内容分成一个个小的矩形区域,当用户滚动时,我们查找并让目标的矩形区块显示,同时我们可以释放不在显示范围内的矩形区域。

有一个内建的CALayer子类 CATiledLayer 帮我们实现了这个分块。它的tileSize属性设置区块的大小。它的drawLayer:inContext: 会在需要一个空的区块时被调用;在图形上下文中调用CGContextGetClipBoundingBox 来截取所需区块的位置。下面我们借用苹果自己提供的PhotoScroller例子来说明:

我们给scroll view添加一个子视图,一个TiledView,这个视图用来存放我们的CATiledLayer图层。TILESIZE是256:

-(void)viewDidLoad {
    CGRect f = CGRectMake(0,0,3*TILESIZE,3*TILESIZE);
    TiledView* content = [[TiledView alloc] initWithFrame:f];
    float tsz = TILESIZE * content.layer.contentsScale;
    [(CATiledLayer*)content.layer setTileSize: CGSizeMake(tsz, tsz)];
    [self.sv addSubview:content];
    [self.sv setContentSize: f.size];
}

下面是TiledView的代码,CATiledLayer是它的根图层,所以TiledView是CATiledLayer的委托。意味着当CATiledLayer的 drawLayer:inContext: 被调用时,TiledView的drawRect:方法也会被调用,而且我们必须使用imageWithContentsOfFile:方法获取图片,而不是imageNamed:,防止系统缓存这张图片:

+ (Class) layerClass {
    return [CATiledLayer class];
}
-(void)drawRect:(CGRect)r {
    CGRect tile = r;
    int x = tile.origin.x/TILESIZE;
    int y = tile.origin.y/TILESIZE;
    NSString *tileName =
        [NSString stringWithFormat:@"CuriousFrog_500_%d_%d", x+3, y];
    NSString *path =
        [[NSBundle mainBundle] pathForResource:tileName ofType:@"png"];
    UIImage *image = [UIImage imageWithContentsOfFile:path];
    [image drawAtPoint:CGPointMake(x*TILESIZE,y*TILESIZE)];
}

上面的代码中并没有明确释放离屏区块,你可以在TiledView中调用setNeedsDisplay 或者setNeedsDisplayInRect: 方法,但是这样并不会清除离屏的区块,我们相信这个CATiledLayer会帮我们处理好。

推荐阅读