summaryrefslogtreecommitdiff
path: root/engine/src/data_types/matrix.rs
blob: 3a29ae25f7e0855fb1da7a5705b7df2e65c45258 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
use crate::vector::Vec3;

#[derive(Debug, Clone)]
pub struct Matrix<Value, const ROWS: usize, const COLUMNS: usize>
{
    /// Items must be layed out this way for it to work with OpenGL shaders.
    items: [[Value; ROWS]; COLUMNS],
}

impl<Value, const ROWS: usize, const COLUMNS: usize> Matrix<Value, ROWS, COLUMNS>
{
    /// Creates a new `ROWS` * `COLUMNS` matrix.
    #[must_use]
    pub fn new() -> Self
    where
        Value: Default,
    {
        Self {
            items: std::array::from_fn(|_| std::array::from_fn(|_| Value::default())),
        }
    }

    /// Sets the value at the specified cell.
    pub fn set_cell(&mut self, row: usize, column: usize, value: Value)
    {
        self.items[column][row] = value;
    }

    /// Returns the internal 2D array as a pointer.
    #[must_use]
    pub fn as_ptr(&self) -> *const Value
    {
        self.items[0].as_ptr()
    }
}

impl<Value, const ROWS: usize, const COLUMNS: usize> Default
    for Matrix<Value, ROWS, COLUMNS>
where
    Value: Default,
{
    fn default() -> Self
    {
        Self::new()
    }
}

impl<const ROWS_COLS: usize> Matrix<f32, ROWS_COLS, ROWS_COLS>
{
    /// Creates a new identity matrix.
    #[must_use]
    pub fn new_identity() -> Self
    {
        let mut index = 0;

        let items = [(); ROWS_COLS].map(|()| {
            let mut columns = [0.0; ROWS_COLS];

            columns[index] = 1.0;

            index += 1;

            columns
        });

        Self { items }
    }
}

impl Matrix<f32, 4, 4>
{
    pub fn translate(&mut self, translation: &Vec3<f32>)
    {
        self.set_cell(0, 3, translation.x);
        self.set_cell(1, 3, translation.y);
        self.set_cell(2, 3, translation.z);
        self.set_cell(3, 3, 1.0);
    }

    pub fn scale(&mut self, scaling: &Vec3<f32>)
    {
        self.set_cell(0, 0, scaling.x);
        self.set_cell(1, 1, scaling.y);
        self.set_cell(2, 2, scaling.z);
        self.set_cell(3, 3, 1.0);
    }

    pub fn look_at(&mut self, eye: &Vec3<f32>, target: &Vec3<f32>, up: &Vec3<f32>)
    {
        let rev_target_direction = (eye - target).normalize();

        let camera_right = up.cross(&rev_target_direction).normalize();

        let camera_up = rev_target_direction.cross(&camera_right);

        self.set_cell(0, 0, camera_right.x);
        self.set_cell(0, 1, camera_right.y);
        self.set_cell(0, 2, camera_right.z);

        self.set_cell(1, 0, camera_up.x);
        self.set_cell(1, 1, camera_up.y);
        self.set_cell(1, 2, camera_up.z);

        self.set_cell(2, 0, rev_target_direction.x);
        self.set_cell(2, 1, rev_target_direction.y);
        self.set_cell(2, 2, rev_target_direction.z);

        // The vector is negated since we want the world to be translated in the opposite
        // direction of where we want the camera to move.
        let camera_pos = -Vec3 {
            x: camera_right.dot(eye),
            y: camera_up.dot(eye),
            z: rev_target_direction.dot(eye),
        };

        self.set_cell(0, 3, camera_pos.x);
        self.set_cell(1, 3, camera_pos.y);
        self.set_cell(2, 3, camera_pos.z);

        self.set_cell(3, 3, 1.0);
    }
}