@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.
Files changed (201) hide show
  1. package/README.md +75 -72
  2. package/package.json +15 -37
  3. package/src/{engine/ai → ai}/movement/dynamic/align.js +9 -9
  4. package/src/{engine/ai → ai}/movement/dynamic/arrive.js +9 -10
  5. package/src/{engine/ai → ai}/movement/dynamic/evade.js +9 -9
  6. package/src/{engine/ai → ai}/movement/dynamic/face.js +5 -6
  7. package/src/{engine/ai/movement/dynamic/seek.js → ai/movement/dynamic/flee.js} +8 -7
  8. package/src/ai/movement/dynamic/look-where-youre-going.js +16 -0
  9. package/src/{engine/ai → ai}/movement/dynamic/match-velocity.js +9 -8
  10. package/src/{engine/ai → ai}/movement/dynamic/pursue.js +9 -9
  11. package/src/{engine/ai/movement/dynamic/flee.js → ai/movement/dynamic/seek.js} +7 -8
  12. package/src/{engine/ai → ai}/movement/dynamic/wander.js +9 -10
  13. package/src/{engine/ai → ai}/movement/kinematic/align.js +7 -7
  14. package/src/{engine/ai → ai}/movement/kinematic/arrive.js +8 -8
  15. package/src/{engine/ai → ai}/movement/kinematic/face.js +5 -6
  16. package/src/{engine/ai → ai}/movement/kinematic/flee.js +5 -5
  17. package/src/{engine/ai → ai}/movement/kinematic/seek.js +5 -5
  18. package/src/{engine/ai → ai}/movement/kinematic/seek.test.js +10 -10
  19. package/src/{engine/ai → ai}/movement/kinematic/wander-as-seek.js +9 -9
  20. package/src/{engine/ai → ai}/movement/kinematic/wander.js +5 -5
  21. package/src/animation/sprite.js +101 -0
  22. package/src/animation/ticker.js +38 -0
  23. package/src/behaviors/camera.js +68 -0
  24. package/src/behaviors/controls/dynamic/modern.js +76 -0
  25. package/src/behaviors/controls/dynamic/shooter.js +84 -0
  26. package/src/behaviors/controls/dynamic/tank.js +69 -0
  27. package/src/behaviors/controls/event-handlers.js +17 -0
  28. package/src/behaviors/controls/kinematic/modern.js +76 -0
  29. package/src/behaviors/controls/kinematic/shooter.js +82 -0
  30. package/src/behaviors/controls/kinematic/tank.js +67 -0
  31. package/src/behaviors/debug/collision.js +29 -0
  32. package/src/behaviors/fps.js +29 -0
  33. package/src/behaviors/fsm.js +33 -0
  34. package/src/{game/decorators → behaviors}/fsm.test.js +15 -22
  35. package/src/behaviors/game.js +15 -0
  36. package/src/behaviors/input/controls.js +37 -0
  37. package/src/behaviors/input/gamepad.js +114 -0
  38. package/src/behaviors/input/input.js +48 -0
  39. package/src/behaviors/input/keyboard.js +64 -0
  40. package/src/behaviors/input/mouse.js +91 -0
  41. package/src/behaviors/physics/bouncy.js +25 -0
  42. package/src/behaviors/physics/clamped.js +36 -0
  43. package/src/{game/decorators/collisions.js → behaviors/physics/collidable.js} +3 -7
  44. package/src/behaviors/physics/jumpable.js +145 -0
  45. package/src/behaviors/ui/button.js +17 -0
  46. package/src/collision/detection.js +110 -0
  47. package/src/core/api.js +34 -0
  48. package/src/core/dev-tools.js +135 -0
  49. package/src/core/engine.js +119 -0
  50. package/src/core/loop.js +15 -0
  51. package/src/{engine/loop → core/loops}/animation-frame.js +1 -2
  52. package/src/{engine/loop → core/loops}/elapsed.js +1 -2
  53. package/src/{engine/loop → core/loops}/fixed.js +1 -2
  54. package/src/{engine/loop → core/loops}/flash.js +1 -2
  55. package/src/{engine/loop → core/loops}/lag.js +1 -2
  56. package/src/core/select.js +26 -0
  57. package/src/core/store.js +178 -0
  58. package/src/core/store.test.js +110 -0
  59. package/src/main.js +7 -2
  60. package/src/{engine/movement → movement}/dynamic/modern.js +3 -6
  61. package/src/{engine/movement → movement}/dynamic/tank.js +9 -9
  62. package/src/{engine/movement → movement}/kinematic/modern.js +3 -3
  63. package/src/movement/kinematic/modern.test.js +27 -0
  64. package/src/{engine/movement → movement}/kinematic/tank.js +5 -5
  65. package/src/physics/bounds.js +138 -0
  66. package/src/physics/position.js +43 -0
  67. package/src/physics/position.test.js +80 -0
  68. package/src/systems/sprite-animation.js +27 -0
  69. package/src/engine/ai/movement/dynamic/look-where-youre-going.js +0 -17
  70. package/src/engine/collision/detection.js +0 -115
  71. package/src/engine/loop.js +0 -15
  72. package/src/engine/movement/kinematic/modern.test.js +0 -27
  73. package/src/engine/store.js +0 -174
  74. package/src/engine/store.test.js +0 -256
  75. package/src/engine.js +0 -74
  76. package/src/game/animation.js +0 -26
  77. package/src/game/bounds.js +0 -66
  78. package/src/game/decorators/character.js +0 -5
  79. package/src/game/decorators/clamp-to-bounds.js +0 -15
  80. package/src/game/decorators/controls/dynamic/modern.js +0 -48
  81. package/src/game/decorators/controls/dynamic/shooter.js +0 -47
  82. package/src/game/decorators/controls/dynamic/tank.js +0 -55
  83. package/src/game/decorators/controls/kinematic/modern.js +0 -49
  84. package/src/game/decorators/controls/kinematic/shooter.js +0 -45
  85. package/src/game/decorators/controls/kinematic/tank.js +0 -52
  86. package/src/game/decorators/debug/collisions.js +0 -32
  87. package/src/game/decorators/double-jump.js +0 -70
  88. package/src/game/decorators/fps.js +0 -30
  89. package/src/game/decorators/fsm.js +0 -27
  90. package/src/game/decorators/game.js +0 -11
  91. package/src/game/decorators/image/image.js +0 -5
  92. package/src/game/decorators/image/sprite.js +0 -5
  93. package/src/game/decorators/image/tilemap.js +0 -5
  94. package/src/game/decorators/input/controls.js +0 -27
  95. package/src/game/decorators/input/gamepad.js +0 -74
  96. package/src/game/decorators/input/input.js +0 -41
  97. package/src/game/decorators/input/keyboard.js +0 -49
  98. package/src/game/decorators/input/mouse.js +0 -65
  99. package/src/game/decorators/jump.js +0 -72
  100. package/src/game/decorators/platform.js +0 -5
  101. package/src/game/decorators/ui/button.js +0 -21
  102. package/src/game/sprite.js +0 -119
  103. package/src/ui/canvas/absolute-position.js +0 -17
  104. package/src/ui/canvas/character.js +0 -35
  105. package/src/ui/canvas/form/button.js +0 -25
  106. package/src/ui/canvas/fps.js +0 -18
  107. package/src/ui/canvas/image/hitmask.js +0 -37
  108. package/src/ui/canvas/image/image.js +0 -37
  109. package/src/ui/canvas/image/sprite.js +0 -49
  110. package/src/ui/canvas/image/tilemap.js +0 -64
  111. package/src/ui/canvas/mouse.js +0 -37
  112. package/src/ui/canvas/shapes/circle.js +0 -31
  113. package/src/ui/canvas/shapes/rectangle.js +0 -31
  114. package/src/ui/canvas.js +0 -81
  115. package/src/ui/react/game/character/character.module.scss +0 -17
  116. package/src/ui/react/game/character/index.jsx +0 -30
  117. package/src/ui/react/game/cursor/cursor.module.scss +0 -47
  118. package/src/ui/react/game/cursor/index.jsx +0 -20
  119. package/src/ui/react/game/form/fields/field/field.module.scss +0 -5
  120. package/src/ui/react/game/form/fields/field/index.jsx +0 -56
  121. package/src/ui/react/game/form/fields/fields.module.scss +0 -48
  122. package/src/ui/react/game/form/fields/index.jsx +0 -12
  123. package/src/ui/react/game/form/form.module.scss +0 -18
  124. package/src/ui/react/game/form/index.jsx +0 -22
  125. package/src/ui/react/game/fps/index.jsx +0 -16
  126. package/src/ui/react/game/game.jsx +0 -71
  127. package/src/ui/react/game/index.jsx +0 -29
  128. package/src/ui/react/game/platform/index.jsx +0 -30
  129. package/src/ui/react/game/platform/platform.module.scss +0 -7
  130. package/src/ui/react/game/scene/index.jsx +0 -25
  131. package/src/ui/react/game/scene/scene.module.scss +0 -9
  132. package/src/ui/react/game/sprite/index.jsx +0 -58
  133. package/src/ui/react/game/sprite/sprite.module.css +0 -3
  134. package/src/ui/react/game/stats/index.jsx +0 -22
  135. package/src/ui/react/hocs/with-absolute-position/index.jsx +0 -20
  136. package/src/ui/react/hocs/with-absolute-position/with-absolute-position.module.scss +0 -5
  137. package/src/ui/react/index.jsx +0 -9
  138. package/src/utils/algorithms/decision-tree.js +0 -24
  139. package/src/utils/algorithms/decision-tree.test.js +0 -102
  140. package/src/utils/algorithms/path-finding.js +0 -155
  141. package/src/utils/algorithms/path-finding.test.js +0 -151
  142. package/src/utils/algorithms/types.d.ts +0 -28
  143. package/src/utils/data-structures/array.js +0 -83
  144. package/src/utils/data-structures/array.test.js +0 -173
  145. package/src/utils/data-structures/board.js +0 -159
  146. package/src/utils/data-structures/board.test.js +0 -242
  147. package/src/utils/data-structures/boolean.js +0 -9
  148. package/src/utils/data-structures/heap.js +0 -164
  149. package/src/utils/data-structures/heap.test.js +0 -103
  150. package/src/utils/data-structures/object.js +0 -102
  151. package/src/utils/data-structures/object.test.js +0 -121
  152. package/src/utils/data-structures/objects.js +0 -48
  153. package/src/utils/data-structures/objects.test.js +0 -99
  154. package/src/utils/data-structures/tree.js +0 -36
  155. package/src/utils/data-structures/tree.test.js +0 -33
  156. package/src/utils/data-structures/types.d.ts +0 -4
  157. package/src/utils/functions/functions.js +0 -19
  158. package/src/utils/functions/functions.test.js +0 -23
  159. package/src/utils/math/geometry/circle.js +0 -117
  160. package/src/utils/math/geometry/circle.test.js +0 -97
  161. package/src/utils/math/geometry/hitmask.js +0 -39
  162. package/src/utils/math/geometry/hitmask.test.js +0 -84
  163. package/src/utils/math/geometry/line.js +0 -35
  164. package/src/utils/math/geometry/line.test.js +0 -49
  165. package/src/utils/math/geometry/platform.js +0 -42
  166. package/src/utils/math/geometry/platform.test.js +0 -133
  167. package/src/utils/math/geometry/point.js +0 -71
  168. package/src/utils/math/geometry/point.test.js +0 -81
  169. package/src/utils/math/geometry/rectangle.js +0 -45
  170. package/src/utils/math/geometry/rectangle.test.js +0 -42
  171. package/src/utils/math/geometry/segment.js +0 -80
  172. package/src/utils/math/geometry/segment.test.js +0 -183
  173. package/src/utils/math/geometry/triangle.js +0 -15
  174. package/src/utils/math/geometry/triangle.test.js +0 -11
  175. package/src/utils/math/geometry/types.d.ts +0 -23
  176. package/src/utils/math/linear-algebra/2d.js +0 -28
  177. package/src/utils/math/linear-algebra/2d.test.js +0 -17
  178. package/src/utils/math/linear-algebra/quaternion.js +0 -22
  179. package/src/utils/math/linear-algebra/quaternion.test.js +0 -25
  180. package/src/utils/math/linear-algebra/quaternions.js +0 -20
  181. package/src/utils/math/linear-algebra/quaternions.test.js +0 -29
  182. package/src/utils/math/linear-algebra/types.d.ts +0 -4
  183. package/src/utils/math/linear-algebra/vector.js +0 -302
  184. package/src/utils/math/linear-algebra/vector.test.js +0 -257
  185. package/src/utils/math/linear-algebra/vectors.js +0 -122
  186. package/src/utils/math/linear-algebra/vectors.test.js +0 -65
  187. package/src/utils/math/numbers.js +0 -90
  188. package/src/utils/math/numbers.test.js +0 -137
  189. package/src/utils/math/rng.js +0 -44
  190. package/src/utils/math/rng.test.js +0 -39
  191. package/src/utils/math/statistics.js +0 -43
  192. package/src/utils/math/statistics.test.js +0 -47
  193. package/src/utils/math/trigonometry.js +0 -89
  194. package/src/utils/math/trigonometry.test.js +0 -52
  195. package/src/utils/physics/acceleration.js +0 -63
  196. package/src/utils/physics/friction.js +0 -30
  197. package/src/utils/physics/friction.test.js +0 -44
  198. package/src/utils/physics/gravity.js +0 -71
  199. package/src/utils/physics/gravity.test.js +0 -80
  200. package/src/utils/physics/jump.js +0 -41
  201. package/src/utils/physics/velocity.js +0 -38
@@ -1,155 +0,0 @@
1
- /**
2
- * @typedef {import("./types").Node} Node
3
- * @typedef {import("./types").Graph} Graph
4
- */
5
-
6
- import {
7
- contains,
8
- push,
9
- remove,
10
- root,
11
- } from "@inglorious/utils/data-structures/heap.js"
12
- import { abs, magnitude } from "@inglorious/utils/math/linear-algebra/vector.js"
13
- import { subtract } from "@inglorious/utils/math/linear-algebra/vectors.js"
14
-
15
- const NO_COST = 0
16
-
17
- /**
18
- * Calculates the cost of a path using Dijkstra's algorithm.
19
- * In this simplified form the cost is simply 0.
20
- *
21
- * @returns {number} - The cost of the path (which is 0).
22
- */
23
- export const dijkstra = () => NO_COST
24
-
25
- /**
26
- * Calculates the Euclidean distance between two nodes.
27
- *
28
- * @param {Node} a - The first node.
29
- * @param {Node} b - The second node.
30
- * @returns {number} - The Euclidean distance between the two nodes.
31
- */
32
- export const eucledianDistance = (a, b) =>
33
- magnitude(subtract(a.position, b.position))
34
-
35
- /**
36
- * Calculates the Manhattan distance between two nodes.
37
- *
38
- * @param {Node} a - The first node.
39
- * @param {Node} b - The second node.
40
- * @returns {number} - The Manhattan distance between the two nodes.
41
- */
42
- export const manhattanDistance = (a, b) => abs(subtract(a.position, b.position))
43
-
44
- /**
45
- * Compares the cost of two nodes.
46
- *
47
- * @param {Node} a - The first node.
48
- * @param {Node} b - The second node.
49
- * @returns {number} - The difference in cost between the two nodes.
50
- */
51
- const compareCost = (a, b) => b.cost - a.cost
52
-
53
- /**
54
- * Compares the total cost of two nodes.
55
- *
56
- * @param {Node} a - The first node.
57
- * @param {Node} b - The second node.
58
- * @returns {number} - The difference in total cost between the two nodes.
59
- */
60
- const compareTotalCost = (a, b) => b.totalCost - a.totalCost
61
-
62
- /**
63
- * Finds the shortest path in a graph from a start node to an end node using a heuristic.
64
- *
65
- * @param {Graph} graph - The graph containing nodes and arcs.
66
- * @param {string} start - The ID of the start node.
67
- * @param {string} end - The ID of the end node.
68
- * @param {(a: Node, b: Node) => number} [heuristic=eucledianDistance] - The heuristic function to estimate the cost.
69
- * @returns {string[]} - An array of node IDs representing the shortest path.
70
- */
71
- export function findPath(graph, start, end, heuristic = eucledianDistance) {
72
- const { nodes, arcs } = adaptGraph(graph)
73
- const findNode = createFindNode(nodes)
74
-
75
- const startNode = findNode(start)
76
- const endNode = findNode(end)
77
- let discoveredNodes = [startNode]
78
- let evaluatedNodes = []
79
- let path = []
80
-
81
- while (discoveredNodes.length) {
82
- let current = root(discoveredNodes, compareTotalCost)
83
-
84
- if (current.id === end) {
85
- while (current != null) {
86
- path = [current.id, ...path]
87
- current = current.previous
88
- }
89
-
90
- return path
91
- }
92
-
93
- discoveredNodes = remove(discoveredNodes, current)
94
- evaluatedNodes = push(evaluatedNodes, current)
95
-
96
- const outgoingArcs = arcs.filter(({ from }) => from === current.id)
97
- const destinations = outgoingArcs.map(({ to }) => findNode(to))
98
-
99
- for (const destination of destinations) {
100
- const totalCost =
101
- current.cost +
102
- destination.cost +
103
- heuristic(current, destination) +
104
- outgoingArcs.find(({ to }) => to === destination.id)?.cost
105
-
106
- let isNewPathFound = false
107
- if (contains(discoveredNodes, destination)) {
108
- if (totalCost < destination.cost) {
109
- destination.cost = totalCost
110
- isNewPathFound = true
111
- }
112
- } else {
113
- destination.cost = totalCost
114
- isNewPathFound = true
115
- discoveredNodes = push(discoveredNodes, destination, compareCost)
116
- }
117
-
118
- if (isNewPathFound) {
119
- destination.totalCost =
120
- destination.cost + heuristic(destination, endNode)
121
- destination.previous = current
122
- }
123
- }
124
- }
125
- }
126
-
127
- /**
128
- * Adapts a graph to ensure all nodes and arcs have a cost property.
129
- *
130
- * @param {Graph} graph - The graph to adapt.
131
- * @returns {Graph} - The adapted graph with cost properties added.
132
- */
133
- function adaptGraph(graph) {
134
- return {
135
- ...graph,
136
- nodes: Array.isArray(graph.nodes)
137
- ? graph.nodes.map((node) => ({ ...node, cost: node.cost ?? NO_COST }))
138
- : Object.entries(graph.nodes).map(([key, value]) => ({
139
- id: key,
140
- position: value,
141
- cost: 0,
142
- })),
143
- arcs: graph.arcs.map((arc) => ({ ...arc, cost: arc.cost ?? NO_COST })),
144
- }
145
- }
146
-
147
- /**
148
- * Creates a function that finds a node by its ID.
149
- *
150
- * @param {Node[]} nodes - The array of nodes.
151
- * @returns {(id: string) => Node} - A function that finds a node by its ID.
152
- */
153
- function createFindNode(nodes) {
154
- return (id) => nodes.find((node) => node.id === id)
155
- }
@@ -1,151 +0,0 @@
1
- import {
2
- createBoard,
3
- down,
4
- downRight,
5
- right,
6
- } from "@inglorious/utils/data-structures/board.js"
7
- import { expect, test } from "vitest"
8
-
9
- import { findPath } from "./path-finding.js"
10
-
11
- test("it should find the shortest path in a trivial example", () => {
12
- const graph = {
13
- nodes: {
14
- A: [0, 0],
15
- B: [2, 0],
16
- },
17
- arcs: [{ from: "A", to: "B", cost: 2 }],
18
- }
19
- const start = "A"
20
- const end = "B"
21
- const expectedResult = ["A", "B"]
22
-
23
- expect(findPath(graph, start, end)).toStrictEqual(expectedResult)
24
- })
25
-
26
- test("it should find the shortest path between two nodes with default heuristic", () => {
27
- const graph = {
28
- nodes: {
29
- A: [0, 0],
30
- B: [2, 0],
31
- C: [3, 2],
32
- D: [4, 4],
33
- E: [1, 1],
34
- F: [2, 3],
35
- G: [0, 4],
36
- },
37
- arcs: [
38
- { from: "A", to: "B" },
39
- { from: "B", to: "C" },
40
- { from: "C", to: "D" },
41
- { from: "A", to: "E" },
42
- { from: "E", to: "F" },
43
- { from: "F", to: "G" },
44
- { from: "G", to: "D" },
45
- ],
46
- }
47
- const start = "A"
48
- const end = "D"
49
- const expectedResult = ["A", "B", "C", "D"]
50
-
51
- expect(findPath(graph, start, end)).toStrictEqual(expectedResult)
52
- })
53
-
54
- test("it should find the shortest path between two nodes with node costs", () => {
55
- const graph = {
56
- nodes: [
57
- { id: "A", position: [0, 0], cost: 0 },
58
- { id: "B", position: [2, 0], cost: 7 },
59
- { id: "C", position: [3, 2], cost: 3 },
60
- { id: "D", position: [4, 4], cost: 2 },
61
- { id: "E", position: [1, 1], cost: 1.5 },
62
- { id: "F", position: [2, 3], cost: 2 },
63
- { id: "G", position: [0, 4], cost: 3 },
64
- ],
65
- arcs: [
66
- { from: "A", to: "B" },
67
- { from: "B", to: "C" },
68
- { from: "C", to: "D" },
69
- { from: "A", to: "E" },
70
- { from: "E", to: "F" },
71
- { from: "F", to: "G" },
72
- { from: "G", to: "D" },
73
- ],
74
- }
75
- const start = "A"
76
- const end = "D"
77
- const expectedResult = ["A", "E", "F", "G", "D"]
78
-
79
- expect(findPath(graph, start, end)).toStrictEqual(expectedResult)
80
- })
81
-
82
- test("it should find the shortest path between two nodes with arc costs", () => {
83
- const graph = {
84
- nodes: {
85
- A: [0, 0],
86
- B: [2, 0],
87
- C: [3, 2],
88
- D: [4, 4],
89
- E: [1, 1],
90
- F: [2, 3],
91
- G: [0, 4],
92
- },
93
- arcs: [
94
- { from: "A", to: "B", cost: 7 },
95
- { from: "B", to: "C", cost: 3 },
96
- { from: "C", to: "D", cost: 2 },
97
- { from: "A", to: "E", cost: 1.5 },
98
- { from: "E", to: "F", cost: 2 },
99
- { from: "F", to: "G", cost: 3 },
100
- { from: "G", to: "D", cost: 4 },
101
- ],
102
- }
103
- const start = "A"
104
- const end = "D"
105
- const expectedResult = ["A", "E", "F", "G", "D"]
106
-
107
- expect(findPath(graph, start, end)).toStrictEqual(expectedResult)
108
- })
109
-
110
- test("it should find the best path in a board graph", () => {
111
- const rows = 5
112
- const columns = 5
113
- const size = [rows, columns]
114
- const filler = (i, j) => ({ id: `${i}${j}`, position: [i, j] })
115
- const board = createBoard(size, filler)
116
-
117
- const graph = {
118
- nodes: board,
119
- arcs: [],
120
- }
121
-
122
- for (let i = 0; i < rows; i++) {
123
- for (let j = 0; j < columns; j++) {
124
- const current = `${i}${j}`
125
- if (i + 1 < rows) {
126
- graph.arcs.push({
127
- from: current,
128
- to: down([i, j], size).join(""),
129
- })
130
- }
131
- if (j + 1 < columns) {
132
- graph.arcs.push({
133
- from: current,
134
- to: right([i, j], size).join(""),
135
- })
136
- }
137
- if (i + 1 < rows && j + 1 < columns) {
138
- graph.arcs.push({
139
- from: current,
140
- to: downRight([i, j], size).join(""),
141
- })
142
- }
143
- }
144
- }
145
-
146
- const start = "00"
147
- const end = "44"
148
- const expectedResult = ["00", "11", "22", "33", "44"]
149
-
150
- expect(findPath(graph, start, end)).toStrictEqual(expectedResult)
151
- })
@@ -1,28 +0,0 @@
1
- export type Conditions = unknown
2
- export type Outcome = string | boolean
3
-
4
- export interface DecisionTree {
5
- test: (conditions: Conditions) => Outcome
6
- [outcome: Outcome]: () => Outcome | DecisionTree
7
- }
8
-
9
- interface NodeMap {
10
- [id: string]: [number, number] // [x, y]
11
- }
12
-
13
- interface Node {
14
- id: string
15
- position: [number, number] // [x, y]
16
- cost?: number
17
- }
18
-
19
- interface Arc {
20
- from: string
21
- to: string
22
- cost?: number
23
- }
24
-
25
- export interface Graph {
26
- nodes: NodeMap | Node[]
27
- arcs: Arc[]
28
- }
@@ -1,83 +0,0 @@
1
- /**
2
- * A default comparator function that compares two numbers.
3
- *
4
- * @param {number} a - The first number.
5
- * @param {number} b - The second number.
6
- * @returns {number} - A positive number if `a` is less than `b`, a negative number if `a` is greater than `b`, or 0 if they are equal.
7
- */
8
- const DEFAULT_COMPARATOR = (a, b) => b - a
9
-
10
- // Threshold constants for comparison results
11
- const LESS_THAN_ZERO = 0
12
- const GREATER_THAN_ZERO = 0
13
-
14
- /**
15
- * Checks if an array contains a specific item.
16
- *
17
- * @param {Array} arr - The array to search.
18
- * @param {*} item - The item to check for.
19
- * @returns {boolean} - `true` if the item is found, otherwise `false`.
20
- */
21
- export function contains(arr, item) {
22
- return arr.includes(item)
23
- }
24
-
25
- /**
26
- * Finds the maximum item in an array based on a comparator function.
27
- *
28
- * @param {Array} arr - The array to search.
29
- * @param {Function} [comparator=DEFAULT_COMPARATOR] - The comparator function to determine the maximum item.
30
- * @returns {*} - The maximum item in the array.
31
- */
32
- export function max(arr, comparator = DEFAULT_COMPARATOR) {
33
- return arr.reduce((acc, item) =>
34
- comparator(item, acc) < LESS_THAN_ZERO ? item : acc,
35
- )
36
- }
37
-
38
- /**
39
- * Finds the minimum item in an array based on a comparator function.
40
- *
41
- * @param {Array} arr - The array to search.
42
- * @param {Function} [comparator=DEFAULT_COMPARATOR] - The comparator function to determine the minimum item.
43
- * @returns {*} - The minimum item in the array.
44
- */
45
- export function min(arr, comparator = DEFAULT_COMPARATOR) {
46
- return arr.reduce((acc, item) =>
47
- comparator(item, acc) > GREATER_THAN_ZERO ? item : acc,
48
- )
49
- }
50
-
51
- /**
52
- * Removes the smallest item from an array, based on a comparator function.
53
- *
54
- * @param {Array} arr - The array to remove the item from.
55
- * @param {Function} [comparator=DEFAULT_COMPARATOR] - The comparator function to determine the smallest item.
56
- * @returns {*} - A new array with the item removed.
57
- */
58
- export function pop(arr, comparator = DEFAULT_COMPARATOR) {
59
- const item = min(arr, comparator)
60
- return remove(arr, item)
61
- }
62
-
63
- /**
64
- * Adds an item to an array in an immutable way.
65
- *
66
- * @param {Array} arr - The array to add the item to.
67
- * @param {*} item - The item to add.
68
- * @returns {Array} - A new array with the item added.
69
- */
70
- export function push(arr, item) {
71
- return [...arr, item]
72
- }
73
-
74
- /**
75
- * Removes a specific item from an array in an immutable way.
76
- *
77
- * @param {Array} arr - The array to remove the item from.
78
- * @param {*} item - The item to remove.
79
- * @returns {Array} - A new array with the item removed.
80
- */
81
- export function remove(arr, item) {
82
- return arr.filter((el) => el !== item)
83
- }
@@ -1,173 +0,0 @@
1
- import { expect, test } from "vitest"
2
-
3
- import { contains, max, min, pop, push, remove } from "./array.js"
4
-
5
- test("it should check if an array contains a value", () => {
6
- const arr = [3, 2, 6, 1, 7, 4, 5]
7
- const item = 1
8
- const expectedResult = true
9
-
10
- expect(contains(arr, item)).toBe(expectedResult)
11
- })
12
-
13
- test("it should check if an array contains an object", () => {
14
- const arr = [
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 = arr[3]
24
- const expectedResult = true
25
-
26
- expect(contains(arr, item)).toBe(expectedResult)
27
- })
28
-
29
- test("it should find the maximum value of an array", () => {
30
- const arr = [3, 2, 6, 1, 7, 4, 5]
31
- const expectedResult = 7
32
-
33
- expect(max(arr)).toBe(expectedResult)
34
- })
35
-
36
- test("it should find the maximum value in an array of objects", () => {
37
- const arr = [
38
- { value: 3 },
39
- { value: 2 },
40
- { value: 6 },
41
- { value: 1 },
42
- { value: 7 },
43
- { value: 4 },
44
- { value: 5 },
45
- ]
46
- const comparator = (a, b) => b.value - a.value
47
- const expectedResult = { value: 7 }
48
-
49
- expect(max(arr, comparator)).toStrictEqual(expectedResult)
50
- })
51
-
52
- test("it should find the minimum value of an array", () => {
53
- const arr = [3, 2, 6, 1, 7, 4, 5]
54
- const expectedResult = 1
55
-
56
- expect(min(arr)).toBe(expectedResult)
57
- })
58
-
59
- test("it should find the minimum value in an array of objects", () => {
60
- const arr = [
61
- { value: 3 },
62
- { value: 2 },
63
- { value: 6 },
64
- { value: 1 },
65
- { value: 7 },
66
- { value: 4 },
67
- { value: 5 },
68
- ]
69
- const comparator = (a, b) => b.value - a.value
70
- const expectedResult = { value: 1 }
71
-
72
- expect(min(arr, comparator)).toStrictEqual(expectedResult)
73
- })
74
-
75
- test("it should remove the minimum value from an array", () => {
76
- const arr = [3, 2, 6, 1, 7, 4, 5]
77
- const expectedResult = [3, 2, 6, 7, 4, 5]
78
-
79
- expect(pop(arr)).toStrictEqual(expectedResult)
80
- })
81
-
82
- test("it should remove the minimum object from an array", () => {
83
- const arr = [
84
- { value: 3 },
85
- { value: 2 },
86
- { value: 6 },
87
- { value: 1 },
88
- { value: 7 },
89
- { value: 4 },
90
- { value: 5 },
91
- ]
92
- const comparator = (a, b) => b.value - a.value
93
- const expectedResult = [
94
- { value: 3 },
95
- { value: 2 },
96
- { value: 6 },
97
- { value: 7 },
98
- { value: 4 },
99
- { value: 5 },
100
- ]
101
-
102
- expect(pop(arr, comparator)).toStrictEqual(expectedResult)
103
- })
104
-
105
- test("it should remove the maximum value from an array", () => {
106
- const arr = [3, 2, 6, 1, 7, 4, 5]
107
- const comparator = (a, b) => a - b
108
- const expectedResult = [3, 2, 6, 1, 4, 5]
109
-
110
- expect(pop(arr, comparator)).toStrictEqual(expectedResult)
111
- })
112
-
113
- test("it should remove the maximum object from an array", () => {
114
- const arr = [
115
- { value: 3 },
116
- { value: 2 },
117
- { value: 6 },
118
- { value: 1 },
119
- { value: 7 },
120
- { value: 4 },
121
- { value: 5 },
122
- ]
123
- const comparator = (a, b) => a.value - b.value
124
- const expectedResult = [
125
- { value: 3 },
126
- { value: 2 },
127
- { value: 6 },
128
- { value: 1 },
129
- { value: 4 },
130
- { value: 5 },
131
- ]
132
-
133
- expect(pop(arr, comparator)).toStrictEqual(expectedResult)
134
- })
135
-
136
- test("it should add a value at the end of the array", () => {
137
- const arr = [3, 2, 6, 1, 7, 4]
138
- const item = 5
139
- const expectedResult = [3, 2, 6, 1, 7, 4, 5]
140
-
141
- expect(push(arr, item)).toStrictEqual(expectedResult)
142
- })
143
-
144
- test("it should remove a primitive value from an array", () => {
145
- const arr = [3, 2, 6, 1, 7, 4, 5]
146
- const item = 1
147
- const expectedResult = [3, 2, 6, 7, 4, 5]
148
-
149
- expect(remove(arr, item)).toStrictEqual(expectedResult)
150
- })
151
-
152
- test("it should remove an object from an array", () => {
153
- const arr = [
154
- { value: 3 },
155
- { value: 2 },
156
- { value: 6 },
157
- { value: 1 },
158
- { value: 7 },
159
- { value: 4 },
160
- { value: 5 },
161
- ]
162
- const item = arr[3]
163
- const expectedResult = [
164
- { value: 3 },
165
- { value: 2 },
166
- { value: 6 },
167
- { value: 7 },
168
- { value: 4 },
169
- { value: 5 },
170
- ]
171
-
172
- expect(remove(arr, item)).toStrictEqual(expectedResult)
173
- })