@inglorious/engine 0.1.1 → 0.3.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 +75 -72
- package/package.json +15 -37
- package/src/{engine/ai → ai}/movement/dynamic/align.js +9 -9
- package/src/{engine/ai → ai}/movement/dynamic/arrive.js +9 -10
- package/src/{engine/ai → ai}/movement/dynamic/evade.js +9 -9
- package/src/{engine/ai → ai}/movement/dynamic/face.js +5 -6
- package/src/{engine/ai/movement/dynamic/seek.js → ai/movement/dynamic/flee.js} +8 -7
- package/src/ai/movement/dynamic/look-where-youre-going.js +16 -0
- package/src/{engine/ai → ai}/movement/dynamic/match-velocity.js +9 -8
- package/src/{engine/ai → ai}/movement/dynamic/pursue.js +9 -9
- package/src/{engine/ai/movement/dynamic/flee.js → ai/movement/dynamic/seek.js} +7 -8
- package/src/{engine/ai → ai}/movement/dynamic/wander.js +9 -10
- package/src/{engine/ai → ai}/movement/kinematic/align.js +7 -7
- package/src/{engine/ai → ai}/movement/kinematic/arrive.js +8 -8
- package/src/{engine/ai → ai}/movement/kinematic/face.js +5 -6
- package/src/{engine/ai → ai}/movement/kinematic/flee.js +5 -5
- package/src/{engine/ai → ai}/movement/kinematic/seek.js +5 -5
- package/src/{engine/ai → ai}/movement/kinematic/seek.test.js +10 -10
- package/src/{engine/ai → ai}/movement/kinematic/wander-as-seek.js +9 -9
- package/src/{engine/ai → ai}/movement/kinematic/wander.js +5 -5
- package/src/animation/sprite.js +101 -0
- package/src/animation/ticker.js +38 -0
- package/src/behaviors/camera.js +68 -0
- package/src/behaviors/controls/dynamic/modern.js +76 -0
- package/src/behaviors/controls/dynamic/shooter.js +84 -0
- package/src/behaviors/controls/dynamic/tank.js +69 -0
- package/src/behaviors/controls/event-handlers.js +17 -0
- package/src/behaviors/controls/kinematic/modern.js +76 -0
- package/src/behaviors/controls/kinematic/shooter.js +82 -0
- package/src/behaviors/controls/kinematic/tank.js +67 -0
- package/src/behaviors/debug/collision.js +29 -0
- package/src/behaviors/fps.js +29 -0
- package/src/behaviors/fsm.js +33 -0
- package/src/{game/decorators → behaviors}/fsm.test.js +15 -22
- package/src/behaviors/game.js +15 -0
- package/src/behaviors/input/controls.js +37 -0
- package/src/behaviors/input/gamepad.js +114 -0
- package/src/behaviors/input/input.js +48 -0
- package/src/behaviors/input/keyboard.js +64 -0
- package/src/behaviors/input/mouse.js +91 -0
- package/src/behaviors/physics/bouncy.js +25 -0
- package/src/behaviors/physics/clamped.js +36 -0
- package/src/{game/decorators/collisions.js → behaviors/physics/collidable.js} +3 -7
- package/src/behaviors/physics/jumpable.js +145 -0
- package/src/behaviors/ui/button.js +17 -0
- package/src/collision/detection.js +110 -0
- package/src/core/api.js +34 -0
- package/src/core/dev-tools.js +135 -0
- package/src/core/engine.js +119 -0
- package/src/core/loop.js +15 -0
- package/src/{engine/loop → core/loops}/animation-frame.js +1 -2
- package/src/{engine/loop → core/loops}/elapsed.js +1 -2
- package/src/{engine/loop → core/loops}/fixed.js +1 -2
- package/src/{engine/loop → core/loops}/flash.js +1 -2
- package/src/{engine/loop → core/loops}/lag.js +1 -2
- package/src/core/select.js +26 -0
- package/src/core/store.js +178 -0
- package/src/core/store.test.js +110 -0
- package/src/main.js +7 -2
- package/src/{engine/movement → movement}/dynamic/modern.js +3 -6
- package/src/{engine/movement → movement}/dynamic/tank.js +9 -9
- package/src/{engine/movement → movement}/kinematic/modern.js +3 -3
- package/src/movement/kinematic/modern.test.js +27 -0
- package/src/{engine/movement → movement}/kinematic/tank.js +5 -5
- package/src/physics/bounds.js +138 -0
- package/src/physics/position.js +43 -0
- package/src/physics/position.test.js +80 -0
- package/src/systems/sprite-animation.js +27 -0
- package/src/engine/ai/movement/dynamic/look-where-youre-going.js +0 -17
- package/src/engine/collision/detection.js +0 -115
- package/src/engine/loop.js +0 -15
- package/src/engine/movement/kinematic/modern.test.js +0 -27
- package/src/engine/store.js +0 -174
- package/src/engine/store.test.js +0 -256
- package/src/engine.js +0 -74
- package/src/game/animation.js +0 -26
- package/src/game/bounds.js +0 -66
- package/src/game/decorators/character.js +0 -5
- package/src/game/decorators/clamp-to-bounds.js +0 -15
- package/src/game/decorators/controls/dynamic/modern.js +0 -48
- package/src/game/decorators/controls/dynamic/shooter.js +0 -47
- package/src/game/decorators/controls/dynamic/tank.js +0 -55
- package/src/game/decorators/controls/kinematic/modern.js +0 -49
- package/src/game/decorators/controls/kinematic/shooter.js +0 -45
- package/src/game/decorators/controls/kinematic/tank.js +0 -52
- package/src/game/decorators/debug/collisions.js +0 -32
- package/src/game/decorators/double-jump.js +0 -70
- package/src/game/decorators/fps.js +0 -30
- package/src/game/decorators/fsm.js +0 -27
- package/src/game/decorators/game.js +0 -11
- package/src/game/decorators/image/image.js +0 -5
- package/src/game/decorators/image/sprite.js +0 -5
- package/src/game/decorators/image/tilemap.js +0 -5
- package/src/game/decorators/input/controls.js +0 -27
- package/src/game/decorators/input/gamepad.js +0 -74
- package/src/game/decorators/input/input.js +0 -41
- package/src/game/decorators/input/keyboard.js +0 -49
- package/src/game/decorators/input/mouse.js +0 -65
- package/src/game/decorators/jump.js +0 -72
- package/src/game/decorators/platform.js +0 -5
- package/src/game/decorators/ui/button.js +0 -21
- package/src/game/sprite.js +0 -119
- package/src/ui/canvas/absolute-position.js +0 -17
- package/src/ui/canvas/character.js +0 -35
- package/src/ui/canvas/form/button.js +0 -25
- package/src/ui/canvas/fps.js +0 -18
- package/src/ui/canvas/image/hitmask.js +0 -37
- package/src/ui/canvas/image/image.js +0 -37
- package/src/ui/canvas/image/sprite.js +0 -49
- package/src/ui/canvas/image/tilemap.js +0 -64
- package/src/ui/canvas/mouse.js +0 -37
- package/src/ui/canvas/shapes/circle.js +0 -31
- package/src/ui/canvas/shapes/rectangle.js +0 -31
- package/src/ui/canvas.js +0 -81
- package/src/ui/react/game/character/character.module.scss +0 -17
- package/src/ui/react/game/character/index.jsx +0 -30
- package/src/ui/react/game/cursor/cursor.module.scss +0 -47
- package/src/ui/react/game/cursor/index.jsx +0 -20
- package/src/ui/react/game/form/fields/field/field.module.scss +0 -5
- package/src/ui/react/game/form/fields/field/index.jsx +0 -56
- package/src/ui/react/game/form/fields/fields.module.scss +0 -48
- package/src/ui/react/game/form/fields/index.jsx +0 -12
- package/src/ui/react/game/form/form.module.scss +0 -18
- package/src/ui/react/game/form/index.jsx +0 -22
- package/src/ui/react/game/fps/index.jsx +0 -16
- package/src/ui/react/game/game.jsx +0 -71
- package/src/ui/react/game/index.jsx +0 -29
- package/src/ui/react/game/platform/index.jsx +0 -30
- package/src/ui/react/game/platform/platform.module.scss +0 -7
- package/src/ui/react/game/scene/index.jsx +0 -25
- package/src/ui/react/game/scene/scene.module.scss +0 -9
- package/src/ui/react/game/sprite/index.jsx +0 -58
- package/src/ui/react/game/sprite/sprite.module.css +0 -3
- package/src/ui/react/game/stats/index.jsx +0 -22
- package/src/ui/react/hocs/with-absolute-position/index.jsx +0 -20
- package/src/ui/react/hocs/with-absolute-position/with-absolute-position.module.scss +0 -5
- package/src/ui/react/index.jsx +0 -9
- package/src/utils/algorithms/decision-tree.js +0 -24
- package/src/utils/algorithms/decision-tree.test.js +0 -102
- 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 -102
- package/src/utils/data-structures/object.test.js +0 -121
- package/src/utils/data-structures/objects.js +0 -48
- 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 -117
- package/src/utils/math/geometry/circle.test.js +0 -97
- package/src/utils/math/geometry/hitmask.js +0 -39
- package/src/utils/math/geometry/hitmask.test.js +0 -84
- package/src/utils/math/geometry/line.js +0 -35
- package/src/utils/math/geometry/line.test.js +0 -49
- package/src/utils/math/geometry/platform.js +0 -42
- package/src/utils/math/geometry/platform.test.js +0 -133
- package/src/utils/math/geometry/point.js +0 -71
- package/src/utils/math/geometry/point.test.js +0 -81
- package/src/utils/math/geometry/rectangle.js +0 -45
- 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 -302
- package/src/utils/math/linear-algebra/vector.test.js +0 -257
- 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/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 -63
- package/src/utils/physics/friction.js +0 -30
- package/src/utils/physics/friction.test.js +0 -44
- package/src/utils/physics/gravity.js +0 -71
- package/src/utils/physics/gravity.test.js +0 -80
- package/src/utils/physics/jump.js +0 -41
- package/src/utils/physics/velocity.js +0 -38
|
@@ -1,159 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Default filler function for board cells.
|
|
3
|
-
* @returns {null} Default value for a cell.
|
|
4
|
-
*/
|
|
5
|
-
const DEFAULT_FILLER = () => null
|
|
6
|
-
|
|
7
|
-
/**
|
|
8
|
-
* Default function to convert a cell to a string.
|
|
9
|
-
* @param {*} cell - The cell value.
|
|
10
|
-
* @returns {string} String representation of the cell.
|
|
11
|
-
*/
|
|
12
|
-
const DEFAULT_CELL_TO_STRING = (cell) => `${cell}`
|
|
13
|
-
|
|
14
|
-
const FIRST_ROW = 0
|
|
15
|
-
const FIRST_COLUMN = 0
|
|
16
|
-
const LAST_ROW_OFFSET = 1
|
|
17
|
-
const LAST_COLUMN_OFFSET = 1
|
|
18
|
-
const MOVE_DOWN = 1
|
|
19
|
-
const MOVE_LEFT = -1
|
|
20
|
-
const MOVE_RIGHT = 1
|
|
21
|
-
const MOVE_UP = -1
|
|
22
|
-
|
|
23
|
-
/**
|
|
24
|
-
* Calculates the index in a one-dimensional array for given row and column.
|
|
25
|
-
* @param {number} row - The row index.
|
|
26
|
-
* @param {number} column - The column index.
|
|
27
|
-
* @param {number} columns - The total number of columns.
|
|
28
|
-
* @returns {number} The calculated index.
|
|
29
|
-
*/
|
|
30
|
-
function getIndex(row, column, columns) {
|
|
31
|
-
return row * columns + column
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
/**
|
|
35
|
-
* Creates a 1D board with the specified dimensions and fills it using the provided filler function.
|
|
36
|
-
* @param {[number, number]} dimensions - Array containing the number of rows and columns.
|
|
37
|
-
* @param {Function} [filler=DEFAULT_FILLER] - Function to fill each cell of the board.
|
|
38
|
-
* @returns {any[]} The created board.
|
|
39
|
-
*/
|
|
40
|
-
export function createBoard([rows, columns], filler = DEFAULT_FILLER) {
|
|
41
|
-
return new Array(rows * columns).fill(null).map((_, index) => {
|
|
42
|
-
const row = Math.floor(index / columns)
|
|
43
|
-
const column = index % columns
|
|
44
|
-
return filler(row, column)
|
|
45
|
-
})
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
/**
|
|
49
|
-
* Moves the given coordinates one step down.
|
|
50
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
51
|
-
* @param {[number]} size - Array containing the number of rows.
|
|
52
|
-
* @returns {[number, number]} The new coordinates after moving down.
|
|
53
|
-
* @throws {Error} If the movement goes out of bounds.
|
|
54
|
-
*/
|
|
55
|
-
export function down([i, j], [rows]) {
|
|
56
|
-
if (i === rows - LAST_ROW_OFFSET) {
|
|
57
|
-
throw new Error()
|
|
58
|
-
}
|
|
59
|
-
return [i + MOVE_DOWN, j]
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
/**
|
|
63
|
-
* Moves the given coordinates one step down and one step left.
|
|
64
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
65
|
-
* @param {[number, number]} size - Array containing the number of rows and columns.
|
|
66
|
-
* @returns {[number, number]} The new coordinates after moving down-left.
|
|
67
|
-
*/
|
|
68
|
-
export function downLeft(coords, size) {
|
|
69
|
-
return down(left(coords, size), size)
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* Moves the given coordinates one step down and one step right.
|
|
74
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
75
|
-
* @param {[number, number]} size - Array containing the number of rows and columns.
|
|
76
|
-
* @returns {[number, number]} The new coordinates after moving down-right.
|
|
77
|
-
*/
|
|
78
|
-
export function downRight(coords, size) {
|
|
79
|
-
return down(right(coords, size), size)
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
/**
|
|
83
|
-
* Moves the given coordinates one step left.
|
|
84
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
85
|
-
* @returns {[number, number]} The new coordinates after moving left.
|
|
86
|
-
* @throws {Error} If the movement goes out of bounds.
|
|
87
|
-
*/
|
|
88
|
-
export function left([i, j]) {
|
|
89
|
-
if (j === FIRST_COLUMN) {
|
|
90
|
-
throw new Error()
|
|
91
|
-
}
|
|
92
|
-
return [i, j + MOVE_LEFT]
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
/**
|
|
96
|
-
* Moves the given coordinates one step right.
|
|
97
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
98
|
-
* @param {[number, number]} size - Array containing the number of rows and columns.
|
|
99
|
-
* @returns {[number, number]} The new coordinates after moving right.
|
|
100
|
-
* @throws {Error} If the movement goes out of bounds.
|
|
101
|
-
*/
|
|
102
|
-
export function right([i, j], [, columns]) {
|
|
103
|
-
if (j === columns - LAST_COLUMN_OFFSET) {
|
|
104
|
-
throw new Error()
|
|
105
|
-
}
|
|
106
|
-
return [i, j + MOVE_RIGHT]
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
/**
|
|
110
|
-
* Converts the board to a string representation.
|
|
111
|
-
* @param {any[]} board - The 1D board to convert.
|
|
112
|
-
* @param {[number, number]} dimensions - Array containing the number of rows and columns.
|
|
113
|
-
* @param {Function} [cellToString=DEFAULT_CELL_TO_STRING] - Function to convert each cell to a string.
|
|
114
|
-
* @returns {string} The string representation of the board.
|
|
115
|
-
*/
|
|
116
|
-
export function toString(
|
|
117
|
-
board,
|
|
118
|
-
[rows, columns],
|
|
119
|
-
cellToString = DEFAULT_CELL_TO_STRING,
|
|
120
|
-
) {
|
|
121
|
-
return Array.from({ length: rows }, (_, i) =>
|
|
122
|
-
Array.from({ length: columns }, (_, j) =>
|
|
123
|
-
cellToString(board[getIndex(i, j, columns)], i, j),
|
|
124
|
-
).join(" "),
|
|
125
|
-
).join("\n")
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
/**
|
|
129
|
-
* Moves the given coordinates one step up.
|
|
130
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
131
|
-
* @returns {[number, number]} The new coordinates after moving up.
|
|
132
|
-
* @throws {Error} If the movement goes out of bounds.
|
|
133
|
-
*/
|
|
134
|
-
export function up([i, j]) {
|
|
135
|
-
if (i === FIRST_ROW) {
|
|
136
|
-
throw new Error()
|
|
137
|
-
}
|
|
138
|
-
return [i + MOVE_UP, j]
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
/**
|
|
142
|
-
* Moves the given coordinates one step up and one step left.
|
|
143
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
144
|
-
* @param {[number, number]} size - Array containing the number of rows and columns.
|
|
145
|
-
* @returns {[number, number]} The new coordinates after moving up-left.
|
|
146
|
-
*/
|
|
147
|
-
export function upLeft(coords, size) {
|
|
148
|
-
return up(left(coords, size), size)
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
/**
|
|
152
|
-
* Moves the given coordinates one step up and one step right.
|
|
153
|
-
* @param {[number, number]} coords - The current coordinates [row, column].
|
|
154
|
-
* @param {[number, number]} size - Array containing the number of rows and columns.
|
|
155
|
-
* @returns {[number, number]} The new coordinates after moving up-right.
|
|
156
|
-
*/
|
|
157
|
-
export function upRight(coords, size) {
|
|
158
|
-
return up(right(coords, size), size)
|
|
159
|
-
}
|
|
@@ -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
|
-
})
|