@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.
- package/LICENSE +9 -0
- package/README.md +72 -0
- package/package.json +76 -0
- package/src/docs/ai/movement/dynamic/align.js +131 -0
- package/src/docs/ai/movement/dynamic/arrive.js +88 -0
- package/src/docs/ai/movement/dynamic/dynamic.mdx +99 -0
- package/src/docs/ai/movement/dynamic/dynamic.stories.js +58 -0
- package/src/docs/ai/movement/dynamic/evade.js +72 -0
- package/src/docs/ai/movement/dynamic/face.js +90 -0
- package/src/docs/ai/movement/dynamic/flee.js +38 -0
- package/src/docs/ai/movement/dynamic/look-where-youre-going.js +114 -0
- package/src/docs/ai/movement/dynamic/match-velocity.js +92 -0
- package/src/docs/ai/movement/dynamic/pursue.js +72 -0
- package/src/docs/ai/movement/dynamic/seek.js +37 -0
- package/src/docs/ai/movement/dynamic/wander.js +71 -0
- package/src/docs/ai/movement/kinematic/align.js +122 -0
- package/src/docs/ai/movement/kinematic/arrive.js +78 -0
- package/src/docs/ai/movement/kinematic/face.js +82 -0
- package/src/docs/ai/movement/kinematic/flee.js +36 -0
- package/src/docs/ai/movement/kinematic/kinematic.mdx +67 -0
- package/src/docs/ai/movement/kinematic/kinematic.stories.js +42 -0
- package/src/docs/ai/movement/kinematic/seek.js +34 -0
- package/src/docs/ai/movement/kinematic/wander-as-seek.js +62 -0
- package/src/docs/ai/movement/kinematic/wander.js +28 -0
- package/src/docs/bounds.js +7 -0
- package/src/docs/code-reuse.js +35 -0
- package/src/docs/collision/circles.js +58 -0
- package/src/docs/collision/collision.mdx +27 -0
- package/src/docs/collision/collision.stories.js +22 -0
- package/src/docs/collision/platform.js +76 -0
- package/src/docs/collision/tilemap.js +181 -0
- package/src/docs/empty.js +1 -0
- package/src/docs/engine.mdx +81 -0
- package/src/docs/engine.stories.js +37 -0
- package/src/docs/event-handlers.js +68 -0
- package/src/docs/framerate.js +37 -0
- package/src/docs/game.jsx +15 -0
- package/src/docs/image/image.js +19 -0
- package/src/docs/image/image.stories.js +22 -0
- package/src/docs/image/sprite.js +39 -0
- package/src/docs/image/tilemap.js +84 -0
- package/src/docs/input/controls.js +67 -0
- package/src/docs/input/gamepad.js +67 -0
- package/src/docs/input/input.mdx +55 -0
- package/src/docs/input/input.stories.js +27 -0
- package/src/docs/input/keyboard.js +58 -0
- package/src/docs/input/mouse.js +32 -0
- package/src/docs/instances.js +49 -0
- package/src/docs/player/dynamic/double-jump.js +90 -0
- package/src/docs/player/dynamic/dynamic.stories.js +32 -0
- package/src/docs/player/dynamic/jump.js +83 -0
- package/src/docs/player/dynamic/modern-controls.js +57 -0
- package/src/docs/player/dynamic/shooter-controls.js +51 -0
- package/src/docs/player/dynamic/tank-controls.js +44 -0
- package/src/docs/player/kinematic/double-jump.js +90 -0
- package/src/docs/player/kinematic/jump.js +82 -0
- package/src/docs/player/kinematic/kinematic.stories.js +32 -0
- package/src/docs/player/kinematic/modern-controls.js +56 -0
- package/src/docs/player/kinematic/shooter-controls.js +48 -0
- package/src/docs/player/kinematic/tank-controls.js +42 -0
- package/src/docs/quick-start/first-game.js +49 -0
- package/src/docs/quick-start/hello-world.js +1 -0
- package/src/docs/quick-start.mdx +127 -0
- package/src/docs/quick-start.stories.js +17 -0
- package/src/docs/recipes/add-and-remove.js +71 -0
- package/src/docs/recipes/add-instance.js +42 -0
- package/src/docs/recipes/decision-tree.js +169 -0
- package/src/docs/recipes/random-instances.js +25 -0
- package/src/docs/recipes/recipes.mdx +81 -0
- package/src/docs/recipes/recipes.stories.js +37 -0
- package/src/docs/recipes/remove-instance.js +52 -0
- package/src/docs/recipes/states.js +64 -0
- package/src/docs/ui/button.js +28 -0
- package/src/docs/ui/form.stories.js +55 -0
- package/src/docs/ui-chooser.jsx +6 -0
- package/src/docs/utils/data-structures/object.mdx +47 -0
- package/src/docs/utils/data-structures/objects.mdx +30 -0
- package/src/docs/utils/functions/functions.mdx +34 -0
- package/src/docs/utils/math/geometry/circle.mdx +55 -0
- package/src/docs/utils/math/geometry/point.mdx +38 -0
- package/src/docs/utils/math/geometry/rectangle.mdx +24 -0
- package/src/docs/utils/math/geometry/segment.mdx +55 -0
- package/src/docs/utils/math/geometry/triangle.mdx +22 -0
- package/src/docs/utils/math/linear-algebra/2d.mdx +22 -0
- package/src/docs/utils/math/linear-algebra/quaternion.mdx +21 -0
- package/src/docs/utils/math/linear-algebra/quaternions.mdx +22 -0
- package/src/docs/utils/math/linear-algebra/vector.mdx +177 -0
- package/src/docs/utils/math/linear-algebra/vectors.mdx +58 -0
- package/src/docs/utils/math/numbers.mdx +76 -0
- package/src/docs/utils/math/random.mdx +35 -0
- package/src/docs/utils/math/statistics.mdx +38 -0
- package/src/docs/utils/math/trigonometry.mdx +85 -0
- package/src/docs/utils/physics/friction.mdx +20 -0
- package/src/docs/utils/physics/gravity.mdx +28 -0
- package/src/engine/ai/movement/dynamic/align.js +63 -0
- package/src/engine/ai/movement/dynamic/arrive.js +43 -0
- package/src/engine/ai/movement/dynamic/evade.js +38 -0
- package/src/engine/ai/movement/dynamic/face.js +20 -0
- package/src/engine/ai/movement/dynamic/flee.js +45 -0
- package/src/engine/ai/movement/dynamic/look-where-youre-going.js +17 -0
- package/src/engine/ai/movement/dynamic/match-velocity.js +50 -0
- package/src/engine/ai/movement/dynamic/pursue.js +38 -0
- package/src/engine/ai/movement/dynamic/seek.js +44 -0
- package/src/engine/ai/movement/dynamic/wander.js +32 -0
- package/src/engine/ai/movement/kinematic/align.js +37 -0
- package/src/engine/ai/movement/kinematic/arrive.js +42 -0
- package/src/engine/ai/movement/kinematic/face.js +20 -0
- package/src/engine/ai/movement/kinematic/flee.js +26 -0
- package/src/engine/ai/movement/kinematic/seek.js +26 -0
- package/src/engine/ai/movement/kinematic/seek.test.js +42 -0
- package/src/engine/ai/movement/kinematic/wander-as-seek.js +31 -0
- package/src/engine/ai/movement/kinematic/wander.js +27 -0
- package/src/engine/collision/detection.js +115 -0
- package/src/engine/loop/animation-frame.js +26 -0
- package/src/engine/loop/elapsed.js +23 -0
- package/src/engine/loop/fixed.js +28 -0
- package/src/engine/loop/flash.js +14 -0
- package/src/engine/loop/lag.js +27 -0
- package/src/engine/loop.js +15 -0
- package/src/engine/movement/dynamic/modern.js +24 -0
- package/src/engine/movement/dynamic/tank.js +43 -0
- package/src/engine/movement/kinematic/modern.js +16 -0
- package/src/engine/movement/kinematic/modern.test.js +27 -0
- package/src/engine/movement/kinematic/tank.js +27 -0
- package/src/engine/store.js +174 -0
- package/src/engine/store.test.js +256 -0
- package/src/engine.js +74 -0
- package/src/game/animation.js +26 -0
- package/src/game/bounds.js +66 -0
- package/src/game/decorators/character.js +5 -0
- package/src/game/decorators/clamp-to-bounds.js +15 -0
- package/src/game/decorators/collisions.js +24 -0
- package/src/game/decorators/controls/dynamic/modern.js +48 -0
- package/src/game/decorators/controls/dynamic/shooter.js +47 -0
- package/src/game/decorators/controls/dynamic/tank.js +55 -0
- package/src/game/decorators/controls/kinematic/modern.js +49 -0
- package/src/game/decorators/controls/kinematic/shooter.js +45 -0
- package/src/game/decorators/controls/kinematic/tank.js +52 -0
- package/src/game/decorators/debug/collisions.js +32 -0
- package/src/game/decorators/double-jump.js +70 -0
- package/src/game/decorators/fps.js +30 -0
- package/src/game/decorators/fsm.js +27 -0
- package/src/game/decorators/fsm.test.js +56 -0
- package/src/game/decorators/game.js +11 -0
- package/src/game/decorators/image/image.js +5 -0
- package/src/game/decorators/image/sprite.js +5 -0
- package/src/game/decorators/image/tilemap.js +5 -0
- package/src/game/decorators/input/controls.js +27 -0
- package/src/game/decorators/input/gamepad.js +74 -0
- package/src/game/decorators/input/input.js +41 -0
- package/src/game/decorators/input/keyboard.js +49 -0
- package/src/game/decorators/input/mouse.js +65 -0
- package/src/game/decorators/jump.js +72 -0
- package/src/game/decorators/platform.js +5 -0
- package/src/game/decorators/ui/button.js +21 -0
- package/src/game/sprite.js +119 -0
- package/src/main.js +5 -0
- package/src/ui/canvas/absolute-position.js +17 -0
- package/src/ui/canvas/character.js +35 -0
- package/src/ui/canvas/form/button.js +25 -0
- package/src/ui/canvas/fps.js +18 -0
- package/src/ui/canvas/image/hitmask.js +37 -0
- package/src/ui/canvas/image/image.js +37 -0
- package/src/ui/canvas/image/sprite.js +49 -0
- package/src/ui/canvas/image/tilemap.js +64 -0
- package/src/ui/canvas/mouse.js +37 -0
- package/src/ui/canvas/shapes/circle.js +31 -0
- package/src/ui/canvas/shapes/rectangle.js +31 -0
- package/src/ui/canvas.js +81 -0
- package/src/ui/react/game/character/character.module.scss +17 -0
- package/src/ui/react/game/character/index.jsx +30 -0
- package/src/ui/react/game/cursor/cursor.module.scss +47 -0
- package/src/ui/react/game/cursor/index.jsx +20 -0
- package/src/ui/react/game/form/fields/field/field.module.scss +5 -0
- package/src/ui/react/game/form/fields/field/index.jsx +56 -0
- package/src/ui/react/game/form/fields/fields.module.scss +48 -0
- package/src/ui/react/game/form/fields/index.jsx +12 -0
- package/src/ui/react/game/form/form.module.scss +18 -0
- package/src/ui/react/game/form/index.jsx +22 -0
- package/src/ui/react/game/fps/index.jsx +16 -0
- package/src/ui/react/game/game.jsx +71 -0
- package/src/ui/react/game/index.jsx +29 -0
- package/src/ui/react/game/platform/index.jsx +30 -0
- package/src/ui/react/game/platform/platform.module.scss +7 -0
- package/src/ui/react/game/scene/index.jsx +25 -0
- package/src/ui/react/game/scene/scene.module.scss +9 -0
- package/src/ui/react/game/sprite/index.jsx +58 -0
- package/src/ui/react/game/sprite/sprite.module.css +3 -0
- package/src/ui/react/game/stats/index.jsx +22 -0
- package/src/ui/react/hocs/with-absolute-position/index.jsx +20 -0
- package/src/ui/react/hocs/with-absolute-position/with-absolute-position.module.scss +5 -0
- package/src/ui/react/index.jsx +9 -0
- package/src/utils/algorithms/decision-tree.js +24 -0
- package/src/utils/algorithms/decision-tree.test.js +102 -0
- package/src/utils/algorithms/path-finding.js +155 -0
- package/src/utils/algorithms/path-finding.test.js +151 -0
- package/src/utils/algorithms/types.d.ts +28 -0
- package/src/utils/data-structures/array.js +83 -0
- package/src/utils/data-structures/array.test.js +173 -0
- package/src/utils/data-structures/board.js +159 -0
- package/src/utils/data-structures/board.test.js +242 -0
- package/src/utils/data-structures/boolean.js +9 -0
- package/src/utils/data-structures/heap.js +164 -0
- package/src/utils/data-structures/heap.test.js +103 -0
- package/src/utils/data-structures/object.js +102 -0
- package/src/utils/data-structures/object.test.js +121 -0
- package/src/utils/data-structures/objects.js +48 -0
- package/src/utils/data-structures/objects.test.js +99 -0
- package/src/utils/data-structures/tree.js +36 -0
- package/src/utils/data-structures/tree.test.js +33 -0
- package/src/utils/data-structures/types.d.ts +4 -0
- package/src/utils/functions/functions.js +19 -0
- package/src/utils/functions/functions.test.js +23 -0
- package/src/utils/math/geometry/circle.js +117 -0
- package/src/utils/math/geometry/circle.test.js +97 -0
- package/src/utils/math/geometry/hitmask.js +39 -0
- package/src/utils/math/geometry/hitmask.test.js +84 -0
- package/src/utils/math/geometry/line.js +35 -0
- package/src/utils/math/geometry/line.test.js +49 -0
- package/src/utils/math/geometry/platform.js +42 -0
- package/src/utils/math/geometry/platform.test.js +133 -0
- package/src/utils/math/geometry/point.js +71 -0
- package/src/utils/math/geometry/point.test.js +81 -0
- package/src/utils/math/geometry/rectangle.js +45 -0
- package/src/utils/math/geometry/rectangle.test.js +42 -0
- package/src/utils/math/geometry/segment.js +80 -0
- package/src/utils/math/geometry/segment.test.js +183 -0
- package/src/utils/math/geometry/triangle.js +15 -0
- package/src/utils/math/geometry/triangle.test.js +11 -0
- package/src/utils/math/geometry/types.d.ts +23 -0
- package/src/utils/math/linear-algebra/2d.js +28 -0
- package/src/utils/math/linear-algebra/2d.test.js +17 -0
- package/src/utils/math/linear-algebra/quaternion.js +22 -0
- package/src/utils/math/linear-algebra/quaternion.test.js +25 -0
- package/src/utils/math/linear-algebra/quaternions.js +20 -0
- package/src/utils/math/linear-algebra/quaternions.test.js +29 -0
- package/src/utils/math/linear-algebra/types.d.ts +4 -0
- package/src/utils/math/linear-algebra/vector.js +302 -0
- package/src/utils/math/linear-algebra/vector.test.js +257 -0
- package/src/utils/math/linear-algebra/vectors.js +122 -0
- package/src/utils/math/linear-algebra/vectors.test.js +65 -0
- package/src/utils/math/numbers.js +90 -0
- package/src/utils/math/numbers.test.js +137 -0
- package/src/utils/math/rng.js +44 -0
- package/src/utils/math/rng.test.js +39 -0
- package/src/utils/math/statistics.js +43 -0
- package/src/utils/math/statistics.test.js +47 -0
- package/src/utils/math/trigonometry.js +89 -0
- package/src/utils/math/trigonometry.test.js +52 -0
- package/src/utils/physics/acceleration.js +63 -0
- package/src/utils/physics/friction.js +30 -0
- package/src/utils/physics/friction.test.js +44 -0
- package/src/utils/physics/gravity.js +71 -0
- package/src/utils/physics/gravity.test.js +80 -0
- package/src/utils/physics/jump.js +41 -0
- package/src/utils/physics/velocity.js +38 -0
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {import("./types").Point} Point
|
|
3
|
+
* @typedef {import("./types").Line} Line
|
|
4
|
+
* @typedef {import("./types").Circle} Circle
|
|
5
|
+
* @typedef {import("./types").Rectangle} Rectangle
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const SQUARED = 2
|
|
9
|
+
const HALF = 2
|
|
10
|
+
|
|
11
|
+
import { distanceFromPoint } from "./line.js"
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Calculates the distance from a point to a line.
|
|
15
|
+
* @param {Point} point - The point as a 3D coordinate [x, y, z].
|
|
16
|
+
* @param {Line} line - The line to calculate the distance from.
|
|
17
|
+
* @returns {number} The distance from the point to the line.
|
|
18
|
+
*/
|
|
19
|
+
export function getDistanceFromLine(point, line) {
|
|
20
|
+
return distanceFromPoint(line, point)
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Checks if two points intersect.
|
|
25
|
+
* @param {Point} point1 - The first point as a 3D coordinate [x, y, z].
|
|
26
|
+
* @param {Point} point2 - The second point as a 3D coordinate [x, y, z].
|
|
27
|
+
* @returns {boolean} True if the points intersect, false otherwise.
|
|
28
|
+
*/
|
|
29
|
+
export function intersectsPoint(point1, point2) {
|
|
30
|
+
const [x1, y1, z1] = point1
|
|
31
|
+
const [x2, y2, z2] = point2
|
|
32
|
+
return x1 === x2 && y1 === y2 && z1 === z2
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Checks if a point intersects with a circle.
|
|
37
|
+
* @param {Point} point - The point as a 3D coordinate [x, y, z].
|
|
38
|
+
* @param {Circle} circle - The circle with a position and radius.
|
|
39
|
+
* @returns {boolean} True if the point intersects the circle, false otherwise.
|
|
40
|
+
*/
|
|
41
|
+
export function intersectsCircle(point, circle) {
|
|
42
|
+
const [x, y, z] = point
|
|
43
|
+
const [left, top, front] = circle.position
|
|
44
|
+
const radius = circle.radius
|
|
45
|
+
|
|
46
|
+
return (
|
|
47
|
+
(x - left) ** SQUARED + (y - top) ** SQUARED + (z - front) ** SQUARED <=
|
|
48
|
+
radius ** SQUARED
|
|
49
|
+
)
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Checks if a point intersects with a rectangle.
|
|
54
|
+
* @param {Point} point - The point as a 3D coordinate [x, y, z].
|
|
55
|
+
* @param {Rectangle} rectangle - The rectangle with a position and size.
|
|
56
|
+
* @returns {boolean} True if the point intersects the rectangle, false otherwise.
|
|
57
|
+
*/
|
|
58
|
+
export function intersectsRectangle(point, rectangle) {
|
|
59
|
+
const [x, y, z] = point
|
|
60
|
+
const [left, top, front] = rectangle.position
|
|
61
|
+
const [width, height, depth] = rectangle.size
|
|
62
|
+
|
|
63
|
+
return (
|
|
64
|
+
x >= left - width / HALF &&
|
|
65
|
+
x <= left + width / HALF &&
|
|
66
|
+
y >= top - height / HALF &&
|
|
67
|
+
y <= top + height / HALF &&
|
|
68
|
+
z >= front - depth / HALF &&
|
|
69
|
+
z <= front + depth / HALF
|
|
70
|
+
)
|
|
71
|
+
}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { expect, test } from "vitest"
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
intersectsCircle,
|
|
5
|
+
intersectsPoint,
|
|
6
|
+
intersectsRectangle,
|
|
7
|
+
} from "./point.js"
|
|
8
|
+
|
|
9
|
+
test("it should prove that two equal points intersect", () => {
|
|
10
|
+
const point1 = [1.5, 1.5, 0]
|
|
11
|
+
const point2 = [1.5, 1.5, 0]
|
|
12
|
+
|
|
13
|
+
expect(intersectsPoint(point1, point2)).toBe(true)
|
|
14
|
+
})
|
|
15
|
+
|
|
16
|
+
test("it should prove that two different points do not intersect", () => {
|
|
17
|
+
const point1 = [1.5, 1.5, 0]
|
|
18
|
+
const point2 = [2, 1, 0]
|
|
19
|
+
|
|
20
|
+
expect(intersectsPoint(point1, point2)).toBe(false)
|
|
21
|
+
})
|
|
22
|
+
|
|
23
|
+
test("it should prove that a point inside of a circle intersects with it", () => {
|
|
24
|
+
const point = [1.5, 1.5, 0]
|
|
25
|
+
const circle = {
|
|
26
|
+
position: [1, 1, 0],
|
|
27
|
+
radius: 1,
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
expect(intersectsCircle(point, circle)).toBe(true)
|
|
31
|
+
})
|
|
32
|
+
|
|
33
|
+
test("it should prove that a point on the border of a circle intersects with it", () => {
|
|
34
|
+
const point = [2, 1, 0]
|
|
35
|
+
const circle = {
|
|
36
|
+
position: [1, 1, 0],
|
|
37
|
+
radius: 1,
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
expect(intersectsCircle(point, circle)).toBe(true)
|
|
41
|
+
})
|
|
42
|
+
|
|
43
|
+
test("it should prove that a point outside of a circle does not intersect with it", () => {
|
|
44
|
+
const point = [2, 2, 0]
|
|
45
|
+
const circle = {
|
|
46
|
+
position: [1, 1, 0],
|
|
47
|
+
radius: 1,
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
expect(intersectsCircle(point, circle)).toBe(false)
|
|
51
|
+
})
|
|
52
|
+
|
|
53
|
+
test("it should prove that a point inside of a rectangle intersects with it", () => {
|
|
54
|
+
const point = [1.5, 1.5, 0]
|
|
55
|
+
const rectangle = {
|
|
56
|
+
position: [1, 1, 0],
|
|
57
|
+
size: [2, 2, 0],
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
expect(intersectsRectangle(point, rectangle)).toBe(true)
|
|
61
|
+
})
|
|
62
|
+
|
|
63
|
+
test("it should prove that a point on the border of a rectangle intersects with it", () => {
|
|
64
|
+
const point = [2, 1, 0]
|
|
65
|
+
const rectangle = {
|
|
66
|
+
position: [1, 1, 0],
|
|
67
|
+
size: [2, 2, 0],
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
expect(intersectsRectangle(point, rectangle)).toBe(true)
|
|
71
|
+
})
|
|
72
|
+
|
|
73
|
+
test("it should prove that a point outside of a rectangle does not intersect with it", () => {
|
|
74
|
+
const point = [2.5, 2.5, 0]
|
|
75
|
+
const rectangle = {
|
|
76
|
+
position: [1, 1, 0],
|
|
77
|
+
size: [2, 2, 0],
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
expect(intersectsRectangle(point, rectangle)).toBe(false)
|
|
81
|
+
})
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {import("./types").Rectangle} Rectangle
|
|
3
|
+
* @typedef {import("./types").Platform} Platform
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { intersectsRectangle as circleIntersectsRectangle } from "./circle.js"
|
|
7
|
+
import { intersectsRectangle as platformIntersectsRectangle } from "./platform.js"
|
|
8
|
+
|
|
9
|
+
export function intersectsCircle(rectangle, circle) {
|
|
10
|
+
return circleIntersectsRectangle(circle, rectangle)
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Determines whether a rectangle intersects another in 3D space.
|
|
15
|
+
*
|
|
16
|
+
* @param {Rectangle} rectangle1 - The rectangle defined by its position (x, y, z) and size (width, height, depth).
|
|
17
|
+
* @param {Rectangle} rectangle2 - The other rectangle defined by its position (z, y, z) and size (width, height, depth).
|
|
18
|
+
* @returns {boolean} True if the two rectangles intersect, false otherwise.
|
|
19
|
+
*/
|
|
20
|
+
export function intersectsRectangle(rectangle1, rectangle2) {
|
|
21
|
+
const [x1, y1, z1] = rectangle1.position
|
|
22
|
+
const [w1, h1, d1] = rectangle1.size
|
|
23
|
+
|
|
24
|
+
const [x2, y2, z2] = rectangle2.position
|
|
25
|
+
const [w2, h2, d2] = rectangle2.size
|
|
26
|
+
|
|
27
|
+
return (
|
|
28
|
+
x1 <= x2 + w2 &&
|
|
29
|
+
x1 + w1 >= x2 &&
|
|
30
|
+
y1 <= y2 + h2 &&
|
|
31
|
+
y1 + h1 >= y2 &&
|
|
32
|
+
z1 <= z2 + d2 &&
|
|
33
|
+
z1 + d1 >= z2
|
|
34
|
+
)
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Checks if a rectangle intersects with a platform.
|
|
39
|
+
* @param {Rectangle} rectangle - The rectangle to check.
|
|
40
|
+
* @param {Platform} platform - The platform to check.
|
|
41
|
+
* @returns {boolean} True if the rectangle intersects the platform, false otherwise.
|
|
42
|
+
*/
|
|
43
|
+
export function intersectsPlatform(rectangle, platform) {
|
|
44
|
+
return platformIntersectsRectangle(platform, rectangle)
|
|
45
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { expect, test } from "vitest"
|
|
2
|
+
|
|
3
|
+
import { intersectsRectangle } from "./rectangle.js"
|
|
4
|
+
|
|
5
|
+
test("it should prove that a rectangle crossing another one intersects with it", () => {
|
|
6
|
+
const rectangle1 = {
|
|
7
|
+
position: [0, 0, 0],
|
|
8
|
+
size: [2, 2, 2],
|
|
9
|
+
}
|
|
10
|
+
const rectangle2 = {
|
|
11
|
+
position: [1, 1, 1],
|
|
12
|
+
size: [2, 2, 2],
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
expect(intersectsRectangle(rectangle1, rectangle2)).toBe(true)
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
test("it should prove that a rectangle right on top of another intersects with it", () => {
|
|
19
|
+
const rectangle1 = {
|
|
20
|
+
position: [0, 0, 0],
|
|
21
|
+
size: [2, 2, 2],
|
|
22
|
+
}
|
|
23
|
+
const rectangle2 = {
|
|
24
|
+
position: [0, 0, 0],
|
|
25
|
+
size: [2, 0, 2],
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
expect(intersectsRectangle(rectangle1, rectangle2)).toBe(true)
|
|
29
|
+
})
|
|
30
|
+
|
|
31
|
+
test("it should prove that a rectangle not crossing another one does not intersect with it", () => {
|
|
32
|
+
const rectangle1 = {
|
|
33
|
+
position: [0, 0, 0],
|
|
34
|
+
size: [2, 2, 2],
|
|
35
|
+
}
|
|
36
|
+
const rectangle2 = {
|
|
37
|
+
position: [3, 3, 3],
|
|
38
|
+
size: [2, 2, 2],
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
expect(intersectsRectangle(rectangle1, rectangle2)).toBe(false)
|
|
42
|
+
})
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {import("./types").Segment} Segment
|
|
3
|
+
* @typedef {import("./types").Point} Point
|
|
4
|
+
* @typedef {import("./types").Circle} Circle
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import {
|
|
8
|
+
magnitude,
|
|
9
|
+
setMagnitude,
|
|
10
|
+
} from "@inglorious/utils/math/linear-algebra/vector.js"
|
|
11
|
+
import {
|
|
12
|
+
distance,
|
|
13
|
+
dot,
|
|
14
|
+
subtract,
|
|
15
|
+
sum,
|
|
16
|
+
} from "@inglorious/utils/math/linear-algebra/vectors.js"
|
|
17
|
+
|
|
18
|
+
const BEFORE_SEGMENT = 0
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Calculates the coefficients [a, b, c] of the line equation ax + bz + c = 0
|
|
22
|
+
* for a given segment in 2D space.
|
|
23
|
+
*
|
|
24
|
+
* @param {Segment} segment - The segment defined by its start (`from`) and end (`to`) points.
|
|
25
|
+
* @returns {[number, number, number]} An array [a, b, c] representing the line equation.
|
|
26
|
+
*/
|
|
27
|
+
export function coefficients(segment) {
|
|
28
|
+
const [x1, , z1] = segment.from
|
|
29
|
+
const [x2, , z2] = segment.to
|
|
30
|
+
return [z1 - z2, x2 - x1, (x1 - x2) * z1 + x1 * (z2 - z1)]
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Finds the closest point on a segment to a given point in 3D space.
|
|
35
|
+
*
|
|
36
|
+
* @param {Segment} segment - The segment defined by its start (`from`) and end (`to`) points.
|
|
37
|
+
* @param {Point} point - The point in 3D space represented as [x, y, z].
|
|
38
|
+
* @returns {Point} The closest point on the segment to the given point.
|
|
39
|
+
*/
|
|
40
|
+
export function closestPoint(segment, point) {
|
|
41
|
+
const shiftedSegment = subtract(segment.to, segment.from)
|
|
42
|
+
const shiftedPoint = subtract(point, segment.from)
|
|
43
|
+
|
|
44
|
+
const projectionLength =
|
|
45
|
+
dot(shiftedSegment, shiftedPoint) / magnitude(shiftedSegment)
|
|
46
|
+
|
|
47
|
+
if (projectionLength < BEFORE_SEGMENT) {
|
|
48
|
+
return segment.from
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
if (projectionLength > magnitude(shiftedSegment)) {
|
|
52
|
+
return segment.to
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const projectedPoint = setMagnitude(shiftedSegment, projectionLength)
|
|
56
|
+
return sum(segment.from, projectedPoint)
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Calculates the shortest distance from a point to a segment in 3D space.
|
|
61
|
+
*
|
|
62
|
+
* @param {Segment} segment - The segment defined by its start (`from`) and end (`to`) points.
|
|
63
|
+
* @param {Point} point - The point in 3D space represented as [x, y, z].
|
|
64
|
+
* @returns {number} The shortest distance from the point to the segment.
|
|
65
|
+
*/
|
|
66
|
+
export function distanceFromPoint(segment, point) {
|
|
67
|
+
const closest = closestPoint(segment, point)
|
|
68
|
+
return distance(point, closest)
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Determines whether a segment intersects with the perimeter of a circle.
|
|
73
|
+
*
|
|
74
|
+
* @param {Segment} segment - The segment defined by its start (`from`) and end (`to`) points.
|
|
75
|
+
* @param {Circle} circle - The circle defined by its position (center) and radius.
|
|
76
|
+
* @returns {boolean} `true` if the segment intersects the circle's perimeter, otherwise `false`.
|
|
77
|
+
*/
|
|
78
|
+
export function intersectsCircle(segment, circle) {
|
|
79
|
+
return distanceFromPoint(segment, circle.position) <= circle.radius
|
|
80
|
+
}
|
|
@@ -0,0 +1,183 @@
|
|
|
1
|
+
import { expect, test } from "vitest"
|
|
2
|
+
|
|
3
|
+
import { isClose } from "../numbers.js"
|
|
4
|
+
import {
|
|
5
|
+
closestPoint,
|
|
6
|
+
coefficients,
|
|
7
|
+
distanceFromPoint,
|
|
8
|
+
intersectsCircle,
|
|
9
|
+
} from "./segment.js"
|
|
10
|
+
|
|
11
|
+
expect.extend({
|
|
12
|
+
toEqualVector(received, expected) {
|
|
13
|
+
const { isNot } = this
|
|
14
|
+
return {
|
|
15
|
+
pass: received.every((coord, index) => isClose(coord, expected[index])),
|
|
16
|
+
message: () =>
|
|
17
|
+
`${received} is ${isNot ? "" : "not"} close to ${expected}`,
|
|
18
|
+
}
|
|
19
|
+
},
|
|
20
|
+
})
|
|
21
|
+
|
|
22
|
+
test("it should retrieve values of *a*, *b*, and *c* from a segment, so it looks like a line in the form *ax + by + c = 0*", () => {
|
|
23
|
+
const segment = {
|
|
24
|
+
from: [0, 0, 0],
|
|
25
|
+
to: [1, 0, 2],
|
|
26
|
+
}
|
|
27
|
+
const expectedResult = [-2, 1, 0]
|
|
28
|
+
|
|
29
|
+
expect(coefficients(segment)).toEqual(expectedResult)
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
test("it should find the closest point in a segment to a point projectable on it", () => {
|
|
33
|
+
const segment = {
|
|
34
|
+
from: [0, 0, 0],
|
|
35
|
+
to: [2, 0, 2],
|
|
36
|
+
}
|
|
37
|
+
const point = [2, 0, 0]
|
|
38
|
+
const expectedResult = [1, 0, 1]
|
|
39
|
+
|
|
40
|
+
expect(closestPoint(segment, point)).toEqualVector(expectedResult)
|
|
41
|
+
})
|
|
42
|
+
|
|
43
|
+
test("it should find the closest point as the point itself when resting on the segment", () => {
|
|
44
|
+
const segment = {
|
|
45
|
+
from: [0, 0, 0],
|
|
46
|
+
to: [2, 0, 2],
|
|
47
|
+
}
|
|
48
|
+
const point = [1, 0, 1]
|
|
49
|
+
const expectedResult = [1, 0, 1]
|
|
50
|
+
|
|
51
|
+
expect(closestPoint(segment, point)).toEqualVector(expectedResult)
|
|
52
|
+
})
|
|
53
|
+
|
|
54
|
+
test("it return the start of a segment when the point is close to its start", () => {
|
|
55
|
+
const segment = {
|
|
56
|
+
from: [0, 0, 0],
|
|
57
|
+
to: [3, 0, 4],
|
|
58
|
+
}
|
|
59
|
+
const point = [-2, 0, 0]
|
|
60
|
+
const expectedResult = [0, 0, 0]
|
|
61
|
+
|
|
62
|
+
expect(closestPoint(segment, point)).toEqualVector(expectedResult)
|
|
63
|
+
})
|
|
64
|
+
|
|
65
|
+
test("it should return the end of a segment when the point is close to its end", () => {
|
|
66
|
+
const segment = {
|
|
67
|
+
from: [0, 0, 0],
|
|
68
|
+
to: [3, 0, 4],
|
|
69
|
+
}
|
|
70
|
+
const point = [5, 0, 4]
|
|
71
|
+
const expectedResult = [3, 0, 4]
|
|
72
|
+
|
|
73
|
+
expect(closestPoint(segment, point)).toEqualVector(expectedResult)
|
|
74
|
+
})
|
|
75
|
+
|
|
76
|
+
test("it should compute the distance between a segment and a point projectable on it", () => {
|
|
77
|
+
const segment = {
|
|
78
|
+
from: [0, 0, 0],
|
|
79
|
+
to: [3, 0, 4],
|
|
80
|
+
}
|
|
81
|
+
const point = [5, 0, 0]
|
|
82
|
+
const expectedResult = 4
|
|
83
|
+
|
|
84
|
+
expect(distanceFromPoint(segment, point)).toBe(expectedResult)
|
|
85
|
+
})
|
|
86
|
+
|
|
87
|
+
test("it should compute the distance between a segment and a point resting on it", () => {
|
|
88
|
+
const segment = {
|
|
89
|
+
from: [0, 0, 0],
|
|
90
|
+
to: [2, 0, 2],
|
|
91
|
+
}
|
|
92
|
+
const point = [1, 0, 1]
|
|
93
|
+
const expectedResult = 0
|
|
94
|
+
|
|
95
|
+
expect(distanceFromPoint(segment, point)).toBeCloseTo(expectedResult)
|
|
96
|
+
})
|
|
97
|
+
|
|
98
|
+
test("it should compute the distance between a segment and a point close to the start", () => {
|
|
99
|
+
const segment = {
|
|
100
|
+
from: [0, 0, 0],
|
|
101
|
+
to: [3, 0, 4],
|
|
102
|
+
}
|
|
103
|
+
const point = [-2, 0, 0]
|
|
104
|
+
const expectedResult = 2
|
|
105
|
+
|
|
106
|
+
expect(distanceFromPoint(segment, point)).toBe(expectedResult)
|
|
107
|
+
})
|
|
108
|
+
|
|
109
|
+
test("it should compute the distance between a segment and a point close to the end", () => {
|
|
110
|
+
const segment = {
|
|
111
|
+
from: [0, 0, 0],
|
|
112
|
+
to: [3, 0, 4],
|
|
113
|
+
}
|
|
114
|
+
const point = [5, 0, 4]
|
|
115
|
+
const expectedResult = 2
|
|
116
|
+
|
|
117
|
+
expect(distanceFromPoint(segment, point)).toBeCloseTo(expectedResult)
|
|
118
|
+
})
|
|
119
|
+
|
|
120
|
+
test("it should prove that a segment that crosses a circle intersects with it", () => {
|
|
121
|
+
const segment = {
|
|
122
|
+
from: [0, 0, 0],
|
|
123
|
+
to: [2, 0, 2],
|
|
124
|
+
}
|
|
125
|
+
const circle = {
|
|
126
|
+
position: [1, 0, 1],
|
|
127
|
+
radius: 1,
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
expect(intersectsCircle(segment, circle)).toBe(true)
|
|
131
|
+
})
|
|
132
|
+
|
|
133
|
+
test("it should prove that a segment tangent to a circle intersects with it", () => {
|
|
134
|
+
const segment = {
|
|
135
|
+
from: [0, 0, 0],
|
|
136
|
+
to: [0, 0, 2],
|
|
137
|
+
}
|
|
138
|
+
const circle = {
|
|
139
|
+
position: [1, 0, 1],
|
|
140
|
+
radius: 1,
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
expect(intersectsCircle(segment, circle)).toBe(true)
|
|
144
|
+
})
|
|
145
|
+
|
|
146
|
+
test("it should prove that a segment inside of a circle intersects with it", () => {
|
|
147
|
+
const segment = {
|
|
148
|
+
from: [0.5, 0, 0.5],
|
|
149
|
+
to: [1.5, 0, 1.5],
|
|
150
|
+
}
|
|
151
|
+
const circle = {
|
|
152
|
+
position: [1, 0, 1],
|
|
153
|
+
radius: 1,
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
expect(intersectsCircle(segment, circle)).toBe(true)
|
|
157
|
+
})
|
|
158
|
+
|
|
159
|
+
test("it should prove that a segment that does not cross a circle does not intersect with it", () => {
|
|
160
|
+
const segment = {
|
|
161
|
+
from: [0, 0, 0],
|
|
162
|
+
to: [-2, 0, 2],
|
|
163
|
+
}
|
|
164
|
+
const circle = {
|
|
165
|
+
position: [1, 0, 1],
|
|
166
|
+
radius: 1,
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
expect(intersectsCircle(segment, circle)).toBe(false)
|
|
170
|
+
})
|
|
171
|
+
|
|
172
|
+
test("it should prove that a segment that reaches outside of a circle does not intersect with it", () => {
|
|
173
|
+
const segment = {
|
|
174
|
+
from: [0, 0, 0],
|
|
175
|
+
to: [-1, 0, -2],
|
|
176
|
+
}
|
|
177
|
+
const circle = {
|
|
178
|
+
position: [1, 0, 1],
|
|
179
|
+
radius: 1,
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
expect(intersectsCircle(segment, circle)).toBe(false)
|
|
183
|
+
})
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Calculates the length of the hypothenuse (or magnitude) for a given set of numbers
|
|
3
|
+
* using the Pythagorean theorem.
|
|
4
|
+
*
|
|
5
|
+
* @param {number[]} nums - A list of numbers representing the sides of a right triangle.
|
|
6
|
+
* @returns {number} The length of the hypothenuse.
|
|
7
|
+
*/
|
|
8
|
+
export function hypothenuse(...nums) {
|
|
9
|
+
return Math.hypot(...nums)
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Alias for the `hypothenuse` function.
|
|
14
|
+
*/
|
|
15
|
+
export const pythagoras = hypothenuse
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { expect, test } from "vitest"
|
|
2
|
+
|
|
3
|
+
import { hypothenuse } from "./triangle.js"
|
|
4
|
+
|
|
5
|
+
test("it should compute the hypothenuse of two catheti (aka pythagoras)", () => {
|
|
6
|
+
const cat1 = 3
|
|
7
|
+
const cat2 = 4
|
|
8
|
+
const expectedResult = 5
|
|
9
|
+
|
|
10
|
+
expect(hypothenuse(cat1, cat2)).toBe(expectedResult)
|
|
11
|
+
})
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
export type Point = [number, number, number] // [x, y, z]
|
|
2
|
+
export type Size = [number, number, number] // [width, height, depth]
|
|
3
|
+
export type Line = [number, number, number] // [a, b, c]
|
|
4
|
+
|
|
5
|
+
export interface Circle {
|
|
6
|
+
position: Point
|
|
7
|
+
radius: number
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export interface Rectangle {
|
|
11
|
+
position: Point
|
|
12
|
+
size: Size
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface Platform {
|
|
16
|
+
position: Point
|
|
17
|
+
size: Size
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export interface Segment {
|
|
21
|
+
from: Point
|
|
22
|
+
to: Point
|
|
23
|
+
}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {import('../../../types').Vector2} Vector2
|
|
3
|
+
* @typedef {import('../../../types').Vector3} Vector3
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
const NO_Y = 0 // Default Y-coordinate value for 3D vectors derived from 2D vectors.
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Converts a 2D vector [x, z] into a 3D vector [x, 0, z].
|
|
10
|
+
*
|
|
11
|
+
* @param {Vector2} vector - A 2D vector represented as [x, z].
|
|
12
|
+
* @returns {Vector3} A 3D vector represented as [x, 0, z].
|
|
13
|
+
*/
|
|
14
|
+
export function from2D(vector) {
|
|
15
|
+
const [x, z] = vector
|
|
16
|
+
return [x, NO_Y, z]
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Converts a 3D vector [x, y, z] into a 2D vector [x, z].
|
|
21
|
+
*
|
|
22
|
+
* @param {Vector3} vector - A 3D vector represented as [x, y, z].
|
|
23
|
+
* @returns {Vector2} A 2D vector represented as [x, z].
|
|
24
|
+
*/
|
|
25
|
+
export function to2D(vector) {
|
|
26
|
+
const [x, , z] = vector
|
|
27
|
+
return [x, z]
|
|
28
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { expect, test } from "vitest"
|
|
2
|
+
|
|
3
|
+
import { from2D, to2D } from "./2d.js"
|
|
4
|
+
|
|
5
|
+
test("build a 3D vector from a 2D one", () => {
|
|
6
|
+
const vector = [3, 4]
|
|
7
|
+
const expectedResult = [3, 0, 4]
|
|
8
|
+
|
|
9
|
+
expect(from2D(vector)).toStrictEqual(expectedResult)
|
|
10
|
+
})
|
|
11
|
+
|
|
12
|
+
test("build a 2D vector from a 3D one", () => {
|
|
13
|
+
const vector = [3, 0, 4]
|
|
14
|
+
const expectedResult = [3, 4]
|
|
15
|
+
|
|
16
|
+
expect(to2D(vector)).toStrictEqual(expectedResult)
|
|
17
|
+
})
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {import('../../../types').Vector3} Vector3
|
|
3
|
+
* @typedef {import('../../../types').Quaternion} Quaternion
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { cos, sin } from "@inglorious/utils/math/trigonometry.js"
|
|
7
|
+
|
|
8
|
+
const DEFAULT_ANGLE = 0
|
|
9
|
+
const HALF = 2 // Constant representing the divisor for halving an angle.
|
|
10
|
+
// eslint-disable-next-line no-magic-numbers
|
|
11
|
+
const Y_AXIS = [0, 1, 0] // Default axis of rotation (Y-axis).
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Computes a quaternion representing a rotation around a given axis.
|
|
15
|
+
*
|
|
16
|
+
* @param {number} [angle=0] - The angle of rotation in radians. Defaults to 0.
|
|
17
|
+
* @param {Vector3} [axis=Y_AXIS] - The axis of rotation as a 3D vector Defaults to the Y-axis.
|
|
18
|
+
* @returns {Quaternion} The quaternion as an array [w, x, y, z].
|
|
19
|
+
*/
|
|
20
|
+
export function quaternion(angle = DEFAULT_ANGLE, axis = Y_AXIS) {
|
|
21
|
+
return [cos(angle / HALF), ...axis.map((coord) => coord * sin(angle / HALF))]
|
|
22
|
+
}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { cos, pi, sin } from "@inglorious/utils/math/trigonometry.js"
|
|
2
|
+
import { expect, test } from "vitest"
|
|
3
|
+
|
|
4
|
+
import { quaternion } from "./quaternion.js"
|
|
5
|
+
|
|
6
|
+
test("it should return the quaternion for no rotation", () => {
|
|
7
|
+
const angle = 0
|
|
8
|
+
const expectedResult = [1, 0, 0, 0]
|
|
9
|
+
|
|
10
|
+
expect(quaternion(angle)).toStrictEqual(expectedResult)
|
|
11
|
+
})
|
|
12
|
+
|
|
13
|
+
test("it should return the quaternion for a rotation of pi/2", () => {
|
|
14
|
+
const angle = pi() / 2
|
|
15
|
+
const expectedResult = [cos(pi() / 4), 0, sin(pi() / 4), 0]
|
|
16
|
+
|
|
17
|
+
expect(quaternion(angle)).toStrictEqual(expectedResult)
|
|
18
|
+
})
|
|
19
|
+
|
|
20
|
+
test("it should return the quaternion for a rotation of negative pi/2", () => {
|
|
21
|
+
const angle = -pi() / 2
|
|
22
|
+
const expectedResult = [cos(-pi() / 4), -0, sin(-pi() / 4), -0]
|
|
23
|
+
|
|
24
|
+
expect(quaternion(angle)).toStrictEqual(expectedResult)
|
|
25
|
+
})
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {import('./types').Quaternion} Quaternion
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { times } from "./vector.js"
|
|
6
|
+
import { cross, dot, sum } from "./vectors.js"
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Combines two quaternions using the Hamilton product.
|
|
10
|
+
*
|
|
11
|
+
* @param {Quaternion} q1 - The first quaternion.
|
|
12
|
+
* @param {Quaternion} q2 - The second quaternion.
|
|
13
|
+
* @returns {Quaternion} - The resulting quaternion after combining q1 and q2.
|
|
14
|
+
*/
|
|
15
|
+
export function combine(q1, q2) {
|
|
16
|
+
const [s, ...v] = q1
|
|
17
|
+
const [t, ...w] = q2
|
|
18
|
+
|
|
19
|
+
return [s * t - dot(v, w), ...sum(times(w, s), times(v, t), cross(v, w))]
|
|
20
|
+
}
|