货拉拉用户端SwiftUI踩坑之旅

1. 前言

在货拉拉用户端适配灵动岛开发过程中,切身感受到了SwiftUI编写界面带来的便捷性,像声明式UI能减少许多初始化代码、更简便更灵活的Flex布局,以及所有iOSer都梦寐以求的Hot Reload功能,都是能大幅度提升编码体验与效率的功能。基于这些优点,对SwiftUI落地进行探索,在项目内找一个页面,使用SwiftUI编写接入,尝试新技术对编码效率的提升感受,以及体验SwiftUI与OC项目的兼容程度,为后续技术选型提供实践经验。本文记录接入过程中一些"坑"与经验,在此与大家分享下这趟踩坑之旅。

另外想了解货拉拉用户iOS端接入灵动岛实践经验的,可以移步这篇文章 货拉拉用户 iOS 端灵动岛实践总结

2. 接入实践

接入SwiftUI条件

  • SwiftUI文件支持的最低系统是iOS13。如果工程最低支持版本在iOS13以下但又想接入,可以使用系统版本判断@available(iOS 13.0, *),在iOS13以下使用常规的UIKit视图,iOS13使用SwiftUI视图。

2.1 创建swiftUI文件

New File -> User Interface -> SwiftUI View创建一个SwiftUI视图文件,如果是纯OC项目,需要创建Swift的桥接文件,如果工程是OC/Swift混编,则不需要再次创建。

也可以创建一个普通的Swift文件,导入头文件 SwiftUI,手动编写SwiftUI视图。

swift 复制代码
import SwiftUI

public struct SwiftUIView: View {
    public var body: some View {
        Text("SwiftUI")
    }
}
struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        SwiftUIView()
    }
}
  • PreviewProvider为实时预览功能,可以根据项目实际情况选择使不使用。

2.2 OC调用SwiftUI视图

OC无法直接使用SwiftUI视图,但是可以通过UIHostingController控制器包裹SwiftUI视图,然后通过push或present等方式打开ViewController,此时打开ViewController中的view则为SwiftUI视图。

  • 由于OC与Swift的访问控制,所以在SwiftUI文件和OC中间需要增加一道Swift的桥接。
swift 复制代码
// 桥接
@objc
public class HDSwiftUIBridg: NSObject {
    @objc public func makeStudyView() -> UIViewController { 
        let vc = UIHostingController(rootView: HDSwiftUIView())
        return vc
    }
}
// SwiftUI
struct HDSwiftUIView:View {
    var body: some View {
        Text("SwiftUI")
    }
}

// OC
@ implementation HDHomeVC
    - (void)buttonClick {
       UIViewController *vc = [[HDSwiftUIBridg new] makeStudyView];
       [self.navigationController pushViewController:vc animated:YES];
    }
@end

2.3 OC模型传递

先在SwiftUI视图中增加属性变量,SwiftUI中就可以直接访问模型属性。属性变量写完以后,在初始化SwiftUI类时,构建方法就会自动增加相应的入参。

swift 复制代码
// SwiftUI
struct HDSwiftUIView:View {

    var model: HDTimeModel
    
    var body: some View {
        Text(model.name ?? "")
    }
}

// 桥接
@objc
public class HDSwiftUIBridg: NSObject {
    @objc public func makeStudyView(model:HDTimeModel) -> UIViewController { 
        let vc = UIHostingController(rootView: HDSwiftUIView(model: model))
        return vc
    }
}

// OC
@ implementation HDHomeVC 
    - (void)buttonClick {
       HDTimeModel *model = [HDTimeModel new];
       UIViewController *vc = [[HDSwiftUIBridg new] makeStudyViewWithModel:model];
       [self.navigationController pushViewController:vc animated:YES];
    }
@end

2.4 使用UIKit视图

由于存在业务场景特殊性的限制,现阶段可能存在需要使用UIKit的视图,比如使用lottie,或者视图与业务已经深绑定,使用SwiftUI重写周期长,这些问题都会影响选用SwiftUI的意愿。

官方早也帮我们想好解决办法。只需要用到 UIViewRepresentable 协议,通过这个协议就可以将UIKit的视图桥接到SwiftUI上。

需要实现两个协议方法,分别是初始化和与更新视图。示例中在创建视图之前已经获取到了数据,并且数据不会动态更改,则在初始化视图的时候就可以直接将数据传入,而不是通过 updateUIView 。

协议内还有许多实用方法,感兴趣的可以自行探索。

scss 复制代码
protocol UIViewRepresentable
@MainActor func makeUIView(context: Self.Context) -> Self.UIViewType
@MainActor func updateUIView(_ uiView: Self.UIViewType, context: Self.Context)
....
@end

示例:

swift 复制代码
// View 桥接
struct HDBridgeViewWrapper: UIViewRepresentable {
    var model:HDUIDataModel
    func makeUIView(context: Context) -> some UIView {
        let view = HDOCViewCell()
        view.reload(model)
        return view
    }
    func updateUIView(_ uiView: UIViewType, context: Context) {
    }
}

// SwiftUI
@available(iOS 13.0, *)
public struct SwiftUIView: View {
    public var body: some View {
        List {
          ForEach(model.list, id: .self) { obj in
               HDBridgeViewWrapper(model: obj)
                    .frame(maxWidth: .infinity, minHeight: 58 ,maxHeight: 58)
                    .listRowBackground(Color.clear)
                    .listRowInsets(.none)
          }
        }
    }
}

// UIKit
@implementation HDOCViewCell
- (instancetype)initWithFrame:(CGRect)frame {
    self = [super initWithFrame:frame];
    if (self) {
    
    }
    return self;
}
@end

2.5 点击事件传递

虽然SwiftUI更推荐响应式编程,但目前iOS主流的编码模式还是面向对象式编程,所以视图虽然是使用SwfitUI编写,动作事件还是不可避免存在大量与原逻辑的交互,而想从SwiftUI视图的点击透传到OC&Swift的逻辑层上,通过iOSer最熟悉的Block即可轻松实现。

实现方式与数据传参一致,在SwiftUI属性中增加闭包属性,初始化方法中同样也会自动增加入参,在桥接文件与OC交互的方法内手动增加闭包入参,就可以实现点击事件的回传。

less 复制代码
@available(iOS 13.0, *)
@objc public class HDSwiftUIBridg: NSObject {
    @objc public func makeStudyView(callBack: @escaping() -> Void) -> UIViewController {
        let vc = UIHostingController(rootView: SwiftUIView(model: model, closeCallBack: {
            callBack()
        }))
        return vc
    }
}

@available(iOS 13.0, *)
public struct SwiftUIView: View {
    
    var model: HLLUCardInfoModel
    var closeCallBack:(()->Void)?
    
    public var body: some View {
        VStack {
            Button(action: {
               closeCallBack?()
            }) {
               Text(model.shopName ?? "")
            }
        }
    }
}

implementation HDHomeVC
    - (void)buttonClick {
       HDDataModel *model = [HDDataModel new];
       UIViewController *vc = [[HDSwiftUIBridg new] makeStudyViewWithModel:model callBack: {
           //原逻辑
       }];
       [self.navigationController pushViewController:vc animated:YES];
    }
@end

完成上述代码后,SwiftUI编写的视图已经能够在项目中正确展示出来了。现在可以开始着手在SwiftUI上编写UI代码,尝试下编写UI最便利的功能------HotReload。相信在语法熟练后,一定能带来比现在更好的编码体验。

3. 踩坑记录与解决

在实践中遇到了一些比较有意思的问题,换一种思路就很好的解决。在此记录一下,与大家分享,如果大家还有更优雅的解决办法,欢迎讨论交流。

  • 无法使用PreviewProvider实时预览功能

    这个问题出现原因不明,在新建的空白工程,能正常运行预览功能,在项目内就无法使用,也没有具体的报错原因,经过多种尝试,最终得到能临时解决的方法。

  1. 如果你的项目也是使用CocoaPods管理的单仓多组件工程,先把SwiftUI移到主目录下,与AppDelegate同级。再次尝试运行预览功能,如果不成功则进行第2步设置。

  2. 导航栏 -> Editor -> Canvas -> Automatically Refresh Canvas取消勾选。再次尝试运行预览功能,我们是在这一步成功运行了实时预览功能,如果还是不行可以进行下面的一下尝试。

  3. 如果是M1电脑,可能会出现 xxx.frameworks not supported x86_64 ,这时候可以在预览的界面左下角,进行模拟器切换(arm64&x86_64都分别尝试,我们项目是使用x86_64的模拟器运行起来的)。

  4. 也可以使用真机进行预览功能,如果在真机上出现了Xcode Previews这个APP,但是项目无法运行起来,在这个时候可以切回使用模拟器再次尝试第3步。

    需要注意的是只需要把PreviewProvider写在主目录(壳工程)下,而SwiftUI视图要放在子Pod内,因为主工程能索引子组件的Swift文件,如果把视图也放在主目录下,子组件内就无法访问到View。

  • SwiftUI主视图的frame与背景色设置不生效

可以在桥接文件内,获取到视图后设置

swift 复制代码
@objc
public class HDSwiftUIBridg: NSObject {
    @objc public func makeStudyView(model:HDTimeModel) -> UIViewController { 
        let vc = UIHostingController(rootView: HDSwiftUIView(model: model))
        vc.view.frame = .init(x: 0, y: 0, width: 300, height: 300)
        vc.view.backgroundColor = .white
        return vc
    }
}
  • SwiftUI视图支持不足

    • 像Scroll的禁止滑动、list隐藏分割线等,部分需要到iOS15+系统才可以支持,这些属性或者功能随着SwiftUI的迭代会有解决办法,但在低版本实现起来非常复杂,如果想简单实现这些功能可以使用UIKit的视图,通过桥接的方式供SwiftUI使用。
  • 圆角与边框同时设置圆角无法处边框消失

    css 复制代码
      struct SwiftUIView: View {
          var body: some View {
              Text("Hello, SwiftUI!")
                          .padding()
                          .border(Color.blue, width: 1)
                          .cornerRadius(40)
          }
      }

可以使用overlay,需要注意的是这个属性最低支持的系统版本是iOS15。

less 复制代码
struct SwiftUIView: View {
    var body: some View {
        Text("Hello, SwiftUI!")
                    .padding()
                    .overlay(
                        RoundedRectangle(cornerRadius: 40, style: .continuous)
                            .stroke(.blue,lineWidth: 1.0)
                    )
    }
}
  • 容易遗漏桥接文件强引用

    • 桥接文件是一个NSObject类,通过桥接类获取持有SwiftUI的ViewController,容易遗漏在当前类强持有桥接文件,会导致传入的block释放,导致回调无响应。
  • 只使用SwiftUI视图

    • 因为桥接类UIHostingController是UIViewController,正常是获取到viewController进行Push,如果只想用view可以通过viewController.view获取视图。
    swift 复制代码
      @objc
      public class HDSwiftUIBridg: NSObject {
          @objc public func makeStudyView(model:HDTimeModel) -> UIView { 
              let vc = UIHostingController(rootView: HDSwiftUIView(model: model))
              return vc.view
          }
      }
  • 数据模型双向绑定不支持OC模型

    • SwifUI采用声明式的布局方式,语言设计上更契合响应式编程,许多Demo都是用模型视图双向绑定的来实现属于与数据的更新,但在使用到OC模型时,双向绑定的关键字不支持在OC上编写,导致无法让视图与模型绑定。
    • 可以参考示例,直接将模型传入,按面向对象编程设计进行编码。

4. 总结

在编码上OC/Swift/SwiftUI混编总体非常丝滑,接入SwiftUI的编码体验与当年OC接入Swift非常类似,通过系统提供的API就能很轻易实现SwiftUI的接入。在业务层上,由于需要考虑最低系统版本的支持、业务的复杂性、UI的高还原性等业务问题,导致SwiftU在业务开发体验上暂时还是不如OC&Swift开发。但是随着官方的力推、系统版本的更新,SwiftU的缺点会逐步解决,使用SwiftUI的编码体验与效率也会直线上升,总的来说现有项目接入SwiftUI从技术层面上是没问题的,在业务上可以考虑渐进的去尝试,相信能给大家带来不一样的编码体验。

我们希望通过分享开发过程中遇到的问题和解决方案,可以帮助到更多的人。如果你有任何问题或者想法,欢迎在评论区留言。期待我们在技术的道路上再次相遇。

相关推荐
CV大师杨某3 小时前
关于H5复制ios没有效果
ios
Swift社区11 小时前
LeetCode - #182 Swift 实现找出重复的电子邮件
算法·leetcode·swift
Batac_蝠猫13 小时前
iOS - runtime总结
ios
大邳草民1 天前
iOS 概述
笔记·ios
打工人你好1 天前
iOS 逆向学习 - iOS Application Publishing:应用发布
学习·ios·cocoa
Batac_蝠猫1 天前
iOS - Objective-C语言的动态性
ios·objective-c·xcode
刘小哈哈哈1 天前
iOS 解决两个tableView.嵌套滚动手势冲突
macos·ios·cocoa
YJlio1 天前
苹果手机(IOS系统)出现安全延迟进行中如何关闭?
ios
Batac_蝠猫1 天前
iOS - 关联对象
ios·cocoa·xcode