SwiftUI - 与UIKit集成

说到与UIKit的集成不免会觉得有些鸡肋,因为现在很难做到只支持iOS13,不过到iOS14时,这种集成就变得必不可少了吧,在此先预热一下咯 ~ 先想想使用场景:

  1. 在现有基于UIKit的App中使用SwiftUI - 这应该是最常见的一种方式;
  2. 在SwiftUI中使用UIKit - 新写的页面是用了SwiftUI,但不免会跳转到原先的页面,或是在SwiftUI要使用UIKit写的一些View,毕竟重写原先所有的view成本有些大,需要逐步替换。

UIKit中使用SwiftUI

还记得我们在SwiftUI 初识 中提到的UIHostingController,它是SwiftUI的容器,同时也是UIViewController的子类。集成方式也就显而易见咯。

@IBAction func showSwiftUIByCode(_ sender: Any) {
  let vc = UIHostingController(rootView: ContentView())
  self.show(vc, sender: nil)
}

若使用的是Segue

@IBSegueAction func openSwiftUI(_ coder: NSCoder) -> UIViewController? {
  return UIHostingController(coder: coder, rootView: ContentView())
}

在SwiftUI中使用UIKit

首先,我们来看看,如何从SwiftUI跳转到一个UIViewController。先写一个最简单的UIViewController:

class ViewController: UIViewController {
  override func viewDidLoad() {
    super.viewDidLoad()
    view.backgroundColor = UIColor.red
  }
}

在SwiftUI中没有提供类似self.show(vc, sender: nil)的方式,需要通过一个wrapper把ViewController包装一下,可以认为这就是一个适配器。SwiftUI提供了UIViewControllerRepresentable协议,来承担适配的功能,写一个自己的ViewControllerRepresentation实现这个协议。

import SwiftUI

struct ViewControllerRepresentation: UIViewControllerRepresentable {
  typealias UIViewControllerType = ViewController
    
  func makeUIViewController(context: UIViewControllerRepresentableContext<ViewControllerRepresentation>) -> ViewControllerRepresentation.UIViewControllerType {
    return UIStoryboard(name: "Storyboard", bundle: nil).instantiateViewController(identifier: "ViewController") as! ViewController
}
    
  func updateUIViewController(_ uiViewController: ViewControllerRepresentation.UIViewControllerType, context: UIViewControllerRepresentableContext<ViewControllerRepresentation>) {}
}

其中的typealias UIViewControllerType = ViewController指明了要包裹的UIViewController的具体类型,并实现两个方法,一个是make,一个是update。其中make顾名思义,就是如何生成我们的ViewController,大家可以用自己喜欢的方式初始化它。update暂时放空,它的作用是如何更新这个VC。我们会在下个示例中来看看它的用法。
接下来,我们要在SwiftUI中来调用我们这里定义的ViewControllerRepresentation

var body: some View {
  NavigationView {
    VStack {
      HStack {
        TargetView(showAlert: $showAlert, rTarget: rTarget, gTarget: gTarget, bTarget: bTarget)
        MatchingView(rGuess: $rGuess, gGuess: $gGuess, bGuess: $bGuess, counter: $counter.counter)
      }
      Button(action: { self.showAlert = true }) { Text("Hit me") }.alert(isPresented: $showAlert) { () -> Alert in
        Alert(title: Text("Your Score"),message: Text(String(computeScore())))
      }.frame(width: nil, height: 35, alignment: .center)
      SlideView(value: $rGuess, textColor: .red)
      SlideView(value: $gGuess, textColor: .green)
      SlideView(value: $bGuess, textColor: .blue)
      NavigationLink(destination: ViewControllerRepresentation()) {
        Text("Play BullsEye").frame(width: nil, height: 31, alignment: .center)
      }.padding(.bottom)
    }
  }
}

在底部我们放了一个NavigationLink,它就是在NavigationView中跳转至其他页面的一个链接容器,指明他的destination就是我们刚刚定义好的ViewControllerRepresentation,run一下,大功告成。

此时,不免会想,如何传递参数呢?这个UIViewController如何将一下返回值回传给SwiftUI呢?我们用下一个例子来说明,这个例子中是在SwiftUI中放置一个UIKit定义的UIView,先看code:

import SwiftUI
import UIKit

struct ColorUISlider: UIViewRepresentable {
  var color: UIColor
  @Binding var value: Double
    
  typealias UIViewType = UISlider
    
  func makeUIView(context: UIViewRepresentableContext<ColorUISlider>) -> UISlider {
    let slider = UISlider(frame: .zero)
    slider.thumbTintColor = color
    slider.value = Float(value)
    slider.addTarget(context.coordinator, action: #selector(Coordinator.valueChanged(_:)), for: .valueChanged)
    return slider
  }
    
  func updateUIView(_ uiView: UISlider, context: UIViewRepresentableContext<ColorUISlider>) {
    uiView.value = Float(value)
  }
    
  class Coordinator: NSObject {
    var value: Binding<Double>
    init(value: Binding<Double>) {
      self.value = value
    }
        
    @objc func valueChanged(_ sender: UISlider) {
      self.value.wrappedValue = Double(sender.value)
    }
  }
    
  func makeCoordinator() -> ColorUISlider.Coordinator {
    return Coordinator(value: $value)
  }
}

code有点多,我们一点点看。首先想用再SwiftUI中使用UIView,同样需要一个适配器,在UIViewController时,用了UIViewControllerRepresentable,UIView提供了同样的协议UIViewRepresentable,需要的方法也如出一辙:

typealias UIViewType = UISlider
func makeUIView(...)
func makeUIView(...)

typealias UIViewType = UISlider指定了需要包装的UIView的类型,make方法要求返回UIView的实例,这里无需指定该view的frame或约束。

还记得我们之前留下的问题么?如何从SwiftUI中传值给UIView(UIViewController是相同的),直接给这个ColorUISlider添加属性,通过struct的默认构造函数,便可以将参数传入:

ColorUISlider(color: .red, value: .constant(0.5))

那么这个ColorUISlider如何影响其父view呢?我们看到了这个@Binding参数@Binding var value: Double(一个引用类型,如果不记得了可以查阅之前的一篇文章Data Binding),但Binding是没有办法在UIView中直接使用的,还需另一个类,Coordinator,如何使用呢?

  1. 添加内部类:
class Coordinator: NSObject {
  var value: Binding<Double>
  init(value: Binding<Double>) {
    self.value = value
  }
        
  @objc func valueChanged(_ sender: UISlider) {
    self.value.wrappedValue = Double(sender.value)
  }
}
  1. 实现UIViewRepresentable的另一个方法makeCoordinator:
func makeCoordinator() -> ColorUISlider.Coordinator {
  return Coordinator(value: $value)
}

当我们初始化UISlider的时候,给它添加了一个行为:

slider.addTarget(context.coordinator, action: #selector(Coordinator.valueChanged(_:)), for: .valueChanged)

在我们滑动滑块的时候,会触发:

@objc func valueChanged(_ sender: UISlider) {
  self.value.wrappedValue = Double(sender.value)
}

从而使Binding起作用,影响外面传入的value,进而作用于父View。

到此为止,我们集成SwiftUI的预热已经结束,还有很多细节没有涉及,比如context。毕竟是预热嘛,用到的时候再来重温咯~

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

推荐阅读更多精彩内容