@reactivex/rxjs
Version:
Reactive Extensions for modern JavaScript
69 lines (59 loc) • 2.25 kB
text/typescript
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();
}
}