ix
Version:
The Interactive Extensions for JavaScript
54 lines (49 loc) • 2.06 kB
text/typescript
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);
};
}