@inglorious/engine 0.1.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 (256) hide show
  1. package/LICENSE +9 -0
  2. package/README.md +72 -0
  3. package/package.json +76 -0
  4. package/src/docs/ai/movement/dynamic/align.js +131 -0
  5. package/src/docs/ai/movement/dynamic/arrive.js +88 -0
  6. package/src/docs/ai/movement/dynamic/dynamic.mdx +99 -0
  7. package/src/docs/ai/movement/dynamic/dynamic.stories.js +58 -0
  8. package/src/docs/ai/movement/dynamic/evade.js +72 -0
  9. package/src/docs/ai/movement/dynamic/face.js +90 -0
  10. package/src/docs/ai/movement/dynamic/flee.js +38 -0
  11. package/src/docs/ai/movement/dynamic/look-where-youre-going.js +114 -0
  12. package/src/docs/ai/movement/dynamic/match-velocity.js +92 -0
  13. package/src/docs/ai/movement/dynamic/pursue.js +72 -0
  14. package/src/docs/ai/movement/dynamic/seek.js +37 -0
  15. package/src/docs/ai/movement/dynamic/wander.js +71 -0
  16. package/src/docs/ai/movement/kinematic/align.js +122 -0
  17. package/src/docs/ai/movement/kinematic/arrive.js +78 -0
  18. package/src/docs/ai/movement/kinematic/face.js +82 -0
  19. package/src/docs/ai/movement/kinematic/flee.js +36 -0
  20. package/src/docs/ai/movement/kinematic/kinematic.mdx +67 -0
  21. package/src/docs/ai/movement/kinematic/kinematic.stories.js +42 -0
  22. package/src/docs/ai/movement/kinematic/seek.js +34 -0
  23. package/src/docs/ai/movement/kinematic/wander-as-seek.js +62 -0
  24. package/src/docs/ai/movement/kinematic/wander.js +28 -0
  25. package/src/docs/bounds.js +7 -0
  26. package/src/docs/code-reuse.js +35 -0
  27. package/src/docs/collision/circles.js +58 -0
  28. package/src/docs/collision/collision.mdx +27 -0
  29. package/src/docs/collision/collision.stories.js +22 -0
  30. package/src/docs/collision/platform.js +76 -0
  31. package/src/docs/collision/tilemap.js +181 -0
  32. package/src/docs/empty.js +1 -0
  33. package/src/docs/engine.mdx +81 -0
  34. package/src/docs/engine.stories.js +37 -0
  35. package/src/docs/event-handlers.js +68 -0
  36. package/src/docs/framerate.js +37 -0
  37. package/src/docs/game.jsx +15 -0
  38. package/src/docs/image/image.js +19 -0
  39. package/src/docs/image/image.stories.js +22 -0
  40. package/src/docs/image/sprite.js +39 -0
  41. package/src/docs/image/tilemap.js +84 -0
  42. package/src/docs/input/controls.js +67 -0
  43. package/src/docs/input/gamepad.js +67 -0
  44. package/src/docs/input/input.mdx +55 -0
  45. package/src/docs/input/input.stories.js +27 -0
  46. package/src/docs/input/keyboard.js +58 -0
  47. package/src/docs/input/mouse.js +32 -0
  48. package/src/docs/instances.js +49 -0
  49. package/src/docs/player/dynamic/double-jump.js +90 -0
  50. package/src/docs/player/dynamic/dynamic.stories.js +32 -0
  51. package/src/docs/player/dynamic/jump.js +83 -0
  52. package/src/docs/player/dynamic/modern-controls.js +57 -0
  53. package/src/docs/player/dynamic/shooter-controls.js +51 -0
  54. package/src/docs/player/dynamic/tank-controls.js +44 -0
  55. package/src/docs/player/kinematic/double-jump.js +90 -0
  56. package/src/docs/player/kinematic/jump.js +82 -0
  57. package/src/docs/player/kinematic/kinematic.stories.js +32 -0
  58. package/src/docs/player/kinematic/modern-controls.js +56 -0
  59. package/src/docs/player/kinematic/shooter-controls.js +48 -0
  60. package/src/docs/player/kinematic/tank-controls.js +42 -0
  61. package/src/docs/quick-start/first-game.js +49 -0
  62. package/src/docs/quick-start/hello-world.js +1 -0
  63. package/src/docs/quick-start.mdx +127 -0
  64. package/src/docs/quick-start.stories.js +17 -0
  65. package/src/docs/recipes/add-and-remove.js +71 -0
  66. package/src/docs/recipes/add-instance.js +42 -0
  67. package/src/docs/recipes/decision-tree.js +169 -0
  68. package/src/docs/recipes/random-instances.js +25 -0
  69. package/src/docs/recipes/recipes.mdx +81 -0
  70. package/src/docs/recipes/recipes.stories.js +37 -0
  71. package/src/docs/recipes/remove-instance.js +52 -0
  72. package/src/docs/recipes/states.js +64 -0
  73. package/src/docs/ui/button.js +28 -0
  74. package/src/docs/ui/form.stories.js +55 -0
  75. package/src/docs/ui-chooser.jsx +6 -0
  76. package/src/docs/utils/data-structures/object.mdx +47 -0
  77. package/src/docs/utils/data-structures/objects.mdx +30 -0
  78. package/src/docs/utils/functions/functions.mdx +34 -0
  79. package/src/docs/utils/math/geometry/circle.mdx +55 -0
  80. package/src/docs/utils/math/geometry/point.mdx +38 -0
  81. package/src/docs/utils/math/geometry/rectangle.mdx +24 -0
  82. package/src/docs/utils/math/geometry/segment.mdx +55 -0
  83. package/src/docs/utils/math/geometry/triangle.mdx +22 -0
  84. package/src/docs/utils/math/linear-algebra/2d.mdx +22 -0
  85. package/src/docs/utils/math/linear-algebra/quaternion.mdx +21 -0
  86. package/src/docs/utils/math/linear-algebra/quaternions.mdx +22 -0
  87. package/src/docs/utils/math/linear-algebra/vector.mdx +177 -0
  88. package/src/docs/utils/math/linear-algebra/vectors.mdx +58 -0
  89. package/src/docs/utils/math/numbers.mdx +76 -0
  90. package/src/docs/utils/math/random.mdx +35 -0
  91. package/src/docs/utils/math/statistics.mdx +38 -0
  92. package/src/docs/utils/math/trigonometry.mdx +85 -0
  93. package/src/docs/utils/physics/friction.mdx +20 -0
  94. package/src/docs/utils/physics/gravity.mdx +28 -0
  95. package/src/engine/ai/movement/dynamic/align.js +63 -0
  96. package/src/engine/ai/movement/dynamic/arrive.js +43 -0
  97. package/src/engine/ai/movement/dynamic/evade.js +38 -0
  98. package/src/engine/ai/movement/dynamic/face.js +20 -0
  99. package/src/engine/ai/movement/dynamic/flee.js +45 -0
  100. package/src/engine/ai/movement/dynamic/look-where-youre-going.js +17 -0
  101. package/src/engine/ai/movement/dynamic/match-velocity.js +50 -0
  102. package/src/engine/ai/movement/dynamic/pursue.js +38 -0
  103. package/src/engine/ai/movement/dynamic/seek.js +44 -0
  104. package/src/engine/ai/movement/dynamic/wander.js +32 -0
  105. package/src/engine/ai/movement/kinematic/align.js +37 -0
  106. package/src/engine/ai/movement/kinematic/arrive.js +42 -0
  107. package/src/engine/ai/movement/kinematic/face.js +20 -0
  108. package/src/engine/ai/movement/kinematic/flee.js +26 -0
  109. package/src/engine/ai/movement/kinematic/seek.js +26 -0
  110. package/src/engine/ai/movement/kinematic/seek.test.js +42 -0
  111. package/src/engine/ai/movement/kinematic/wander-as-seek.js +31 -0
  112. package/src/engine/ai/movement/kinematic/wander.js +27 -0
  113. package/src/engine/collision/detection.js +115 -0
  114. package/src/engine/loop/animation-frame.js +26 -0
  115. package/src/engine/loop/elapsed.js +23 -0
  116. package/src/engine/loop/fixed.js +28 -0
  117. package/src/engine/loop/flash.js +14 -0
  118. package/src/engine/loop/lag.js +27 -0
  119. package/src/engine/loop.js +15 -0
  120. package/src/engine/movement/dynamic/modern.js +24 -0
  121. package/src/engine/movement/dynamic/tank.js +43 -0
  122. package/src/engine/movement/kinematic/modern.js +16 -0
  123. package/src/engine/movement/kinematic/modern.test.js +27 -0
  124. package/src/engine/movement/kinematic/tank.js +27 -0
  125. package/src/engine/store.js +174 -0
  126. package/src/engine/store.test.js +256 -0
  127. package/src/engine.js +74 -0
  128. package/src/game/animation.js +26 -0
  129. package/src/game/bounds.js +66 -0
  130. package/src/game/decorators/character.js +5 -0
  131. package/src/game/decorators/clamp-to-bounds.js +15 -0
  132. package/src/game/decorators/collisions.js +24 -0
  133. package/src/game/decorators/controls/dynamic/modern.js +48 -0
  134. package/src/game/decorators/controls/dynamic/shooter.js +47 -0
  135. package/src/game/decorators/controls/dynamic/tank.js +55 -0
  136. package/src/game/decorators/controls/kinematic/modern.js +49 -0
  137. package/src/game/decorators/controls/kinematic/shooter.js +45 -0
  138. package/src/game/decorators/controls/kinematic/tank.js +52 -0
  139. package/src/game/decorators/debug/collisions.js +32 -0
  140. package/src/game/decorators/double-jump.js +70 -0
  141. package/src/game/decorators/fps.js +30 -0
  142. package/src/game/decorators/fsm.js +27 -0
  143. package/src/game/decorators/fsm.test.js +56 -0
  144. package/src/game/decorators/game.js +11 -0
  145. package/src/game/decorators/image/image.js +5 -0
  146. package/src/game/decorators/image/sprite.js +5 -0
  147. package/src/game/decorators/image/tilemap.js +5 -0
  148. package/src/game/decorators/input/controls.js +27 -0
  149. package/src/game/decorators/input/gamepad.js +74 -0
  150. package/src/game/decorators/input/input.js +41 -0
  151. package/src/game/decorators/input/keyboard.js +49 -0
  152. package/src/game/decorators/input/mouse.js +65 -0
  153. package/src/game/decorators/jump.js +72 -0
  154. package/src/game/decorators/platform.js +5 -0
  155. package/src/game/decorators/ui/button.js +21 -0
  156. package/src/game/sprite.js +119 -0
  157. package/src/main.js +5 -0
  158. package/src/ui/canvas/absolute-position.js +17 -0
  159. package/src/ui/canvas/character.js +35 -0
  160. package/src/ui/canvas/form/button.js +25 -0
  161. package/src/ui/canvas/fps.js +18 -0
  162. package/src/ui/canvas/image/hitmask.js +37 -0
  163. package/src/ui/canvas/image/image.js +37 -0
  164. package/src/ui/canvas/image/sprite.js +49 -0
  165. package/src/ui/canvas/image/tilemap.js +64 -0
  166. package/src/ui/canvas/mouse.js +37 -0
  167. package/src/ui/canvas/shapes/circle.js +31 -0
  168. package/src/ui/canvas/shapes/rectangle.js +31 -0
  169. package/src/ui/canvas.js +81 -0
  170. package/src/ui/react/game/character/character.module.scss +17 -0
  171. package/src/ui/react/game/character/index.jsx +30 -0
  172. package/src/ui/react/game/cursor/cursor.module.scss +47 -0
  173. package/src/ui/react/game/cursor/index.jsx +20 -0
  174. package/src/ui/react/game/form/fields/field/field.module.scss +5 -0
  175. package/src/ui/react/game/form/fields/field/index.jsx +56 -0
  176. package/src/ui/react/game/form/fields/fields.module.scss +48 -0
  177. package/src/ui/react/game/form/fields/index.jsx +12 -0
  178. package/src/ui/react/game/form/form.module.scss +18 -0
  179. package/src/ui/react/game/form/index.jsx +22 -0
  180. package/src/ui/react/game/fps/index.jsx +16 -0
  181. package/src/ui/react/game/game.jsx +71 -0
  182. package/src/ui/react/game/index.jsx +29 -0
  183. package/src/ui/react/game/platform/index.jsx +30 -0
  184. package/src/ui/react/game/platform/platform.module.scss +7 -0
  185. package/src/ui/react/game/scene/index.jsx +25 -0
  186. package/src/ui/react/game/scene/scene.module.scss +9 -0
  187. package/src/ui/react/game/sprite/index.jsx +58 -0
  188. package/src/ui/react/game/sprite/sprite.module.css +3 -0
  189. package/src/ui/react/game/stats/index.jsx +22 -0
  190. package/src/ui/react/hocs/with-absolute-position/index.jsx +20 -0
  191. package/src/ui/react/hocs/with-absolute-position/with-absolute-position.module.scss +5 -0
  192. package/src/ui/react/index.jsx +9 -0
  193. package/src/utils/algorithms/decision-tree.js +24 -0
  194. package/src/utils/algorithms/decision-tree.test.js +102 -0
  195. package/src/utils/algorithms/path-finding.js +155 -0
  196. package/src/utils/algorithms/path-finding.test.js +151 -0
  197. package/src/utils/algorithms/types.d.ts +28 -0
  198. package/src/utils/data-structures/array.js +83 -0
  199. package/src/utils/data-structures/array.test.js +173 -0
  200. package/src/utils/data-structures/board.js +159 -0
  201. package/src/utils/data-structures/board.test.js +242 -0
  202. package/src/utils/data-structures/boolean.js +9 -0
  203. package/src/utils/data-structures/heap.js +164 -0
  204. package/src/utils/data-structures/heap.test.js +103 -0
  205. package/src/utils/data-structures/object.js +102 -0
  206. package/src/utils/data-structures/object.test.js +121 -0
  207. package/src/utils/data-structures/objects.js +48 -0
  208. package/src/utils/data-structures/objects.test.js +99 -0
  209. package/src/utils/data-structures/tree.js +36 -0
  210. package/src/utils/data-structures/tree.test.js +33 -0
  211. package/src/utils/data-structures/types.d.ts +4 -0
  212. package/src/utils/functions/functions.js +19 -0
  213. package/src/utils/functions/functions.test.js +23 -0
  214. package/src/utils/math/geometry/circle.js +117 -0
  215. package/src/utils/math/geometry/circle.test.js +97 -0
  216. package/src/utils/math/geometry/hitmask.js +39 -0
  217. package/src/utils/math/geometry/hitmask.test.js +84 -0
  218. package/src/utils/math/geometry/line.js +35 -0
  219. package/src/utils/math/geometry/line.test.js +49 -0
  220. package/src/utils/math/geometry/platform.js +42 -0
  221. package/src/utils/math/geometry/platform.test.js +133 -0
  222. package/src/utils/math/geometry/point.js +71 -0
  223. package/src/utils/math/geometry/point.test.js +81 -0
  224. package/src/utils/math/geometry/rectangle.js +45 -0
  225. package/src/utils/math/geometry/rectangle.test.js +42 -0
  226. package/src/utils/math/geometry/segment.js +80 -0
  227. package/src/utils/math/geometry/segment.test.js +183 -0
  228. package/src/utils/math/geometry/triangle.js +15 -0
  229. package/src/utils/math/geometry/triangle.test.js +11 -0
  230. package/src/utils/math/geometry/types.d.ts +23 -0
  231. package/src/utils/math/linear-algebra/2d.js +28 -0
  232. package/src/utils/math/linear-algebra/2d.test.js +17 -0
  233. package/src/utils/math/linear-algebra/quaternion.js +22 -0
  234. package/src/utils/math/linear-algebra/quaternion.test.js +25 -0
  235. package/src/utils/math/linear-algebra/quaternions.js +20 -0
  236. package/src/utils/math/linear-algebra/quaternions.test.js +29 -0
  237. package/src/utils/math/linear-algebra/types.d.ts +4 -0
  238. package/src/utils/math/linear-algebra/vector.js +302 -0
  239. package/src/utils/math/linear-algebra/vector.test.js +257 -0
  240. package/src/utils/math/linear-algebra/vectors.js +122 -0
  241. package/src/utils/math/linear-algebra/vectors.test.js +65 -0
  242. package/src/utils/math/numbers.js +90 -0
  243. package/src/utils/math/numbers.test.js +137 -0
  244. package/src/utils/math/rng.js +44 -0
  245. package/src/utils/math/rng.test.js +39 -0
  246. package/src/utils/math/statistics.js +43 -0
  247. package/src/utils/math/statistics.test.js +47 -0
  248. package/src/utils/math/trigonometry.js +89 -0
  249. package/src/utils/math/trigonometry.test.js +52 -0
  250. package/src/utils/physics/acceleration.js +63 -0
  251. package/src/utils/physics/friction.js +30 -0
  252. package/src/utils/physics/friction.test.js +44 -0
  253. package/src/utils/physics/gravity.js +71 -0
  254. package/src/utils/physics/gravity.test.js +80 -0
  255. package/src/utils/physics/jump.js +41 -0
  256. package/src/utils/physics/velocity.js +38 -0
@@ -0,0 +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
+ }
@@ -0,0 +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
+ })
@@ -0,0 +1,102 @@
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
+ * Converts an object or array to a formatted string representation.
66
+ * @param {*} obj - The object or array to convert.
67
+ * @param {number} [indentationLevel=INITIAL_LEVEL] - The current indentation level (used for nested structures).
68
+ * @returns {string} A string representation of the input object or array.
69
+ */
70
+ export function toString(obj, indentationLevel = INITIAL_LEVEL) {
71
+ if (Array.isArray(obj)) {
72
+ return `[
73
+ ${obj
74
+ .map(
75
+ (item) =>
76
+ " ".repeat(indentationLevel + NEXT_LEVEL) +
77
+ toString(item, indentationLevel + NEXT_LEVEL),
78
+ )
79
+ .join(",\n")}
80
+ ${" ".repeat(indentationLevel)}]`
81
+ }
82
+
83
+ if (typeof obj === "object" && obj != null) {
84
+ return `{
85
+ ${Object.entries(obj)
86
+ .map(
87
+ ([key, value]) =>
88
+ `${" ".repeat(indentationLevel + NEXT_LEVEL)}${key}: ${toString(
89
+ value,
90
+ indentationLevel + NEXT_LEVEL,
91
+ )}`,
92
+ )
93
+ .join(",\n")}
94
+ ${" ".repeat(indentationLevel)}}`
95
+ }
96
+
97
+ if (typeof obj === "string") {
98
+ return `"${obj}"`
99
+ }
100
+
101
+ return obj
102
+ }
@@ -0,0 +1,121 @@
1
+ import { expect, test } from "vitest"
2
+
3
+ import { clone, filter, find, isObject, map, toString } from "./object.js"
4
+
5
+ test("it should deep clone an object", () => {
6
+ const obj = {
7
+ primitive: 1,
8
+ array: [2, 3],
9
+ object: { a: 1, b: 2 },
10
+ }
11
+ const expectedResult = {
12
+ primitive: 1,
13
+ array: [2, 3],
14
+ object: { a: 1, b: 2 },
15
+ }
16
+
17
+ const result = clone(obj)
18
+
19
+ expect(result).toStrictEqual(expectedResult)
20
+ expect(result).not.toBe(expectedResult)
21
+ expect(result.primitive).toBe(expectedResult.primitive)
22
+ expect(result.array).not.toBe(expectedResult.array)
23
+ expect(result.object).not.toBe(expectedResult.object)
24
+ })
25
+
26
+ test("it should behave like Array.prototype.filter, but on an object", () => {
27
+ const obj = {
28
+ key1: "value1",
29
+ key2: "value2",
30
+ key3: "value3",
31
+ }
32
+ const callback = (key) => ["key2", "key3"].includes(key)
33
+ const expectedResult = {
34
+ key2: "value2",
35
+ key3: "value3",
36
+ }
37
+
38
+ expect(filter(obj, callback)).toStrictEqual(expectedResult)
39
+ })
40
+
41
+ test("it should behave like Array.prototype.find, but on an object", () => {
42
+ const obj = {
43
+ key1: "value1",
44
+ key2: "value2",
45
+ key3: "value3",
46
+ }
47
+ const callback = (key) => ["key2", "key3"].includes(key)
48
+ const expectedResult = {
49
+ key2: "value2",
50
+ }
51
+
52
+ expect(find(obj, callback)).toStrictEqual(expectedResult)
53
+ })
54
+
55
+ test("it correctly should check if a value is an object", () => {
56
+ expect(isObject(1)).toBe(false)
57
+ expect(isObject("a")).toBe(false)
58
+ expect(isObject([])).toBe(false)
59
+ expect(isObject(null)).toBe(false)
60
+ expect(isObject(new Date())).toBe(false)
61
+ expect(isObject({})).toBe(true)
62
+ })
63
+
64
+ test("it should behave like Array.prototype.map, but on an object", () => {
65
+ const obj = {
66
+ key1: "value1",
67
+ key2: "value2",
68
+ key3: "value3",
69
+ }
70
+ const callback = (key, value) => value.toUpperCase()
71
+ const expectedResult = {
72
+ key1: "VALUE1",
73
+ key2: "VALUE2",
74
+ key3: "VALUE3",
75
+ }
76
+
77
+ expect(map(obj, callback)).toStrictEqual(expectedResult)
78
+ })
79
+
80
+ test("it should return a string representation of a shallow object", () => {
81
+ const obj = {
82
+ key1: "value1",
83
+ key2: "value2",
84
+ key3: "value3",
85
+ }
86
+
87
+ expect(toString(obj)).toBe(`{
88
+ key1: "value1",
89
+ key2: "value2",
90
+ key3: "value3"
91
+ }`)
92
+ })
93
+
94
+ test("it should return a string representation of a nested object", () => {
95
+ const obj = {
96
+ a: 1,
97
+ b: [7, 3],
98
+ c: { d: 4, h: 8 },
99
+ e: [{ f: 5, i: 9 }],
100
+ g: 6,
101
+ }
102
+
103
+ expect(toString(obj)).toBe(`{
104
+ a: 1,
105
+ b: [
106
+ 7,
107
+ 3
108
+ ],
109
+ c: {
110
+ d: 4,
111
+ h: 8
112
+ },
113
+ e: [
114
+ {
115
+ f: 5,
116
+ i: 9
117
+ }
118
+ ],
119
+ g: 6
120
+ }`)
121
+ })
@@ -0,0 +1,48 @@
1
+ import { isObject } from "./object.js"
2
+
3
+ /**
4
+ * Extends a destination object by merging it with one or more source objects.
5
+ * Performs a deep merge for nested objects and arrays.
6
+ *
7
+ * @param {Object} dest - The destination object to extend.
8
+ * @param {...Object} sources - The source objects to merge into the destination object.
9
+ * @returns {Object} - The extended destination object.
10
+ */
11
+ export function extend(dest, ...sources) {
12
+ return merge({}, dest, ...sources)
13
+ }
14
+
15
+ /**
16
+ * Merges multiple source objects into a destination object.
17
+ * Performs a deep merge for nested objects and arrays.
18
+ *
19
+ * @param {Object} dest - The destination object to merge into.
20
+ * @param {...Object} sources - The source objects to merge from.
21
+ * @returns {Object} - The merged destination object.
22
+ */
23
+ export function merge(dest, ...sources) {
24
+ return sources
25
+ .filter((source) => source != null)
26
+ .reduce((acc, source) => deepMerge(acc, source), dest)
27
+ }
28
+
29
+ /**
30
+ * Recursively merges the properties of the source object into the destination object.
31
+ *
32
+ * @param {Object} dest - The destination object to merge into.
33
+ * @param {Object} source - The source object to merge from.
34
+ * @returns {Object} - The merged destination object.
35
+ */
36
+ function deepMerge(dest, source) {
37
+ for (const [key, value] of Object.entries(source)) {
38
+ if (Array.isArray(value) || isObject(value)) {
39
+ if (dest[key] === undefined) {
40
+ dest[key] = new value.__proto__.constructor()
41
+ }
42
+ dest[key] = deepMerge(dest[key], value)
43
+ } else {
44
+ dest[key] = value
45
+ }
46
+ }
47
+ return dest
48
+ }
@@ -0,0 +1,99 @@
1
+ import { expect, test } from "vitest"
2
+
3
+ import { extend, merge } from "./objects.js"
4
+
5
+ test("it should extend an object with another, producing a new object as a result", () => {
6
+ const obj1 = {
7
+ primitiveKept: 1,
8
+ primitiveMerged: 2,
9
+ primitiveArrayKept: [1, 2],
10
+ primitiveArrayMerged: [3, 4],
11
+ primitiveObjectKept: { a: 1 },
12
+ primitiveObjectMerged: { b: 2, c: 3 },
13
+ nestedArrayKept: [{ a: 1 }],
14
+ nestedArrayMerged: [{ b: 2, c: 3 }],
15
+ nestedObjectKept: { a: { b: 2 } },
16
+ nestedObjectMerged: { c: { d: 4 } },
17
+ }
18
+ const obj2 = {
19
+ primitiveMerged: 3,
20
+ primitiveAdded: 4,
21
+ primitiveArrayMerged: [5],
22
+ primitiveArrayAdded: [6, 7],
23
+ primitiveObjectMerged: { c: 4 },
24
+ primitiveObjectAdded: { d: 4 },
25
+ nestedArrayMerged: [{ d: 4 }],
26
+ nestedArrayAdded: [{ e: 5 }],
27
+ nestedObjectMerged: { c: { e: 5 } },
28
+ nestedObjectAdded: { f: { g: 7 } },
29
+ }
30
+ const expectedResult = {
31
+ primitiveKept: 1,
32
+ primitiveMerged: 3,
33
+ primitiveAdded: 4,
34
+ primitiveArrayKept: [1, 2],
35
+ primitiveArrayMerged: [5, 4],
36
+ primitiveArrayAdded: [6, 7],
37
+ primitiveObjectKept: { a: 1 },
38
+ primitiveObjectMerged: { b: 2, c: 4 },
39
+ primitiveObjectAdded: { d: 4 },
40
+ nestedArrayKept: [{ a: 1 }],
41
+ nestedArrayMerged: [{ b: 2, c: 3, d: 4 }],
42
+ nestedArrayAdded: [{ e: 5 }],
43
+ nestedObjectKept: { a: { b: 2 } },
44
+ nestedObjectMerged: { c: { d: 4, e: 5 } },
45
+ nestedObjectAdded: { f: { g: 7 } },
46
+ }
47
+
48
+ const result = extend(obj1, obj2)
49
+ expect(result).toStrictEqual(expectedResult)
50
+ expect(result).not.toBe(obj1)
51
+ })
52
+
53
+ test("it should deep merge an two objects, changing the first object in place", () => {
54
+ const obj1 = {
55
+ primitiveKept: 1,
56
+ primitiveMerged: 2,
57
+ primitiveArrayKept: [1, 2],
58
+ primitiveArrayMerged: [3, 4],
59
+ primitiveObjectKept: { a: 1 },
60
+ primitiveObjectMerged: { b: 2, c: 3 },
61
+ nestedArrayKept: [{ a: 1 }],
62
+ nestedArrayMerged: [{ b: 2, c: 3 }],
63
+ nestedObjectKept: { a: { b: 2 } },
64
+ nestedObjectMerged: { c: { d: 4 } },
65
+ }
66
+ const obj2 = {
67
+ primitiveMerged: 3,
68
+ primitiveAdded: 4,
69
+ primitiveArrayMerged: [5],
70
+ primitiveArrayAdded: [6, 7],
71
+ primitiveObjectMerged: { c: 4 },
72
+ primitiveObjectAdded: { d: 4 },
73
+ nestedArrayMerged: [{ d: 4 }],
74
+ nestedArrayAdded: [{ e: 5 }],
75
+ nestedObjectMerged: { c: { e: 5 } },
76
+ nestedObjectAdded: { f: { g: 7 } },
77
+ }
78
+ const expectedResult = {
79
+ primitiveKept: 1,
80
+ primitiveMerged: 3,
81
+ primitiveAdded: 4,
82
+ primitiveArrayKept: [1, 2],
83
+ primitiveArrayMerged: [5, 4],
84
+ primitiveArrayAdded: [6, 7],
85
+ primitiveObjectKept: { a: 1 },
86
+ primitiveObjectMerged: { b: 2, c: 4 },
87
+ primitiveObjectAdded: { d: 4 },
88
+ nestedArrayKept: [{ a: 1 }],
89
+ nestedArrayMerged: [{ b: 2, c: 3, d: 4 }],
90
+ nestedArrayAdded: [{ e: 5 }],
91
+ nestedObjectKept: { a: { b: 2 } },
92
+ nestedObjectMerged: { c: { d: 4, e: 5 } },
93
+ nestedObjectAdded: { f: { g: 7 } },
94
+ }
95
+
96
+ const result = merge(obj1, obj2)
97
+ expect(result).toStrictEqual(expectedResult)
98
+ expect(result).toBe(obj1)
99
+ })
@@ -0,0 +1,36 @@
1
+ /** @typedef {import('./types').Tree} Tree */
2
+
3
+ /**
4
+ * Performs a breadth-first search (BFS) traversal on a tree structure.
5
+ * @param {Tree} tree - The root node of the tree.
6
+ * @returns {Array} An array of values in BFS order.
7
+ */
8
+ export function bfs(tree) {
9
+ const result = []
10
+ const queue = [tree]
11
+
12
+ while (queue.length) {
13
+ const node = queue.shift()
14
+ result.push(node.value)
15
+ if (node.children) {
16
+ queue.push(...node.children)
17
+ }
18
+ }
19
+
20
+ return result
21
+ }
22
+
23
+ /**
24
+ * Performs a depth-first search (DFS) traversal on a tree structure.
25
+ * @param {Tree} tree - The root node of the tree.
26
+ * @returns {Array} An array of values in DFS order.
27
+ */
28
+ export function dfs(tree) {
29
+ const result = [tree.value]
30
+
31
+ if (tree.children) {
32
+ result.push(...tree.children.flatMap(dfs))
33
+ }
34
+
35
+ return result
36
+ }
@@ -0,0 +1,33 @@
1
+ import { expect, test } from "vitest"
2
+
3
+ import { bfs, dfs } from "./tree.js"
4
+
5
+ test("it should perform Breadth-First Search on a tree", () => {
6
+ const tree = {
7
+ value: 1,
8
+ children: [
9
+ { value: 2, children: [{ value: 4 }, { value: 5 }] },
10
+ { value: 3, children: [{ value: 6 }, { value: 7 }] },
11
+ ],
12
+ }
13
+ const expectedResult = [1, 2, 3, 4, 5, 6, 7]
14
+
15
+ const result = bfs(tree)
16
+
17
+ expect(result).toStrictEqual(expectedResult)
18
+ })
19
+
20
+ test("it should perform Depth-First Search on a tree", () => {
21
+ const tree = {
22
+ value: 1,
23
+ children: [
24
+ { value: 2, children: [{ value: 3 }, { value: 4 }] },
25
+ { value: 5, children: [{ value: 6 }, { value: 7 }] },
26
+ ],
27
+ }
28
+ const expectedResult = [1, 2, 3, 4, 5, 6, 7]
29
+
30
+ const result = dfs(tree)
31
+
32
+ expect(result).toStrictEqual(expectedResult)
33
+ })