ios - UIView 作为@EnvironmentObject 的订阅者

我有一个在工作线程中更新的@EnvironmentObject,并且几个 SwiftUI View 订阅了对已发布值的更改。

这一切都很好。

但我正在努力让 UIView 订阅@EnvironmentObject 中的更改。

给定

@EnvironmentObject var settings: Settings 

where Settings is:

final class Settings {
    @Published var bar: Int = 0
    @Published var beat: Int = 1
    etc.
}

SwiftUI View 根据发布的值变化进行更新。

但是现在,我想声明一个接收器,该接收器在符合 UIViewRepresentable 的 UIView 中接收发布的值。

我一直在研究 Combine 这本书,并认为我可以用类似的东西声明一个 .sink 闭包:

   func subscribeToBeatChanges() {
        settings.$bar
            .sink(receiveValue: {
                bar in
                self.bar = bar
                print("Bar = \(bar)")
            } )
        settings.$beat
            .sink(receiveValue: {
                beat in
                self.beat = beat
                print("Beat = \(beat)")
                self.setNeedsDisplay()
            } )
    }
 

不幸的是,闭包仅在调用 subscribeToBeatChanges() 时被调用一次。我想要的是每次 @EnvironmentObject 值中的 @Published 属性更改时调用闭包。

我还尝试在 UIViewRepresentable 包装器内部订阅,在 makeUIView 方法中使用一些东西,但没有成功。

我显然犯了一些相当简单和根本性的错误,我肯定会很感激你朝正确的方向前进,因为我正试图弄清楚这个问题!

谢谢!

最佳答案

您必须存储对从 .sink 收到的 AnyCancellable 的引用:

private var cancellables = Set<AnyCancellable>()

func subscribeToBeatChanges() {
    settings.$bar
        .sink(receiveValue: {
            bar in
            self.bar = bar
            print("Bar = \(bar)")
        } )
        .store(in: &cancellables) // <-- store the instance
    settings.$beat
        .sink(receiveValue: {
            beat in
            self.beat = beat
            print("Beat = \(beat)")
            self.setNeedsDisplay()
        } )
        .store(in: &cancellables) // <-- store the instance
}

如果您不存储它,AnyCancellable 实例将在 deinit 时自动取消订阅。

https://stackoverflow.com/questions/62585953/

相关文章:

html - 如何在选择下拉菜单上设置 Font Awesome 图标?

javascript - Angular ngrx : TypeError: Cannot free

r - 在 Windows 上安装 odbc R 包时出现问题

java - 将内部 HashMap 类型强制转换为 Map

postgresql - postgres SQL 状态 : 22P02

python - DBSCAN 聚类甚至无法处理 40k 数据,但可以使用 python 和 skl

java - Open Api Generator 插件不适用于 Maven

vue.js - 当对象没有(可识别的)变化时触发 Vue 监视

string - 将字符串拆分为最少数量的字符串,其中每个字母重复奇数次

java - SonarQube Scanner - 插件下载之间的长时间停顿