From 07827eee879f3dbc5da3d7942a4ea3cf927f5d1a Mon Sep 17 00:00:00 2001 From: OJ Kwon Date: Fri, 19 Feb 2016 11:42:14 -0800 Subject: [PATCH] fix(forkJoin): accpets observables emitting null or undefined closes #1362 --- spec/observables/forkJoin-spec.js | 12 ++++++++++ src/observable/ForkJoinObservable.ts | 35 ++++++++++++++-------------- 2 files changed, 30 insertions(+), 17 deletions(-) diff --git a/spec/observables/forkJoin-spec.js b/spec/observables/forkJoin-spec.js index 4b7531a037..056cefb334 100644 --- a/spec/observables/forkJoin-spec.js +++ b/spec/observables/forkJoin-spec.js @@ -13,6 +13,18 @@ describe('Observable.forkJoin', function () { var expected = '--------------(x|)'; expectObservable(e1).toBe(expected, {x: ['d', 'b', '3']}); + + //Hack - just adding one new test cases in here or either jasmin-is-weird-spec.js, + //one of test breaks under publish-spec.js + var e2 = Observable.forkJoin( + hot('--a--b--c--d--|', { d: null }), + hot('(b|)'), + hot('--1--2--3--|'), + hot('-----r--t--u--|', { u: undefined }) + ); + var expected2 = '--------------(x|)'; + + expectObservable(e2).toBe(expected2, {x: [null, 'b', '3', undefined]}); }); it('should join the last values of the provided observables with selector', function () { diff --git a/src/observable/ForkJoinObservable.ts b/src/observable/ForkJoinObservable.ts index a9df73b307..4ac2134620 100644 --- a/src/observable/ForkJoinObservable.ts +++ b/src/observable/ForkJoinObservable.ts @@ -40,7 +40,12 @@ export class ForkJoinObservable extends Observable { const sources = this.sources; const len = sources.length; - const context = { completed: 0, total: len, values: emptyArray(len), selector: this.resultSelector }; + const context = { completed: 0, + total: len, + values: new Array(len), + haveValues: new Array(len), + selector: this.resultSelector }; + for (let i = 0; i < len; i++) { let source = sources[i]; if (isPromise(source)) { @@ -52,39 +57,43 @@ export class ForkJoinObservable extends Observable { } class AllSubscriber extends Subscriber { - private _value: T = null; constructor(destination: Subscriber, private index: number, private context: { completed: number, total: number, values: any[], + haveValues: any[], selector: (...values: Array) => any }) { super(destination); } protected _next(value: T): void { - this._value = value; + const context = this.context; + const index = this.index; + + context.values[index] = value; + context.haveValues[index] = true; } protected _complete(): void { const destination = this.destination; + const context = this.context; - if (this._value == null) { + if (!context.haveValues[this.index]) { destination.complete(); } - const context = this.context; context.completed++; - context.values[this.index] = this._value; + const values = context.values; if (context.completed !== values.length) { return; } - if (values.every(hasValue)) { - let value = context.selector ? context.selector.apply(this, values) : + if (context.haveValues.every(hasValue)) { + const value = context.selector ? context.selector.apply(this, values) : values; destination.next(value); } @@ -94,13 +103,5 @@ class AllSubscriber extends Subscriber { } function hasValue(x: any): boolean { - return x !== null; -} - -function emptyArray(len: number): any[] { - let arr: any[] = []; - for (let i = 0; i < len; i++) { - arr.push(null); - } - return arr; + return x === true; }