SwiftUI 接收自定义事件
Posted
技术标签:
【中文标题】SwiftUI 接收自定义事件【英文标题】:SwiftUI receive custom Event 【发布时间】:2021-02-17 14:52:26 【问题描述】:我有一个ObservableObject
,它使用@Published
属性包装器发布一些值。该对象还包含一个计时器。
问题是,如何在计时器执行后立即触发事件并在 SwiftUI 的视图中处理该事件(我更喜欢使用 onReceive
之类的东西)?
使用组合框架来发布变化的值,我想正确地实现这个事件触发/处理。但是到目前为止,我所读到的关于 Combine 的所有内容都是关于处理值变化的。但就我而言,它只是一个简单的事件(没有任何值)。
我知道我可以简单地使用闭包并在计时器到期时调用它,如果没有更好的类似组合的解决方案,我会这样做。
这是一个非常简单的问题的概念性问题,所以我认为它可以自我解释,而无需我提出代码示例?
【问题讨论】:
这是否回答了您的问题***.com/a/63239900/12299030? 谢谢,但不,一点也不。这只是在 swiftUI 中直接使用计时器。但在我的场景中,应该从我正在谈论的可观察对象实例中触发事件。计时器只是一个例子,也可能是其他原因。简而言之,我正在寻找一种方法来触发观察对象中的事件并在 swiftUI 中正确捕获/处理该事件。 您查看过“.onReceive”订阅者吗?这是一种 SwiftUI + 组合方法。 没错,这就是我在问题中提到它的原因。我想使用 onRecieve 来捕捉事件。但是我怎样才能从我观察到的对象中触发一个事件呢?这就是问题所在。 是否可以有一个由定时器触发的Bool
属性,然后在onReceive 中重置? (或增加一个 Int ?)感觉很hack-y,但可能有用吗?
【参考方案1】:
SwiftUI 使用 Combine 的方式是通过 .onReceive
,它需要一个发布者。对象可以将发布者(无论是 Timer
还是其他东西)公开为属性。
联合发布者通过发出值来工作,如果您只需要发出事件发生的信号,您可以发出 ()
aka Void
值。
顺便说一下,这个对象不必是ObservableObject
,但它可以是。
class Foo: ObservableObject
let timer = Timer.publish(every: 1, on: .main, in: .default)
.autoconnect()
.map _ in // map to Void
.eraseToAnyPublisher() // optional, but a good practice
现在,您可以使用.onReceive
订阅timer
事件:
struct ContentView: View
@StateObject var foo = Foo()
@State var int: Int = 0
var body: some View
Text("\(int)")
.onReceive(timer)
self.int += 1
当然,您不仅限于 TimerPublisher。例如,如果发生一些随机事件,您可以使用 PassthroughSubject
发布值:
class Foo
let eventA: AnyPublisher<Void, Never>
private let subject = PassthroughSubject<Void, Never>()
init()
eventA = subject.eraseToAnyPublisher()
let delay = Double.random(in: 10...100)
DispatchQueue.main.asyncAfter(deadline: .now() + delay) [weak self] in
// something random happened, notify
self?.subject.send()
【讨论】:
第二部分正是我正在寻找的解决方案!谢谢!【参考方案2】:只是一个想法,但您可以将特定线程专用于该对象吗?然后专门在该队列上监听并触发事件。我以前从未这样做过,但这似乎是一种正确的想法。
https://developer.apple.com/documentation/dispatch/dispatchqueue
https://www.freecodecamp.org/news/ios-concurrency/
【讨论】:
嘿,感谢您的想法和提供的链接!我肯定会对此进行研究,但乍一看,不得不切换到不同的线程对我来说似乎有点过于复杂。我的意思是,触发和接收事件是如此简单的任务,而组合框架似乎是此类的理想框架,所以我真的想知道它是否不以某种轻松(但不知何故未提及:D)的方式支持这一点..跨度>以上是关于SwiftUI 接收自定义事件的主要内容,如果未能解决你的问题,请参考以下文章