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