使用UIStackView來簡化iOS的界面布局

前言

在過去iOS頁面布局較為傳統(tǒng),大多數(shù)人使用Frame或者AutoLayout來布局,在iOS9以后,引入了UIStackView。UIStackView是用于線性布局的控件,可以自動管理子視圖布局,自動填充。它借鑒了前端的布局算法Flexbox,可以簡便地實現(xiàn)各種頁面布局。

UIStackView雖然已經(jīng)不是新控件了,但還是有很多同學(xué)并沒有使用起來。通常有時改別人的代碼看到亂糟糟的布局代碼就有很多槽點(diǎn)。所以這也是寫這篇文章的目的所在,真的推薦大家使用StackView。事半功倍,省下來的時間摸魚不香嘛。

回歸正題,不管是使用Frame或者AutoLayout來布局,我們都需要對所有的控件的位置、大小進(jìn)行設(shè)置,Frame需要指定位置布局,AutoLayout需要指定約束布局;而UIStackView布局方式凸顯它的優(yōu)勢在于不需要設(shè)置排列視圖(即子視圖)的位置,大小(不是必須的),而是通過自身的排列、分布方式自動完成布局。對比起來,使用UIStackView更高效,我們可以通過嵌套UIStackView快速完成各式各樣的布局。

UIStackView布局思想

UIStackView的初衷就是為了簡化的界面布局,適用于列或行中布局視圖集合。

StackView使用自動布局(AutoLayout)來定位和設(shè)置其排列視圖的大小。StackView將第一個和最后一個排列的視圖與其沿堆棧軸的邊緣對齊。在水平堆棧中,這意味著第一個排列視圖的前緣被固定在StackView的前緣上,而最后一個排列視圖的后緣被固定在StackView的后緣上;在垂直堆棧中,頂部和底部邊緣分別固定在堆棧的頂部和底部邊緣上。

StackView會根據(jù)自身的布局規(guī)則進(jìn)行填充排列視圖。

distribution:

distribution 即排列方式:

  • fill根據(jù)抗拉伸、壓縮優(yōu)先級填充(默認(rèn)拉伸第一個排列視圖)
  • fillEqually在排列方向上的填充大小相同(即橫向布局寬度相同,縱向布局高度相同)
  • fillProportionally根據(jù)排列視圖的大小按比例填充
  • equalSpacing均勻地填充視圖之間的間距
  • equalCentering根據(jù)排列視圖中心點(diǎn)之間的相同間隔填充

alignment:

alignment即對齊方式:(垂直于排列方向)

  • fill填充排列視圖到StackView的可用空間
  • topStackView的頂部排列(與之相似的是leading
  • bottomStackView的尾部排列(與之相似的是trailing
  • centerStackView的中間排列
  • firstBaseline以第一個基準(zhǔn)線排列
  • lastBaseline以最后一個的基準(zhǔn)線排列

如需設(shè)置排列視圖之間的間距可以通過設(shè)置space屬性,若是排列視圖之間的間距不同,可以使用方法指定某個排列視圖的間距(此方法iOS11以上使用),或者使用一個無用的view插入在視圖之間替代間隙,此view僅作為間距使用(使用xib、Storyboard會經(jīng)常使用此類方法,可以參照)。

當(dāng)你真的了解UIStackView的這些布局思想之后,你就會知道它能幫你解決很多繁瑣的布局。(如一個多變的底部操作欄、一行大小各異的控件等等)

從上面的布局思想中,不難看出,其實我們僅需要確定StackView的排列方向,以及排列方式、對其方式,就能大體上對整個排列視圖初步的布局,而后在根據(jù)不同的視圖進(jìn)行大小上的調(diào)整以及間距的調(diào)整即可。

使用UIStackView來自動布局子視圖,你只需要每個子視圖關(guān)注自身的大小即可。

以此類操作欄為例,舉個栗子??:


先說說我們常用的布局方式,可能還是會有一部分人會選擇Frame布局,或者AutoLayout布局。

但此類UI在設(shè)計之初,通常會有很多狀態(tài)、特征,每一種狀態(tài)下,控件都會變化。

那么Frame布局在這種布局容易變化的情況下,就顯得有非常的繁瑣,布局代碼非常的多,并且狀態(tài)很多的時候不好維護(hù)。
同樣AutoLayout也是如此,需要寫很多的更新布局約束。

這個時候,借用UIStackView的思想,我們可以很簡單的實現(xiàn)這個布局。每個控件只關(guān)注自身大小,不會對其他的空間產(chǎn)生依賴關(guān)系,在需要時顯示出來,不需要時隱藏起來。

我們先以文本輸入入口“說點(diǎn)什么”小試牛刀。

下面就是用StackView布局的效果。


這里我是使用的xib結(jié)合StackView。如果我們平時使用的代碼布局,也可以使用代碼結(jié)合StackView布局,這樣也會減少很多代碼量,可以自行腦補(bǔ)。

UIStackView用法

初始化

與其他控件一樣的初始化方式;

- (instancetype)initWithFrame:(CGRect)frame NS_DESIGNATED_INITIALIZER;
- (instancetype)initWithCoder:(NSCoder *)coder NS_DESIGNATED_INITIALIZER;

當(dāng)然也可以選擇專屬的初始化方式;

- (instancetype)initWithArrangedSubviews:(NSArray<__kindof UIView *> *)views;

添加、刪除子視圖

- (void)addArrangedSubview:(UIView *)view;
- (void)removeArrangedSubview:(UIView *)view;

排列方向

@property(nonatomic) UILayoutConstraintAxis axis;

typedef NS_ENUM(NSInteger, UILayoutConstraintAxis) {
    UILayoutConstraintAxisHorizontal = 0, // 水平方向
    UILayoutConstraintAxisVertical = 1    // 垂直方向
};

布局方式

@property(nonatomic) UIStackViewDistribution distribution;

typedef NS_ENUM(NSInteger, UIStackViewDistribution) {
    UIStackViewDistributionFill = 0, //子視圖填充滿指定方向,優(yōu)先拉伸第一個控件
    UIStackViewDistributionFillEqually, //每個子視圖填充大小相等,
    UIStackViewDistributionFillProportionally, //根據(jù)每個子視圖里面內(nèi)容的尺寸來進(jìn)行填充操作
    UIStackViewDistributionEqualSpacing, //每個子視圖之間的間距相等
    UIStackViewDistributionEqualCentering, //每個子視圖中心直接的間距相等
} API_AVAILABLE(ios(9.0));

對齊方式

@property(nonatomic) UIStackViewAlignment alignment;

typedef NS_ENUM(NSInteger, UIStackViewAlignment) { 
    UIStackViewAlignmentFill, //水平:subView的上下和StackView的上下邊距 相等   垂直: subView的左右邊距和 StackView的所有相等 
    UIStackViewAlignmentLeading,//垂直有效 :左對齊
    UIStackViewAlignmentTop = UIStackViewAlignmentLeading, // 水平有效 上對齊
    UIStackViewAlignmentFirstBaseline,//水平有效,第一行基準(zhǔn)線對齊。
    UIStackViewAlignmentCenter, //中心基準(zhǔn)線對齊 1.水平 高度中點(diǎn)對齊 2.垂直:寬度中點(diǎn)對齊
    UIStackViewAlignmentTrailing,  //垂直有效,右邊界對齊。
    UIStackViewAlignmentBottom = UIStackViewAlignmentTrailing,// 水平有效 ,下邊界對齊。
    UIStackViewAlignmentLastBaseline,//水平有效,最后一行基準(zhǔn)線對齊。
} API_AVAILABLE(9_0);

間距

@property(nonatomic) NSInteger space; //排列視圖相鄰邊緣之間的距離。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容