このタスクでは、ReactiveSwiftに組み込み演算子はありません。代わりに、拡張書き込み、次のアプローチを使用することができます。
import Foundation
import ReactiveSwift
import Result
public extension Signal {
public func completeAfter(after: TimeInterval, onScheduler : DateSchedulerProtocol = QueueScheduler()) -> Signal {
let pipe : (Signal<(), NoError>, ReactiveSwift.Observer<(), NoError>) = Signal<(), NoError>.pipe()
onScheduler.schedule(after: Date(timeIntervalSinceNow: after)) {
pipe.1.sendCompleted()
}
return Signal { observer in
return self.observe { event in
switch event {
case let .value(value):
observer.send(value: value)
case .completed:
observer.sendCompleted()
case let .failed(error):
observer.send(error: error)
case .interrupted:
observer.sendInterrupted()
}
}
}.take(until: pipe.0)
}
public func collectUntil(until: TimeInterval) -> Signal<[Value], Error> {
return self.completeAfter(after: until).collect()
}
}
をそしてsignal.collectUntil(5)
メソッドを使用します。
別の方法は、ReactiveSwiftのtimer
機能を使用することです。例(上記のように、同じ拡張子に追加):それは内部信号を抽出SignalProducer
タイプの性質を偽造ているため
public func collectUntil2(until: TimeInterval) -> Signal<[Value], Error> {
var signal: Signal<(), NoError>? = nil
timer(interval: until, on: QueueScheduler()).startWithSignal { innerSignal, _ in
signal = innerSignal.map { _ in() }.take(first: 1)
}
return self.take(until: signal!).collect()
}
私は、しかし、このアプローチは好きではありません。
Signal
タイプ自体もtimeout
機能ですが、エラーが発生しているので使いにくいです。それを使用する方法の例(まだ、同じ拡張子に追加):
public func completeOnError() -> Signal<Value, Error> {
return Signal { observer in
return self.observe { event in
switch(event) {
case .value(let v): observer.send(value: v)
case .failed(_): observer.sendCompleted()
case .interrupted: observer.sendInterrupted()
case .completed: observer.sendCompleted()
}
}
}
}
public func collectUntil3(until: TimeInterval) -> Signal<[Value], Error> {
return self
.timeout(after: until,
raising: NSError() as! Error,
on: QueueScheduler())
.completeOnError()
.collect()
}
P.S. 3つのオプションのいずれかを選択することにより、正しいスケジューラーを渡すか、ソリューションを正しいスケジューラーでパラメトリックにすることを心がけてください。
あなたの答えは正しいとは言いましたが、それは私が探していたものではありませんでしたが、あなたは正しい方向に私を指摘しました。私は私の問題を解決するために書いた拡張を含む自分自身の答えを追加しました。 –