@inglorious/engine 0.2.0 → 0.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +76 -75
- package/package.json +14 -25
- package/src/{engine/ai → ai}/movement/dynamic/align.js +63 -63
- package/src/{engine/ai → ai}/movement/dynamic/arrive.js +42 -42
- package/src/{engine/ai → ai}/movement/dynamic/evade.js +38 -38
- package/src/{engine/ai → ai}/movement/dynamic/face.js +19 -19
- package/src/{engine/ai → ai}/movement/dynamic/flee.js +45 -45
- package/src/{engine/ai → ai}/movement/dynamic/look-where-youre-going.js +16 -16
- package/src/{engine/ai → ai}/movement/dynamic/match-velocity.js +51 -51
- package/src/{engine/ai → ai}/movement/dynamic/pursue.js +38 -38
- package/src/{engine/ai → ai}/movement/dynamic/seek.js +44 -44
- package/src/{engine/ai → ai}/movement/dynamic/wander.js +31 -31
- package/src/{engine/ai → ai}/movement/kinematic/align.js +37 -37
- package/src/{engine/ai → ai}/movement/kinematic/arrive.js +42 -42
- package/src/{engine/ai → ai}/movement/kinematic/face.js +19 -19
- package/src/{engine/ai → ai}/movement/kinematic/flee.js +26 -26
- package/src/{engine/ai → ai}/movement/kinematic/seek.js +26 -26
- package/src/{engine/ai → ai}/movement/kinematic/seek.test.js +42 -42
- package/src/{engine/ai → ai}/movement/kinematic/wander-as-seek.js +31 -31
- package/src/{engine/ai → ai}/movement/kinematic/wander.js +27 -27
- package/src/{engine/animation → animation}/sprite.js +101 -101
- package/src/{engine/animation → animation}/ticker.js +38 -38
- package/src/{engine/behaviors → behaviors}/camera.js +68 -68
- package/src/{engine/behaviors → behaviors}/controls/dynamic/modern.js +76 -76
- package/src/{engine/behaviors → behaviors}/controls/dynamic/shooter.js +84 -84
- package/src/{engine/behaviors → behaviors}/controls/dynamic/tank.js +69 -69
- package/src/{engine/behaviors → behaviors}/controls/event-handlers.js +17 -17
- package/src/{engine/behaviors → behaviors}/controls/kinematic/modern.js +76 -76
- package/src/{engine/behaviors → behaviors}/controls/kinematic/shooter.js +82 -82
- package/src/{engine/behaviors → behaviors}/controls/kinematic/tank.js +67 -67
- package/src/behaviors/debug/collision.js +29 -0
- package/src/{engine/behaviors → behaviors}/fps.js +29 -29
- package/src/{engine/behaviors → behaviors}/fsm.js +33 -33
- package/src/{engine/behaviors → behaviors}/fsm.test.js +49 -49
- package/src/{engine/behaviors → behaviors}/game.js +15 -15
- package/src/{engine/behaviors → behaviors}/input/controls.js +37 -37
- package/src/{engine/behaviors → behaviors}/input/gamepad.js +114 -114
- package/src/{engine/behaviors → behaviors}/input/input.js +48 -48
- package/src/{engine/behaviors → behaviors}/input/keyboard.js +64 -64
- package/src/{engine/behaviors → behaviors}/input/mouse.js +91 -91
- package/src/{engine/behaviors → behaviors}/physics/bouncy.js +25 -25
- package/src/{engine/behaviors → behaviors}/physics/clamped.js +36 -36
- package/src/{engine/behaviors → behaviors}/physics/collidable.js +20 -20
- package/src/{engine/behaviors → behaviors}/physics/jumpable.js +145 -145
- package/src/{engine/behaviors → behaviors}/ui/button.js +17 -17
- package/src/{engine/collision → collision}/detection.js +110 -110
- package/src/{engine/core → core}/dev-tools.js +135 -135
- package/src/{engine/core → core}/engine.js +119 -119
- package/src/{engine/core → core}/loop.js +15 -15
- package/src/{engine/core → core}/loops/animation-frame.js +25 -25
- package/src/{engine/core → core}/loops/elapsed.js +22 -22
- package/src/{engine/core → core}/loops/fixed.js +27 -27
- package/src/{engine/core → core}/loops/flash.js +13 -13
- package/src/{engine/core → core}/loops/lag.js +26 -26
- package/src/main.js +10 -10
- package/src/{engine/movement → movement}/dynamic/modern.js +21 -21
- package/src/{engine/movement → movement}/dynamic/tank.js +43 -43
- package/src/{engine/movement → movement}/kinematic/modern.js +16 -16
- package/src/{engine/movement → movement}/kinematic/modern.test.js +27 -27
- package/src/{engine/movement → movement}/kinematic/tank.js +27 -27
- package/src/{engine/physics → physics}/bounds.js +138 -138
- package/src/{engine/physics → physics}/position.js +43 -43
- package/src/{engine/physics → physics}/position.test.js +80 -80
- package/src/{engine/systems → systems}/sprite-animation.js +27 -27
- package/src/engine/behaviors/debug/collision.js +0 -35
- package/src/engine/core/api.js +0 -34
- package/src/engine/core/select.js +0 -26
- package/src/engine/core/store.js +0 -178
- package/src/engine/core/store.test.js +0 -110
- package/src/renderers/canvas/absolute-position.js +0 -18
- package/src/renderers/canvas/camera.js +0 -13
- package/src/renderers/canvas/canvas-renderer.js +0 -68
- package/src/renderers/canvas/character.js +0 -38
- package/src/renderers/canvas/form/button.js +0 -25
- package/src/renderers/canvas/fps.js +0 -18
- package/src/renderers/canvas/image/hitmask.js +0 -51
- package/src/renderers/canvas/image/image.js +0 -34
- package/src/renderers/canvas/image/sprite.js +0 -49
- package/src/renderers/canvas/image/tilemap.js +0 -66
- package/src/renderers/canvas/mouse.js +0 -37
- package/src/renderers/canvas/rendering-system.js +0 -79
- package/src/renderers/canvas/shapes/circle.js +0 -29
- package/src/renderers/canvas/shapes/rectangle.js +0 -27
- package/src/renderers/react/game/character/character.module.scss +0 -17
- package/src/renderers/react/game/character/index.jsx +0 -20
- package/src/renderers/react/game/cursor/cursor.module.scss +0 -47
- package/src/renderers/react/game/cursor/index.jsx +0 -20
- package/src/renderers/react/game/form/fields/field/field.module.scss +0 -5
- package/src/renderers/react/game/form/fields/field/index.jsx +0 -56
- package/src/renderers/react/game/form/fields/fields.module.scss +0 -48
- package/src/renderers/react/game/form/fields/index.jsx +0 -12
- package/src/renderers/react/game/form/form.module.scss +0 -18
- package/src/renderers/react/game/form/index.jsx +0 -22
- package/src/renderers/react/game/fps/index.jsx +0 -16
- package/src/renderers/react/game/game.jsx +0 -72
- package/src/renderers/react/game/index.jsx +0 -29
- package/src/renderers/react/game/platform/index.jsx +0 -30
- package/src/renderers/react/game/platform/platform.module.scss +0 -7
- package/src/renderers/react/game/scene/index.jsx +0 -27
- package/src/renderers/react/game/scene/scene.module.scss +0 -9
- package/src/renderers/react/game/sprite/index.jsx +0 -60
- package/src/renderers/react/game/sprite/sprite.module.css +0 -3
- package/src/renderers/react/game/stats/index.jsx +0 -22
- package/src/renderers/react/hocs/with-absolute-position/index.jsx +0 -20
- package/src/renderers/react/hocs/with-absolute-position/with-absolute-position.module.scss +0 -5
- package/src/renderers/react/index.jsx +0 -9
- package/src/utils/algorithms/decision-tree.js +0 -24
- package/src/utils/algorithms/decision-tree.test.js +0 -153
- package/src/utils/algorithms/path-finding.js +0 -155
- package/src/utils/algorithms/path-finding.test.js +0 -151
- package/src/utils/algorithms/types.d.ts +0 -28
- package/src/utils/data-structures/array.js +0 -83
- package/src/utils/data-structures/array.test.js +0 -173
- package/src/utils/data-structures/board.js +0 -159
- package/src/utils/data-structures/board.test.js +0 -242
- package/src/utils/data-structures/boolean.js +0 -9
- package/src/utils/data-structures/heap.js +0 -164
- package/src/utils/data-structures/heap.test.js +0 -103
- package/src/utils/data-structures/object.js +0 -138
- package/src/utils/data-structures/object.test.js +0 -218
- package/src/utils/data-structures/objects.js +0 -66
- package/src/utils/data-structures/objects.test.js +0 -99
- package/src/utils/data-structures/tree.js +0 -36
- package/src/utils/data-structures/tree.test.js +0 -33
- package/src/utils/data-structures/types.d.ts +0 -4
- package/src/utils/functions/functions.js +0 -19
- package/src/utils/functions/functions.test.js +0 -23
- package/src/utils/math/geometry/circle.js +0 -70
- package/src/utils/math/geometry/circle.test.js +0 -97
- package/src/utils/math/geometry/hitmask.js +0 -70
- package/src/utils/math/geometry/hitmask.test.js +0 -155
- package/src/utils/math/geometry/line.js +0 -35
- package/src/utils/math/geometry/line.test.js +0 -49
- package/src/utils/math/geometry/point.js +0 -78
- package/src/utils/math/geometry/point.test.js +0 -81
- package/src/utils/math/geometry/rectangle.js +0 -76
- package/src/utils/math/geometry/rectangle.test.js +0 -42
- package/src/utils/math/geometry/segment.js +0 -80
- package/src/utils/math/geometry/segment.test.js +0 -183
- package/src/utils/math/geometry/triangle.js +0 -15
- package/src/utils/math/geometry/triangle.test.js +0 -11
- package/src/utils/math/geometry/types.d.ts +0 -23
- package/src/utils/math/linear-algebra/2d.js +0 -28
- package/src/utils/math/linear-algebra/2d.test.js +0 -17
- package/src/utils/math/linear-algebra/quaternion.js +0 -22
- package/src/utils/math/linear-algebra/quaternion.test.js +0 -25
- package/src/utils/math/linear-algebra/quaternions.js +0 -20
- package/src/utils/math/linear-algebra/quaternions.test.js +0 -29
- package/src/utils/math/linear-algebra/types.d.ts +0 -4
- package/src/utils/math/linear-algebra/vector.js +0 -327
- package/src/utils/math/linear-algebra/vector.test.js +0 -265
- package/src/utils/math/linear-algebra/vectors.js +0 -122
- package/src/utils/math/linear-algebra/vectors.test.js +0 -65
- package/src/utils/math/linear-interpolation.js +0 -9
- package/src/utils/math/numbers.js +0 -90
- package/src/utils/math/numbers.test.js +0 -137
- package/src/utils/math/rng.js +0 -44
- package/src/utils/math/rng.test.js +0 -39
- package/src/utils/math/statistics.js +0 -43
- package/src/utils/math/statistics.test.js +0 -47
- package/src/utils/math/trigonometry.js +0 -89
- package/src/utils/math/trigonometry.test.js +0 -52
- package/src/utils/physics/acceleration.js +0 -61
- package/src/utils/physics/friction.js +0 -28
- package/src/utils/physics/friction.test.js +0 -42
- package/src/utils/physics/gravity.js +0 -69
- package/src/utils/physics/gravity.test.js +0 -77
- package/src/utils/physics/jump.js +0 -31
- package/src/utils/physics/velocity.js +0 -36
|
@@ -1,242 +0,0 @@
|
|
|
1
|
-
import { mod } from "@inglorious/utils/math/numbers.js"
|
|
2
|
-
import { expect, test } from "vitest"
|
|
3
|
-
|
|
4
|
-
import {
|
|
5
|
-
createBoard,
|
|
6
|
-
down,
|
|
7
|
-
downLeft,
|
|
8
|
-
downRight,
|
|
9
|
-
left,
|
|
10
|
-
right,
|
|
11
|
-
toString,
|
|
12
|
-
up,
|
|
13
|
-
upLeft,
|
|
14
|
-
upRight,
|
|
15
|
-
} from "./board.js"
|
|
16
|
-
|
|
17
|
-
test("it should create an empty board", () => {
|
|
18
|
-
const size = [4, 4]
|
|
19
|
-
const expectedResult = [
|
|
20
|
-
null,
|
|
21
|
-
null,
|
|
22
|
-
null,
|
|
23
|
-
null,
|
|
24
|
-
null,
|
|
25
|
-
null,
|
|
26
|
-
null,
|
|
27
|
-
null,
|
|
28
|
-
null,
|
|
29
|
-
null,
|
|
30
|
-
null,
|
|
31
|
-
null,
|
|
32
|
-
null,
|
|
33
|
-
null,
|
|
34
|
-
null,
|
|
35
|
-
null,
|
|
36
|
-
]
|
|
37
|
-
|
|
38
|
-
expect(createBoard(size)).toStrictEqual(expectedResult)
|
|
39
|
-
})
|
|
40
|
-
|
|
41
|
-
test("it should create a checkerboard", () => {
|
|
42
|
-
const size = [4, 4]
|
|
43
|
-
const filler = (i, j) => (!mod(i + j, 2) ? 1 : 0)
|
|
44
|
-
const expectedResult = [1, 0, 1, 0, 0, 1, 0, 1, 1, 0, 1, 0, 0, 1, 0, 1]
|
|
45
|
-
|
|
46
|
-
expect(createBoard(size, filler)).toStrictEqual(expectedResult)
|
|
47
|
-
})
|
|
48
|
-
|
|
49
|
-
test("it should return the coordinates of the cell downward", () => {
|
|
50
|
-
const size = [8, 8]
|
|
51
|
-
const coordinates = [3, 4]
|
|
52
|
-
const expectedResult = [4, 4]
|
|
53
|
-
|
|
54
|
-
expect(down(coordinates, size)).toStrictEqual(expectedResult)
|
|
55
|
-
})
|
|
56
|
-
|
|
57
|
-
test("it should throw going down if cell is downmost", () => {
|
|
58
|
-
const size = [8, 8]
|
|
59
|
-
const coordinates = [7, 4]
|
|
60
|
-
|
|
61
|
-
expect(() => down(coordinates, size)).toThrow()
|
|
62
|
-
})
|
|
63
|
-
|
|
64
|
-
test("it should return the coordinates of the cell down left", () => {
|
|
65
|
-
const size = [8, 8]
|
|
66
|
-
const coordinates = [3, 4]
|
|
67
|
-
const expectedResult = [4, 3]
|
|
68
|
-
|
|
69
|
-
expect(downLeft(coordinates, size)).toStrictEqual(expectedResult)
|
|
70
|
-
})
|
|
71
|
-
|
|
72
|
-
test("it should throw going down left if cell is downmost", () => {
|
|
73
|
-
const size = [8, 8]
|
|
74
|
-
const coordinates = [7, 4]
|
|
75
|
-
|
|
76
|
-
expect(() => downLeft(coordinates, size)).toThrow()
|
|
77
|
-
})
|
|
78
|
-
|
|
79
|
-
test("it should throw going down left if cell is downmost", () => {
|
|
80
|
-
const size = [8, 8]
|
|
81
|
-
const coordinates = [3, 0]
|
|
82
|
-
|
|
83
|
-
expect(() => downLeft(coordinates, size)).toThrow()
|
|
84
|
-
})
|
|
85
|
-
|
|
86
|
-
test("it should throw going down left if cell is on the angle", () => {
|
|
87
|
-
const size = [8, 8]
|
|
88
|
-
const coordinates = [7, 0]
|
|
89
|
-
|
|
90
|
-
expect(() => downLeft(coordinates, size)).toThrow()
|
|
91
|
-
})
|
|
92
|
-
|
|
93
|
-
test("it should return the coordinates of the cell down right", () => {
|
|
94
|
-
const size = [8, 8]
|
|
95
|
-
const coordinates = [3, 4]
|
|
96
|
-
const expectedResult = [4, 5]
|
|
97
|
-
|
|
98
|
-
expect(downRight(coordinates, size)).toStrictEqual(expectedResult)
|
|
99
|
-
})
|
|
100
|
-
|
|
101
|
-
test("it should throw going down right if cell is downmost", () => {
|
|
102
|
-
const size = [8, 8]
|
|
103
|
-
const coordinates = [7, 4]
|
|
104
|
-
|
|
105
|
-
expect(() => downRight(coordinates, size)).toThrow()
|
|
106
|
-
})
|
|
107
|
-
|
|
108
|
-
test("it should throw going down right if cell is downmost", () => {
|
|
109
|
-
const size = [8, 8]
|
|
110
|
-
const coordinates = [3, 7]
|
|
111
|
-
|
|
112
|
-
expect(() => downRight(coordinates, size)).toThrow()
|
|
113
|
-
})
|
|
114
|
-
|
|
115
|
-
test("it should throw going down right if cell is on the angle", () => {
|
|
116
|
-
const size = [8, 8]
|
|
117
|
-
const coordinates = [7, 7]
|
|
118
|
-
|
|
119
|
-
expect(() => downRight(coordinates, size)).toThrow()
|
|
120
|
-
})
|
|
121
|
-
|
|
122
|
-
test("it should return the coordinates of the cell to the left", () => {
|
|
123
|
-
const size = [8, 8]
|
|
124
|
-
const coordinates = [3, 4]
|
|
125
|
-
const expectedResult = [3, 3]
|
|
126
|
-
|
|
127
|
-
expect(left(coordinates, size)).toStrictEqual(expectedResult)
|
|
128
|
-
})
|
|
129
|
-
|
|
130
|
-
test("it should not return coordinates to the left if cell is leftmost", () => {
|
|
131
|
-
const size = [8, 8]
|
|
132
|
-
const coordinates = [3, 0]
|
|
133
|
-
|
|
134
|
-
expect(() => left(coordinates, size)).toThrow()
|
|
135
|
-
})
|
|
136
|
-
|
|
137
|
-
test("it should return the coordinates of the cell to the right", () => {
|
|
138
|
-
const size = [8, 8]
|
|
139
|
-
const coordinates = [3, 4]
|
|
140
|
-
const expectedResult = [3, 5]
|
|
141
|
-
|
|
142
|
-
expect(right(coordinates, size)).toStrictEqual(expectedResult)
|
|
143
|
-
})
|
|
144
|
-
|
|
145
|
-
test("it should not return coordinates to the right if cell is rightmost", () => {
|
|
146
|
-
const size = [8, 8]
|
|
147
|
-
const coordinates = [3, 7]
|
|
148
|
-
|
|
149
|
-
expect(() => right(coordinates, size)).toThrow()
|
|
150
|
-
})
|
|
151
|
-
|
|
152
|
-
test("it should return a string representation of the given board", () => {
|
|
153
|
-
const board = [1, 0, 1, 0, 0, 1, 0, 1, 1, 0, 1, 0, 0, 1, 0, 1]
|
|
154
|
-
|
|
155
|
-
expect(toString(board, [4, 4])).toBe(`1 0 1 0
|
|
156
|
-
0 1 0 1
|
|
157
|
-
1 0 1 0
|
|
158
|
-
0 1 0 1`)
|
|
159
|
-
})
|
|
160
|
-
|
|
161
|
-
test("it should return a custom string representation of the given board", () => {
|
|
162
|
-
const board = [1, 0, 1, 0, 0, 1, 0, 1, 1, 0, 1, 0, 0, 1, 0, 1]
|
|
163
|
-
const cellToString = (cell) => cell.toFixed(1)
|
|
164
|
-
|
|
165
|
-
expect(toString(board, [4, 4], cellToString)).toBe(`1.0 0.0 1.0 0.0
|
|
166
|
-
0.0 1.0 0.0 1.0
|
|
167
|
-
1.0 0.0 1.0 0.0
|
|
168
|
-
0.0 1.0 0.0 1.0`)
|
|
169
|
-
})
|
|
170
|
-
|
|
171
|
-
test("it should return the coordinates of the cell upward", () => {
|
|
172
|
-
const size = [8, 8]
|
|
173
|
-
const coordinates = [3, 4]
|
|
174
|
-
const expectedResult = [2, 4]
|
|
175
|
-
|
|
176
|
-
expect(up(coordinates, size)).toStrictEqual(expectedResult)
|
|
177
|
-
})
|
|
178
|
-
|
|
179
|
-
test("it should throw going upward if cell is topmost", () => {
|
|
180
|
-
const size = [8, 8]
|
|
181
|
-
const coordinates = [0, 4]
|
|
182
|
-
|
|
183
|
-
expect(() => up(coordinates, size)).toThrow()
|
|
184
|
-
})
|
|
185
|
-
|
|
186
|
-
test("it should return the coordinates of the cell up left", () => {
|
|
187
|
-
const size = [8, 8]
|
|
188
|
-
const coordinates = [3, 4]
|
|
189
|
-
const expectedResult = [2, 3]
|
|
190
|
-
|
|
191
|
-
expect(upLeft(coordinates, size)).toStrictEqual(expectedResult)
|
|
192
|
-
})
|
|
193
|
-
|
|
194
|
-
test("it should throw going up left if cell is upmost", () => {
|
|
195
|
-
const size = [8, 8]
|
|
196
|
-
const coordinates = [0, 4]
|
|
197
|
-
|
|
198
|
-
expect(() => upLeft(coordinates, size)).toThrow()
|
|
199
|
-
})
|
|
200
|
-
|
|
201
|
-
test("it should throw going up left if cell is leftmost", () => {
|
|
202
|
-
const size = [8, 8]
|
|
203
|
-
const coordinates = [3, 0]
|
|
204
|
-
|
|
205
|
-
expect(() => upLeft(coordinates, size)).toThrow()
|
|
206
|
-
})
|
|
207
|
-
|
|
208
|
-
test("it should throw going up left if cell is on the angle", () => {
|
|
209
|
-
const size = [8, 8]
|
|
210
|
-
const coordinates = [0, 0]
|
|
211
|
-
|
|
212
|
-
expect(() => upLeft(coordinates, size)).toThrow()
|
|
213
|
-
})
|
|
214
|
-
|
|
215
|
-
test("it should return the coordinates of the cell up right", () => {
|
|
216
|
-
const size = [8, 8]
|
|
217
|
-
const coordinates = [3, 4]
|
|
218
|
-
const expectedResult = [2, 5]
|
|
219
|
-
|
|
220
|
-
expect(upRight(coordinates, size)).toStrictEqual(expectedResult)
|
|
221
|
-
})
|
|
222
|
-
|
|
223
|
-
test("it should throw going up right if cell is upmost", () => {
|
|
224
|
-
const size = [8, 8]
|
|
225
|
-
const coordinates = [0, 4]
|
|
226
|
-
|
|
227
|
-
expect(() => upRight(coordinates, size)).toThrow()
|
|
228
|
-
})
|
|
229
|
-
|
|
230
|
-
test("it should throw going down right if cell is rightmost", () => {
|
|
231
|
-
const size = [8, 8]
|
|
232
|
-
const coordinates = [3, 7]
|
|
233
|
-
|
|
234
|
-
expect(() => upRight(coordinates, size)).toThrow()
|
|
235
|
-
})
|
|
236
|
-
|
|
237
|
-
test("it should throw going down right if cell is on the angle", () => {
|
|
238
|
-
const size = [8, 8]
|
|
239
|
-
const coordinates = [0, 7]
|
|
240
|
-
|
|
241
|
-
expect(() => upRight(coordinates, size)).toThrow()
|
|
242
|
-
})
|
|
@@ -1,9 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Converts a boolean value to its string representation.
|
|
3
|
-
*
|
|
4
|
-
* @param {boolean} bool - The boolean value to convert.
|
|
5
|
-
* @returns {string} The string "true" if the input is true, otherwise "false".
|
|
6
|
-
*/
|
|
7
|
-
export function toString(bool) {
|
|
8
|
-
return bool ? "true" : "false"
|
|
9
|
-
}
|
|
@@ -1,164 +0,0 @@
|
|
|
1
|
-
// @see https://stackfull.dev/heaps-in-javascript
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Default comparator for a max-heap.
|
|
5
|
-
* @param {*} a - First element.
|
|
6
|
-
* @param {*} b - Second element.
|
|
7
|
-
* @returns {number} Positive if `a` should come before `b`, negative otherwise.
|
|
8
|
-
*/
|
|
9
|
-
const DEFAULT_COMPARATOR = (a, b) => b - a
|
|
10
|
-
|
|
11
|
-
// Constants for heap operations
|
|
12
|
-
const ROOT_INDEX = 0 // Index of the root element in the heap.
|
|
13
|
-
const SINGLE_ELEMENT = 1 // Heap size when it contains a single element.
|
|
14
|
-
const LEFT_CHILD_MULTIPLIER = 2 // Multiplier to calculate the left child index.
|
|
15
|
-
const RIGHT_CHILD_MULTIPLIER = 2 // Multiplier to calculate the right child index.
|
|
16
|
-
const PARENT_DIVISOR = 2 // Divisor to calculate the parent index.
|
|
17
|
-
const LEFT_CHILD_OFFSET = 1 // Offset to calculate the left child index.
|
|
18
|
-
const RIGHT_CHILD_OFFSET = 2 // Offset to calculate the right child index.
|
|
19
|
-
const LAST_ELEMENT_OFFSET = 1 // Offset to get the last element in the heap.
|
|
20
|
-
const INDEX_ADJUSTMENT = 1 // Adjustment for zero-based indexing.
|
|
21
|
-
const COMPARATOR_THRESHOLD = 0 // Threshold for comparator results.
|
|
22
|
-
|
|
23
|
-
/**
|
|
24
|
-
* Checks if a heap contains a specific item.
|
|
25
|
-
* @param {Array} heap - The heap array.
|
|
26
|
-
* @param {*} item - The item to check for.
|
|
27
|
-
* @returns {boolean} True if the item exists in the heap, false otherwise.
|
|
28
|
-
*/
|
|
29
|
-
export function contains(heap, item) {
|
|
30
|
-
return heap.includes(item)
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
/**
|
|
34
|
-
* Converts an array into a heap.
|
|
35
|
-
* @param {Array} arr - The input array.
|
|
36
|
-
* @param {Function} [comparator=DEFAULT_COMPARATOR] - Comparator function to define heap order.
|
|
37
|
-
* @returns {Array} The heapified array.
|
|
38
|
-
*/
|
|
39
|
-
export function heapify(arr, comparator = DEFAULT_COMPARATOR) {
|
|
40
|
-
let heap = []
|
|
41
|
-
for (const item of arr) {
|
|
42
|
-
heap = push(heap, item, comparator)
|
|
43
|
-
}
|
|
44
|
-
return heap
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
/**
|
|
48
|
-
* Adds an item to the heap and maintains the heap property.
|
|
49
|
-
* @param {Array} heap - The heap array.
|
|
50
|
-
* @param {*} item - The item to add.
|
|
51
|
-
* @param {Function} [comparator=DEFAULT_COMPARATOR] - Comparator function to define heap order.
|
|
52
|
-
* @returns {Array} The updated heap array.
|
|
53
|
-
*/
|
|
54
|
-
export function push(heap, item, comparator = DEFAULT_COMPARATOR) {
|
|
55
|
-
const h = [...heap, item]
|
|
56
|
-
|
|
57
|
-
if (h.length === SINGLE_ELEMENT) {
|
|
58
|
-
return h
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
let index = h.length - LAST_ELEMENT_OFFSET
|
|
62
|
-
let parentIndex = parent(index)
|
|
63
|
-
|
|
64
|
-
while (
|
|
65
|
-
index > ROOT_INDEX &&
|
|
66
|
-
comparator(h[index], h[parentIndex]) > COMPARATOR_THRESHOLD
|
|
67
|
-
) {
|
|
68
|
-
;[h[index], h[parentIndex]] = [h[parentIndex], h[index]]
|
|
69
|
-
|
|
70
|
-
index = parentIndex
|
|
71
|
-
parentIndex = parent(index)
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
return h
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
/**
|
|
78
|
-
* Calculates the index of the left child of a given node.
|
|
79
|
-
* @param {number} index - The index of the parent node.
|
|
80
|
-
* @returns {number} The index of the left child.
|
|
81
|
-
*/
|
|
82
|
-
export function left(index) {
|
|
83
|
-
return LEFT_CHILD_MULTIPLIER * index + LEFT_CHILD_OFFSET
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
/**
|
|
87
|
-
* Calculates the index of the parent of a given node.
|
|
88
|
-
* @param {number} index - The index of the child node.
|
|
89
|
-
* @returns {number} The index of the parent node.
|
|
90
|
-
*/
|
|
91
|
-
export function parent(index) {
|
|
92
|
-
return Math.floor((index - INDEX_ADJUSTMENT) / PARENT_DIVISOR)
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
/**
|
|
96
|
-
* Removes and returns the root element of the heap.
|
|
97
|
-
* @param {Array} heap - The heap array.
|
|
98
|
-
* @param {Function} [comparator=DEFAULT_COMPARATOR] - Comparator function to define heap order.
|
|
99
|
-
* @returns {Array} The updated heap array after removing the root.
|
|
100
|
-
*/
|
|
101
|
-
export function pop(heap, comparator = DEFAULT_COMPARATOR) {
|
|
102
|
-
const h = [...heap]
|
|
103
|
-
;[h[ROOT_INDEX], h[h.length - LAST_ELEMENT_OFFSET]] = [
|
|
104
|
-
h[h.length - LAST_ELEMENT_OFFSET],
|
|
105
|
-
h[ROOT_INDEX],
|
|
106
|
-
]
|
|
107
|
-
|
|
108
|
-
h.pop()
|
|
109
|
-
|
|
110
|
-
let index = ROOT_INDEX
|
|
111
|
-
let leftIndex = left(index)
|
|
112
|
-
let rightIndex = right(index)
|
|
113
|
-
let newMinFound = false
|
|
114
|
-
|
|
115
|
-
while (leftIndex < h.length && !newMinFound) {
|
|
116
|
-
const minIndex =
|
|
117
|
-
rightIndex < h.length &&
|
|
118
|
-
comparator(h[rightIndex], h[leftIndex]) > COMPARATOR_THRESHOLD
|
|
119
|
-
? rightIndex
|
|
120
|
-
: leftIndex
|
|
121
|
-
newMinFound = comparator(h[minIndex], h[index]) > COMPARATOR_THRESHOLD
|
|
122
|
-
|
|
123
|
-
if (newMinFound) {
|
|
124
|
-
;[h[minIndex], h[index]] = [h[index], h[minIndex]]
|
|
125
|
-
|
|
126
|
-
index = minIndex
|
|
127
|
-
leftIndex = left(index)
|
|
128
|
-
rightIndex = right(index)
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
return h
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
/**
|
|
136
|
-
* Removes the root element and reorders the heap.
|
|
137
|
-
* @param {Array} heap - The heap array.
|
|
138
|
-
* @returns {Array} The updated heap array.
|
|
139
|
-
*/
|
|
140
|
-
export function remove(heap) {
|
|
141
|
-
const [, ...rest] = heap
|
|
142
|
-
return heapify([
|
|
143
|
-
...rest.slice(-LAST_ELEMENT_OFFSET),
|
|
144
|
-
...rest.slice(ROOT_INDEX, -LAST_ELEMENT_OFFSET),
|
|
145
|
-
])
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
/**
|
|
149
|
-
* Calculates the index of the right child of a given node.
|
|
150
|
-
* @param {number} index - The index of the parent node.
|
|
151
|
-
* @returns {number} The index of the right child.
|
|
152
|
-
*/
|
|
153
|
-
export function right(index) {
|
|
154
|
-
return RIGHT_CHILD_MULTIPLIER * index + RIGHT_CHILD_OFFSET
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
/**
|
|
158
|
-
* Retrieves the root element of the heap.
|
|
159
|
-
* @param {Array} heap - The heap array.
|
|
160
|
-
* @returns {*} The root element of the heap.
|
|
161
|
-
*/
|
|
162
|
-
export function root(heap) {
|
|
163
|
-
return heap[ROOT_INDEX]
|
|
164
|
-
}
|
|
@@ -1,103 +0,0 @@
|
|
|
1
|
-
import { expect, test } from "vitest"
|
|
2
|
-
|
|
3
|
-
import { contains, heapify, pop, push, remove, root } from "./heap.js"
|
|
4
|
-
|
|
5
|
-
test("it should check if a heap contains a value", () => {
|
|
6
|
-
const heap = [3, 2, 6, 1, 7, 4, 5]
|
|
7
|
-
const item = 1
|
|
8
|
-
const expectedResult = true
|
|
9
|
-
|
|
10
|
-
expect(contains(heap, item)).toBe(expectedResult)
|
|
11
|
-
})
|
|
12
|
-
|
|
13
|
-
test("it should check if a heap contains an object", () => {
|
|
14
|
-
const heap = [
|
|
15
|
-
{ value: 3 },
|
|
16
|
-
{ value: 2 },
|
|
17
|
-
{ value: 6 },
|
|
18
|
-
{ value: 1 },
|
|
19
|
-
{ value: 7 },
|
|
20
|
-
{ value: 4 },
|
|
21
|
-
{ value: 5 },
|
|
22
|
-
]
|
|
23
|
-
const item = heap[3]
|
|
24
|
-
const expectedResult = true
|
|
25
|
-
|
|
26
|
-
expect(contains(heap, item)).toBe(expectedResult)
|
|
27
|
-
})
|
|
28
|
-
|
|
29
|
-
test("it should not convert an array in existing heap form into a heap", () => {
|
|
30
|
-
const arr = [1, 2, 3, 4, 5, 6, 7]
|
|
31
|
-
const expectedResult = [1, 2, 3, 4, 5, 6, 7]
|
|
32
|
-
|
|
33
|
-
expect(heapify(arr)).toStrictEqual(expectedResult)
|
|
34
|
-
})
|
|
35
|
-
|
|
36
|
-
test("it should convert an array into a heap", () => {
|
|
37
|
-
const arr = [7, 6, 5, 4, 3, 2, 1]
|
|
38
|
-
const expectedResult = [1, 4, 2, 7, 5, 6, 3]
|
|
39
|
-
|
|
40
|
-
expect(heapify(arr)).toStrictEqual(expectedResult)
|
|
41
|
-
})
|
|
42
|
-
|
|
43
|
-
test("it should convert an array into a maxheap", () => {
|
|
44
|
-
const arr = [1, 2, 3, 4, 5, 6, 7]
|
|
45
|
-
const comparator = (a, b) => a - b
|
|
46
|
-
const expectedResult = [7, 4, 6, 1, 3, 2, 5]
|
|
47
|
-
|
|
48
|
-
expect(heapify(arr, comparator)).toStrictEqual(expectedResult)
|
|
49
|
-
})
|
|
50
|
-
|
|
51
|
-
test("it should find the minimum value in a min heap", () => {
|
|
52
|
-
const heap = [1, 4, 2, 7, 5, 6, 3]
|
|
53
|
-
const expectedResult = 1
|
|
54
|
-
|
|
55
|
-
expect(root(heap)).toBe(expectedResult)
|
|
56
|
-
})
|
|
57
|
-
|
|
58
|
-
test("it should push the first element of the heap", () => {
|
|
59
|
-
const heap = []
|
|
60
|
-
const value = 7
|
|
61
|
-
const expectedResult = [7]
|
|
62
|
-
|
|
63
|
-
expect(push(heap, value)).toStrictEqual(expectedResult)
|
|
64
|
-
})
|
|
65
|
-
|
|
66
|
-
test("it should push the biggest value at the end of the heap", () => {
|
|
67
|
-
const heap = [3, 13, 19, 33, 42, 23, 21]
|
|
68
|
-
const value = 50
|
|
69
|
-
const expectedResult = [3, 13, 19, 33, 42, 23, 21, 50]
|
|
70
|
-
|
|
71
|
-
expect(push(heap, value)).toStrictEqual(expectedResult)
|
|
72
|
-
})
|
|
73
|
-
|
|
74
|
-
test("it should push a value in the right spot of a heap", () => {
|
|
75
|
-
const heap = [3, 13, 19, 33, 42, 23, 21]
|
|
76
|
-
const value = 7
|
|
77
|
-
const expectedResult = [3, 7, 19, 13, 42, 23, 21, 33]
|
|
78
|
-
|
|
79
|
-
expect(push(heap, value)).toStrictEqual(expectedResult)
|
|
80
|
-
})
|
|
81
|
-
|
|
82
|
-
test("it should pop the lowest value from the end of the heap", () => {
|
|
83
|
-
const heap = [3, 13, 19, 33, 42, 23, 21]
|
|
84
|
-
const expectedResult = [13, 21, 19, 33, 42, 23]
|
|
85
|
-
|
|
86
|
-
expect(pop(heap)).toStrictEqual(expectedResult)
|
|
87
|
-
})
|
|
88
|
-
|
|
89
|
-
test("it should push the biggest value at the beginning of a maxheap", () => {
|
|
90
|
-
const heap = [3, 13, 19, 33, 42, 23, 21]
|
|
91
|
-
const comparator = (a, b) => a - b
|
|
92
|
-
const value = 50
|
|
93
|
-
const expectedResult = [50, 3, 19, 13, 42, 23, 21, 33]
|
|
94
|
-
|
|
95
|
-
expect(push(heap, value, comparator)).toStrictEqual(expectedResult)
|
|
96
|
-
})
|
|
97
|
-
|
|
98
|
-
test("it should remove the heap's root element", () => {
|
|
99
|
-
const heap = [3, 13, 19, 33, 42, 23, 21]
|
|
100
|
-
const expectedResult = [13, 21, 19, 33, 42, 23]
|
|
101
|
-
|
|
102
|
-
expect(remove(heap)).toStrictEqual(expectedResult)
|
|
103
|
-
})
|
|
@@ -1,138 +0,0 @@
|
|
|
1
|
-
const INITIAL_LEVEL = 0
|
|
2
|
-
const NEXT_LEVEL = 2
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
* Creates a deep clone of the given object.
|
|
6
|
-
* @param {Object} obj - The object to clone.
|
|
7
|
-
* @returns {Object} A deep clone of the input object.
|
|
8
|
-
*/
|
|
9
|
-
export function clone(obj) {
|
|
10
|
-
return JSON.parse(JSON.stringify(obj))
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
/**
|
|
14
|
-
* Filters the properties of an object based on a callback function.
|
|
15
|
-
* @param {Object} obj - The object to filter.
|
|
16
|
-
* @param {Function} callback - A function that determines whether a property should be included.
|
|
17
|
-
* Receives (key, value, obj) as arguments.
|
|
18
|
-
* @returns {Object} A new object with the filtered properties.
|
|
19
|
-
*/
|
|
20
|
-
export function filter(obj, callback) {
|
|
21
|
-
return Object.fromEntries(
|
|
22
|
-
Object.entries(obj).filter(([key, value], obj) =>
|
|
23
|
-
callback(key, value, obj),
|
|
24
|
-
),
|
|
25
|
-
)
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
/**
|
|
29
|
-
* Finds the first property in an object that satisfies the callback function.
|
|
30
|
-
* @param {Object} obj - The object to search.
|
|
31
|
-
* @param {Function} callback - A function that determines whether a property matches.
|
|
32
|
-
* Receives (key, value, obj) as arguments.
|
|
33
|
-
* @returns {Object} An object containing the first matching property, or an empty object if none match.
|
|
34
|
-
*/
|
|
35
|
-
export function find(obj, callback) {
|
|
36
|
-
return Object.fromEntries([
|
|
37
|
-
Object.entries(obj).find(([key, value], obj) => callback(key, value, obj)),
|
|
38
|
-
])
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
/**
|
|
42
|
-
* Checks if a value is a plain object.
|
|
43
|
-
* @param {*} obj - The value to check.
|
|
44
|
-
* @returns {boolean} True if the value is a plain object, false otherwise.
|
|
45
|
-
*/
|
|
46
|
-
export function isObject(obj) {
|
|
47
|
-
return obj != null && obj.constructor === Object
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
/**
|
|
51
|
-
* Maps the properties of an object using a callback function.
|
|
52
|
-
* @param {Object} obj - The object to map.
|
|
53
|
-
* @param {Function} callback - A function that transforms each property.
|
|
54
|
-
* Receives (key, value, obj) as arguments.
|
|
55
|
-
* @returns {Object} A new object with the mapped properties.
|
|
56
|
-
*/
|
|
57
|
-
export function map(obj, callback) {
|
|
58
|
-
return Object.entries(obj).reduce((acc, [key, value]) => {
|
|
59
|
-
acc[key] = callback(key, value, obj)
|
|
60
|
-
return acc
|
|
61
|
-
}, {})
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
/**
|
|
65
|
-
* A utility function inspired by Immer's `produce` API. It provides a convenient
|
|
66
|
-
* way to work with immutable data structures by allowing "mutations" on a
|
|
67
|
-
* temporary draft.
|
|
68
|
-
*
|
|
69
|
-
* **Important:** Unlike Immer, which uses structural sharing via proxies for
|
|
70
|
-
* high performance, this implementation performs a full deep clone of the base
|
|
71
|
-
* state on every call using `JSON.parse(JSON.stringify())`. This can be
|
|
72
|
-
* inefficient for large or complex states. It is intended for simple use cases
|
|
73
|
-
* where the convenience of the API outweighs the performance cost.
|
|
74
|
-
*
|
|
75
|
-
* The recipe function receives a draft copy of the state. It can either
|
|
76
|
-
* mutate the draft and return nothing (`undefined`), or it can return a
|
|
77
|
-
* completely new value, which will become the next state.
|
|
78
|
-
*
|
|
79
|
-
* Can be called in two ways:
|
|
80
|
-
* - **Standard:** `produce(baseState, recipe, ...args)`
|
|
81
|
-
* - **Curried:** `produce(recipe)` returns a new function `(baseState, ...args) => newState`
|
|
82
|
-
*
|
|
83
|
-
* @template T
|
|
84
|
-
* @param {T|function(T, ...*): (T|void)} baseState The initial state, or a recipe for currying.
|
|
85
|
-
* @param {function(T, ...*): (T|void)} [recipe] The recipe function.
|
|
86
|
-
* @param {...*} args Additional arguments to pass to the recipe.
|
|
87
|
-
* @returns {T | function(T, ...*): T} A new state, or a producer function if curried.
|
|
88
|
-
*/
|
|
89
|
-
export function produce(baseState, recipe, ...args) {
|
|
90
|
-
if (typeof baseState === "function" && recipe === undefined) {
|
|
91
|
-
const recipeFn = baseState
|
|
92
|
-
return (state, ...recipeArgs) => produce(state, recipeFn, ...recipeArgs)
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
const draft = clone(baseState)
|
|
96
|
-
const result = recipe(draft, ...args)
|
|
97
|
-
return result === undefined ? draft : result
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
/**
|
|
101
|
-
* Converts an object or array to a formatted string representation.
|
|
102
|
-
* @param {*} obj - The object or array to convert.
|
|
103
|
-
* @param {number} [indentationLevel=INITIAL_LEVEL] - The current indentation level (used for nested structures).
|
|
104
|
-
* @returns {string} A string representation of the input object or array.
|
|
105
|
-
*/
|
|
106
|
-
export function toString(obj, indentationLevel = INITIAL_LEVEL) {
|
|
107
|
-
if (Array.isArray(obj)) {
|
|
108
|
-
return `[
|
|
109
|
-
${obj
|
|
110
|
-
.map(
|
|
111
|
-
(item) =>
|
|
112
|
-
" ".repeat(indentationLevel + NEXT_LEVEL) +
|
|
113
|
-
toString(item, indentationLevel + NEXT_LEVEL),
|
|
114
|
-
)
|
|
115
|
-
.join(",\n")}
|
|
116
|
-
${" ".repeat(indentationLevel)}]`
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
if (typeof obj === "object" && obj != null) {
|
|
120
|
-
return `{
|
|
121
|
-
${Object.entries(obj)
|
|
122
|
-
.map(
|
|
123
|
-
([key, value]) =>
|
|
124
|
-
`${" ".repeat(indentationLevel + NEXT_LEVEL)}${key}: ${toString(
|
|
125
|
-
value,
|
|
126
|
-
indentationLevel + NEXT_LEVEL,
|
|
127
|
-
)}`,
|
|
128
|
-
)
|
|
129
|
-
.join(",\n")}
|
|
130
|
-
${" ".repeat(indentationLevel)}}`
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
if (typeof obj === "string") {
|
|
134
|
-
return `"${obj}"`
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
return obj
|
|
138
|
-
}
|