SwiftUI - 界面布局知识点

前言

SwiftUI采用的布局方式是和Flutter一样是弹性布局,而不是iOS之前的坐标轴的方式布局,不用准确的设置出位置大小,只需要设置当前视图大小及视图间排布的方式。灵活性增强,布局操作简便,学完Flutter再来搞SwiftUI,布局上得心应手,原理都是一样的。

1、Vstack 、Hstack 、Zstack

SwiftUI里的三大容器视图,在布局中是基本的容器类视图,子视图(如Text、Image、Button等)放在它们里面按一定的规则排序。子视图之间默认的spacing=8(即不设置时候间距为8),子视图默认的padding=16(需要设置.padding()才会有),如Text,默认的.padding()=.padding(16)。容器视图的区域为所有子视图所占的矩形空间,如果只有一个子视图,那大小就是子视图的大小。
Vstack:纵向布局容器,容器内子视图呈纵向排列,从上往下排列。
Hstack:横向布局容器,容器内子视图呈横向排列,从左往右排列。
Zstack:深度布局容器,容器内子视图呈前后排列,从里到外排列(屏幕为参照)。

2、Spacer

Spacer():一个看似透明的视图,在布局中起重要作用,它起一个撑满的作用,比如Hstack中的一个Text想在屏幕左边,那么右边添加一个Spacer即可,Spacer就会将右边剩余部分撑满,Text就会被撑到左边。在Vstack中同样可以控制一个视图在纵向的位置。如果给它设置宽度或者高度,那效果也会不一样。

HStack() {
            Text("Hello Lcr").padding(20).background(.red)
            Spacer()
        }

Spacer

注意:Text的背景颜色设置是.background,而不是.backgroundColor,background是在底部新建一个View。而且它与padding的顺序上也是有讲究的,谁在前谁在后效果都是不一样的。不妨可以试试看。

3、Devider()

SwiftUI中的表示分割线的一条线,在容器内以交叉轴方向做延伸,在不设置长度的情况下会撑满容器的最大可显示区域交叉轴。这样容器类的区域也会随着Devider去放大。当然也可以给它设置相应的宽或高来满足我们的需求。

4、LazyStack
struct customView: View {
    var text: String
    var body: some View {
        Text(text)
    }
    init(_ text: String) {
        print("create", text)
        self.text = text
    }
}
ScrollView{
     VStack {
          ForEach(0..<100){
              customView("Cell \($0)").font(.title)
          }
    }
 }

之前讲ScrollView时候我们用的是VStack,发现并没有用到懒加载机制,今天我们把VStack替换成LazyStack,发现只加载屏幕上需要展示的View,当滑动时才去展示更多的View,即触发了懒加载机制。
当我们去掉ScrollView后,发现无法触发懒加载。
当我们把ForEach替换成用Group包装的多个组后,也不能实现懒加载效果。
所以LazyStack想要触发懒加载机制,ScrollView及ForEach缺一不可。

5、Group
Group相关控件

GroupBox:字面意思看是一个“分组盒子”,示例如下:

GroupBox("Group") {
            Text("hsshhsh")
            Text("hsshhsh")
            GroupBox("Group1"){
                Text("jiuijj")
            }.padding().colorMultiply(.red)
        }.padding().colorMultiply(.yellow)

GroupBox

可见GroupBox就是一个分组的盒子,而且可以嵌套使用,图中外Box显示全黄色以及内Box显示全红色的效果使用的是colorMultiply而不是background,因为background只是在底部添加一个View,colorMultiply则是在最顶部也就是屏幕最外面添加一个遮罩层,就像在做颜色混合计算一样,覆盖上去,会影响子视图显示的颜色(如果子视图设置了别的颜色,此处未设置,所以随Box.colorMultiply颜色)。

OutlineGroup:类似文件夹的分层效果,可实现树状结构的分层效果,可折叠,可展开。

DisClosureGroup: 可折叠的分组,类似于List里的.listStyle(.sidebar)样式,是GroupBox中的子视图可折叠可展开样式。嵌套使用时候即可实现OutlineGroup分层结构效果,树状结构效果上个人感觉比OutlineGroup效果更好。

ControlGroup:类似于UIKit中的Segmented的样式。如果想改变样式,可以更改.controlgRgoupStyle()

6、overlay

在实现前后顺序的功能,布局上除了ZStack,我们还可以使用overlay
如系统计算器里按钮上的文字就可以使用overlay来实现。

Button(action: {
        }){
            Text("")
        }.frame(width: 50, height: 50).background(.red).cornerRadius(25)
        .overlay(){
            Image(systemName: "person")
        }
overlay

就很简单的实现了按钮上添加图片的功能。默认图片展示在按钮中心点上。
利用ZStack实现相同功能如下:

ZStack{
            Button(action: {
                
            }){
                Text("")
            }.frame(width: 50, height: 50).background(.red).cornerRadius(25)
            //.overlay(){
            Image(systemName: "person")
            //}
        }

而且区别就是.overlay是按钮的一个Modifier,而ZStack是一个容器。具体的还是要根据项目实际功能来选择哪种方式。

7、ViewBuilder

上面1中的三大容器类也是一个View,查看底层初始化构造代码:

@inlinable public init(alignment: Alignment = .center, @ViewBuilder content: () -> Content)

初始化函数里有个带修饰符ViewBuilder的闭包,查看ViewBuilder修饰符:

@available(iOS 13.0, macOS 10.15, tvOS 13.0, watchOS 6.0, *)
@resultBuilder public struct ViewBuilder {

    /// Builds an empty view from a block containing no statements.
    public static func buildBlock() -> EmptyView

    /// Passes a single view written as a child view through unmodified.
    ///
    /// An example of a single view written as a child view is
    /// `{ Text("Hello") }`.
    public static func buildBlock<Content>(_ content: Content) -> Content where Content : View
}

@resultBuilder的意思就是结果创建,也就是会返回一个结果值,此处会返回一个some View类型的View。
也就是可以理解为VStack{}大括号内的所有子视图会被打包成一个结果值返回给到VStack。所有子视图组成的组合,会被resultBuilder捕获。

VStack{
            Text("Hello Lcr")
            Image(systemName: "person")
            Button {
                print(type(of: self.body))
            } label: {
                Text("打印")
            }
        }

打印结果:

VStack<TupleView<(Text, Image, Button<Text>)>>

TupleView就是将所有子视图包装的View。

8、绝对位置、相对位置

position(x:,y:):绝对位置,设置视图的中心点在距离左上角(x,y)的位置。

//Positions the center of this view at the
//specified coordinates in its parent's coordinate space.
@inlinable public func position(x: CGFloat = 0, y: CGFloat = 0) -> some View

结合下面三段代码:

//第一段
Text("Hello Lcr")
            .padding()
            .font(.title)
            .background(.red)
            .position(x:100,y:100)
//第二段
Text("Hello Lcr")
            .padding()
            .font(.title)
            .position(x:100,y:100)
            .background(.red)
//第三段
Text("Hello Lcr")
            .padding()
            .font(.title)
            .background(.red)
            .position(x:100,y:100)
            .background(.green)

可以看出增加position后显示区域感觉变大了,原因是position会新建一个View作为Text的父视图,所以position之后的颜色设置的是position所返回的View的背景颜色。

offset():相对位置,相对position来说不会新建一个父视图,而是直接将中心点按offset所标大小移动。只是去改变显示的位置。
注意:从这里我们可以看出函数响应式代码的细节,要注意顺序的前后对结果的影响。

对于SwiftUI布局来说,还是需要多使用,多查找资料,多注意一些细节

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

推荐阅读更多精彩内容