首页 > 其他分享 >iOS开发_Masonry使用

iOS开发_Masonry使用

时间:2022-11-29 15:11:13浏览次数:77  
标签:mas iOS make equalTo 约束 开发 Masonry view left

  • 本文主要会讲到masonry英文文档(见上面的git地址)中提及到的使用说明,以及个人使用过程中的一些经验,仅适用一些刚使用masonry的新手,大牛可以忽略,当然也可以进来指点。

1、常用的属性与常量

  • 1.1 MASViewAttribute 以对应的系统类型

    MASViewAttribute	NSLayoutAttribute
    view.mas_left	    NSLayoutAttributeLeft
    view.mas_right	    NSLayoutAttributeRight
    view.mas_top	    NSLayoutAttributeTop
    view.mas_bottom	    NSLayoutAttributeBottom
    view.mas_leading	NSLayoutAttributeLeading
    view.mas_trailing	NSLayoutAttributeTrailing
    view.mas_width	    NSLayoutAttributeWidth
    view.mas_height	    NSLayoutAttributeHeight
    view.mas_centerX	NSLayoutAttributeCenterX
    view.mas_centerY	NSLayoutAttributeCenterY
    view.mas_baseline	NSLayoutAttributeBaseline
    
  • 1.2 UIView

    • 先来一波最为常用的使用方法,大家可以看一下大致语法,下面会细讲使用
    //.分别设置各个相对边距(superview为view的父类视图,下同)
    make.left.mas_equalTo(superView.mas_left).mas_offset(10);
    make.right.mas_equalTo(superView.mas_right).mas_offset(-10);
    make.top.mas_equalTo(superView.mas_top).mas_offset(10);
    make.bottom.mas_equalTo(superView.mas_bottom).offset(-10);
    
    //直接连接使用left大于等于每个值
    make.left.mas_greaterThanOrEqualTo(10);
    
    //设置宽和高
    make.width.mas_equalTo(60);
    make.height.mas_equalTo(60);
    
    //.设置center和款高比
    make.center.mas_equalTo(superView);
    make.width.mas_equalTo(superView).multipliedBy(1.00/3);
    make.height.mas_equalTo(superView).multipliedBy(0.25);
    
    //.关于约束优先级,此处要注意约束冲突的问题,统一约束优先级大的生效
    make.left.mas_equalTo(100);
    make.left.mas_equalTo(view.superview.mas_left).offset(10);
    make.left.mas_equalTo(20).priority(700);
    make.left.mas_equalTo(40).priorityHigh();
    make.left.mas_equalTo(60).priorityMedium();
    make.left.mas_equalTo(80).priorityLow();
    
    //.如果你想让view的(x坐标)左边大于等于label的左边,以下两个约束的写法效果一样
    make.left.greaterThanOrEqualTo(label);
    make.left.greaterThanOrEqualTo(label.mas_left);
    
    • 注:约束的链式写法中,不包含其他相对的view时,默认为其superview,即make.left.mas_equalTo(100);等价于make.left.mas_equalTo(view.superview.mas_left).offset(10);和make.left.mas_equalTo(view.superview).offset(10);
  • 1.3 NSNumber

    • 自动布局允许使用常量去设置宽或高,如果你想通过一个数字设置一个view的最小和最大的width,可以用equality blocks,如下:
    //width >= 200 && width <= 400
    make.width.greaterThanOrEqualTo(@200);
    make.width.lessThanOrEqualTo(@400)
    
    • 然而自动布局不允许对齐属性的约束(如:left,right,centerY等)设置为常量值,你可以使用NSNumber来设置相对于父类view这些约束属性,如:
    // creates view.left = view.superview.left + 10
    make.left.lessThanOrEqualTo(@10)
    
    • 如果你不想使用NSNumber来设置,也可以用如下结构来创建你的约束,如:
    make.top.mas_equalTo(42);
    make.height.mas_equalTo(20);
    make.size.mas_equalTo(CGSizeMake(50, 100));
    make.edges.mas_equalTo(UIEdgeInsetsMake(10, 0, 10, 0));
    make.left.mas_equalTo(view).mas_offset(UIEdgeInsetsMake(10, 0, 10, 0));
    
    • 以上用法默认添加mas_前缀,如果你不想添加此前缀,但还想使用常量值设置约束,需要在导入Masonry头文件前,
    • 设置宏定义MAS_SHORTHAND_GLOBALS,至于为什么,去masonry代码中搜索一下MAS_SHORTHAND_GLOBALS便知。
  • 1.4 NSArray

    • 用数组添加集中不同类的约束,如:
    make.height.equalTo(@[view1.mas_height, view2.mas_height]);
    make.height.equalTo(@[view1, view2]);
    make.left.equalTo(@[view1, @100, view3.right]);
    

2、约束的优先级属性

.priority       允许你设置一个非常准确的的约束优先级(0-1000)
.priorityHigh   相当于系统的 UILayoutPriorityDefaultHigh
.priorityMedium 介于 high and low之间的优先级
.priorityLow    相当于系统的 UILayoutPriorityDefaultLow
  • 注:默认通过mas_make添加的约束不设置优先级时,默认都是最高(1000)
  • 优先级属性可以放在约束链的末端使用,如:
make.left.greaterThanOrEqualTo(label.mas_left).with.priorityLow();
make.top.equalTo(label.mas_top).with.priority(600);

3、更加便利的约束方法

  • Masonry提供了一些便利的方法供我们同时创建多个不同的约束,他们被称为MASCompositeConstraints,如:
  • 3.1 edges

    // 使一个view的top, left, bottom, right 等于view2的
    make.edges.equalTo(view2);
    //相对于superviewde上左下右边距分别为5,10,15,20
    make.edges.equalTo(superview).insets(UIEdgeInsetsMake(5, 10, 15, 20))
    
  • 3.2 size

    // 使得宽度和高度大于等于 titleLabel
    make.size.greaterThanOrEqualTo(titleLabel)
    // 相对于superview宽度大100,高度小50
    make.size.equalTo(superview).sizeOffset(CGSizeMake(100, -50))
    
  • 3.3 center

    //中心与button1对齐
    make.center.equalTo(button1)
    //水平方向中心相对向左偏移5,竖直方向中心向下偏移10
    make.center.equalTo(superview).centerOffset(CGPointMake(-5, 10))
    你可以在约束链里添加相应的view来增加代码的可读性:
    // 除了top,所有的边界与superview对齐
    make.left.right.and.bottom.equalTo(superview);
    make.top.equalTo(otherView);
    

4、关于如何修改约束

  • 有时候,你为了实现动画或者移除替换一些约束时,你需要去修改一些已经存在的约束,Masonry提供了一些不同的方法去更新约束,你也可以将多个约束存在数组里
  • 4.1 References

    • 你可以持有某个特定的约束,让其成为成员变量或者属性
    //设置为公共或私接口
    @property (nonatomic, strong) MASConstraint *topConstraint;
    ...
    
    // 添加约束
    [view1 mas_makeConstraints:^(MASConstraintMaker *make) {
    	self.topConstraint = make.top.equalTo(superview.mas_top).with.offset(padding.top);
    	make.left.equalTo(superview.mas_left).with.offset(padding.left);
    }];
    ...
    
    // 然后可以调用
    //该约束移除
    [self.topConstraint uninstall];
    //重新设置value,最常用
    self.topConstraint.mas_equalTo(20);
    //该约束失效
    [self.topConstraint deactivate];
    //该约束生效
    [self.topConstraint activate];
    
  • 4.2 mas_updateConstraints

    • 如果你只是想更新一下view对应的约束,可以使用 mas_updateConstraints 方法代替 mas_makeConstraints方法
    //这是苹果推荐的添加或者更新约束的地方
    // 在响应setNeedsUpdateConstraints方法时,这个方法会被调用多次
    // 此方法会被UIKit内部调用,或者在你触发约束更新时调用
    - (void)updateConstraints {
    	[self.growingButton mas_updateConstraints:^(MASConstraintMaker *make) {
    		make.center.equalTo(self);
    		make.width.equalTo(@(self.buttonSize.width)).priorityLow();
    		make.height.equalTo(@(self.buttonSize.height)).priorityLow();
    		make.width.lessThanOrEqualTo(self);
    		make.height.lessThanOrEqualTo(self);
    	}];
    	//调用super
    	[super updateConstraints];
    }
    
  • 4.3 mas_remakeConstraints

    • mas_updateConstraints只是去更新一些约束,然而有些时候修改一些约束值是没用的,这时候mas_remakeConstraints就可以派上用场了
    • mas_remakeConstraints某些程度相似于mas_updateConstraints,但不同于mas_updateConstraints去更新约束值,他会移除之前的view的所有约束,然后再去添加约束
    - (void)changeButtonPosition {
    	[self.button mas_remakeConstraints:^(MASConstraintMaker *make) {
    		make.size.equalTo(self.buttonSize);
    		if (topLeft) {
    			make.top.and.left.offset(10);
    		} else {
    			make.bottom.and.right.offset(-10);
    		}
    	}];
    }
    

5、在哪创建我的约束

  • 贴一个官方说明的例子:
@implementation DIYCustomView
- (id)init {
	self = [super init];
	if (!self) return nil;
	// --- Create your views here ---
	self.button = [[UIButton alloc] init];
	return self;
}

// tell UIKit that you are using AutoLayout
+ (BOOL)requiresConstraintBasedLayout {
	return YES;
}

// this is Apple's recommended place for adding/updating constraints
- (void)updateConstraints {
	// --- remake/update constraints here
	[self.button remakeConstraints:^(MASConstraintMaker *make) {
		make.width.equalTo(@(self.buttonSize.width));
		make.height.equalTo(@(self.buttonSize.height));
	}];
	//according to apple super should be called at end of method
	[super updateConstraints];
}

- (void)didTapButton:(UIButton *)button {
	// --- Do your changes ie change variables that affect your layout etc ---
	self.buttonSize = CGSize(200, 200);
	// tell constraints they need updating
	[self setNeedsUpdateConstraints];
}

6、Layout必备知识

  • AutoLayout关于更新的几个方法的区别
  • setNeedsLayout:告知页面需要更新,但是不会立刻开始更新。执行后会立刻调用layoutSubviews。
  • layoutIfNeeded:告知页面布局立刻更新。所以一般都会和setNeedsLayout一起使用。如果希望立刻生成新的frame需要调用此方法,利用这点一般布局动画可以在更新布局后直接使用这个方法让动画生效。
  • layoutSubviews:系统重写布局
  • setNeedsUpdateConstraints:告知需要更新约束,但是不会立刻开始
  • updateConstraintsIfNeeded:告知立刻更新约束
  • updateConstraints:系统更新约束

7、使用tips

  • 给view添加约束时,必须已经添加到其superview上面

  • 不需要设置view.translatesAutoresizingMaskIntoConstraints = NO;,masonry内部已经帮我设置过了

  • 手写布局时,合理使用约束,尽量约束冲突问题

  • 因为iOS中原点在左上角所以注意使用offset时注意right和bottom用负数

  • 推荐两个大牛写的关于Masonry源码解析的博客

  • DemoA --- DemoB

  • 布局约束SDAutoLayout

标签:mas,iOS,make,equalTo,约束,开发,Masonry,view,left
From: https://www.cnblogs.com/CH520/p/9541550.html

相关文章

  • 团队开发冲刺第八天(实现评论功能)
      昨天在设计评论的UI,以及查阅网址看看如何添加评论今天完成了评论的功能遇到的问题:多线程并发的问题,导致第一次输出评论没有及时的显示在屏幕上,之后在进入该页面时,之前的......
  • 团队冲刺开发第二天
    今天花费将近3个多小时的时间,将以前的Android知识回顾了一遍,同时自己针对自己的任务做了一个小Demo。主要用到的是ViewModel来存储页面数据,Respositry仓库来进行数据的增删......
  • android 之ndk开发
    1、AndroidNDK简介NDK全称为nativedevelopmentkit本地语言(C&C++)开发包。而对应的是经常接触的Android-SDK,(softwaredevelopmentkit)软件开发包(只支持java语言开发)。简单......
  • react 通过axios获取数据 出现promise为pending——使用await获取Promise对象的Promis
    1.await必须在async定义的方法中使用lete=await78910;console.log(e); 2.当await右边为一个PromiseState为fulfilled的Promise对象时,则返回其PromiseResult值asyncfun......
  • 应用 Serverless 化,让业务开发心无旁骛
    我们希望让用户做得更少而收获更多,通过Serverless化,用云就像用电一样简单。”张建锋表示,Serverless让云计算从一种资源真正变成一种能力,未来云将全面Serverless化,更加......
  • tcpdum离线安装(alios8版)
    一、下载rpm包链接:https://pan.baidu.com/s/11eTaFi5fGNtSI_RfPrQtcQ提取码:tl8q二、执行安装命令rpm-ivhtcpdump-4.9.3-3.el8.x86_64.rpm ......
  • 开发申请单——流程表单有转交时确保开发人组件值随着转交人变化而变化
    参考宜搭平台接口:https://www.yuque.com/yida/support/aql605说明:只写了一个方法 getProcessData,有页面自动调用函数didMount调用(当前页面-设置-生命周期-页面加载......
  • SpringBoot 实际项目开发中工厂模式的巧妙使用
    简单工厂模式:     简单工厂模式是创建型模式,创建型模式顾名思义,也就是说在创建对象的时候,遇到了瓶颈才会选择的设计模式。那么该什么情况使用呢。  简单工厂模式......
  • 设计模式在实际开发中的应用
    我相信有些人学了设计模式,依然不知道设计模式的应用场景,感觉我们日常开发中,仅仅写些逻辑,调下接口,连接下数据库就完事儿了。感觉设计模式都被框架运用的淋淋尽致,似乎没有我们......
  • 解决程序开发过程中的 cannot open shared object file 问题
    解决程序开发过程中的cannotopensharedobjectfile问题目录解决程序开发过程中的cannotopensharedobjectfile问题问题描述问题分析问题解决参考文献问题描述......