UNPKG

ix

Version:

The Interactive Extensions for JavaScript

69 lines (64 loc) 3.11 kB
import { AsyncIterableX } from './asynciterablex.js'; import { sleep } from './_sleep.js'; import { throwIfAborted } from '../aborterror.js'; class GenerateTimeAsyncIterable<TState, TResult> extends AsyncIterableX<TResult> { private _initialState: TState; private _condition: (value: TState, signal?: AbortSignal) => boolean | Promise<boolean>; private _iterate: (value: TState, signal?: AbortSignal) => TState | Promise<TState>; private _resultSelector: (value: TState, signal?: AbortSignal) => TResult | Promise<TResult>; private _timeSelector: (value: TState, signal?: AbortSignal) => number | Promise<number>; constructor( initialState: TState, condition: (value: TState, signal?: AbortSignal) => boolean | Promise<boolean>, iterate: (value: TState, signal?: AbortSignal) => TState | Promise<TState>, resultSelector: (value: TState, signal?: AbortSignal) => TResult | Promise<TResult>, timeSelector: (value: TState, signal?: AbortSignal) => number | Promise<number> ) { super(); this._initialState = initialState; this._condition = condition; this._iterate = iterate; this._resultSelector = resultSelector; this._timeSelector = timeSelector; } async *[Symbol.asyncIterator](signal?: AbortSignal) { throwIfAborted(signal); for ( let i = this._initialState; await this._condition(i, signal); i = await this._iterate(i, signal) ) { const time = await this._timeSelector(i, signal); await sleep(time, signal); yield await this._resultSelector(i, signal); } } } /** * Generates an async-iterable sequence by running a time-based state-driven loop producing the sequence's elements. * * @template TState The type of the state used in the generator loop. * @template TResult The type of the elements in the produced sequence. * @param {TState} initialState The initial state. * @param {((value: TState, signal?: AbortSignal) => boolean | Promise<boolean>)} condition Condition to terminate generation (upon returning false). * @param {((value: TState, signal?: AbortSignal) => TState | Promise<TState>)} iterate Iteration step function. * @param {((value: TState, signal?: AbortSignal) => TResult | Promise<TResult>)} resultSelector Selector function for results produced in * the sequence. * @param {((value: TState, signal?: AbortSignal) => number | Promise<number>)} timeSelector Selector function for how much time to wait. * @returns {AsyncIterableX<TResult>} The generated async-iterable sequence. */ export function generateTime<TState, TResult>( initialState: TState, condition: (value: TState, signal?: AbortSignal) => boolean | Promise<boolean>, iterate: (value: TState, signal?: AbortSignal) => TState | Promise<TState>, resultSelector: (value: TState, signal?: AbortSignal) => TResult | Promise<TResult>, timeSelector: (value: TState, signal?: AbortSignal) => number | Promise<number> ): AsyncIterableX<TResult> { return new GenerateTimeAsyncIterable<TState, TResult>( initialState, condition, iterate, resultSelector, timeSelector ); }