SwiftUI 5.0(iOS 17.0)触摸反馈“震荡波”与触发器模式趣谈

devtools/2024/9/25 10:24:35/

在这里插入图片描述

概览

要想创作出一款精彩绝伦的 App,绚丽的界面和灵动的动画并不是唯一吸引用户的要素。有时我们还希望让用户真切的感受到操作引发的触觉反馈,直击使用者的灵魂。

在这里插入图片描述

所幸的是新版 SwiftUI 原生提供了实现触觉震动反馈的机制。在介绍它之后我们还将进一步展开讨论该机制基于的触发器模式,并“青出于蓝而胜于绿”的设计我们自己的触发器实现。

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

相信学完本课后,小伙伴们对于 SwiftUI 中触觉反馈与触发器开发模式会有更深刻的领悟,从而能够更加游刃有余的使用它们。

那还等什么呢?We will rock you!!!😉


1. “震荡波”来袭

除了从视觉上强势吸引用户眼球之外,我们的 App 还可以用“更立体”的方式让用户爱不释手。是滴,我们就是要用触觉反馈震动他们“久逢甘露”的双手,用“震荡波”激荡他们的心灵。

震动反馈Haptic )是 Apple 对于移动设备提供的一种加强用户体验的机制,它最早诞生于 UIKit。它的体验有点类似于之前 iPhone 中 3D Touch 的功能。

Haptic 被广泛应用在 iOS/iPadOS 中,Apple 系统在用户交互中大量使用了震动反馈,比如在锁屏状态下点击 iPhone 屏幕左下角的手电筒按钮:

在这里插入图片描述

或者 iPhone 隔空投送完成时给于用户的提示反馈,以及 AppleWatch 上的通知提醒等等。

更多 Haptic 撸码的相关介绍,请小伙伴们移步 Apple 官方开发网站观赏进一步内容:

  • Playing a single-tap haptic pattern

在这里插入图片描述

在这里插入图片描述

前面说过 Haptic 最先是在 UIKit 中得到很好支持的,从 SwiftUI 5.0iOS 17.0)开始苹果终于推出了 SwiftUI 里 Haptic 的原生实现 SensoryFeedback

在这里插入图片描述

如上所示,SensoryFeedback 结构的“借花献佛”是通过视图扩展方法 sensoryFeedback 来完成的:

@available(iOS 17.0, macOS 14.0, tvOS 17.0, watchOS 10.0, *)
@available(visionOS, unavailable)
extension View {/// - Parameters:///   - feedback: Which type of feedback to play.///   - trigger: A value to monitor for changes to determine when to play.public func sensoryFeedback<T>(_ feedback: SensoryFeedback, trigger: T) -> some View where T : Equatable/// - Parameters:///   - feedback: Which type of feedback to play.///   - trigger: A value to monitor for changes to determine when to play.///   - condition: A closure to determine whether to play the feedback when///     `trigger` changes.public func sensoryFeedback<T>(_ feedback: SensoryFeedback, trigger: T, condition: @escaping (_ oldValue: T, _ newValue: T) -> Bool) -> some View where T : Equatable/// - Parameters:///   - trigger: A value to monitor for changes to determine when to play.///   - feedback: A closure to determine whether to play the feedback and///     what type of feedback to play when `trigger` changes.public func sensoryFeedback<T>(trigger: T, _ feedback: @escaping (_ oldValue: T, _ newValue: T) -> SensoryFeedback?) -> some View where T : Equatable}

可以看到 sensoryFeedback 方法拥有多个重写形式,但它们都毫无例外的使用了触发器模式(Trigger Mode)。

2. 触发器模式

什么是触发器模式呢?在 Apple 的开发中大家可能对观察者模式早就有所耳闻,触发器模式与此类似也属于苹果开发中的一种设计模式。触发器模式就是让状态的改变触发代码的执行。

在以状态驱动的 SwiftUI 王国中,触发器模式的使用更显得“如鱼得水”,仿佛天造地设一般。

struct ContentView: View {@State private var store = Store()var body: some View {NavigationStack {List(store.results, id: \.self) { result inText(result.title)}.searchable(text: $store.query).sensoryFeedback(.success, trigger: store.results)}}
}

在上面的示例代码中,我们使用 sensoryFeedback 修改器方法为视图添加了震动反馈。其中可以看到:Haptic 产生的触发器是 store.results 状态,即当用户引起 Store 搜索结果发生改变时,我们纤细指尖才会喜提激荡震动着的触觉洗礼。

除了感受系统内置震动效果对“心灵的冲击”以外,我们还可以让震动更加“变幻莫测”:

VStack {}
.sensoryFeedback(.impact(weight: .heavy, intensity: 0.9),trigger: trigger
)

类似的,大家还可以根据状态实际的值来决定到底使用何种 Haptic 效果,比如在下面的代码中我们就根据搜索是否成功来决定采用 .error 还是 .success 震动反馈类型:

List(store.results, id: \.self) { result inText(result)}.searchable(text: $store.query).sensoryFeedback(trigger: store.results) { oldValue, newValue inreturn newValue.isEmpty ? .error : .success}

注意,上面所有 Haptic 效果只有在触发器对应状态发生改变时才会产生,所以我们不用担心视图创建时触发器导致不希望的“副作用”。

3. SwiftUI 触发器模式的其它应用

除了 Haptic 对应的实现以外,在 SwiftUI 中还有很多其它功能也大量适配触发器模式。比如 scrollIndicatorsFlash 修改器方法:

在这里插入图片描述

scrollIndicatorsFlash 方法用来在指定状态发生改变时来“闪烁”可滚动视图中的滚动条:

struct TriggerValueExample: View {let messages: [String]var body: some View {List(messages, id: \.self) { message inText(verbatim: message)}.scrollIndicatorsFlash(trigger: messages)}
}

在上面的代码中,当有新消息到来时我们会“闪烁”列表的滚动条以提示用户。

4. 自定义触发器模式

通过上面的介绍,想必大家对于何为触发器模式以及它的工作原理已经了然于心了。触发器模式在 SwiftUI 中被广泛地使用着,那我们能不能“百尺竿头更进一步”打造自己的触发器呢?

答案是肯定的!

正如观察者模式那样,设计模式意味着提供充分可定制的灵活性,除了使用系统框架提供的触发器以外,我们当然可以随心所欲地创建自己的触发器

假如我们希望在 SwiftUI 中当某一状态发生改变时播放指定的声音,这可以恰如其分的用触发器模式来实现:

struct PlaySoundViewModifier<Trigger: Equatable>: ViewModifier {let sound: URLlet trigger: Triggerfunc body(content: Content) -> some View {content.onChange(of: trigger) {if let player = try? AVAudioPlayer(contentsOf: sound) {player.play()}}}
}extension View {func playSound(_ sound: URL, trigger: some Equatable) -> some View {self.modifier(PlaySoundViewModifier(sound: sound, trigger: trigger))}
}

在上面的示例代码中,我们创建了 PlaySoundViewModifier 修改器方法,并绑定了一个遵守 Equatable 协议,类型为 Trigger 的属性,当 trigger 发生变化时,我们就利用 AVAudioPlayer 对象从容地播放想要的声音文件。

小伙伴们可以这样使用上面创建的 PlaySoundViewModifier 修改器方法:

struct SoundFeedbackExample: View {let messages: [String]var body: some View {List(messages, id: \.self) { message inText(verbatim: message)}.playSound(Bundle.main.url(forResource: "sound", withExtension: "wav")!,trigger: messages)}
}

看到了吗?触发器模式就是这么单刀直入,让代码实现变得如此直接了当。从此小伙伴们开发兵器库中又多了一件神兵利器,棒棒哒!💯


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

在这里插入图片描述

  • Swift 语言开发精讲

总结

在本篇博文中,我们介绍了 SwiftUI 5.0iOS 17.0)中触觉反馈(Haptic)机制的实现,并由此抛砖引玉讨论了开发模式中的触发器模式,最后我们看到了实现自己心仪的触发器是多么的简单。

感谢观赏,再会!😎


http://www.ppmy.cn/devtools/11151.html

相关文章

若依前端分离版中使用二维码功能

一、安装 在前端项目工程目录&#xff0c;远端执行如下命令 // npm npm install vue-qr --save // yarn yarn add vue-qr 二、引入组件 在main.js文件中增加如下的内容 // vue2.x import VueQr from vue-qr //注册组件 Vue.component(VueQr, VueQr)// vue3.x import vueQr f…

linux特殊权限

SUID 在 Linux 系统中&#xff0c;其实除了 rwx 权限&#xff0c;还会用到 s 权限&#xff0c;例如&#xff1a; 原本表示文件所有者权限中的 x 权限位&#xff0c;却出现了 s 权限&#xff0c;此种权限通常称为 SetUID &#xff0c;简称 SUID 特殊权限。 SUID 特殊权限…

AI热潮下,公链基础设施赛道都有哪些变化?

最近在一级市场&#xff0c;最火热的赛道无疑是AI&#xff0c;其次是BTC&#xff0c;每天聊的项目80%都集中在这两个赛道&#xff0c;我个人最多的时候一天可以聊5&#xff0c;6个AI项目。 可以预见的是AI泡沫会在明后年达到顶峰&#xff0c;随着数以百计的AI新项目上线&#…

Java 源码-多级时间轮TimingWheel

多级时间轮TimingWheel 一、时间轮是什么 类似现实中的钟表&#xff0c;由多个环形数组组成&#xff0c;每个环形数组包含20个时间单位&#xff0c;表示一个时间维度&#xff08;一轮&#xff09;&#xff0c;如&#xff1a;第一层时间轮&#xff0c;数组中的每个元素代表1m…

C++list类(个人笔记)

list类 1.熟悉list接口以及使用1.1list的构造1.2list iterator的使用1.3list capacity1.4list element access1.5list modifiers1.6list的迭代器失效 2.list的模拟实现 1.熟悉list接口以及使用 list的C官方文档 1.1list的构造 构造函数 constructor接口说明list (size_type …

MetaAI发布Llama 3模型:4000亿参数的AI新标杆

开源和闭源大战 前几天&#xff0c;百度李彦宏说闭源才有意义。 周鸿祎在哈佛论坛演讲上说&#xff0c;别听李彦宏胡说八道&#xff0c;没有开源就没有Linux&#xff0c;就没有今天的互联网&#xff0c;李彦宏自己都是借助开源的力量走到今天&#xff0c;现在反手说开源不好了…

Github 2024-04-20 Rust开源项目日报 Top10

根据Github Trendings的统计,今日(2024-04-20统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Rust项目10Cuda项目1HTML项目1Sway: 将现代语言开发和性能带入区块链生态系统 创建周期:1187 天开发语言:Rust协议类型:Apache License 2.0S…

Indicator-Sysmonitor|Ubuntu 标题栏显示实时网速、CPU使用率等

分享一款在Ubuntu系统中非常实用的工具&#xff0c;名为Indicator-Sysmonitor。这是一款开源软件&#xff0c;你可以在GitHub上找到它&#xff0c;它能够让你在Ubuntu的任务栏中实时监控CPU和内存使用情况。虽然Ubuntu并没有像Windows端的TrafficMonitor那样广为人知的实时监控…