@inglorious/engine 0.1.1 → 0.2.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.
Files changed (201) hide show
  1. package/README.md +39 -36
  2. package/package.json +10 -22
  3. package/src/engine/ai/movement/dynamic/align.js +63 -63
  4. package/src/engine/ai/movement/dynamic/arrive.js +42 -43
  5. package/src/engine/ai/movement/dynamic/evade.js +38 -38
  6. package/src/engine/ai/movement/dynamic/face.js +19 -20
  7. package/src/engine/ai/movement/dynamic/flee.js +45 -45
  8. package/src/engine/ai/movement/dynamic/look-where-youre-going.js +16 -17
  9. package/src/engine/ai/movement/dynamic/match-velocity.js +51 -50
  10. package/src/engine/ai/movement/dynamic/pursue.js +38 -38
  11. package/src/engine/ai/movement/dynamic/seek.js +44 -44
  12. package/src/engine/ai/movement/dynamic/wander.js +31 -32
  13. package/src/engine/ai/movement/kinematic/align.js +37 -37
  14. package/src/engine/ai/movement/kinematic/arrive.js +42 -42
  15. package/src/engine/ai/movement/kinematic/face.js +19 -20
  16. package/src/engine/ai/movement/kinematic/flee.js +26 -26
  17. package/src/engine/ai/movement/kinematic/seek.js +26 -26
  18. package/src/engine/ai/movement/kinematic/seek.test.js +42 -42
  19. package/src/engine/ai/movement/kinematic/wander-as-seek.js +31 -31
  20. package/src/engine/ai/movement/kinematic/wander.js +27 -27
  21. package/src/engine/animation/sprite.js +101 -0
  22. package/src/engine/animation/ticker.js +38 -0
  23. package/src/engine/behaviors/camera.js +68 -0
  24. package/src/engine/behaviors/controls/dynamic/modern.js +76 -0
  25. package/src/engine/behaviors/controls/dynamic/shooter.js +84 -0
  26. package/src/engine/behaviors/controls/dynamic/tank.js +69 -0
  27. package/src/engine/behaviors/controls/event-handlers.js +17 -0
  28. package/src/engine/behaviors/controls/kinematic/modern.js +76 -0
  29. package/src/engine/behaviors/controls/kinematic/shooter.js +82 -0
  30. package/src/engine/behaviors/controls/kinematic/tank.js +67 -0
  31. package/src/engine/behaviors/debug/collision.js +35 -0
  32. package/src/engine/behaviors/fps.js +29 -0
  33. package/src/engine/behaviors/fsm.js +33 -0
  34. package/src/{game/decorators → engine/behaviors}/fsm.test.js +49 -56
  35. package/src/engine/behaviors/game.js +15 -0
  36. package/src/engine/behaviors/input/controls.js +37 -0
  37. package/src/engine/behaviors/input/gamepad.js +114 -0
  38. package/src/engine/behaviors/input/input.js +48 -0
  39. package/src/engine/behaviors/input/keyboard.js +64 -0
  40. package/src/engine/behaviors/input/mouse.js +91 -0
  41. package/src/engine/behaviors/physics/bouncy.js +25 -0
  42. package/src/engine/behaviors/physics/clamped.js +36 -0
  43. package/src/{game/decorators/collisions.js → engine/behaviors/physics/collidable.js} +20 -24
  44. package/src/engine/behaviors/physics/jumpable.js +145 -0
  45. package/src/engine/behaviors/ui/button.js +17 -0
  46. package/src/engine/collision/detection.js +110 -115
  47. package/src/engine/core/api.js +34 -0
  48. package/src/engine/core/dev-tools.js +135 -0
  49. package/src/engine/core/engine.js +119 -0
  50. package/src/engine/core/loop.js +15 -0
  51. package/src/engine/{loop → core/loops}/animation-frame.js +25 -26
  52. package/src/engine/{loop → core/loops}/elapsed.js +22 -23
  53. package/src/engine/{loop → core/loops}/fixed.js +27 -28
  54. package/src/engine/{loop → core/loops}/flash.js +13 -14
  55. package/src/engine/{loop → core/loops}/lag.js +26 -27
  56. package/src/engine/core/select.js +26 -0
  57. package/src/engine/core/store.js +178 -0
  58. package/src/engine/core/store.test.js +110 -0
  59. package/src/engine/movement/dynamic/modern.js +21 -24
  60. package/src/engine/movement/dynamic/tank.js +43 -43
  61. package/src/engine/movement/kinematic/modern.js +16 -16
  62. package/src/engine/movement/kinematic/modern.test.js +27 -27
  63. package/src/engine/movement/kinematic/tank.js +27 -27
  64. package/src/engine/physics/bounds.js +138 -0
  65. package/src/engine/physics/position.js +43 -0
  66. package/src/engine/physics/position.test.js +80 -0
  67. package/src/engine/systems/sprite-animation.js +27 -0
  68. package/src/main.js +10 -5
  69. package/src/renderers/canvas/absolute-position.js +18 -0
  70. package/src/renderers/canvas/camera.js +13 -0
  71. package/src/renderers/canvas/canvas-renderer.js +68 -0
  72. package/src/{ui → renderers}/canvas/character.js +38 -35
  73. package/src/{ui → renderers}/canvas/form/button.js +25 -25
  74. package/src/{ui → renderers}/canvas/fps.js +18 -18
  75. package/src/renderers/canvas/image/hitmask.js +51 -0
  76. package/src/{ui → renderers}/canvas/image/image.js +34 -37
  77. package/src/{ui → renderers}/canvas/image/sprite.js +49 -49
  78. package/src/{ui → renderers}/canvas/image/tilemap.js +66 -64
  79. package/src/{ui → renderers}/canvas/mouse.js +37 -37
  80. package/src/renderers/canvas/rendering-system.js +79 -0
  81. package/src/{ui → renderers}/canvas/shapes/circle.js +29 -31
  82. package/src/{ui → renderers}/canvas/shapes/rectangle.js +27 -31
  83. package/src/renderers/react/game/character/index.jsx +20 -0
  84. package/src/{ui → renderers}/react/game/cursor/index.jsx +20 -20
  85. package/src/{ui → renderers}/react/game/form/fields/field/index.jsx +56 -56
  86. package/src/{ui → renderers}/react/game/form/fields/index.jsx +12 -12
  87. package/src/{ui → renderers}/react/game/form/index.jsx +22 -22
  88. package/src/{ui → renderers}/react/game/fps/index.jsx +16 -16
  89. package/src/{ui → renderers}/react/game/game.jsx +72 -71
  90. package/src/{ui → renderers}/react/game/index.jsx +29 -29
  91. package/src/{ui → renderers}/react/game/platform/index.jsx +30 -30
  92. package/src/{ui → renderers}/react/game/scene/index.jsx +27 -25
  93. package/src/{ui → renderers}/react/game/sprite/index.jsx +60 -58
  94. package/src/{ui → renderers}/react/game/stats/index.jsx +22 -22
  95. package/src/{ui → renderers}/react/hocs/with-absolute-position/index.jsx +20 -20
  96. package/src/{ui → renderers}/react/index.jsx +9 -9
  97. package/src/utils/algorithms/decision-tree.js +24 -24
  98. package/src/utils/algorithms/decision-tree.test.js +153 -102
  99. package/src/utils/algorithms/path-finding.js +155 -155
  100. package/src/utils/algorithms/path-finding.test.js +151 -151
  101. package/src/utils/data-structures/array.js +83 -83
  102. package/src/utils/data-structures/array.test.js +173 -173
  103. package/src/utils/data-structures/board.js +159 -159
  104. package/src/utils/data-structures/board.test.js +242 -242
  105. package/src/utils/data-structures/boolean.js +9 -9
  106. package/src/utils/data-structures/heap.js +164 -164
  107. package/src/utils/data-structures/heap.test.js +103 -103
  108. package/src/utils/data-structures/object.js +138 -102
  109. package/src/utils/data-structures/object.test.js +218 -121
  110. package/src/utils/data-structures/objects.js +66 -48
  111. package/src/utils/data-structures/objects.test.js +99 -99
  112. package/src/utils/data-structures/tree.js +36 -36
  113. package/src/utils/data-structures/tree.test.js +33 -33
  114. package/src/utils/functions/functions.js +19 -19
  115. package/src/utils/functions/functions.test.js +23 -23
  116. package/src/utils/math/geometry/circle.js +70 -117
  117. package/src/utils/math/geometry/circle.test.js +97 -97
  118. package/src/utils/math/geometry/hitmask.js +70 -39
  119. package/src/utils/math/geometry/hitmask.test.js +155 -84
  120. package/src/utils/math/geometry/line.js +35 -35
  121. package/src/utils/math/geometry/line.test.js +49 -49
  122. package/src/utils/math/geometry/point.js +78 -71
  123. package/src/utils/math/geometry/point.test.js +81 -81
  124. package/src/utils/math/geometry/rectangle.js +76 -45
  125. package/src/utils/math/geometry/rectangle.test.js +42 -42
  126. package/src/utils/math/geometry/segment.js +80 -80
  127. package/src/utils/math/geometry/segment.test.js +183 -183
  128. package/src/utils/math/geometry/triangle.js +15 -15
  129. package/src/utils/math/geometry/triangle.test.js +11 -11
  130. package/src/utils/math/linear-algebra/2d.js +28 -28
  131. package/src/utils/math/linear-algebra/2d.test.js +17 -17
  132. package/src/utils/math/linear-algebra/quaternion.js +22 -22
  133. package/src/utils/math/linear-algebra/quaternion.test.js +25 -25
  134. package/src/utils/math/linear-algebra/quaternions.js +20 -20
  135. package/src/utils/math/linear-algebra/quaternions.test.js +29 -29
  136. package/src/utils/math/linear-algebra/vector.js +327 -302
  137. package/src/utils/math/linear-algebra/vector.test.js +265 -257
  138. package/src/utils/math/linear-algebra/vectors.js +122 -122
  139. package/src/utils/math/linear-algebra/vectors.test.js +65 -65
  140. package/src/utils/math/linear-interpolation.js +9 -0
  141. package/src/utils/math/numbers.js +90 -90
  142. package/src/utils/math/numbers.test.js +137 -137
  143. package/src/utils/math/rng.js +44 -44
  144. package/src/utils/math/rng.test.js +39 -39
  145. package/src/utils/math/statistics.js +43 -43
  146. package/src/utils/math/statistics.test.js +47 -47
  147. package/src/utils/math/trigonometry.js +89 -89
  148. package/src/utils/math/trigonometry.test.js +52 -52
  149. package/src/utils/physics/acceleration.js +61 -63
  150. package/src/utils/physics/friction.js +28 -30
  151. package/src/utils/physics/friction.test.js +42 -44
  152. package/src/utils/physics/gravity.js +69 -71
  153. package/src/utils/physics/gravity.test.js +77 -80
  154. package/src/utils/physics/jump.js +31 -41
  155. package/src/utils/physics/velocity.js +36 -38
  156. package/src/engine/loop.js +0 -15
  157. package/src/engine/store.js +0 -174
  158. package/src/engine/store.test.js +0 -256
  159. package/src/engine.js +0 -74
  160. package/src/game/animation.js +0 -26
  161. package/src/game/bounds.js +0 -66
  162. package/src/game/decorators/character.js +0 -5
  163. package/src/game/decorators/clamp-to-bounds.js +0 -15
  164. package/src/game/decorators/controls/dynamic/modern.js +0 -48
  165. package/src/game/decorators/controls/dynamic/shooter.js +0 -47
  166. package/src/game/decorators/controls/dynamic/tank.js +0 -55
  167. package/src/game/decorators/controls/kinematic/modern.js +0 -49
  168. package/src/game/decorators/controls/kinematic/shooter.js +0 -45
  169. package/src/game/decorators/controls/kinematic/tank.js +0 -52
  170. package/src/game/decorators/debug/collisions.js +0 -32
  171. package/src/game/decorators/double-jump.js +0 -70
  172. package/src/game/decorators/fps.js +0 -30
  173. package/src/game/decorators/fsm.js +0 -27
  174. package/src/game/decorators/game.js +0 -11
  175. package/src/game/decorators/image/image.js +0 -5
  176. package/src/game/decorators/image/sprite.js +0 -5
  177. package/src/game/decorators/image/tilemap.js +0 -5
  178. package/src/game/decorators/input/controls.js +0 -27
  179. package/src/game/decorators/input/gamepad.js +0 -74
  180. package/src/game/decorators/input/input.js +0 -41
  181. package/src/game/decorators/input/keyboard.js +0 -49
  182. package/src/game/decorators/input/mouse.js +0 -65
  183. package/src/game/decorators/jump.js +0 -72
  184. package/src/game/decorators/platform.js +0 -5
  185. package/src/game/decorators/ui/button.js +0 -21
  186. package/src/game/sprite.js +0 -119
  187. package/src/ui/canvas/absolute-position.js +0 -17
  188. package/src/ui/canvas/image/hitmask.js +0 -37
  189. package/src/ui/canvas.js +0 -81
  190. package/src/ui/react/game/character/index.jsx +0 -30
  191. package/src/utils/math/geometry/platform.js +0 -42
  192. package/src/utils/math/geometry/platform.test.js +0 -133
  193. /package/src/{ui → renderers}/react/game/character/character.module.scss +0 -0
  194. /package/src/{ui → renderers}/react/game/cursor/cursor.module.scss +0 -0
  195. /package/src/{ui → renderers}/react/game/form/fields/field/field.module.scss +0 -0
  196. /package/src/{ui → renderers}/react/game/form/fields/fields.module.scss +0 -0
  197. /package/src/{ui → renderers}/react/game/form/form.module.scss +0 -0
  198. /package/src/{ui → renderers}/react/game/platform/platform.module.scss +0 -0
  199. /package/src/{ui → renderers}/react/game/scene/scene.module.scss +0 -0
  200. /package/src/{ui → renderers}/react/game/sprite/sprite.module.css +0 -0
  201. /package/src/{ui → renderers}/react/hocs/with-absolute-position/with-absolute-position.module.scss +0 -0
@@ -1,164 +1,164 @@
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
+ // @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 +1,103 @@
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
+ 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
+ })