Skip to content

Commit 089a5a6

Browse files
committed
feat(take): add higher-order lettable version of take
1 parent 2640184 commit 089a5a6

File tree

3 files changed

+90
-47
lines changed

3 files changed

+90
-47
lines changed

src/operator/take.ts

+2-47
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,5 @@
1-
import { Operator } from '../Operator';
2-
import { Subscriber } from '../Subscriber';
3-
import { ArgumentOutOfRangeError } from '../util/ArgumentOutOfRangeError';
4-
import { EmptyObservable } from '../observable/EmptyObservable';
51
import { Observable } from '../Observable';
6-
import { TeardownLogic } from '../Subscription';
2+
import { take as higherOrder } from '../operators/take';
73

84
/**
95
* Emits only the first `count` values emitted by the source Observable.
@@ -39,46 +35,5 @@ import { TeardownLogic } from '../Subscription';
3935
* @owner Observable
4036
*/
4137
export function take<T>(this: Observable<T>, count: number): Observable<T> {
42-
if (count === 0) {
43-
return new EmptyObservable<T>();
44-
} else {
45-
return this.lift(new TakeOperator(count));
46-
}
47-
}
48-
49-
class TakeOperator<T> implements Operator<T, T> {
50-
constructor(private total: number) {
51-
if (this.total < 0) {
52-
throw new ArgumentOutOfRangeError;
53-
}
54-
}
55-
56-
call(subscriber: Subscriber<T>, source: any): TeardownLogic {
57-
return source.subscribe(new TakeSubscriber(subscriber, this.total));
58-
}
59-
}
60-
61-
/**
62-
* We need this JSDoc comment for affecting ESDoc.
63-
* @ignore
64-
* @extends {Ignored}
65-
*/
66-
class TakeSubscriber<T> extends Subscriber<T> {
67-
private count: number = 0;
68-
69-
constructor(destination: Subscriber<T>, private total: number) {
70-
super(destination);
71-
}
72-
73-
protected _next(value: T): void {
74-
const total = this.total;
75-
const count = ++this.count;
76-
if (count <= total) {
77-
this.destination.next(value);
78-
if (count === total) {
79-
this.destination.complete();
80-
this.unsubscribe();
81-
}
82-
}
83-
}
38+
return higherOrder(count)(this);
8439
}

src/operators/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,7 @@ export { subscribeOn } from './subscribeOn';
6969
export { switchAll } from './switchAll';
7070
export { switchMap } from './switchMap';
7171
export { switchMapTo } from './switchMapTo';
72+
export { take } from './take';
7273
export { takeLast } from './takeLast';
7374
export { tap } from './tap';
7475
export { timestamp } from './timestamp';

src/operators/take.ts

+87
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
1+
import { Operator } from '../Operator';
2+
import { Subscriber } from '../Subscriber';
3+
import { ArgumentOutOfRangeError } from '../util/ArgumentOutOfRangeError';
4+
import { EmptyObservable } from '../observable/EmptyObservable';
5+
import { Observable } from '../Observable';
6+
import { TeardownLogic } from '../Subscription';
7+
import { MonoTypeOperatorFunction } from '../interfaces';
8+
9+
/**
10+
* Emits only the first `count` values emitted by the source Observable.
11+
*
12+
* <span class="informal">Takes the first `count` values from the source, then
13+
* completes.</span>
14+
*
15+
* <img src="./img/take.png" width="100%">
16+
*
17+
* `take` returns an Observable that emits only the first `count` values emitted
18+
* by the source Observable. If the source emits fewer than `count` values then
19+
* all of its values are emitted. After that, it completes, regardless if the
20+
* source completes.
21+
*
22+
* @example <caption>Take the first 5 seconds of an infinite 1-second interval Observable</caption>
23+
* var interval = Rx.Observable.interval(1000);
24+
* var five = interval.take(5);
25+
* five.subscribe(x => console.log(x));
26+
*
27+
* @see {@link takeLast}
28+
* @see {@link takeUntil}
29+
* @see {@link takeWhile}
30+
* @see {@link skip}
31+
*
32+
* @throws {ArgumentOutOfRangeError} When using `take(i)`, it delivers an
33+
* ArgumentOutOrRangeError to the Observer's `error` callback if `i < 0`.
34+
*
35+
* @param {number} count The maximum number of `next` values to emit.
36+
* @return {Observable<T>} An Observable that emits only the first `count`
37+
* values emitted by the source Observable, or all of the values from the source
38+
* if the source emits fewer than `count` values.
39+
* @method take
40+
* @owner Observable
41+
*/
42+
export function take<T>(count: number): MonoTypeOperatorFunction<T> {
43+
return (source: Observable<T>) => {
44+
if (count === 0) {
45+
return new EmptyObservable<T>();
46+
} else {
47+
return source.lift(new TakeOperator(count));
48+
}
49+
};
50+
}
51+
52+
class TakeOperator<T> implements Operator<T, T> {
53+
constructor(private total: number) {
54+
if (this.total < 0) {
55+
throw new ArgumentOutOfRangeError;
56+
}
57+
}
58+
59+
call(subscriber: Subscriber<T>, source: any): TeardownLogic {
60+
return source.subscribe(new TakeSubscriber(subscriber, this.total));
61+
}
62+
}
63+
64+
/**
65+
* We need this JSDoc comment for affecting ESDoc.
66+
* @ignore
67+
* @extends {Ignored}
68+
*/
69+
class TakeSubscriber<T> extends Subscriber<T> {
70+
private count: number = 0;
71+
72+
constructor(destination: Subscriber<T>, private total: number) {
73+
super(destination);
74+
}
75+
76+
protected _next(value: T): void {
77+
const total = this.total;
78+
const count = ++this.count;
79+
if (count <= total) {
80+
this.destination.next(value);
81+
if (count === total) {
82+
this.destination.complete();
83+
this.unsubscribe();
84+
}
85+
}
86+
}
87+
}

0 commit comments

Comments
 (0)