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

2024-04-23 1205阅读

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

概览

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

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

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

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

  • 概览
  • 1. “震荡波”来袭
  • 2. 触发器模式
  • 3. SwiftUI 触发器模式的其它应用
  • 4. 自定义触发器模式
  • 总结

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

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


    1. “震荡波”来袭

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

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

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

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

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

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

    • Playing a single-tap haptic pattern

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

      • Core Haptics

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

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

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

        如上所示,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(_ 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(_ 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(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 in
                        Text(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 in
                Text(result)
            }
            .searchable(text: $store.query)
            .sensoryFeedback(trigger: store.results) { oldValue, newValue in
                return newValue.isEmpty ? .error : .success
            }
        

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

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

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

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

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

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

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

        4. 自定义触发器模式

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

        答案是肯定的!

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

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

        struct PlaySoundViewModifier: ViewModifier {
            let sound: URL
            let trigger: Trigger
            func 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 in
                    Text(verbatim: message)
                }
                .playSound(
                    Bundle.main.url(forResource: "sound", withExtension: "wav")!,
                    trigger: messages
                )
            }
        }
        

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


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

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

        • Swift 语言开发精讲

          总结

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

          感谢观赏,再会!😎

VPS购买请点击我

免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们,邮箱:ciyunidc@ciyunshuju.com。本站只作为美观性配图使用,无任何非法侵犯第三方意图,一切解释权归图片著作权方,本站不承担任何责任。如有恶意碰瓷者,必当奉陪到底严惩不贷!

目录[+]