@reactivex/rxjs
Version:
Reactive Extensions for modern JavaScript
278 lines (242 loc) • 7.99 kB
text/typescript
import {Subscriber} from '../Subscriber';
import {Subscription} from '../Subscription';
import {Observable} from '../Observable';
import {Operator} from '../Operator';
import {Subject} from '../Subject';
import {Map} from '../util/Map';
import {FastMap} from '../util/FastMap';
/**
* Groups the items emitted by an Observable according to a specified criterion,
* and emits these grouped items as `GroupedObservables`, one
* {@link GroupedObservable} per group.
*
* <img src="./img/groupBy.png" width="100%">
*
* @param {function(value: T): K} keySelector a function that extracts the key
* for each item.
* @param {function(value: T): R} [elementSelector] a function that extracts the
* return element for each item.
* @param {function(grouped: GroupedObservable<K,R>): Observable<any>} [durationSelector]
* a function that returns an Observable to determine how long each group should
* exist.
* @return {Observable<GroupedObservable<K,R>>} an Observable that emits
* GroupedObservables, each of which corresponds to a unique key value and each
* of which emits those items from the source Observable that share that key
* value.
* @method groupBy
* @owner Observable
*/
export function groupBy<T, K, R>(keySelector: (value: T) => K,
elementSelector?: (value: T) => R,
durationSelector?: (grouped: GroupedObservable<K, R>) => Observable<any>): Observable<GroupedObservable<K, R>> {
return this.lift(new GroupByOperator(this, keySelector, elementSelector, durationSelector));
}
/* tslint:disable:max-line-length */
export interface GroupBySignature<T> {
<K>(keySelector: (value: T) => K): Observable<GroupedObservable<K, T>>;
<K>(keySelector: (value: T) => K, elementSelector: void, durationSelector: (grouped: GroupedObservable<K, T>) => Observable<any>): Observable<GroupedObservable<K, T>>;
<K, R>(keySelector: (value: T) => K, elementSelector?: (value: T) => R, durationSelector?: (grouped: GroupedObservable<K, R>) => Observable<any>): Observable<GroupedObservable<K, R>>;
}
/* tslint:enable:max-line-length */
export interface RefCountSubscription {
count: number;
unsubscribe: () => void;
isUnsubscribed: boolean;
attemptedToUnsubscribe: boolean;
}
class GroupByOperator<T, K, R> implements Operator<T, GroupedObservable<K, R>> {
constructor(public source: Observable<T>,
private keySelector: (value: T) => K,
private elementSelector?: (value: T) => R,
private durationSelector?: (grouped: GroupedObservable<K, R>) => Observable<any>) {
}
call(subscriber: Subscriber<GroupedObservable<K, R>>, source: any): any {
return source._subscribe(new GroupBySubscriber(
subscriber, this.keySelector, this.elementSelector, this.durationSelector
));
}
}
/**
* We need this JSDoc comment for affecting ESDoc.
* @ignore
* @extends {Ignored}
*/
class GroupBySubscriber<T, K, R> extends Subscriber<T> implements RefCountSubscription {
private groups: Map<K, Subject<T|R>> = null;
public attemptedToUnsubscribe: boolean = false;
public count: number = 0;
constructor(destination: Subscriber<GroupedObservable<K, R>>,
private keySelector: (value: T) => K,
private elementSelector?: (value: T) => R,
private durationSelector?: (grouped: GroupedObservable<K, R>) => Observable<any>) {
super();
this.destination = destination;
this.add(destination);
}
protected _next(value: T): void {
let key: K;
try {
key = this.keySelector(value);
} catch (err) {
this.error(err);
return;
}
this._group(value, key);
}
private _group(value: T, key: K) {
let groups = this.groups;
if (!groups) {
groups = this.groups = typeof key === 'string' ? new FastMap() : new Map();
}
let group = groups.get(key);
if (!group) {
groups.set(key, group = new Subject<R>());
const groupedObservable = new GroupedObservable(key, group, this);
if (this.durationSelector) {
this._selectDuration(key, group);
}
this.destination.next(groupedObservable);
}
if (this.elementSelector) {
this._selectElement(value, group);
} else {
this.tryGroupNext(value, group);
}
}
private _selectElement(value: T, group: Subject<T | R>) {
let result: R;
try {
result = this.elementSelector(value);
} catch (err) {
this.error(err);
return;
}
this.tryGroupNext(result, group);
}
private _selectDuration(key: K, group: any) {
let duration: any;
try {
duration = this.durationSelector(new GroupedObservable<K, R>(key, group));
} catch (err) {
this.error(err);
return;
}
this.add(duration.subscribe(new GroupDurationSubscriber(key, group, this)));
}
private tryGroupNext(value: T|R, group: Subject<T | R>): void {
if (!group.isUnsubscribed) {
group.next(value);
}
}
protected _error(err: any): void {
const groups = this.groups;
if (groups) {
groups.forEach((group, key) => {
group.error(err);
});
groups.clear();
}
this.destination.error(err);
}
protected _complete(): void {
const groups = this.groups;
if (groups) {
groups.forEach((group, key) => {
group.complete();
});
groups.clear();
}
this.destination.complete();
}
removeGroup(key: K): void {
this.groups.delete(key);
}
unsubscribe() {
if (!this.isUnsubscribed && !this.attemptedToUnsubscribe) {
this.attemptedToUnsubscribe = true;
if (this.count === 0) {
super.unsubscribe();
}
}
}
}
/**
* We need this JSDoc comment for affecting ESDoc.
* @ignore
* @extends {Ignored}
*/
class GroupDurationSubscriber<K, T> extends Subscriber<T> {
constructor(private key: K,
private group: Subject<T>,
private parent: GroupBySubscriber<any, K, T>) {
super();
}
protected _next(value: T): void {
this.tryComplete();
}
protected _error(err: any): void {
this.tryError(err);
}
protected _complete(): void {
this.tryComplete();
}
private tryError(err: any): void {
const group = this.group;
if (!group.isUnsubscribed) {
group.error(err);
}
this.parent.removeGroup(this.key);
}
private tryComplete(): void {
const group = this.group;
if (!group.isUnsubscribed) {
group.complete();
}
this.parent.removeGroup(this.key);
}
}
/**
* An Observable representing values belonging to the same group represented by
* a common key. The values emitted by a GroupedObservable come from the source
* Observable. The common key is available as the field `key` on a
* GroupedObservable instance.
*
* @class GroupedObservable<K, T>
*/
export class GroupedObservable<K, T> extends Observable<T> {
constructor(public key: K,
private groupSubject: Subject<T>,
private refCountSubscription?: RefCountSubscription) {
super();
}
protected _subscribe(subscriber: Subscriber<T>) {
const subscription = new Subscription();
const {refCountSubscription, groupSubject} = this;
if (refCountSubscription && !refCountSubscription.isUnsubscribed) {
subscription.add(new InnerRefCountSubscription(refCountSubscription));
}
subscription.add(groupSubject.subscribe(subscriber));
return subscription;
}
}
/**
* We need this JSDoc comment for affecting ESDoc.
* @ignore
* @extends {Ignored}
*/
class InnerRefCountSubscription extends Subscription {
constructor(private parent: RefCountSubscription) {
super();
parent.count++;
}
unsubscribe() {
const parent = this.parent;
if (!parent.isUnsubscribed && !this.isUnsubscribed) {
super.unsubscribe();
parent.count -= 1;
if (parent.count === 0 && parent.attemptedToUnsubscribe) {
parent.unsubscribe();
}
}
}
}