UNPKG

ix

Version:

The Interactive Extensions for JavaScript

54 lines (49 loc) 2.06 kB
import { AsyncIterableX } from '../asynciterablex.js'; import { toArray } from '../toarray.js'; import { OperatorAsyncFunction } from '../../interfaces.js'; import { ScanOptions } from './scanoptions.js'; import { throwIfAborted } from '../../aborterror.js'; /** @ignore */ export class ScanRightAsyncIterable<T, R> extends AsyncIterableX<R> { private _source: AsyncIterable<T>; private _fn: (acc: R, x: T, index: number, signal?: AbortSignal) => R | Promise<R>; private _seed?: T | R; private _hasSeed: boolean; constructor(source: AsyncIterable<T>, options: ScanOptions<T, R>) { super(); this._source = source; this._fn = options['callback']; this._hasSeed = options.hasOwnProperty('seed'); this._seed = options['seed']; } async *[Symbol.asyncIterator](signal?: AbortSignal) { throwIfAborted(signal); let hasValue = false; let acc = this._seed; const source = await toArray(this._source, signal); for (let offset = source.length - 1; offset >= 0; offset--) { const item = source[offset]; if (hasValue || (hasValue = this._hasSeed)) { acc = await this._fn(<R>acc, item, offset, signal); yield acc; } else { acc = item; hasValue = true; } } } } /** * Applies an accumulator function over an async-iterable sequence from the right and returns each intermediate result. * The specified seed value, if given, is used as the initial accumulator value. * * @template T The type of the elements in the source sequence. * @template R The type of the result of the aggregation. * @param {ScanOptions<T, R>} options The options including the accumulator function and seed. * @returns {OperatorAsyncFunction<T, R>} An async-enumerable sequence containing the accumulated values from the right. */ export function scanRight<T, R = T>(options: ScanOptions<T, R>): OperatorAsyncFunction<T, R> { return function scanRightOperatorFunction(source: AsyncIterable<T>): AsyncIterableX<R> { return new ScanRightAsyncIterable(source, options); }; }