UNPKG

@naturalcycles/js-lib

Version:

Standard library for universal (browser + Node.js) javascript

90 lines (82 loc) 2.5 kB
import { AsyncIterable2 } from '../iter/asyncIterable2.js' import { Iterable2 } from '../iter/iterable2.js' import type { Integer, Primitive } from '../types.js' /** * Returns an array with ranges from `from` up to (but not including) `to`. * * Right bound is Exclusive (not Inclusive), to comply with lodash _.range * * @example * range(3) // [0, 1, 2] * range(3, 6) // [ 3, 4, 5 ] * range(1, 10, 2) // [ 1, 3, 5, 7, 9 ] */ export function _range(toExcl: Integer): number[] export function _range(fromIncl: Integer, toExcl: Integer, step?: number): number[] export function _range(fromIncl: Integer, toExcl?: Integer, step = 1): number[] { if (toExcl === undefined) { toExcl = fromIncl fromIncl = 0 } const a: number[] = [] for (let i = fromIncl; i < toExcl; i += step) { a.push(i) } return a } /** * Returns an array of `length` filled with `fill` primitive value. * Performance-optimized implementation. * `Array.from({ length }, () => fill)` shows ~25x perf regression in benchmarks * * Fill is Primitive, because it's safe to shallow-copy. * If it was an object - it'll paste the same object reference, which can create bugs. */ export function _arrayFilled<T extends Primitive>(length: Integer, fill: T): T[] { // biome-ignore lint/style/useConsistentBuiltinInstantiation: ok return Array(length).fill(fill) } /** * Like _range, but returns an Iterable2. */ export function _rangeIterable(toExcl: Integer): Iterable2<number> export function _rangeIterable(fromIncl: Integer, toExcl: Integer, step?: number): Iterable2<number> export function _rangeIterable(fromIncl: Integer, toExcl?: Integer, step = 1): Iterable2<number> { if (toExcl === undefined) { toExcl = fromIncl fromIncl = 0 } return Iterable2.of({ *[Symbol.iterator]() { for (let i = fromIncl; i < toExcl; i += step) { yield i } }, }) } /** * Like _range, but returns an AsyncIterable2. */ export function _rangeAsyncIterable(toExcl: Integer): AsyncIterable2<number> export function _rangeAsyncIterable( fromIncl: Integer, toExcl: Integer, step?: number, ): AsyncIterable2<number> export function _rangeAsyncIterable( fromIncl: Integer, toExcl?: Integer, step = 1, ): AsyncIterable2<number> { if (toExcl === undefined) { toExcl = fromIncl fromIncl = 0 } return AsyncIterable2.of({ async *[Symbol.asyncIterator]() { for (let i = fromIncl; i < toExcl; i += step) { yield i } }, }) }