UNPKG

@reactivex/rxjs

Version:

Reactive Extensions for modern JavaScript

69 lines (59 loc) 2.25 kB
import {Operator} from '../Operator'; import {Subscriber} from '../Subscriber'; import {ArgumentOutOfRangeError} from '../util/ArgumentOutOfRangeError'; import {Observable} from '../Observable'; /** * Returns an Observable that emits the item at the specified index in the source Observable. * If default is given, missing indices will output this value on next; otherwise, outputs error. * @throws {ArgumentOutOfRangeError} When using `elementAt(i)`, it delivers an * ArgumentOutOrRangeError to the Observer's `error` callback if `i < 0` or the * Observable has completed before emitting the i-th `next` notification. * @param {number} index the index of the value to be retrieved. * @param {any} [defaultValue] the default value returned for missing indices. * @return {Observable} an Observable that emits a single item, if it is found. Otherwise, will emit the default value if given. * @method elementAt * @owner Observable */ export function elementAt<T>(index: number, defaultValue?: T): Observable<T> { return this.lift(new ElementAtOperator(index, defaultValue)); } export interface ElementAtSignature<T> { (index: number, defaultValue?: T): Observable<T>; } class ElementAtOperator<T> implements Operator<T, T> { constructor(private index: number, private defaultValue?: T) { if (index < 0) { throw new ArgumentOutOfRangeError; } } call(subscriber: Subscriber<T>, source: any): any { return source._subscribe(new ElementAtSubscriber(subscriber, this.index, this.defaultValue)); } } /** * We need this JSDoc comment for affecting ESDoc. * @ignore * @extends {Ignored} */ class ElementAtSubscriber<T> extends Subscriber<T> { constructor(destination: Subscriber<T>, private index: number, private defaultValue?: T) { super(destination); } protected _next(x: T) { if (this.index-- === 0) { this.destination.next(x); this.destination.complete(); } } protected _complete() { const destination = this.destination; if (this.index >= 0) { if (typeof this.defaultValue !== 'undefined') { destination.next(this.defaultValue); } else { destination.error(new ArgumentOutOfRangeError); } } destination.complete(); } }