index.js
1.87 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
'use strict'
class Hoopy extends Array {
constructor (size) {
let index, isIndexOverflowed
if (! isPositiveInteger(size)) {
throw new TypeError('Argument `size` must be a positive integer.')
}
super(size)
this.grow = by => {
if (! isPositiveInteger(by)) {
throw new TypeError('Argument `by` must be a positive integer.')
}
let i
const newSize = size + by
for (i = size; i < newSize; ++i) {
this[i] = undefined
}
if (isIndexOverflowed) {
for (i = 0; i <= index; ++i) {
let j = size + i
if (j >= newSize) {
j %= newSize
}
this[j] = this[i]
this[i] = undefined
}
}
size = newSize
}
return new Proxy(this, {
get (target, key) {
if (isInteger(key)) {
return target[getIndex(key, size)]
}
return target[key]
},
set (target, key, value) {
if (isInteger(key)) {
index = getIndex(key, size)
target[index] = value
if (Math.abs(key) >= size) {
isIndexOverflowed = true
} else {
isIndexOverflowed = false
}
} else {
target[key] = value
}
return true
}
})
}
}
function isPositiveInteger (thing) {
return isInteger(thing) && thing > 0
}
function isInteger (thing) {
try {
return +thing % 1 === 0
} catch (error) {
// Coercing symbols to numbers throws an error
}
return false
}
function getIndex (key, size) {
if (key === 0) {
return 0
}
if (key < 0) {
return (size - Math.abs(key)) % size
}
return key % size
}
function nop () {
throw new Error('Not implemented')
}
Hoopy.prototype.push = nop
Hoopy.prototype.pop = nop
Hoopy.prototype.shift = nop
Hoopy.prototype.unshift = nop
module.exports = Hoopy