Skip to content

Commit 8ab0914

Browse files
committed
feat(pluck): add higher-order lettable version of pluck
1 parent 595e588 commit 8ab0914

File tree

3 files changed

+57
-23
lines changed

3 files changed

+57
-23
lines changed

src/operator/pluck.ts

+2-23
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { Observable } from '../Observable';
2-
import { map } from './map';
2+
import { pluck as higherOrder } from '../operators/pluck';
33

44
/**
55
* Maps each source value (an object) to its specified nested property.
@@ -28,26 +28,5 @@ import { map } from './map';
2828
* @owner Observable
2929
*/
3030
export function pluck<T, R>(this: Observable<T>, ...properties: string[]): Observable<R> {
31-
const length = properties.length;
32-
if (length === 0) {
33-
throw new Error('list of properties cannot be empty.');
34-
}
35-
return map.call(this, plucker(properties, length));
36-
}
37-
38-
function plucker(props: string[], length: number): (x: string) => any {
39-
const mapper = (x: string) => {
40-
let currentProp = x;
41-
for (let i = 0; i < length; i++) {
42-
const p = currentProp[props[i]];
43-
if (typeof p !== 'undefined') {
44-
currentProp = p;
45-
} else {
46-
return undefined;
47-
}
48-
}
49-
return currentProp;
50-
};
51-
52-
return mapper;
31+
return higherOrder(...properties)(this);
5332
}

src/operators/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@ export { observeOn } from './observeOn';
4848
export { onErrorResumeNext } from './onErrorResumeNext';
4949
export { pairwise } from './pairwise';
5050
export { partition } from './partition';
51+
export { pluck } from './pluck';
5152
export { publish } from './publish';
5253
export { race } from './race';
5354
export { reduce } from './reduce';

src/operators/pluck.ts

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
import { Observable } from '../Observable';
2+
import { map } from './map';
3+
import { OperatorFunction } from '../interfaces';
4+
5+
/**
6+
* Maps each source value (an object) to its specified nested property.
7+
*
8+
* <span class="informal">Like {@link map}, but meant only for picking one of
9+
* the nested properties of every emitted object.</span>
10+
*
11+
* <img src="./img/pluck.png" width="100%">
12+
*
13+
* Given a list of strings describing a path to an object property, retrieves
14+
* the value of a specified nested property from all values in the source
15+
* Observable. If a property can't be resolved, it will return `undefined` for
16+
* that value.
17+
*
18+
* @example <caption>Map every click to the tagName of the clicked target element</caption>
19+
* var clicks = Rx.Observable.fromEvent(document, 'click');
20+
* var tagNames = clicks.pluck('target', 'tagName');
21+
* tagNames.subscribe(x => console.log(x));
22+
*
23+
* @see {@link map}
24+
*
25+
* @param {...string} properties The nested properties to pluck from each source
26+
* value (an object).
27+
* @return {Observable} A new Observable of property values from the source values.
28+
* @method pluck
29+
* @owner Observable
30+
*/
31+
export function pluck<T, R>(...properties: string[]): OperatorFunction<T, R> {
32+
const length = properties.length;
33+
if (length === 0) {
34+
throw new Error('list of properties cannot be empty.');
35+
}
36+
return (source: Observable<T>) => map(plucker(properties, length))(source as any);
37+
}
38+
39+
function plucker(props: string[], length: number): (x: string) => any {
40+
const mapper = (x: string) => {
41+
let currentProp = x;
42+
for (let i = 0; i < length; i++) {
43+
const p = currentProp[props[i]];
44+
if (typeof p !== 'undefined') {
45+
currentProp = p;
46+
} else {
47+
return undefined;
48+
}
49+
}
50+
return currentProp;
51+
};
52+
53+
return mapper;
54+
}

0 commit comments

Comments
 (0)