/** * @license * Copyright Google Inc. All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import {NgModuleFactory, ɵisObservable as isObservable, ɵisPromise as isPromise} from '@angular/core'; import {Observable} from 'rxjs/Observable'; import {fromPromise} from 'rxjs/observable/fromPromise'; import {of } from 'rxjs/observable/of'; import {concatAll} from 'rxjs/operator/concatAll'; import {every} from 'rxjs/operator/every'; import * as l from 'rxjs/operator/last'; import {map} from 'rxjs/operator/map'; import {mergeAll} from 'rxjs/operator/mergeAll'; import {PRIMARY_OUTLET} from '../shared'; export function shallowEqualArrays(a: any[], b: any[]): boolean { if (a.length !== b.length) return false; for (let i = 0; i < a.length; ++i) { if (!shallowEqual(a[i], b[i])) return false; } return true; } export function shallowEqual(a: {[x: string]: any}, b: {[x: string]: any}): boolean { const k1 = Object.keys(a); const k2 = Object.keys(b); if (k1.length != k2.length) { return false; } let key: string; for (let i = 0; i < k1.length; i++) { key = k1[i]; if (a[key] !== b[key]) { return false; } } return true; } export function flatten(arr: T[][]): T[] { return Array.prototype.concat.apply([], arr); } export function last(a: T[]): T|null { return a.length > 0 ? a[a.length - 1] : null; } export function and(bools: boolean[]): boolean { return !bools.some(v => !v); } export function forEach(map: {[key: string]: V}, callback: (v: V, k: string) => void): void { for (const prop in map) { if (map.hasOwnProperty(prop)) { callback(map[prop], prop); } } } export function waitForMap( obj: {[k: string]: A}, fn: (k: string, a: A) => Observable): Observable<{[k: string]: B}> { if (Object.keys(obj).length === 0) { return of ({}) } const waitHead: Observable[] = []; const waitTail: Observable[] = []; const res: {[k: string]: B} = {}; forEach(obj, (a: A, k: string) => { const mapped = map.call(fn(k, a), (r: B) => res[k] = r); if (k === PRIMARY_OUTLET) { waitHead.push(mapped); } else { waitTail.push(mapped); } }); const concat$ = concatAll.call(of (...waitHead, ...waitTail)); const last$ = l.last.call(concat$); return map.call(last$, () => res); } export function andObservables(observables: Observable>): Observable { const merged$ = mergeAll.call(observables); return every.call(merged$, (result: any) => result === true); } export function wrapIntoObservable(value: T | NgModuleFactory| Promise| Observable): Observable { if (isObservable(value)) { return value; } if (isPromise(value)) { // Use `Promise.resolve()` to wrap promise-like instances. // Required ie when a Resolver returns a AngularJS `$q` promise to correctly trigger the // change detection. return fromPromise(Promise.resolve(value)); } return of (value); }