UIScrollView 约束剖析

UIScrollView 约束设置曾一直困扰着iOS开发小伙伴们。趁着这次和几位小伙伴们分享我在做UIScrollView约束的一些心得,顺便写出来以便大家参考。

先看展示效果:

111.gif

为了方便大家理解后续的内容,首先我们思考以下几点:

  1. 我们在为UIScrollView设置的上下左右约束,是设置的它Content约束还是UIScrollView本身边界的约束?
  2. UIScrollView是如何计算出contentSize的?
  3. UIScrollView是否一定需要一个容器作为参考?

对于第一个问题,我们先做个测试:新建一个UIViewController带上Xib,在其View中我们放置一个UIScrollView,如下图所示:

屏幕快照

上图中我们的左右边界都超出了View20个像素。如果为UIScrollView设置的左右约束,是设置的它Content相对约束。那么我们可以推断出contentSize应该是超过View宽度的大小也就是说左右可以滑动。

运行结果:

contentSize.gif

我们打印下ViewFrame宽度,UIScrollViewFrame宽度和contentSize的宽度进一步确认下:(我用的是iPhone 6模拟器)

-(void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];
    CGFloat viewWidth = CGRectGetWidth(self.view.bounds);
    CGFloat scrollViewWidth = CGRectGetWidth(self.showScrollView.bounds);
    CGFloat contentSizeWidth = self.showScrollView.contentSize.width;
    NSLog(@"viewWidth:%f--- scrollViewWidth:%f-----contentSizeWidth:%f",viewWidth,scrollViewWidth,contentSizeWidth);
}

终端Log:

ScrollViewLayout[24613:15651720] viewWidth:375.000000--- 

scrollViewWidth:415.000000-----contentSizeWidth:0.000000

我们看到contentSize为0,UIScrollViewFrame宽度为415即为375+20+20。由此对于第一个问题,我们有了答案:

我们在为UIScrollView设置的上下左右约束,是设置的UIScrollView本身边界的约束。而非设置的它Content约束

对于后面两个问题,我们先看几种设置UIScrollView 约束的方法,这里我们只展示上下滑动的情况。

添加一个容器作为参考视图contentSize自动计算

xib中添加子视图的步骤我就不详细说明了,我们简单的添加几张图片和一个Label到容器视图中具体如下图所示:

屏幕快照

上图中当你拖动containerViewUIScrollView中时,设置完前后左右的约束和宽度约束后会报错,不过不要担心,那是因为此时UIScrollView还不能计算出它的contentSize如果你看着不舒服,可以把containerView的高度约束先固定,不过记得后面要删除。

这里需要特别注意,如果你想要高度自动计算,你必须的设置最下面的控件Labelbottom约束到containerView之间的距离。

添加一个容器作为参考视图contentSize手动计算

前面一个例子中,我们必须得设置最下面的控件Labelbottom约束。试想下如果我们界面中最下面一个控件例如Label 可能离底部很长,但是我们又不能设置到底部的距离大于或等于某个值,因为如果这样contentSize的高度是不确定的,会报错。我们又想Label可以根据内容向下扩展,如果文字很多,也许会超过屏幕底部。这时我们该如何设置UIScrollView的约束呢,这时我们需要设置containerView的高度约束并作为一个属性变量来手动更改。界面如下:

屏幕快照

在我们需要改变contentSize时,我们手动计算并赋值即可例如:

-(void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];
    self.containerViewHeightConstraint.constant = CGRectGetMaxY(self.detailLabel.frame);
}

不添加容器直接在UIScrollView中添加子视图

UIScrollView中的子视图添加过程不详细介绍了,添加完毕后的视图如下所示:
我们希望黄色View的高度可以随其里面的Label内容高度而改变。设置完成后运行结果可以看前面展示效果Gif中的无容器视图+高度自动

屏幕快照

实例结论

前面两个例子中我们都设置了容器的宽度约束和UIScrollView的宽度相同而UIScrollView的宽度和UIViewControllerView相同从而推断出容器的宽度。UIScrollView的高度,第一个例子是根据容器里面的子视图计算出容器的高度,第二个例子高度是手动计算。因此都是通过子视图的宽度和高度来推断出UIScrollViewcontentSize。第三个例子中我们也设置了最后一个View的宽度和UIScrollView的宽度相同,高度也是由UIScrollView子视图推断而出,虽然这里黄色View的高度又是由其子视图推断而出。

因此我们得出结论UIScrollViewcontentSize是由其子视图决定。

对于第三个问题:我们可以通过例子三中可以看出,UIScrollView可以不需要一个与之边界相同的容器作为参考。但是其子视图一定得能够计算出其宽度和高度。

纯代码编写多页面滑动

这里有两种方法,一种是需要容器,一种不要容器,代码如下。

- (void)viewDidLoad {
    [super viewDidLoad];
    self.extendedLayoutIncludesOpaqueBars = NO;
    self.edgesForExtendedLayout = UIRectEdgeNone;
    self.automaticallyAdjustsScrollViewInsets = NO;
    
    self.view.backgroundColor = [UIColor whiteColor];
    self.scrollView = [[UIScrollView alloc]init];
    self.scrollView.translatesAutoresizingMaskIntoConstraints = NO;
    self.scrollView.pagingEnabled = YES;
    [self.view addSubview:self.scrollView];
    self.scrollView.backgroundColor = [UIColor greenColor];
    [self.view zhc_pinAllEdgesOfSubview:self.scrollView];
    //[self addSubViewToScrollView];
    
     self.containerView = [[UIView alloc]init];
     self.containerView.translatesAutoresizingMaskIntoConstraints = NO;
     self.containerView.backgroundColor = [UIColor yellowColor];
     [self.scrollView addSubview:self.containerView];
     [self.scrollView zhc_pinAllEdgesOfSubview:self.containerView];
     [self.scrollView zhc_pinSubview:self.containerView toEdge:NSLayoutAttributeHeight withConstant:0.0];
    [self addSubViewToContainerView];
     // Do any additional setup after loading the view.

    
}



-(void)addSubViewToScrollView
{
    UIView *lastView = self.scrollView;
    UIView *superView = self.scrollView;
    NSInteger count = 5;
    for (NSInteger i=0; i< count; i++) {
        UIImageView *imgView = [[UIImageView alloc]init];
        imgView.backgroundColor = [UIColor blueColor];
        imgView.contentMode = UIViewContentModeScaleAspectFit;
        imgView.translatesAutoresizingMaskIntoConstraints = NO;
        NSString *imgName = [NSString stringWithFormat:@"img%ld",(i+1)];
        imgView.image = [UIImage imageNamed:imgName];
        [superView addSubview:imgView];
       
        [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeTop withConstant:0.0];
        [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeBottom withConstant:0.0];
        [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeHeight withConstant:0.0];
        [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeWidth withConstant:0.0];
        if (i==0) {
            [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeLeft withConstant:0.0];
        }else{
            NSLayoutConstraint *constraint = [NSLayoutConstraint constraintWithItem:imgView attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:lastView attribute:NSLayoutAttributeTrailing multiplier:1.0 constant:0.0];
            [superView addConstraint:constraint];
        }
        if (i == (count-1)){
            [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeTrailing withConstant:0.0];
        }
        lastView = imgView;
        
    }
}


-(void)addSubViewToContainerView
{
    UIView *lastView = self.containerView;
    UIView *superView = self.containerView;
    NSInteger count = 5;
    CGFloat imgWidth = [UIScreen mainScreen].bounds.size.width;
    for (NSInteger i=0; i< count; i++) {
        UIImageView *imgView = [[UIImageView alloc]init];
        imgView.backgroundColor = [UIColor blueColor];
        imgView.contentMode = UIViewContentModeScaleAspectFit;
        imgView.translatesAutoresizingMaskIntoConstraints = NO;
        NSString *imgName = [NSString stringWithFormat:@"img%ld",(i+1)];
        imgView.image = [UIImage imageNamed:imgName];
        [superView addSubview:imgView];
        
        [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeTop withConstant:0.0];
        [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeBottom withConstant:0.0];
        [imgView zhc_pinSelfToEdge:NSLayoutAttributeWidth withConstant:imgWidth];
        if (i==0) {
            [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeLeft withConstant:0.0];
        }else{
            NSLayoutConstraint *constraint = [NSLayoutConstraint constraintWithItem:imgView attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:lastView attribute:NSLayoutAttributeTrailing multiplier:1.0 constant:0.0];
            [superView addConstraint:constraint];
        }
        if (i == (count-1)){
            [superView zhc_pinSubview:imgView toEdge:NSLayoutAttributeTrailing withConstant:0.0];
        }
        lastView = imgView;
        
    }

}

Demo下载

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

推荐阅读更多精彩内容