SwiftUI 6.0(Xcode 16)全新 @Entry 和 @Previewable 宏让开发妙趣横生

news2025/1/12 0:51:42

在这里插入图片描述

概览

如火如荼的 WWDC 2024 已进入第五天,苹果开发平台中众多海量新功能都争先恐后的喷薄欲出。

在这里插入图片描述

在这里就让我们从中挑两个轻松有趣的新功能展示给小伙伴们吧:它们分别是 全新的 @Entry 和 @Previewable 宏。

在这里插入图片描述

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

  • 概览
  • 1. 用 @Entry 宏简化环境变量定义
  • 2. @Previewable 让 Xcode 预览调试安闲自得
  • 总结

读完本篇后,相信小伙伴们一定会对全新的 @Entry 和 @Previewable 宏相见恨晚!

那还等什么呢?马上和大熊猫侯佩一起开始 WWDC24 大冒险吧!

Let’s go!!!😉


1. 用 @Entry 宏简化环境变量定义

在 SwiftUI 6.0(iOS 18)之前,要想自己创建自定义环境变量需要以下 3 步。

首先,我们需要创建环境变量类型:

enum SuperPower: CustomStringConvertible {
    case timeStop
    case invisibility
    case predictTheFuture
    case immortal
    case teleportation
    
    var description: String {
        switch self {
        case .immortal:
            "永生"
        case .invisibility:
            "隐身"
        case .predictTheFuture:
            "预知未来"
        case .timeStop:
            "时间停止"
        case .teleportation:
            "瞬间移动"
        }
    }
}

接着,我们需要创建环境变量对应的键(EnvironmentKey):

struct HideSuperPower: EnvironmentKey {
    static var defaultValue: SuperPower = .immortal
}

最后,我们还需要扩展 EnvironmentValues 以便插入我们的环境变量:

extension EnvironmentValues {
    var hideSuperPower: SuperPower {
        get { self[HideSuperPower.self] }
        set { self[HideSuperPower.self] = newValue }
    }
}

为 SwiftUI 增加环境变量这点小事都要如此地大费周章,这不禁让我们这些秃头码农们唏嘘不已。

好消息来了!从 SwiftUI 6.0 开始仅用全新的 @Entry 宏我们即能蜻蜓点水似得创建自定环境变量了。

在这里插入图片描述

有了 @Entry 宏,之前那几坨代码现在可以如此这般简化了:

extension EnvironmentValues {
    @Entry var hideSuperPower: SuperPower = .immortal
}

是不是养眼了不少?

但是不管如何,使用 hideSuperPower 环境变量的方式还和以前是一毛一样滴:

struct ContentView: View {
    
    @Environment(\.hideSuperPower) var power
    
    var body: some View {
        NavigationStack {
            VStack {
                Text("当前超能力:\n\(Text("#\(power)#").foregroundStyle(.red.gradient))")
                    .font(.system(size: 55, weight: .heavy))
                    .foregroundStyle(.gray)
            }
            .navigationTitle("超能力大冒险")
            .toolbar {
                Text("大熊猫侯佩 @ \(Text("CSDN").foregroundStyle(.red))")
                    .font(.headline.weight(.bold))
                    .foregroundStyle(.gray)
            }
        }
    }
}

代码运行效果如下图所示:

在这里插入图片描述

@Entry 宏不仅能够用在环境变量的定义中,它同样可以用来简化 Transaction Values、Container Values 以及 Focused Values 等类型的定义:

extension Transaction {
    @Entry var myCustomValue: String = "Default value"
}

extension ContainerValues {
    @Entry var myCustomValue: String = "Default value"
}

extension FocusedValues {
    @Entry var myCustomValue: String?
}

更多 @Entry 的“玩法”请小伙伴们移步苹果开发者官网恣意研究。

2. @Previewable 让 Xcode 预览调试安闲自得

除了苹果各个开发框架的重磅更新以外,每年的 WWDC 也都会让果粉必备的开发集成环境 Xcode 如日方升,今年的 WWDC 24 自然也不例外。

我们知道 Xcode 中预览(Preview)和 SwiftUI 的界面调试真何谓是“天作之合”。不过 SwiftUI 6.0 之前,如果我们希望在预览中调试需要传入额外状态的视图就会变得“捉襟见肘”:

struct Hero: Identifiable {
    var id = UUID()
    var name: String
    var superpower: SuperPower
    var isInHellMode: Bool
    
    static var previewHeros: [Hero] = {
       [
        Hero(name: "孙悟空", superpower: .immortal, isInHellMode: false),
        Hero(name: "钢铁侠", superpower: .immortal, isInHellMode: false),
        Hero(name: "闪电侠", superpower: .teleportation, isInHellMode: false),
        Hero(name: "吉良吉影", superpower: .predictTheFuture, isInHellMode: false),
        Hero(name: "灭霸", superpower: .timeStop, isInHellMode: true)
       ]
    }()
}

struct HerosView: View {
    
    @Binding var heroList: [Hero]
    
    var body: some View {
        NavigationStack {
            List($heroList) { $hero in
                VStack(alignment: .leading) {
                    HStack {
                        TextField("英雄名字", text: $hero.name)
                            .font(.title.weight(.black))
                        
                        Toggle("地狱模式", isOn: $hero.isInHellMode)
                            
                    }
                    
                    HStack {
                        Text(hero.superpower.description)
                            .font(.headline)
                            .foregroundStyle(.gray)
                        
                        Spacer()
                        
                        Text(hero.id.uuidString.suffix(8))
                            .font(.title2.weight(.heavy))
                            .foregroundStyle(.purple)
                            
                    }
                }
            }
            .navigationTitle("英雄列表")
            .toolbar {
                Text("大熊猫侯佩 @ \(Text("CSDN").foregroundStyle(.red))")
                    .font(.headline.weight(.bold))
                    .foregroundStyle(.gray)
            }
        }
    }
}

如上代码所示:我们希望在 Xcode 预览中调试的 HerosView 视图会被要求传入一个可变 heroList 状态,它的类型是 [Hero]。

在 SwiftUI 6.0(Xcode 16)之前,要想预览与 HerosView 翩翩起舞我们可能需要大费周章地另外写一个包装器视图,在该视图中创建一个 [Hero] 类型的状态,然后再把它传递给 HerosView。


除了用包装器的方式调试 HerosView 视图以外,我们还可以使用 #Preview + @Observable 宏的组合构造可变 @Binding 实参来向 HerosView 传递状态 。

更多细节请小伙伴们移步如下链接观赏进一步精彩的内容:

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

而现在 WWDC24 为我们送来了全新的 @Previewable 宏专注于解决此事:

在这里插入图片描述

有了 @Previewable 宏,我们即可怡然自得的在 #Preview 宏预览闭包中直接向被调试的 SwiftUI 视图传入可变状态了:

#Preview("英雄列表") {
    @Previewable @State var heros = Hero.previewHeros
    HerosView(heroList: $heros)
}

现在,我们在 Xcode 16 中可以易如反掌的调试需要传入可变状态的 SwiftUI 子视图了,棒棒哒💯:

在这里插入图片描述

总结

在本篇博文中,我们介绍了如何在最新的 SwiftUI 6.0(Xcode 16)中利用 WWDC24 中新祭出的 @Entry 和 @Previewable 宏让环境变量定义和 Xcode 界面预览调试更加得心应手,充满乐趣!

感谢观赏,再会!😎

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1830165.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

用React编写一个密码组件表单

theme: condensed-night-purple highlight: atelier-cave-light 背景介绍 我们在使用网站或者应用程序的登录界面或创建帐户界面时,往往避免不了需要用户输入密码这一步骤,而用户是否可以选择看见他们输入的密码是十分重要的一项功能。尤其是在当输入的…

Focusky是什么软件

Focusky是一款基于HTML5技术的多媒体演示软件,可以轻松地制作出生动有趣的PPT演示文稿、动画宣传片以及微课。与其他软件相比,Focusky拥有丰富的多媒体资源和动画效果,可以让演示内容更加生动。本文将为您详细介绍Focusky软件的功能&#xff…

awd工具安装

fscan(漏洞扫描) 下载 下载地址: Releases shadow1ng/fscan GitHub 把下载的文件放到指定文件目录里, 在文件的位置打开cmd 输入 fscan64.exe -h 192.168.1.1/24 ok了 接下来说说fscan的使用 使用 1.信息搜集: 存活探测(icmp) 端口扫描 2.爆破功能: 各类服务爆破(…

MongoDB~高可用集群介绍:复制集群(副本集)、分片集群

背景 MongoDB 的集群主要包括副本集(Replica Set)和分片集群(Sharded Cluster)两种类型。 副本集 组成:通常由一个主节点(Primary)和多个从节点(Secondary)构成。 功…

UniVue更新日志:使用ObservableList优化LoopList/LoopGrid组件的使用

github仓库 稳定版本仓库:https://github.com/Avalon712/UniVue 开发版本仓库:https://github.com/Avalon712/UniVue-Develop UniVue扩展框架-UniVue源生成器仓库:https://github.com/Avalon712/UniVue-SourceGenerator 更新说明 如果大家…

Django REST framework视图集与路由详解:深入理解ViewSet、ModelViewSet与路由映射器

系列文章目录 Django入门全攻略:从零搭建你的第一个Web项目Django ORM入门指南:从概念到实践,掌握模型创建、迁移与视图操作Django ORM实战:模型字段与元选项配置,以及链式过滤与QF查询详解Django ORM深度游&#xff…

【java】数学运算考试系统

目录 一、登录界面: 二、管理员界面: 三、学生考试界面: 面向小学低年级学生,随机生成两个整数的加减法算式要求学生解答。要求有用 户登录、注册等 GUI 界面,用户数据存入文件,体现面向对象编程思想。 …

windows11子系统Ubuntu 22.04.4子安装图形化界面

1、windows11家庭版本设置 打开虚拟机安装许可 2、Microsoft Store下载安装ubuntu 我使用的是22.04.4 LTS版本 3、 打开ubuntu 命令窗口 1、打开win11的命令行,在下拉三角下标,打开,可以看到有Ubuntu 的选项,点击即可进入linux命…

【Android面试八股文】你说一说什么是双亲委托机制?为什么需要双亲委托机制?

一、双亲委托机制 1.1 双亲委托机制概述 双亲委托机制是指当一个类加载器收到一个类加载请求时, 该类加载器首先会把请求委派给父类加载器。 如果父类加载器还存在父类加载器,则会一直向上委派,直至最终交由顶层的启动类加载器完成类加载, 每个类加载器都是如此,只有在所…

RIP解决不连续子网问题

#交换设备 RIP解决不连续子网问题 一、不连续子网的概念 相同主网下的子网,被另一个主网分割,例如下面实验拓扑在某公司的网络整改项目中,原先R1 和RS 属于同一主网络 10.0.0.0/8,现被 R2、R3、R4 分离,整网采用了 …

Docker 安装 MySQL5.7 和 MySQL8

文章目录 安装 MySQL5.7拉取镜像前期准备启动容器 安装MySQL8.0拉取镜像查看镜像前期准备启动容器 安装 MySQL5.7 拉取镜像 docker pull mysql:5.7拉下来镜像后 执行 docker images 此时我们已经有这个镜像了。 前期准备 在根目录下创建 app , 在 app 目录下创建…

牛客周赛 Round 47 解题报告 | 珂学家

前言 题解 这真的是牛客周赛? 哭了 欢迎关注 珂朵莉 牛客周赛专栏 珂朵莉 牛客小白月赛专栏 A. 小红的葫芦 签到题 但是写起来有点变扭,方法应该蛮多的 统计分组 有2组一组长度为2,一组长度为3 def check(arr):arr.sort()if arr[0] …

数据结构试题 20-21

真需要就死记吧 二叉树遍历-先序(非递归)【图解代码】_哔哩哔哩_bilibili 解释一下步骤: 一个循环为: 1.取节点 2.放右子树 3.放左子树 每次循环,都要从栈里取出一个节点 先放右子树,再放左子树 那这道题就是,先放1&am…

内网Docker镜像无法使用?Debian/Ubuntu离线安装Dokcer

离线安装Docker 卸载冲突的包 for pkg in docker.io docker-doc docker-compose podman-docker containerd runc; do sudo apt-get remove $pkg; done先删除docker sudo apt-get purge docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin d…

DSP28335:独立按键控制LED灯

做任何事情不可操之过急,虽然我们可能在之前的单片机学过相关的原理,但是一个新的单片机依然有他的学习的地方,之前我觉得很简单,就跳过这个学习,结果到后面就很浮躁,导致后面的内容与这一章相连接的时候&a…

调用第三方系统的签名设计与校验实例讲解与实践

在现代软件开发中,调用第三方系统API已经成为常见需求。为了保证数据传输的安全性和完整性,许多API采用了签名机制。本文将详细讲解如何设计与校验调用第三方系统的签名,以确保双方通信的安全和可靠。 #### 一、签名机制的意义 签名机制主要…

.gitignore文件忽略的内容不生效问题解决

文章目录 ①:现象②:原因③:解决 ①:现象 在已经提交过的git管理的项目中, 新增加一个.gitignore文件,文件内忽略内容不生效或者修改.gitignore文件之后,文件内新增的忽略内容不生效 ②&#…

Dell戴尔灵越Inspiron 16 Plus 7640/7630笔记本电脑原装Windows11下载,恢复出厂开箱状态预装OEM系统

灵越16P-7630系统包: 链接:https://pan.baidu.com/s/1Rve5_PF1VO8kAKnAQwP22g?pwdjyqq 提取码:jyqq 灵越16P-7640系统包: 链接:https://pan.baidu.com/s/1B8LeIEKM8IF1xbpMVjy3qg?pwdy9qj 提取码:y9qj 戴尔原装WIN11系…

1.华为路由器-三层交换机-二层交换机组网连接

AR1配置GE 0/0/0接口IP [Huawei]int g0/0/0 [Huawei-GigabitEthernet0/0/0] [Huawei-GigabitEthernet0/0/0]ip add 1.1.1.1 24 [Huawei]iP route-static 192.168.0.0 16 1.1.1.2三层交换机配置如下 创建vlan [Huawei]vlan batch 10 20配置接口ip [Huawei]int g0/0/1 [Huawei…

【数据结构】排序(下)

个人主页~ 排序(上) 栈和队列 排序 二、常见排序的实现8、快速排序的优化9、非递归快速排序(1)基本思想(2)代码实现(3)时间复杂度(4)空间复杂度 10、归并排序…