iOS UI布局神器Masonry

我们在iOS开发中,一定会牵扯到我们的布局的开发。一般的来讲,布局分为三种方式。

  • 第一种,就是我们直接在StoryBoard里面“画出”我们的布局,这种所见即所得的方式,是苹果公司推荐开发者的方
    式,所以,在我们最新的XCode里面创建的项目,都会默认包含一个叫做Main.storyboard的文件,在这个文件上面,我们就可以画出我们自己的布局。甚至有时候我们都可以让我们的美工给我们做出Storyboard文件。例如下图。
Screen Shot 2016-01-26 at 5.26.25 PM.png
  • 第二种方式就是我们的xib,或者叫nib文件,这种方式也是一种图像化的布局方式。只是我们的布局是和每个View Controller对应的。所以,这种方式比我们的Storyboard更加灵活,做起来也比较轻量,没有Storyboard中的场景的概念,也可以根据你自己的需求选择需要创建nib文件的类。如下:
Paste_Image.png
  • 第三种,就是通过手写代码来实现我们的UI界面。这种方式在移动开发刚兴起时,比较流行,因为有人说通过手写的代码,比你在Storyboard或者是xib中做的界面,加载起来要快,因为我们的所见即所得的界面,还是要在运行的时候,转换成本地的代码。而随着现在苹果公司对于Interface builder的慢慢优化,我们使用手写UI的性能优势已经越来越小了。我们今天仍然有很多公司使用手写UI,主要是因为遗留的代码问题,还有团队协作的方便性。在手写代码的时候,我们经常会发现一个简单的界面,会有很多代码需要编写,尤其是我们UI组件之间的Layout constraits。

Masonry是什么

刚才说了,我们的手写UI,不可避免的面对一个问题,我们的UI可能会在不同的环境下有不同的显示效果,所以,我们要让我们写出的界面能够在各种不同的设备上显示正确。我们在iOS上解决这个问题,是通过给不同的组件上添加各种各样的约束。约束非常好理解。例如,我们有这么一个样式。

Paste_Image.png

如果我们想在我们的屏幕上写这么一个布局。我们可能就要做一些约束,保证这个界面能够在不同的手机上显示,例如:

  • 让绿色的左边和上边的边界和父视图一样,流出15个像素的边界
  • 让绿色的高度和蓝色的高度一样
  • 让绿色的宽度和红色的一样
    ...
    等等
    同理,我们的红色快和蓝色块也有相应的约束需要添加,当我们确定好我们所需要的所有约束,然后就可以使用Masonry来帮我们写这些约束了,那么我们不用Masonry不行么?

为什么要用Masonry?

如果要实现上面的这些约束,我们iOS里面自带了一个类,叫做NSLayoutConstraints,通过这个类相关的API,我们可以让我们的视图对象创建相应的约束,例如,我们有一个非常简单的需求,想在视图里面创建一个子视图,我们子视图的Rect上下左右比外面的视图都要少10dp。那么,使用NSLayoutConstraints,我们的代码大概是这个样子的。
UIView *superview = self;

  UIView *view1 = [[UIView alloc] init];
  view1.translatesAutoresizingMaskIntoConstraints = NO;
  view1.backgroundColor = [UIColor greenColor];
  [superview addSubview:view1];

  UIEdgeInsets padding = UIEdgeInsetsMake(10, 10, 10, 10);

  [superview addConstraints:@[

    //view1 constraints
    [NSLayoutConstraint constraintWithItem:view1
                                 attribute:NSLayoutAttributeTop
                                 relatedBy:NSLayoutRelationEqual
                                    toItem:superview
                                 attribute:NSLayoutAttributeTop
                                multiplier:1.0
                                  constant:padding.top],

    [NSLayoutConstraint constraintWithItem:view1
                                 attribute:NSLayoutAttributeLeft
                                 relatedBy:NSLayoutRelationEqual
                                    toItem:superview
                                 attribute:NSLayoutAttributeLeft
                                multiplier:1.0
                                  constant:padding.left],

    [NSLayoutConstraint constraintWithItem:view1
                                 attribute:NSLayoutAttributeBottom
                                 relatedBy:NSLayoutRelationEqual
                                    toItem:superview
                                 attribute:NSLayoutAttributeBottom
                                multiplier:1.0
                                  constant:-padding.bottom],

    [NSLayoutConstraint constraintWithItem:view1
                                 attribute:NSLayoutAttributeRight
                                 relatedBy:NSLayoutRelationEqual
                                    toItem:superview
                                 attribute:NSLayoutAttributeRight
                                multiplier:1
                                  constant:-padding.right],

  ]];

就会产生这么一大坨代码,而这还只是一个最简单的例子,如果说你的UI里面有很多的子视图,他们之间有各种关系,那么,你的代码如果这样写下去,基本就没人能读得懂了。
不过没关系,我们不是有Masonry么,看看在Masonry中是如何处理这个约束的。
UIEdgeInsets padding = UIEdgeInsetsMake(10, 10, 10, 10);

    [view1 mas_makeConstraints:^(MASConstraintMaker *make) {
        make.top.equalTo(superview.mas_top).with.offset(padding.top); //with is an optional semantic filler
        make.left.equalTo(superview.mas_left).with.offset(padding.left);
        make.bottom.equalTo(superview.mas_bottom).with.offset(-padding.bottom);
        make.right.equalTo(superview.mas_right).with.offset(-padding.right);
    }];

或者更短:
[view1 mas_makeConstraints:^(MASConstraintMaker *make) {
make.edges.equalTo(superview).with.insets(padding);
}];
是不是清爽多了,他的写法也很清晰,其实就是通过block来实现我们的约束,你的每一个view就是block里面的make,然后他有很多属性,比如说top.left.right.bottom,centerX, centerY等等,然后,你使用一个equalTo来和其他视图建立约束,然后在每个视图上有一些属性,类似于mas_top,就可以使用上了。是不是很爽啊。

样例

例如,我们现在需要写一个tableview cell,这个cell是包含四个按钮,每个按钮中间有个Image view,然后每个Image view正下方有个label,如图所示:


我们就可以通过如下方式来写cell:
// DRDiscoverCategoryCell.h
#import <UIKit/UIKit.h>

@interface DRDiscoverCategoryCell : UITableViewCell

@end

// DRDiscoverCategoryCell.m
#import "DRDiscoverCategoryCell.h"

@implementation DRDiscoverCategoryCell

- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier {
    self = [super initWithStyle:style reuseIdentifier:reuseIdentifier];
    if (self) {
        self.contentView.backgroundColor = [SLColor backgroundColor];
        self.selectionStyle = UITableViewCellSelectionStyleNone;
        CGFloat buttonWidth = [UIScreen mainScreen].bounds.size.width / 4;
        for (int i = 0 ; i < 4; i++) {
            UIButton *mainButton = [[UIButton alloc] initWithFrame:CGRectMake(i * buttonWidth, 0, buttonWidth, 80)];
            mainButton.backgroundColor = [SLColor c9Color];
            
            UIImage *image =
                [UIImage imageWithData:
                [NSData dataWithContentsOfURL:
                [NSURL URLWithString:@"http://forum-dev.dianrong.com/static/image/discover/icon-finance.png"]]];
            UIImageView *imageView = [[UIImageView alloc] initWithImage:image];
            [mainButton addSubview:imageView];
            [imageView mas_makeConstraints:^(MASConstraintMaker *make) {
                make.centerX.equalTo(mainButton);
                make.centerY.equalTo(mainButton.mas_centerY).offset(-10);
                make.width.height.equalTo(@45);
            }];
            
            UILabel *categoryTitleLabel = [[UILabel alloc] init];
            categoryTitleLabel.text = @"点融黑帮";
            categoryTitleLabel.font = [UIFont systemFontOfSize:11];
            categoryTitleLabel.textAlignment = NSTextAlignmentCenter;
            [mainButton addSubview:categoryTitleLabel];
            [categoryTitleLabel mas_makeConstraints:^(MASConstraintMaker *make) {
                make.centerX.equalTo(mainButton);
                make.width.equalTo(@45);
                make.top.equalTo(imageView.mas_bottom).offset(5);
            }];
            [self.contentView addSubview:mainButton];
        }
    }
    return self;
}

@end

是不是很简单啊,如果在使用Masonry的过程中有什么问题,请欢迎随时和我交流。我的微信公众账号:马哥水果派, 可以扫描下面的二维码。

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

推荐阅读更多精彩内容