//
|
// TakeLast.swift
|
// RxSwift
|
//
|
// Created by Tomi Koskinen on 25/10/15.
|
// Copyright © 2015 Krunoslav Zaher. All rights reserved.
|
//
|
|
extension ObservableType {
|
|
/**
|
Returns a specified number of contiguous elements from the end of an observable sequence.
|
|
This operator accumulates a buffer with a length enough to store elements count elements. Upon completion of the source sequence, this buffer is drained on the result sequence. This causes the elements to be delayed.
|
|
- seealso: [takeLast operator on reactivex.io](http://reactivex.io/documentation/operators/takelast.html)
|
|
- parameter count: Number of elements to take from the end of the source sequence.
|
- returns: An observable sequence containing the specified number of elements from the end of the source sequence.
|
*/
|
public func takeLast(_ count: Int)
|
-> Observable<Element> {
|
TakeLast(source: self.asObservable(), count: count)
|
}
|
}
|
|
final private class TakeLastSink<Observer: ObserverType>: Sink<Observer>, ObserverType {
|
typealias Element = Observer.Element
|
typealias Parent = TakeLast<Element>
|
|
private let parent: Parent
|
|
private var elements: Queue<Element>
|
|
init(parent: Parent, observer: Observer, cancel: Cancelable) {
|
self.parent = parent
|
self.elements = Queue<Element>(capacity: parent.count + 1)
|
super.init(observer: observer, cancel: cancel)
|
}
|
|
func on(_ event: Event<Element>) {
|
switch event {
|
case .next(let value):
|
self.elements.enqueue(value)
|
if self.elements.count > self.parent.count {
|
_ = self.elements.dequeue()
|
}
|
case .error:
|
self.forwardOn(event)
|
self.dispose()
|
case .completed:
|
for e in self.elements {
|
self.forwardOn(.next(e))
|
}
|
self.forwardOn(.completed)
|
self.dispose()
|
}
|
}
|
}
|
|
final private class TakeLast<Element>: Producer<Element> {
|
private let source: Observable<Element>
|
fileprivate let count: Int
|
|
init(source: Observable<Element>, count: Int) {
|
if count < 0 {
|
rxFatalError("count can't be negative")
|
}
|
self.source = source
|
self.count = count
|
}
|
|
override func run<Observer: ObserverType>(_ observer: Observer, cancel: Cancelable) -> (sink: Disposable, subscription: Disposable) where Observer.Element == Element {
|
let sink = TakeLastSink(parent: self, observer: observer, cancel: cancel)
|
let subscription = self.source.subscribe(sink)
|
return (sink: sink, subscription: subscription)
|
}
|
}
|