/* 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, defer, finalize, scan, startWith, tap } from "rxjs"; 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)); } /** * 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; }