SwiftUI

SwiftUI是什么?
管方定义:
SwiftUI is a modern way to declare user interfaces for any Apple platform.
SwiftUI是Apple全平台声明式构建UI的开发工具。

苹果开发SwiftUI的原因?
Api采用声明式UI,具有以下优点
->高内聚,可复用性强
->所有UI代码都在body统一位置调用,不需要关注过多的生命周期

struct CustomeView: View {
    struct CustomeView: View {
    @State var title: String = "Hello, World!"
    var body: some View {
        Text(self.title)
    }
}
        let label = UILabel()
        self.addSubview(label)
        label.text = "Hello, World!"
        label.frame = CGRect(x: 0, y: 0, width: 100, height: 100)

SwiftUI 开发
1,页面开发

1,元素组件:Button,Label,Image,List
2,布局组件:VStack垂直布局,HStack水平布局,ZStack重叠布局,padding设置间距,Spacer自动撑满多余间距,frame

2,绑定数据更新
@State:SwiftUI 将会把使用过 @State 修饰器的属性存储到一个特殊的内存区域,并且这个区域和 View struct 是隔离的. 当 @State 装饰过的属性发生了变化,SwiftUI 会根据新的属性值重新创建视图


struct Product: Identifiable {

    var id: String
}

struct ProductsView: View {
    let products: [Product] = [Product(id: "测试1")]

    @State private var showFavorited: Bool = false
    
    var body: some View {
        List {
            Button(
                action: {
                    
                    self.showFavorited = !self.showFavorited
                },
                label: { Text("Change filter") }
            )

            ForEach(products) { product in
                if !self.showFavorited {
                    Text(product.id)
                }
            }
        }
    }
}

@Binding :修饰器修饰后,属性变成了一个引用类型,

struct FilterView: View {
    @Binding var showFavorited: Bool

    var body: some View {
        Button(
            action: {
                self.showFavorited = !self.showFavorited
            },
            label: { Text("Change filter") }
        )
    }
}

struct ProductsBindView: View {
    let products: [Product] = [Product(id: "测试1")]
    
    @State private var showFavorited: Bool = false

    var body: some View {
        List {
            FilterView(showFavorited: $showFavorited)

            ForEach(products) { product in
                if !self.showFavorited {
                    Text(product.id)
                }
            }
        }
    }
}

@ObservableObject和@Published:@ObservedObject 的用处和 @State 非常相似,从名字看来它是来修饰一个对象的,这个对象可以给多个独立的 View 使用。如果你用 @ObservedObject 来修饰一个对象,那么那个对象必须要实现 ObservableObject 协议,然后用 @Published 修饰对象里属性,表示这个属性是需要被 SwiftUI 监听的

final class PodcastPlayer: ObservableObject {
    @Published private(set) var isPlaying: Bool = false

    func play() {
        isPlaying = true
    }

    func pause() {
        isPlaying = false
    }
}

struct EpisodesView: View {
    @ObservedObject var player: PodcastPlayer
    
    var body: some View {
        List {
            Button(
                action: {
                    if self.player.isPlaying {
                        self.player.pause()
                    } else {
                        self.player.play()
                    }
            }, label: {
                    Text(player.isPlaying ? "Pause": "Play")
                }
            )
        }
    }
}

struct EpisodesView_Previews: PreviewProvider {
    static var previews: some View {
        let player = PodcastPlayer()
        EpisodesView(player: player)
    }
}

@StateObject和@ObservedObject:这两个类似都是用于修饰一个对象,区别是声明对象的所有权不同:创建对象的视图必须使用 @StateObject,对于某些复杂场景,多个View公用一个对象处理数据,这时候只有创建的视图使用@StateObject,其他的使用ObservedObject

struct EpisodesViewState: View {
    @StateObject var player: PodcastPlayer = PodcastPlayer()
    
    var body: some View {
        List {
            Button(
                action: {
                    if self.player.isPlaying {
                        self.player.pause()
                    } else {
                        self.player.play()
                    }
            }, label: {
                    Text(player.isPlaying ? "Pause": "Play")
                }
            )
        }
    }
}

@EnvironmentObject:@StateObject和@ObservedObject类似,区别是他用做修饰一个全局的对象

@main
struct swiftUIStateDemoApp: App {
    let persistenceController = PersistenceController.shared

    var body: some Scene {
        let player = PodcastPlayer()
        WindowGroup {
            EpisodesViewEnvironmentObject()
                .environmentObject(player)
        }
    }
}

struct EpisodesViewEnvironmentObject: View {
    @EnvironmentObject var player: PodcastPlayer
    
    var body: some View {
        List {
            Button(
                action: {
                    if self.player.isPlaying {
                        self.player.pause()
                    } else {
                        self.player.play()
                    }
            }, label: {
                    Text(player.isPlaying ? "Pause": "Play")
                }
            )
        }
    }
}

@Environment :用作读取系统的全局变量,也可以读取keyvalue,自定义的全局变量

struct OtherView: View {
    @Environment(\.presentationMode) private var presentationMode: Binding<PresentationMode>
    var body: some View {
        VStack {
            Button("退出登陆") {
                print("loginou")
                self.presentationMode.wrappedValue.dismiss()
            }.foregroundColor(.black)
        }
        .frame(minWidth: 0,maxWidth: .infinity,minHeight: 0,maxHeight: .infinity)
        .background(.red)
    }
}
struct RootPresentationModeKey: EnvironmentKey {
    static let defaultValue: Binding<RootPresentationMode> = .constant(RootPresentationMode())
}

extension EnvironmentValues {
    var rootPresentationMode: Binding<RootPresentationMode> {
        get { return self[RootPresentationModeKey.self] }
        set { self[RootPresentationModeKey.self] = newValue }
    }
}

typealias RootPresentationMode = Bool

extension RootPresentationMode {
    
    public mutating func dismiss() {
        self.toggle()
    }
}

struct WelcomeView: View {
    @State private var isActive : Bool = false
    var body: some View {
        return NavigationView {
            mainView()
        }
        .navigationViewStyle(StackNavigationViewStyle())
        .environment(\.rootPresentationMode, self.$isActive)
    }
}
struct OtherView: View {
    @Environment(\.rootPresentationMode) private var rootPresentationMode: Binding<RootPresentationMode>
    var body: some View {
        VStack {
            Button("退出登陆") {
                print("loginou")
                self.rootPresentationMode.wrappedValue.dismiss()
            }.foregroundColor(.black)
        }
        .frame(minWidth: 0,maxWidth: .infinity,minHeight: 0,maxHeight: .infinity)
        .background(.red)
    }
}

Swift和SwiftUI桥接
通过UIViewRepresentable协议桥接UIKit控件

import SwiftUI
import WebKit

struct ContentView : UIViewRepresentable {
    
    func makeUIView(context: UIViewRepresentableContext<ContentView>) -> WKWebView {
        return WKWebView()
    }
    
    func updateUIView(_ uiView: WKWebView, context: UIViewRepresentableContext<ContentView>) {
        let request = URLRequest(url:URL(string: "https://apple.com")!)
        uiView.load(request)
    }
}

通过UIHostingController调用SwiftUI控件

let contentView = ContentView()
let tempVc = UIHostingController(rootView: contentView)

ViewModifier控件
作用:封装控件

struct ContentView : View {
    
    var body: some View {
        VStack(alignment: .center, spacing: 40){
            Image("avarta1").modifier(myImageStyle())
            Image("avarta2").modifier(myImageStyle())
            Image("avarta3").modifier(myImageStyle())
        }
        .padding()
    }
}

struct myImageStyle: ViewModifier {
    func body(content: Content) -> some View {
        content
            .frame(width: 200, height: 200, alignment: .leading)
            .cornerRadius(100)
            .clipped()
            .saturation(0.0)
    }
}

缺点
一些常用功能不支持:
->富文本不支持点击效果,解决方案:侨接UIKit富文本解决问题
->ScrollView不能监听滚动数值,解决方案:桥接UIKit控件,强行拿到ScrollView,通过KVO监听
->不支持播放视频,解决方案:桥接swift原生控件
->NavagationView 不支持侧滑手势:解决方案:还是需要桥接UIKit的UINavagation控件,但这样做感觉UIKit侵入太重了。。。
->TextField在iOS16才支持类似UITextView的功能,iOS15支持自动聚焦功能
->不支持WebView,需要桥接UIKit控件
Demo地址
https://github.com/riceForChina/SwiftUIDemo.git
参考
https://www.modb.pro/db/170297
https://juejin.cn/post/6844903924084768776
https://blog.csdn.net/Forever_wj/article/details/121981007
https://github.com/fzhlee/SwiftUI-Guide#-%E7%AC%AC20%E8%8A%82image-web-

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

推荐阅读更多精彩内容

  • 学习文章 文集:Hacking with iOS: SwiftUI Edition[https://www.jia...
    xmb阅读 4,460评论 3 14
  • SwiftUI要求 iOS13.0+ 快捷键 control + option + 点击:出现属性编辑器 comm...
    余青松阅读 6,211评论 1 11
  • TextField TextField就相当于UIKit中的UITextField的,单行文本输入框。比如登录用户...
    摇滚马农阅读 7,813评论 0 9
  • 什么是 SwiftUI?[1][#fn1] 官方的定义非常明确: SwiftUI is a user interf...
    yyggzc521阅读 1,826评论 0 2
  • 目前swiftUI最低支持iOS13,苹果还在不断升级完善,目前还不成熟,因为现在widget必须用swiftUI...
    IOSMan阅读 4,202评论 3 16