Skip to content

Commit

Permalink
v1.0.0
Browse files Browse the repository at this point in the history
  • Loading branch information
GreLI committed Mar 29, 2021
1 parent 39bc8ae commit 12357e6
Show file tree
Hide file tree
Showing 7 changed files with 2,724 additions and 0 deletions.
6 changes: 6 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
node_modules/
*.log
.DS_Store
.idea
.vscode
.npmrc
1 change: 1 addition & 0 deletions .npmignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
test.js
110 changes: 110 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
# ES Range generator for javascript

ES Range generator is zero-dependency tiny module that provides `range()` function
similar to one that's in Python. Its main purpose is to use in `for-of` loops
with a far more convenient syntax than common hacks like

```javascript
Array.from(Array(10), (_, i) => i + 1).forEach(i => { /*...*/ })
```

Usage:
```javascript
import range from "es-range-generator";

for (let i of range(1, 10)) {
console.log(i);
}
// → 1
// → 2
// → 3
// ...
```

It supports 1-argument syntax, higher-to-lower values ranges, floating point values, bigints
and built upon ES2015 generators:

```javascript
[...range(5)] // → [0, 1, 2, 3, 4]
[...range(-5)] // → [0, -1, -2, -3, -4]
[...range(3, 0, -1)] // → [3, 2, 1]
[...range(10, 5)] // → [10, 9, 8, 7, 6]
[...range(.25, 1, .125)] // → [.25, .375, .5, .625, .75, .875]
[...range(1e16, 1e16 + 100)] // → [10000000000000000n, 10000000000000001n, ...]
```

## Why another generator?

There are plenty of other python-like range functions in the wild,
however, all of them lack noticable features.

Almost none of them uses ES2015 generators, producing an extra array.
This one returns iterator which produces values lazily, only when needed.
It is the nature of iterators.

Many of them doesn't support negative steps, noone support BigInt.
They even can lead to infinite loop in certain conditions!
In a browser this would cause tab to hang for a while and crash.

Some other are bloated with functions, while this one takes only
around 600 bytes in total.

## How it works

As stated in its name, range generator uses ES2015 generator
returning an iterator, which is the most natural way for implementing.
Just like it's done in Python. Of course this module is an ES module.
It made as simple as possible.

Following with JS way, its params are converted to numbers.
The only exception is when numbers are larger than `MAX_SAFE_INTEGER`
(or lower than `MIN_SAFE_INTEGER`, accordingly)—in that case
they are casted to BigInt. BigInt input is respected as well.
In such a case BigInt is an essential requirement.
However, it's not needed if that is not the case.

While this module is written to prevent possible infinite loops
one still can pass large values which will be iterated
above the reasonable time. Such a case is beyond the scope.
Be careful.

No transpilation or polyfill is used in this module. Common, it's 2021!
Microsoft already replaces Internet Explorer in Windows updates.
If you need support for such an old browser you should already know what to do.

## Usage

Usage of range is quite intuitive, but nevertheless here it is:

```
range(stop)
range(start, stop, [step])
```

Range is iterated from `start` to `stop` (excluding) with increment equal to `step`.
When only one argument is passed it's considered as `stop`
with default `start` value equal `0`. Usually `step` defaults to `1`,
however in couple of cases it can be equal `-1`.
It's somewhat that is not in Python implementation
(wonder, why it's not there):

```javascript
[...range(-5)] // → [0, -1, -2, -3, -4]
[...range(10, 5)] // → [10, 9, 8, 7, 6]
```

The function can be used everywhere an iterator fits: in `for-of` loops,
with array spread operator, in `Array.from()`, function rest operator:

```javascript
import range from "es-range-generator"

for (let i of range(1, 10)) {
/* ... */
}
let arr1 = [...range(10)];
let arr2 = Array.from(range(10));
someFunc('list', ...range(2, 10, 2));
```

…and so on.
21 changes: 21 additions & 0 deletions index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
const N = Number,
{isNaN: i, MAX_SAFE_INTEGER: M} = N,
{abs, sign} = Math,
E = Error

export default function *range(n, m, s = 1) {
if (arguments.length < 2)
m = n, n = 0, s = sign(m)
else if (arguments.length < 3 && m < n)
s = -1
if (typeof n == "bigint" || typeof m == "bigint" || abs(n) > M || abs(m) > M)
n = BigInt(n), m = BigInt(m), s = BigInt(s)
else
n = N(n), m = N(m), s = N(s)
if (i(n) || i(m) || i(s)) throw new E("range() arguments must be numbers")
if (s == 0) throw new E("range() argument 3 must be not zero")
if (s > 0)
for (; n < m; n += s) yield n
else
for (; n > m; n += s) yield n
}
Loading

0 comments on commit 12357e6

Please sign in to comment.