@inglorious/engine 0.1.1 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +75 -72
- package/package.json +15 -37
- package/src/{engine/ai → ai}/movement/dynamic/align.js +9 -9
- package/src/{engine/ai → ai}/movement/dynamic/arrive.js +9 -10
- package/src/{engine/ai → ai}/movement/dynamic/evade.js +9 -9
- package/src/{engine/ai → ai}/movement/dynamic/face.js +5 -6
- package/src/{engine/ai/movement/dynamic/seek.js → ai/movement/dynamic/flee.js} +8 -7
- package/src/ai/movement/dynamic/look-where-youre-going.js +16 -0
- package/src/{engine/ai → ai}/movement/dynamic/match-velocity.js +9 -8
- package/src/{engine/ai → ai}/movement/dynamic/pursue.js +9 -9
- package/src/{engine/ai/movement/dynamic/flee.js → ai/movement/dynamic/seek.js} +7 -8
- package/src/{engine/ai → ai}/movement/dynamic/wander.js +9 -10
- package/src/{engine/ai → ai}/movement/kinematic/align.js +7 -7
- package/src/{engine/ai → ai}/movement/kinematic/arrive.js +8 -8
- package/src/{engine/ai → ai}/movement/kinematic/face.js +5 -6
- package/src/{engine/ai → ai}/movement/kinematic/flee.js +5 -5
- package/src/{engine/ai → ai}/movement/kinematic/seek.js +5 -5
- package/src/{engine/ai → ai}/movement/kinematic/seek.test.js +10 -10
- package/src/{engine/ai → ai}/movement/kinematic/wander-as-seek.js +9 -9
- package/src/{engine/ai → ai}/movement/kinematic/wander.js +5 -5
- package/src/animation/sprite.js +101 -0
- package/src/animation/ticker.js +38 -0
- package/src/behaviors/camera.js +68 -0
- package/src/behaviors/controls/dynamic/modern.js +76 -0
- package/src/behaviors/controls/dynamic/shooter.js +84 -0
- package/src/behaviors/controls/dynamic/tank.js +69 -0
- package/src/behaviors/controls/event-handlers.js +17 -0
- package/src/behaviors/controls/kinematic/modern.js +76 -0
- package/src/behaviors/controls/kinematic/shooter.js +82 -0
- package/src/behaviors/controls/kinematic/tank.js +67 -0
- package/src/behaviors/debug/collision.js +29 -0
- package/src/behaviors/fps.js +29 -0
- package/src/behaviors/fsm.js +33 -0
- package/src/{game/decorators → behaviors}/fsm.test.js +15 -22
- package/src/behaviors/game.js +15 -0
- package/src/behaviors/input/controls.js +37 -0
- package/src/behaviors/input/gamepad.js +114 -0
- package/src/behaviors/input/input.js +48 -0
- package/src/behaviors/input/keyboard.js +64 -0
- package/src/behaviors/input/mouse.js +91 -0
- package/src/behaviors/physics/bouncy.js +25 -0
- package/src/behaviors/physics/clamped.js +36 -0
- package/src/{game/decorators/collisions.js → behaviors/physics/collidable.js} +3 -7
- package/src/behaviors/physics/jumpable.js +145 -0
- package/src/behaviors/ui/button.js +17 -0
- package/src/collision/detection.js +110 -0
- package/src/core/api.js +34 -0
- package/src/core/dev-tools.js +135 -0
- package/src/core/engine.js +119 -0
- package/src/core/loop.js +15 -0
- package/src/{engine/loop → core/loops}/animation-frame.js +1 -2
- package/src/{engine/loop → core/loops}/elapsed.js +1 -2
- package/src/{engine/loop → core/loops}/fixed.js +1 -2
- package/src/{engine/loop → core/loops}/flash.js +1 -2
- package/src/{engine/loop → core/loops}/lag.js +1 -2
- package/src/core/select.js +26 -0
- package/src/core/store.js +178 -0
- package/src/core/store.test.js +110 -0
- package/src/main.js +7 -2
- package/src/{engine/movement → movement}/dynamic/modern.js +3 -6
- package/src/{engine/movement → movement}/dynamic/tank.js +9 -9
- package/src/{engine/movement → movement}/kinematic/modern.js +3 -3
- package/src/movement/kinematic/modern.test.js +27 -0
- package/src/{engine/movement → movement}/kinematic/tank.js +5 -5
- package/src/physics/bounds.js +138 -0
- package/src/physics/position.js +43 -0
- package/src/physics/position.test.js +80 -0
- package/src/systems/sprite-animation.js +27 -0
- package/src/engine/ai/movement/dynamic/look-where-youre-going.js +0 -17
- package/src/engine/collision/detection.js +0 -115
- package/src/engine/loop.js +0 -15
- package/src/engine/movement/kinematic/modern.test.js +0 -27
- package/src/engine/store.js +0 -174
- package/src/engine/store.test.js +0 -256
- package/src/engine.js +0 -74
- package/src/game/animation.js +0 -26
- package/src/game/bounds.js +0 -66
- package/src/game/decorators/character.js +0 -5
- package/src/game/decorators/clamp-to-bounds.js +0 -15
- package/src/game/decorators/controls/dynamic/modern.js +0 -48
- package/src/game/decorators/controls/dynamic/shooter.js +0 -47
- package/src/game/decorators/controls/dynamic/tank.js +0 -55
- package/src/game/decorators/controls/kinematic/modern.js +0 -49
- package/src/game/decorators/controls/kinematic/shooter.js +0 -45
- package/src/game/decorators/controls/kinematic/tank.js +0 -52
- package/src/game/decorators/debug/collisions.js +0 -32
- package/src/game/decorators/double-jump.js +0 -70
- package/src/game/decorators/fps.js +0 -30
- package/src/game/decorators/fsm.js +0 -27
- package/src/game/decorators/game.js +0 -11
- package/src/game/decorators/image/image.js +0 -5
- package/src/game/decorators/image/sprite.js +0 -5
- package/src/game/decorators/image/tilemap.js +0 -5
- package/src/game/decorators/input/controls.js +0 -27
- package/src/game/decorators/input/gamepad.js +0 -74
- package/src/game/decorators/input/input.js +0 -41
- package/src/game/decorators/input/keyboard.js +0 -49
- package/src/game/decorators/input/mouse.js +0 -65
- package/src/game/decorators/jump.js +0 -72
- package/src/game/decorators/platform.js +0 -5
- package/src/game/decorators/ui/button.js +0 -21
- package/src/game/sprite.js +0 -119
- package/src/ui/canvas/absolute-position.js +0 -17
- package/src/ui/canvas/character.js +0 -35
- package/src/ui/canvas/form/button.js +0 -25
- package/src/ui/canvas/fps.js +0 -18
- package/src/ui/canvas/image/hitmask.js +0 -37
- package/src/ui/canvas/image/image.js +0 -37
- package/src/ui/canvas/image/sprite.js +0 -49
- package/src/ui/canvas/image/tilemap.js +0 -64
- package/src/ui/canvas/mouse.js +0 -37
- package/src/ui/canvas/shapes/circle.js +0 -31
- package/src/ui/canvas/shapes/rectangle.js +0 -31
- package/src/ui/canvas.js +0 -81
- package/src/ui/react/game/character/character.module.scss +0 -17
- package/src/ui/react/game/character/index.jsx +0 -30
- package/src/ui/react/game/cursor/cursor.module.scss +0 -47
- package/src/ui/react/game/cursor/index.jsx +0 -20
- package/src/ui/react/game/form/fields/field/field.module.scss +0 -5
- package/src/ui/react/game/form/fields/field/index.jsx +0 -56
- package/src/ui/react/game/form/fields/fields.module.scss +0 -48
- package/src/ui/react/game/form/fields/index.jsx +0 -12
- package/src/ui/react/game/form/form.module.scss +0 -18
- package/src/ui/react/game/form/index.jsx +0 -22
- package/src/ui/react/game/fps/index.jsx +0 -16
- package/src/ui/react/game/game.jsx +0 -71
- package/src/ui/react/game/index.jsx +0 -29
- package/src/ui/react/game/platform/index.jsx +0 -30
- package/src/ui/react/game/platform/platform.module.scss +0 -7
- package/src/ui/react/game/scene/index.jsx +0 -25
- package/src/ui/react/game/scene/scene.module.scss +0 -9
- package/src/ui/react/game/sprite/index.jsx +0 -58
- package/src/ui/react/game/sprite/sprite.module.css +0 -3
- package/src/ui/react/game/stats/index.jsx +0 -22
- package/src/ui/react/hocs/with-absolute-position/index.jsx +0 -20
- package/src/ui/react/hocs/with-absolute-position/with-absolute-position.module.scss +0 -5
- package/src/ui/react/index.jsx +0 -9
- package/src/utils/algorithms/decision-tree.js +0 -24
- package/src/utils/algorithms/decision-tree.test.js +0 -102
- package/src/utils/algorithms/path-finding.js +0 -155
- package/src/utils/algorithms/path-finding.test.js +0 -151
- package/src/utils/algorithms/types.d.ts +0 -28
- package/src/utils/data-structures/array.js +0 -83
- package/src/utils/data-structures/array.test.js +0 -173
- package/src/utils/data-structures/board.js +0 -159
- package/src/utils/data-structures/board.test.js +0 -242
- package/src/utils/data-structures/boolean.js +0 -9
- package/src/utils/data-structures/heap.js +0 -164
- package/src/utils/data-structures/heap.test.js +0 -103
- package/src/utils/data-structures/object.js +0 -102
- package/src/utils/data-structures/object.test.js +0 -121
- package/src/utils/data-structures/objects.js +0 -48
- package/src/utils/data-structures/objects.test.js +0 -99
- package/src/utils/data-structures/tree.js +0 -36
- package/src/utils/data-structures/tree.test.js +0 -33
- package/src/utils/data-structures/types.d.ts +0 -4
- package/src/utils/functions/functions.js +0 -19
- package/src/utils/functions/functions.test.js +0 -23
- package/src/utils/math/geometry/circle.js +0 -117
- package/src/utils/math/geometry/circle.test.js +0 -97
- package/src/utils/math/geometry/hitmask.js +0 -39
- package/src/utils/math/geometry/hitmask.test.js +0 -84
- package/src/utils/math/geometry/line.js +0 -35
- package/src/utils/math/geometry/line.test.js +0 -49
- package/src/utils/math/geometry/platform.js +0 -42
- package/src/utils/math/geometry/platform.test.js +0 -133
- package/src/utils/math/geometry/point.js +0 -71
- package/src/utils/math/geometry/point.test.js +0 -81
- package/src/utils/math/geometry/rectangle.js +0 -45
- package/src/utils/math/geometry/rectangle.test.js +0 -42
- package/src/utils/math/geometry/segment.js +0 -80
- package/src/utils/math/geometry/segment.test.js +0 -183
- package/src/utils/math/geometry/triangle.js +0 -15
- package/src/utils/math/geometry/triangle.test.js +0 -11
- package/src/utils/math/geometry/types.d.ts +0 -23
- package/src/utils/math/linear-algebra/2d.js +0 -28
- package/src/utils/math/linear-algebra/2d.test.js +0 -17
- package/src/utils/math/linear-algebra/quaternion.js +0 -22
- package/src/utils/math/linear-algebra/quaternion.test.js +0 -25
- package/src/utils/math/linear-algebra/quaternions.js +0 -20
- package/src/utils/math/linear-algebra/quaternions.test.js +0 -29
- package/src/utils/math/linear-algebra/types.d.ts +0 -4
- package/src/utils/math/linear-algebra/vector.js +0 -302
- package/src/utils/math/linear-algebra/vector.test.js +0 -257
- package/src/utils/math/linear-algebra/vectors.js +0 -122
- package/src/utils/math/linear-algebra/vectors.test.js +0 -65
- package/src/utils/math/numbers.js +0 -90
- package/src/utils/math/numbers.test.js +0 -137
- package/src/utils/math/rng.js +0 -44
- package/src/utils/math/rng.test.js +0 -39
- package/src/utils/math/statistics.js +0 -43
- package/src/utils/math/statistics.test.js +0 -47
- package/src/utils/math/trigonometry.js +0 -89
- package/src/utils/math/trigonometry.test.js +0 -52
- package/src/utils/physics/acceleration.js +0 -63
- package/src/utils/physics/friction.js +0 -30
- package/src/utils/physics/friction.test.js +0 -44
- package/src/utils/physics/gravity.js +0 -71
- package/src/utils/physics/gravity.test.js +0 -80
- package/src/utils/physics/jump.js +0 -41
- package/src/utils/physics/velocity.js +0 -38
|
@@ -1,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
|
-
})
|