Files
ShadowsocksX-NG/Pods/RxSwift/RxSwift/Observables/Sample.swift

143 lines
4.1 KiB
Swift
Raw Normal View History

2017-03-17 23:08:52 +08:00
//
// Sample.swift
// RxSwift
//
// Created by Krunoslav Zaher on 5/1/15.
// Copyright © 2015 Krunoslav Zaher. All rights reserved.
//
extension ObservableType {
/**
Samples the source observable sequence using a sampler observable sequence producing sampling ticks.
Upon each sampling tick, the latest element (if any) in the source sequence during the last sampling interval is sent to the resulting sequence.
**In case there were no new elements between sampler ticks, no element is sent to the resulting sequence.**
- seealso: [sample operator on reactivex.io](http://reactivex.io/documentation/operators/sample.html)
- parameter sampler: Sampling tick sequence.
- returns: Sampled observable sequence.
*/
public func sample<O: ObservableType>(_ sampler: O)
-> Observable<E> {
return Sample(source: self.asObservable(), sampler: sampler.asObservable())
}
}
final fileprivate class SamplerSink<O: ObserverType, SampleType>
2017-03-17 23:08:52 +08:00
: ObserverType
, LockOwnerType
, SynchronizedOnType {
typealias E = SampleType
typealias Parent = SampleSequenceSink<O, SampleType>
fileprivate let _parent: Parent
2017-03-20 21:26:25 +08:00
var _lock: RecursiveLock {
2017-03-17 23:08:52 +08:00
return _parent._lock
}
init(parent: Parent) {
_parent = parent
}
func on(_ event: Event<E>) {
synchronizedOn(event)
}
func _synchronized_on(_ event: Event<E>) {
switch event {
case .next:
if let element = _parent._element {
_parent._element = nil
_parent.forwardOn(.next(element))
}
if _parent._atEnd {
_parent.forwardOn(.completed)
_parent.dispose()
}
case .error(let e):
_parent.forwardOn(.error(e))
_parent.dispose()
case .completed:
if let element = _parent._element {
_parent._element = nil
_parent.forwardOn(.next(element))
}
if _parent._atEnd {
_parent.forwardOn(.completed)
_parent.dispose()
}
}
}
}
final fileprivate class SampleSequenceSink<O: ObserverType, SampleType>
2017-03-17 23:08:52 +08:00
: Sink<O>
, ObserverType
, LockOwnerType
, SynchronizedOnType {
typealias Element = O.E
typealias Parent = Sample<Element, SampleType>
fileprivate let _parent: Parent
2017-03-20 21:26:25 +08:00
let _lock = RecursiveLock()
2017-03-17 23:08:52 +08:00
// state
fileprivate var _element = nil as Element?
fileprivate var _atEnd = false
fileprivate let _sourceSubscription = SingleAssignmentDisposable()
init(parent: Parent, observer: O, cancel: Cancelable) {
_parent = parent
super.init(observer: observer, cancel: cancel)
}
func run() -> Disposable {
_sourceSubscription.setDisposable(_parent._source.subscribe(self))
let samplerSubscription = _parent._sampler.subscribe(SamplerSink(parent: self))
return Disposables.create(_sourceSubscription, samplerSubscription)
}
func on(_ event: Event<Element>) {
synchronizedOn(event)
}
func _synchronized_on(_ event: Event<Element>) {
switch event {
case .next(let element):
_element = element
case .error:
forwardOn(event)
dispose()
case .completed:
_atEnd = true
_sourceSubscription.dispose()
}
}
}
final fileprivate class Sample<Element, SampleType> : Producer<Element> {
2017-03-17 23:08:52 +08:00
fileprivate let _source: Observable<Element>
fileprivate let _sampler: Observable<SampleType>
init(source: Observable<Element>, sampler: Observable<SampleType>) {
_source = source
_sampler = sampler
}
override func run<O: ObserverType>(_ observer: O, cancel: Cancelable) -> (sink: Disposable, subscription: Disposable) where O.E == Element {
let sink = SampleSequenceSink(parent: self, observer: observer, cancel: cancel)
let subscription = sink.run()
return (sink: sink, subscription: subscription)
}
}