iOS第三方库 ATStackView 2.0的介绍

之前的写 ATStackView 1.0 版本由于是继承UIStackView的,考虑该组件的缺陷而引发的一些性能上、实用性、可扩展性上的问题,决定对该组件进行底部重构。升级2.0摆脱了UIStackView的包袱,同时不再使用约束进行布局。优化布局算法等操作。

ATStackView 1.0 介绍

ATStackView 1.0缺陷:

  1. UIStackView是iOS9.0推出的,这将意味着无法支持更低版本的iOS系统。

  2. ATStackView 1.0 和 UIStackView都是基于自动布局的,频繁地使用它们会产生大量约束代码,而约束代码最终通过线性方程求解得出frame值,虽然苹果公司在iOS12对自动布局算法进行了优化,但是还是不能解决低版本的iOS系统下约束性能的问题。

  3. 继承UIStackView虽然带来了一些布局上的便利,但是它的可扩展性很低,并且由于它的源码是未知的,使得添加新特性比较困难。比如说,ATStackView 1.0 有个方法可以将元素从尾部开始添加,这个过程的实现实际上是通过添加一个约束优先级比较低的自伸长view,将头部和尾部连接(撑)起来,由于它的高度是0,所以用户感觉不到它。这个功能看似很酷,但是本来只要添加两个view的操作,结果添加了三个view。万一开发者需要调用subviews进行操作,那后果岂不是不堪设想...。

基于这些问题的存在,决定重构ATStackView,ATStackView 2.0将继承NSObject,进行轻量级改造。

github链接: https://github.com/AutoJiang/ATStackView

ATStackView 2.0的使用介绍:

1. 导入头文件

#import "UIView+ATStack.h" 

2. 快速创建一个横向布局或者纵向布局的栈(ATVerStack或 ATHorStack)。

ATVerStack *stack = [self.view getStackVer];

创建一个和 self.view 等大,纵向排列的栈。

/**快速创建一个垂直方向、子控件从上到下布局的栈*/
-(ATVerStack *)getStackVer;

/**快速创建一个水平方向、子控件从左到右布局的栈*/
-(ATHorStack *)getStackHor;

/**快速创建一个水平方向、子控件均分的栈*/
-(ATHorStack *)getStackHorEqual;

/**快速创建一个垂直方向、子控件均分的栈*/
-(ATVerStack *)getStackVerEqual;

当然,如果我们不想stack和view一样大,我们可以调用以下方法:

//inset代表内间距
-(ATVerStack *)getStackVerWithInset:(UIEdgeInsets)inset;

-(ATHorStack *)getStackHorWithInset:(UIEdgeInsets)inset;

-(ATHorStack *)getStackHorEqualWithInset:(UIEdgeInsets)inset;

-(ATVerStack *)getStackVerEqualWithInset:(UIEdgeInsets)inset;

通过设置inset来控制stack和view的间距。(注:这里的stack并不像V1.0那样是一个UIStackView实体,它实际上只是一个虚拟的 frame,所以它并没有在view上面添加这个stack)

通过调用view分类的方法生成一个stack,
接下来,我们往这个stack里面添加元素即可。

3. 添加元素(添加控件,或者其他栈)

这种方式直接添加view,view内部必须有高度上面的约束。

[stack addArrangedSubview:view];

推荐使用以下方式:(height 代表的约束height = 30 , isFill = true代表宽度和stack一样宽,isFill = false代表控件使用自适应的宽)

UIView *view = [UIView new];
[stack addArrangedSubview:view height:30 isFill:true];

4. 添加spacing

默认的ATStackView的spacing是0,一旦设置了stack.spacing = 10之后,栈内所有元素之间的间距都是10。

ATVerStackView *stack = [ATVerStackView getStackVer];
stack.spacing = 10;

当然你还可以添加额外的间距。以下代码代表下一个元素和上一个元素之间的间距为30。如果你同时设置了stack.spacing = 10,那么它们的间距就是40。

[stack addSpacing:30];

如果你不想从头部插入元素,而是想从其他位置插入时,你可以调用以下方法:

-(void)addArrangedSubview:(UIView*)view height:(CGFloat)height isFill:(BOOL)isFill position:(ATStackViewPosition)position;
typedef enum ATStackViewPosition: NSUInteger {
/**从头部添加元素*/
    ATStackViewPositionHead = 0,
/**从中间添加元素*/
    ATStackViewPositionCenter,
/**从尾部添加元素*/
    ATStackViewPositionTail,
} ATStackViewPosition;

ATSackView支持从头部,中间,或尾部添加元素。
通过设置 position = ATStackViewPositionTail 属性,可以将元素添加至头部,或者尾部(默认是添加至头部)。

想要ATStackView能从底部或中间添加元素必须得满足以下1个条件:

  1. 父视图不是UIScrollView类型。
    因为UIScrollView内部是无限大的,ATStackView并不知道它的底部在哪儿。如果一定要用,布局
    会当作内部大小和UIScrollView的frame一致计算。

5. 调用布局方法

    [stack layoutFrame];

在添加完元素以后,只有调用了layoutFrame方法,才能对内部元素开始进行布局。当内部有多层嵌套栈的时候,只需要调用最外层的layoutFrame方法即可,布局算法会遍历所有元素去执行layoutFrame。

实战:

微信-发现

50多行代码轻松实现“微信-发现”页面的简单布局。

微信-发现代码截图.png

运行结果:


image.png

九宫格:

九宫格代码截图

运行结果:


image.png

(UIView+ATStack.h)扩展方法和属性介绍:

//添加分割线
-(UIView *)addLineSeparate;
//返回的分割线可自定义颜色和高度
-(UIView *)addLineSeparateWithLelfPadding:(CGFloat)leftPadding;

//隐藏元素,同时改变其他元素的位置
-(void)setAt_hidden:(BOOL)hidden;

-(BOOL)at_hidden;

-(void)setAt_width:(CGFloat)width;

-(CGFloat)at_width;

-(void)setAt_height:(CGFloat)height;

-(CGFloat)at_height;

熟悉UIStackView的人肯定知道,子控件设置hidden = true的时候,这个控件不仅会隐藏,后面的元素位置也会自动向前移动。这里的at_hidden属性就是这样的作用。

demo演示:

demo.gif

demo地址

希望这一套布局方案能为大家的开发带来便利!😄

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

推荐阅读更多精彩内容