Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

04-27 1469阅读 0评论

Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

概览

从 Xcode 15 开始,苹果推出了新的 #Preview 宏预览机制,它无论从语法还是灵活性上都远远超过之前的预览方式。#Preview 不但可以实时预览 SwiftUI 视图,而且对 UIKit 的界面预览也是信手拈来。

Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

想学习新 #Preview 预览的一些超实用调试小妙招吗?那就“如意如意”随小伙伴们的心意吧!

在本篇博文中,您将学到如下内容:

  • 概览
  • 1. Xcode 15.0 新预览机制简介
  • 2. #Preview 让状态初始化如此轻松!
  • 3. 为什么 #Preview 中不能直接嵌入可变状态?
  • 4 #Preview + @Observable 宏构造可变 @Binding 实参
  • 总结

    相信学完本课后,大家对于 Xcode 15+ 预览的使用以及 SwiftUI 界面调试会更加的轻车熟路!

    那还等什么呢?让我们马上开始吧!Let‘s preview!!!😉


    本文对应的视频课在此,欢迎恣意观赏 😃

    Xcode 15.0新 #Preview 预览让调试悠然自得

    1. Xcode 15.0 新预览机制简介

    从 Xcode 15 开始,苹果借助于 Swift 5.9 宏(Macro)的“东风”,也为我们带来了全新的 #Preview 预览机制。你猜的没错,它其实就是一个宏:

    Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

    如上所示:我们将 #Preview 宏定义展开为了其原始代码的实现,大家可以清楚的看到 #Preview 宏背地里到底做了些神马。

    在 Xcode 15 之前,小伙伴们需要使用遵循 PreviewProvider 协议的 Previews 结构来帮助我们预览指定的 SwiftUI 视图:

    struct LaunchView_Previews: PreviewProvider {
        static var previews: some View {
            LaunchView()
                .environmentObject(Model())
        }
    }
    

    而现在,只需一个 #Preview 即可搞定所有,岂不呜呼快哉:

    #Preview {
        LaunchView()
            .environmentObject(Model())
    }
    

    为了方便起见,我们还可以非常 nice 的将多个定制的 #Preview 预览内容混合在一起显示:

    Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

    如上所示,为了便于观察我们在 #Preview 中指定了不同预览名称以及预览设备的方向和明暗主题等特性,简直小菜一碟。

    2. #Preview 让状态初始化如此轻松!

    “理想很骨感,现实却很残酷”。

    在实际开发中,不可能所有视图都如此简单。在现实的 App 中视图多半都会与模型(数据)相绑定,这意味着我们在预览它们之前需要创建对应的数据,否则预览就不会达到预期效果,显示将是一片“空空如也”。

    比如在 SwiftUI 里我们有一个分类选择视图(V2_ChallengeClassSelectView),所有内置(Built in)的分类都是从数据库中读取的,但前提是我们在数据库中已经初始化了这些分类,这是通过调用如下方法来完成的:

    V2_ChallengeClassification.initializeData()
    

    所以,我们可能会写出下面的代码以期待 #Preview 预览可以正常工作:

    @available(iOS 17.0, *)
    #Preview {
        V2_ChallengeClassSelectView(selecting: .constant(nil))
            .onAppear {
                try? V2_ChallengeClassification.initializeData()
            }
    }
    

    不过可惜的是,以上实现无法得偿所愿。原因是我们 V2_ChallengeClassSelectView 视图中的分类数据必须在其 body 显示之前就准备就绪:

    @available(iOS 17.0, *)
    struct V2_ChallengeClassSelectView: View {
        
        @Binding var selecting: V2_ChallengeClassification?
        
        let builtInClasses = try? V2_ChallengeClassification.allBuiltInClassifications()
    }
    

    对于这种情况,#Preview 宏有一个非常简单的解决方案:我们只需在预览内容之前直接调用初始化代码即可:

    @available(iOS 17.0, *)
    #Preview {
        try? V2_ChallengeClassification.initializeData()
        
        return V2_ChallengeClassSelectView(selecting: .constant(nil))
    }
    

    运行可以看到,我们已经能够在预览中正确显示初始化之后的所有内置分类了:

    Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

    3. 为什么 #Preview 中不能直接嵌入可变状态?

    大家可能已经发现了,上面示例中的 V2_ChallengeClassSelectView 视图包含一个 selecting 绑定状态:

    struct V2_ChallengeClassSelectView: View {   
        @Binding var selecting: V2_ChallengeClassification?
    }
    

    但在我们的预览中,为了“偷懒”实际向其传入的是一个绑定常量:

    V2_ChallengeClassSelectView(selecting: .constant(nil))
    

    这样做的后果是:我们无法在预览中改变 selecting 属性的值,也就无法观察到视图中选择所产生的变化了。

    小伙伴们可能会觉得,下面的实现可以帮我们摆脱这一问题:

    @available(iOS 17.0, *)
    #Preview {
        
        @State var selecting: V2_ChallengeClassification?
        
        try? V2_ChallengeClassification.initializeData()
        
        return V2_ChallengeClassSelectView(selecting: $selecting)
           
    }
    

    遗憾的是这样做“然并卵”,毫无用处:

    Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

    其原因是:与 Xcode 15 之前的旧预览机制类似,嵌入在预览结构中的简单状态实际上是无法被改变的,即使它被 @State 等(可变)限定符所修饰时也是如此。

    那么我们如何解决呢?

    答案很简单:将可变状态放到 #Preview 外面去!

    4 #Preview + @Observable 宏构造可变 @Binding 实参

    从 Xcode 15 (Swift 5.9)开始,苹果推出了新的 @Observable 宏帮我们便捷的创建可观察对象。


    更多关于 @Observable 宏以及 Observation 框架的详细介绍,小伙伴们可以移步到下面的博文中进一步观赏:

    • Swift 5.9 与 SwiftUI 5.0 中新 Observation 框架应用之深入浅出
    • Swift 5.9 新 @Observable 对象在 SwiftUI 使用中的陷阱与解决

      简单来说,我们可以在 #Preview 之外利用 @Observable 宏包裹我们的可变状态,从而可以将其通过绑定传入到对应的视图中去:

      @available(iOS 17.0, *)
      @Observable
      class PreviewModel {
          var selecting: V2_ChallengeClassification?
      }
      @available(iOS 17.0, *)
      #Preview {
          @State var model = PreviewModel()
          
          try? V2_ChallengeClassification.initializeData()
          
          return V2_ChallengeClassSelectView(selecting: $model.selecting)
      }
      

      注意,在上面的代码示例中我们实际向 V2_ChallengeClassSelectView 视图传递的绑定是 @Observable 可观察对象 model 中的属性。虽然 model 作为 @State 放在了预览内部,不过由于它是一个可观察对象,所以它仍然可以变化自如。

      编译运行修改后的代码,我们现在可以在 #Preview 预览界面中顺畅自如的测试 selecting 分类属性改变时的显示逻辑了:

      Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

      至此,我们通过上面几个小“栗子”对 Xcode 15 中新的 #Preview 预览机制又有了更深刻的领悟,小伙伴们还不赶快给自己点一个大大的赞吧!👍🏻


      想要系统学习 Swift 语言的小伙伴们,千万不要错过我的《Swift 语言开发精讲》专栏哦:

      Xcode 15.0 新 #Preview 预览让 SwiftUI 界面调试更加悠然自得

      • Swift 语言开发精讲

        总结

        在本篇博文中,我们介绍了 Xcode 15+ 中新的 #Preview 预览机制,并讨论了如何利用 #Preview + @Observable 宏让 SwiftUI 界面调试更加“如虎添翼”。

        感谢观赏,再会啦!😎


免责声明
本网站所收集的部分公开资料来源于AI生成和互联网,转载的目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。
文章版权声明:除非注明,否则均为主机测评原创文章,转载或复制请以超链接形式并注明出处。

发表评论

快捷回复: 表情:
评论列表 (暂无评论,1469人围观)

还没有评论,来说两句吧...

目录[+]