/* Copyright 2023, 2024 New Vector Ltd. SPDX-License-Identifier: AGPL-3.0-only OR LicenseRef-Element-Commercial Please see LICENSE in the repository root for full details. */ import { type Observable, audit, combineLatest, concat, defer, filter, finalize, map, of, scan, startWith, takeWhile, tap, withLatestFrom, } from "rxjs"; import { type Behavior } from "../state/Behavior"; const nothing = Symbol("nothing"); /** * RxJS operator that invokes a callback when the Observable is finalized, * passing the most recently emitted value. If no value was emitted, the * callback will not be invoked. */ export function finalizeValue(callback: (finalValue: T) => void) { return (source$: Observable): Observable => defer(() => { let finalValue: T | typeof nothing = nothing; return source$.pipe( tap((value) => (finalValue = value)), finalize(() => { if (finalValue !== nothing) callback(finalValue); }), ); }); } /** * RxJS operator that accumulates a state from a source of events. This is like * scan, except it emits an initial value immediately before any events arrive. */ export function accumulate( initial: State, update: (state: State, event: Event) => State, ) { return (events$: Observable): Observable => events$.pipe(scan(update, initial), startWith(initial)); } const switchSymbol = Symbol("switch"); /** * RxJS operator which behaves like the input Observable (A) until it emits a * value satisfying the given predicate, then behaves like Observable B. * * The switch is immediate; the value that triggers the switch will not be * present in the output. */ export function switchWhen( predicate: (a: A, index: number) => boolean, b$: Observable, ) { return (a$: Observable): Observable => concat( a$.pipe( map((a, index) => (predicate(a, index) ? switchSymbol : a)), takeWhile((a) => a !== switchSymbol), ) as Observable, b$, ); } /** * Reads the current value of a state Observable without reacting to future * changes. * * This function exists to help with certain cases of bridging Observables into * React, where an initial value is needed. You should never use it to create an * Observable derived from another Observable; use reactive operators instead. */ export function getValue(state$: Observable): T { let value: T | typeof nothing = nothing; state$.subscribe((x) => (value = x)).unsubscribe(); if (value === nothing) throw new Error("Not a state Observable"); return value; } /** * Creates an Observable that has a value of true whenever all its inputs are * true. */ export function and$(...inputs: Observable[]): Observable { return combineLatest(inputs, (...flags) => flags.every((flag) => flag)); } /** * RxJS operator that pauses all changes in the input value whenever a Behavior * is true. When the Behavior returns to being false, the most recently * suppressed change is emitted as the most recent value. */ export function pauseWhen(pause$: Behavior) { return (value$: Observable): Observable => value$.pipe( withLatestFrom(pause$), audit(([, pause]) => pause ? pause$.pipe(filter((pause) => !pause)) : of(null), ), map(([value]) => value), ); }