Skip to content

Commit 9974fc2

Browse files
committed
feat(operators): higher-order lettables of reduce, min, max and defaultIfEmpty added
1 parent cd7e7dd commit 9974fc2

File tree

9 files changed

+249
-111
lines changed

9 files changed

+249
-111
lines changed

src/operator/defaultIfEmpty.ts

+3-38
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
1-
import { Operator } from '../Operator';
1+
22
import { Observable } from '../Observable';
3-
import { Subscriber } from '../Subscriber';
3+
import { defaultIfEmpty as higherOrder } from '../operators';
44

55
/* tslint:disable:max-line-length */
66
export function defaultIfEmpty<T>(this: Observable<T>, defaultValue?: T): Observable<T>;
@@ -38,40 +38,5 @@ export function defaultIfEmpty<T, R>(this: Observable<T>, defaultValue?: R): Obs
3838
* @owner Observable
3939
*/
4040
export function defaultIfEmpty<T, R>(this: Observable<T>, defaultValue: R = null): Observable<T | R> {
41-
return this.lift(new DefaultIfEmptyOperator(defaultValue));
42-
}
43-
44-
class DefaultIfEmptyOperator<T, R> implements Operator<T, T | R> {
45-
46-
constructor(private defaultValue: R) {
47-
}
48-
49-
call(subscriber: Subscriber<T | R>, source: any): any {
50-
return source.subscribe(new DefaultIfEmptySubscriber(subscriber, this.defaultValue));
51-
}
52-
}
53-
54-
/**
55-
* We need this JSDoc comment for affecting ESDoc.
56-
* @ignore
57-
* @extends {Ignored}
58-
*/
59-
class DefaultIfEmptySubscriber<T, R> extends Subscriber<T> {
60-
private isEmpty: boolean = true;
61-
62-
constructor(destination: Subscriber<T | R>, private defaultValue: R) {
63-
super(destination);
64-
}
65-
66-
protected _next(value: T): void {
67-
this.isEmpty = false;
68-
this.destination.next(value);
69-
}
70-
71-
protected _complete(): void {
72-
if (this.isEmpty) {
73-
this.destination.next(this.defaultValue);
74-
}
75-
this.destination.complete();
76-
}
41+
return higherOrder<T, R>(defaultValue)(this);
7742
}

src/operator/max.ts

+2-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { Observable } from '../Observable';
2-
import { ReduceOperator } from './reduce';
2+
import { max as higherOrderMax } from '../operators';
33

44
/**
55
* The Max operator operates on an Observable that emits numbers (or items that can be compared with a provided function),
@@ -33,8 +33,5 @@ import { ReduceOperator } from './reduce';
3333
* @owner Observable
3434
*/
3535
export function max<T>(this: Observable<T>, comparer?: (x: T, y: T) => number): Observable<T> {
36-
const max: (x: T, y: T) => T = (typeof comparer === 'function')
37-
? (x, y) => comparer(x, y) > 0 ? x : y
38-
: (x, y) => x > y ? x : y;
39-
return this.lift(new ReduceOperator(max));
36+
return higherOrderMax(comparer)(this);
4037
}

src/operator/min.ts

+2-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { Observable } from '../Observable';
2-
import { ReduceOperator } from './reduce';
2+
import { min as higherOrderMin } from '../operators';
33

44
/**
55
* The Min operator operates on an Observable that emits numbers (or items that can be compared with a provided function),
@@ -33,8 +33,5 @@ import { ReduceOperator } from './reduce';
3333
* @owner Observable
3434
*/
3535
export function min<T>(this: Observable<T>, comparer?: (x: T, y: T) => number): Observable<T> {
36-
const min: (x: T, y: T) => T = (typeof comparer === 'function')
37-
? (x, y) => comparer(x, y) < 0 ? x : y
38-
: (x, y) => x < y ? x : y;
39-
return this.lift(new ReduceOperator(min));
36+
return higherOrderMin(comparer)(this);
4037
}

src/operator/reduce.ts

+3-63
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
import { Observable } from '../Observable';
2-
import { Operator } from '../Operator';
3-
import { Subscriber } from '../Subscriber';
2+
import { reduce as higherOrderReduce } from '../operators';
43

54
/* tslint:disable:max-line-length */
65
export function reduce<T>(this: Observable<T>, accumulator: (acc: T, value: T, index: number) => T, seed?: T): Observable<T>;
@@ -53,73 +52,14 @@ export function reduce<T, R>(this: Observable<T>, accumulator: (acc: R, value: T
5352
* @owner Observable
5453
*/
5554
export function reduce<T, R>(this: Observable<T>, accumulator: (acc: R, value: T, index?: number) => R, seed?: R): Observable<R> {
56-
let hasSeed = false;
5755
// providing a seed of `undefined` *should* be valid and trigger
5856
// hasSeed! so don't use `seed !== undefined` checks!
5957
// For this reason, we have to check it here at the original call site
6058
// otherwise inside Operator/Subscriber we won't know if `undefined`
6159
// means they didn't provide anything or if they literally provided `undefined`
6260
if (arguments.length >= 2) {
63-
hasSeed = true;
61+
return higherOrderReduce(accumulator, seed)(this);
6462
}
6563

66-
return this.lift(new ReduceOperator(accumulator, seed, hasSeed));
67-
}
68-
69-
export class ReduceOperator<T, R> implements Operator<T, R> {
70-
constructor(private accumulator: (acc: R, value: T, index?: number) => R, private seed?: R, private hasSeed: boolean = false) {}
71-
72-
call(subscriber: Subscriber<R>, source: any): any {
73-
return source.subscribe(new ReduceSubscriber(subscriber, this.accumulator, this.seed, this.hasSeed));
74-
}
75-
}
76-
77-
/**
78-
* We need this JSDoc comment for affecting ESDoc.
79-
* @ignore
80-
* @extends {Ignored}
81-
*/
82-
export class ReduceSubscriber<T, R> extends Subscriber<T> {
83-
private index: number = 0;
84-
private acc: T | R;
85-
private hasValue: boolean = false;
86-
87-
constructor(destination: Subscriber<R>,
88-
private accumulator: (acc: R, value: T, index?: number) => R,
89-
seed: R,
90-
private hasSeed: boolean) {
91-
super(destination);
92-
this.acc = seed;
93-
94-
if (!this.hasSeed) {
95-
this.index++;
96-
}
97-
}
98-
99-
protected _next(value: T) {
100-
if (this.hasValue || (this.hasValue = this.hasSeed)) {
101-
this._tryReduce(value);
102-
} else {
103-
this.acc = value;
104-
this.hasValue = true;
105-
}
106-
}
107-
108-
private _tryReduce(value: T) {
109-
let result: any;
110-
try {
111-
result = this.accumulator(<R>this.acc, value, this.index++);
112-
} catch (err) {
113-
this.destination.error(err);
114-
return;
115-
}
116-
this.acc = result;
117-
}
118-
119-
protected _complete() {
120-
if (this.hasValue || this.hasSeed) {
121-
this.destination.next(this.acc);
122-
}
123-
this.destination.complete();
124-
}
64+
return higherOrderReduce(accumulator)(this);
12565
}

src/operators/defaultIfEmpty.ts

+80
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
import { Operator } from '../Operator';
2+
import { Observable } from '../Observable';
3+
import { Subscriber } from '../Subscriber';
4+
import { OperatorFunction } from '../interfaces';
5+
6+
/* tslint:disable:max-line-length */
7+
export function defaultIfEmpty<T>(defaultValue?: T): OperatorFunction<T, T>;
8+
export function defaultIfEmpty<T, R>(defaultValue?: R): OperatorFunction<T, T | R>;
9+
/* tslint:enable:max-line-length */
10+
11+
/**
12+
* Emits a given value if the source Observable completes without emitting any
13+
* `next` value, otherwise mirrors the source Observable.
14+
*
15+
* <span class="informal">If the source Observable turns out to be empty, then
16+
* this operator will emit a default value.</span>
17+
*
18+
* <img src="./img/defaultIfEmpty.png" width="100%">
19+
*
20+
* `defaultIfEmpty` emits the values emitted by the source Observable or a
21+
* specified default value if the source Observable is empty (completes without
22+
* having emitted any `next` value).
23+
*
24+
* @example <caption>If no clicks happen in 5 seconds, then emit "no clicks"</caption>
25+
* var clicks = Rx.Observable.fromEvent(document, 'click');
26+
* var clicksBeforeFive = clicks.takeUntil(Rx.Observable.interval(5000));
27+
* var result = clicksBeforeFive.defaultIfEmpty('no clicks');
28+
* result.subscribe(x => console.log(x));
29+
*
30+
* @see {@link empty}
31+
* @see {@link last}
32+
*
33+
* @param {any} [defaultValue=null] The default value used if the source
34+
* Observable is empty.
35+
* @return {Observable} An Observable that emits either the specified
36+
* `defaultValue` if the source Observable emits no items, or the values emitted
37+
* by the source Observable.
38+
* @method defaultIfEmpty
39+
* @owner Observable
40+
*/
41+
export function defaultIfEmpty<T, R>(defaultValue: R = null): OperatorFunction<T, T | R> {
42+
return function defaultIfEmptyOperatorFunction(source: Observable<T>) {
43+
return source.lift(new DefaultIfEmptyOperator(defaultValue));
44+
};
45+
}
46+
47+
class DefaultIfEmptyOperator<T, R> implements Operator<T, T | R> {
48+
49+
constructor(private defaultValue: R) {
50+
}
51+
52+
call(subscriber: Subscriber<T | R>, source: any): any {
53+
return source.subscribe(new DefaultIfEmptySubscriber(subscriber, this.defaultValue));
54+
}
55+
}
56+
57+
/**
58+
* We need this JSDoc comment for affecting ESDoc.
59+
* @ignore
60+
* @extends {Ignored}
61+
*/
62+
class DefaultIfEmptySubscriber<T, R> extends Subscriber<T> {
63+
private isEmpty: boolean = true;
64+
65+
constructor(destination: Subscriber<T | R>, private defaultValue: R) {
66+
super(destination);
67+
}
68+
69+
protected _next(value: T): void {
70+
this.isEmpty = false;
71+
this.destination.next(value);
72+
}
73+
74+
protected _complete(): void {
75+
if (this.isEmpty) {
76+
this.destination.next(this.defaultValue);
77+
}
78+
this.destination.complete();
79+
}
80+
}

src/operators/index.ts

+4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,11 @@
11
export { concatMap } from './concatMap';
2+
export { defaultIfEmpty } from './defaultIfEmpty';
23
export { filter } from './filter';
34
export { map } from './map';
5+
export { max } from './max';
46
export { mergeMap } from './mergeMap';
7+
export { min } from './min';
8+
export { reduce } from './reduce';
59
export { scan } from './scan';
610
export { switchMap } from './switchMap';
711
export { takeLast } from './takeLast';

src/operators/max.ts

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import { reduce } from './reduce';
2+
import { OperatorFunction } from '../interfaces';
3+
4+
/**
5+
* The Max operator operates on an Observable that emits numbers (or items that can be compared with a provided function),
6+
* and when source Observable completes it emits a single item: the item with the largest value.
7+
*
8+
* <img src="./img/max.png" width="100%">
9+
*
10+
* @example <caption>Get the maximal value of a series of numbers</caption>
11+
* Rx.Observable.of(5, 4, 7, 2, 8)
12+
* .max()
13+
* .subscribe(x => console.log(x)); // -> 8
14+
*
15+
* @example <caption>Use a comparer function to get the maximal item</caption>
16+
* interface Person {
17+
* age: number,
18+
* name: string
19+
* }
20+
* Observable.of<Person>({age: 7, name: 'Foo'},
21+
* {age: 5, name: 'Bar'},
22+
* {age: 9, name: 'Beer'})
23+
* .max<Person>((a: Person, b: Person) => a.age < b.age ? -1 : 1)
24+
* .subscribe((x: Person) => console.log(x.name)); // -> 'Beer'
25+
* }
26+
*
27+
* @see {@link min}
28+
*
29+
* @param {Function} [comparer] - Optional comparer function that it will use instead of its default to compare the
30+
* value of two items.
31+
* @return {Observable} An Observable that emits item with the largest value.
32+
* @method max
33+
* @owner Observable
34+
*/
35+
export function max<T>(comparer?: (x: T, y: T) => number): OperatorFunction<T, T> {
36+
const max: (x: T, y: T) => T = (typeof comparer === 'function')
37+
? (x, y) => comparer(x, y) > 0 ? x : y
38+
: (x, y) => x > y ? x : y;
39+
40+
return reduce(max);
41+
}

src/operators/min.ts

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
import { reduce } from './reduce';
2+
import { OperatorFunction } from '../interfaces';
3+
4+
/**
5+
* The Min operator operates on an Observable that emits numbers (or items that can be compared with a provided function),
6+
* and when source Observable completes it emits a single item: the item with the smallest value.
7+
*
8+
* <img src="./img/min.png" width="100%">
9+
*
10+
* @example <caption>Get the minimal value of a series of numbers</caption>
11+
* Rx.Observable.of(5, 4, 7, 2, 8)
12+
* .min()
13+
* .subscribe(x => console.log(x)); // -> 2
14+
*
15+
* @example <caption>Use a comparer function to get the minimal item</caption>
16+
* interface Person {
17+
* age: number,
18+
* name: string
19+
* }
20+
* Observable.of<Person>({age: 7, name: 'Foo'},
21+
* {age: 5, name: 'Bar'},
22+
* {age: 9, name: 'Beer'})
23+
* .min<Person>( (a: Person, b: Person) => a.age < b.age ? -1 : 1)
24+
* .subscribe((x: Person) => console.log(x.name)); // -> 'Bar'
25+
* }
26+
*
27+
* @see {@link max}
28+
*
29+
* @param {Function} [comparer] - Optional comparer function that it will use instead of its default to compare the
30+
* value of two items.
31+
* @return {Observable<R>} An Observable that emits item with the smallest value.
32+
* @method min
33+
* @owner Observable
34+
*/
35+
export function min<T>(comparer?: (x: T, y: T) => number): OperatorFunction<T, T> {
36+
const min: (x: T, y: T) => T = (typeof comparer === 'function')
37+
? (x, y) => comparer(x, y) < 0 ? x : y
38+
: (x, y) => x < y ? x : y;
39+
return reduce(min);
40+
}

0 commit comments

Comments
 (0)