/** * Copyright 2019 Google Inc. All Rights Reserved. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * http://www.apache.org/licenses/LICENSE-2.0 * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { Observable } from "../types.ts"; /** * Combines items from multiple observables. * The resulting observable emits array tuples whenever any of the given * observables emit, as long as every observable has emitted at least once. * The tuples contain the last emitted item from each observable. * * @typeparam T Type of items emitted by the observables. * @param os Observables to combine. * @returns Observable that emits tuples of items. */ export function combineLatest( o1: Observable, o2: Observable, ): Observable<[T1, T2]>; export function combineLatest( o1: Observable, o2: Observable, o3: Observable, ): Observable<[T1, T2, T3]>; export function combineLatest(...os: Array>): Observable; export function combineLatest(...os: Array>): Observable { const NONE = Symbol(); const latestValue: Array = os.map(() => NONE); return new ReadableStream( { async start(controller) { const forwarders = os.map(async (o, idx) => { const reader = o.getReader(); while (true) { const { value, done } = await reader.read(); if (done) { return; } latestValue[idx] = value!; if (!latestValue.includes(NONE)) { controller.enqueue([...(latestValue as T[])]); } } }); await Promise.all(forwarders); controller.close(); }, }, { highWaterMark: 0 }, ); }