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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
use crate::vertex_attributes::MultiIndexedVertexAttributes;
use crate::BlenderMesh;

/// Indicates an error while calculating the tangents for a mesh's verticies
#[derive(Debug, Fail)]
pub enum TangentError {
    #[fail(display = "Cannot calculate vertex tangents for a mesh with no uvs")]
    NoVertexUvs,
}

impl BlenderMesh {
    /// Calculate the tangent for each face in the mesh - useful for normal mapping where you'll
    /// typically want to do lighting calculations in tangent space.
    ///
    /// We look at the first, second and third vertex position and uv coordinate in the
    /// face in order to calculate the tangent.
    ///
    /// This is useful for normal mapping.
    ///
    /// We'll push tangents to `.face_tangents`
    ///
    /// Then later in when combining indices we'll use these `.face_tangents` in order to
    /// generate `per_vertex_tangents`
    pub(crate) fn calculate_face_tangents(&self) -> Result<Vec<f32>, TangentError> {
        let multi = &self.multi_indexed_vertex_attributes;

        let MultiIndexedVertexAttributes {
            vertices_in_each_face,
            positions,
            uvs,
            ..
        } = multi;

        if uvs.is_none() {
            return Err(TangentError::NoVertexUvs)?;
        }
        let uvs = uvs.as_ref().unwrap();

        let mut total_indices_processed = 0;

        let mut face_tangents = vec![];

        // Iterate over each face and calculate the tangent for that face.
        for vertices_in_face in vertices_in_each_face.iter() {
            let vertices_in_face = *vertices_in_face;

            let idx = total_indices_processed as usize;

            // Get the first three vertex indices for this face
            let pos_idx_0 = positions.indices[idx];
            let pos_idx_1 = positions.indices[idx + 1];
            let pos_idx_2 = positions.indices[idx + 2];

            // Get the three UV indices for this face
            let uv_idx_0 = uvs.indices[idx];
            let uv_idx_1 = uvs.indices[idx + 1];
            let uv_idx_2 = uvs.indices[idx + 2];

            let pos0 = positions.attribute.data_at_idx(pos_idx_0);
            let pos1 = positions.attribute.data_at_idx(pos_idx_1);
            let pos2 = positions.attribute.data_at_idx(pos_idx_2);

            let uv0 = uvs.attribute.data_at_idx(uv_idx_0);
            let uv1 = uvs.attribute.data_at_idx(uv_idx_1);
            let uv2 = uvs.attribute.data_at_idx(uv_idx_2);

            let edge1 = (pos1[0] - pos0[0], pos1[1] - pos0[1], pos1[2] - pos0[2]);
            let edge2 = (pos2[0] - pos1[0], pos2[1] - pos1[1], pos2[2] - pos1[2]);

            let delta_uv1 = (uv1[0] - uv0[0], uv1[1] - uv0[1]);
            let delta_uv2 = (uv2[0] - uv1[0], uv2[1] - uv1[1]);

            let f = 1.0 / ((delta_uv1.0 * delta_uv2.1) - (delta_uv2.0 * delta_uv1.1));

            let tangent_x = f * ((delta_uv2.1 * edge1.0) - (delta_uv1.1 * edge2.0));
            let tangent_y = f * ((delta_uv2.1 * edge1.1) - (delta_uv1.1 * edge2.1));
            let tangent_z = f * ((delta_uv2.1 * edge1.2) - (delta_uv1.1 * edge2.2));

            face_tangents.push(tangent_x);
            face_tangents.push(tangent_y);
            face_tangents.push(tangent_z);

            total_indices_processed += vertices_in_face as u16;
        }

        Ok(face_tangents)
    }
}

/// Given a face idx, get the corresponding tangent
pub(crate) fn face_tangent_at_idx(face_tangents: &[f32], face_idx: usize) -> (f32, f32, f32) {
    (
        face_tangents[face_idx * 3],
        face_tangents[face_idx * 3 + 1],
        face_tangents[face_idx * 3 + 2],
    )
}

// Numbers in these tests were not verified by hand.
// Instead, we took this common tangent calculation formula wrote tests, and verified
// that the rendered models looked visually correct (meaning that our test values are also correct).
#[cfg(test)]
mod tests {
    use super::*;
    use crate::concat_vecs;
    use crate::test_utils::*;

    /// Ensure that a mesh with no uvs returns TangentError::NoVertexUvs
    #[test]
    fn no_vertex_uvs() {
        let mesh: BlenderMesh = BlenderMesh::default();

        match mesh.calculate_face_tangents() {
            Ok(_) => unreachable!(),
            Err(TangentError::NoVertexUvs) => {}
        }
    }

    /// Properly calculates tangents for a mesh that has one triangle
    #[test]
    fn calculate_tangents_1_triangle() {
        let mesh: BlenderMesh = BlenderMesh {
            multi_indexed_vertex_attributes: MultiIndexedVertexAttributes {
                positions: (
                    vec![0, 1, 2],
                    (
                        concat_vecs!(v(0), vec![1.0, 0.0, 0.0], vec![1.0, 1.0, 0.0]),
                        3,
                    )
                        .into(),
                )
                    .into(),
                uvs: Some(
                    (
                        vec![0, 1, 2],
                        (concat_vecs!(v2(0), vec![0.5, 0.0], v2(1)), 2).into(),
                    )
                        .into(),
                ),
                vertices_in_each_face: vec![3],
                ..MultiIndexedVertexAttributes::default()
            },
            ..BlenderMesh::default()
        };

        mesh.calculate_face_tangents().unwrap();

        assert_eq!(
            &mesh.calculate_face_tangents().unwrap(),
            // One face (a triangle) so only one face tangent vector
            &vec![2., 0., 0.]
        );
    }

    #[test]
    fn calculate_tangents_2_triangle() {
        let mesh: BlenderMesh = BlenderMesh {
            multi_indexed_vertex_attributes: MultiIndexedVertexAttributes {
                positions: (
                    vec![0, 1, 2, 0, 2, 3],
                    (
                        concat_vecs!(
                            v(0),
                            vec![1.0, 0.0, 0.0],
                            vec![1.0, 1.0, 0.0],
                            vec![0., 1., 0.]
                        ),
                        3,
                    )
                        .into(),
                )
                    .into(),
                uvs: Some(
                    (
                        vec![0, 1, 2, 0, 2, 3],
                        (concat_vecs!(v2(0), vec![0.5, 0.0], v2(1), vec![0., 1.]), 2).into(),
                    )
                        .into(),
                ),
                vertices_in_each_face: vec![3, 3],
                ..MultiIndexedVertexAttributes::default()
            },
            ..BlenderMesh::default()
        };

        mesh.calculate_face_tangents().unwrap();

        assert_eq!(
            &mesh.calculate_face_tangents().unwrap(),
            // Two faces (two triangles) so two tangent vectors
            &vec![2., 0., 0., 1., 0., 0.]
        );
    }

    #[test]
    fn calculate_tangents_1_quad() {
        let mesh: BlenderMesh = BlenderMesh {
            multi_indexed_vertex_attributes: MultiIndexedVertexAttributes {
                positions: (
                    vec![0, 1, 2, 3],
                    (
                        concat_vecs!(
                            v(0),
                            vec![1.0, 0.0, 0.0],
                            vec![1.0, 1.0, 0.0],
                            vec![0., 1., 0.]
                        ),
                        3,
                    )
                        .into(),
                )
                    .into(),
                uvs: Some(
                    (
                        vec![0, 1, 2, 3],
                        (concat_vecs!(v2(0), vec![0.5, 0.0], v2(1), vec![0., 1.]), 2).into(),
                    )
                        .into(),
                ),
                vertices_in_each_face: vec![4],
                ..MultiIndexedVertexAttributes::default()
            },
            ..BlenderMesh::default()
        };

        mesh.calculate_face_tangents().unwrap();

        assert_eq!(
            &mesh.calculate_face_tangents().unwrap(),
            // Two faces (two triangles) so two tangent vectors
            &vec![2., 0., 0.]
        );
    }
}