2024-01-28 23:18:11 +00:00
|
|
|
#![allow(
|
|
|
|
unsafe_op_in_unsafe_fn,
|
|
|
|
clippy::all,
|
2024-02-11 23:19:36 +00:00
|
|
|
clippy::undocumented_unsafe_blocks,
|
2024-09-20 19:16:42 +00:00
|
|
|
clippy::ptr_cast_constness,
|
|
|
|
// FIXME(15321): solve CI failures, then replace with `#![expect()]`.
|
|
|
|
missing_docs
|
2024-01-28 23:18:11 +00:00
|
|
|
)]
|
2024-03-23 02:22:52 +00:00
|
|
|
#![cfg_attr(docsrs, feature(doc_auto_cfg))]
|
2024-03-25 18:52:50 +00:00
|
|
|
#![doc(
|
|
|
|
html_logo_url = "https://bevyengine.org/assets/icon.png",
|
|
|
|
html_favicon_url = "https://bevyengine.org/assets/icon.png"
|
|
|
|
)]
|
2022-05-31 22:53:54 +00:00
|
|
|
|
|
|
|
use glam::{Vec2, Vec3};
|
|
|
|
|
|
|
|
mod generated;
|
|
|
|
|
|
|
|
/// The interface by which mikktspace interacts with your geometry.
|
|
|
|
pub trait Geometry {
|
|
|
|
/// Returns the number of faces.
|
|
|
|
fn num_faces(&self) -> usize;
|
|
|
|
|
|
|
|
/// Returns the number of vertices of a face.
|
|
|
|
fn num_vertices_of_face(&self, face: usize) -> usize;
|
|
|
|
|
|
|
|
/// Returns the position of a vertex.
|
|
|
|
fn position(&self, face: usize, vert: usize) -> [f32; 3];
|
|
|
|
|
|
|
|
/// Returns the normal of a vertex.
|
|
|
|
fn normal(&self, face: usize, vert: usize) -> [f32; 3];
|
|
|
|
|
|
|
|
/// Returns the texture coordinate of a vertex.
|
|
|
|
fn tex_coord(&self, face: usize, vert: usize) -> [f32; 2];
|
|
|
|
|
|
|
|
/// Sets the generated tangent for a vertex.
|
|
|
|
/// Leave this function unimplemented if you are implementing
|
|
|
|
/// `set_tangent_encoded`.
|
|
|
|
fn set_tangent(
|
|
|
|
&mut self,
|
|
|
|
tangent: [f32; 3],
|
|
|
|
_bi_tangent: [f32; 3],
|
|
|
|
_f_mag_s: f32,
|
|
|
|
_f_mag_t: f32,
|
|
|
|
bi_tangent_preserves_orientation: bool,
|
|
|
|
face: usize,
|
|
|
|
vert: usize,
|
|
|
|
) {
|
|
|
|
let sign = if bi_tangent_preserves_orientation {
|
|
|
|
1.0
|
|
|
|
} else {
|
|
|
|
-1.0
|
|
|
|
};
|
|
|
|
self.set_tangent_encoded([tangent[0], tangent[1], tangent[2], sign], face, vert);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the generated tangent for a vertex with its bi-tangent encoded as the 'W' (4th)
|
|
|
|
/// component in the tangent. The 'W' component marks if the bi-tangent is flipped. This
|
|
|
|
/// is called by the default implementation of `set_tangent`; therefore, this function will
|
|
|
|
/// not be called by the crate unless `set_tangent` is unimplemented.
|
|
|
|
fn set_tangent_encoded(&mut self, _tangent: [f32; 4], _face: usize, _vert: usize) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Generates tangents for the input geometry.
|
|
|
|
///
|
|
|
|
/// # Errors
|
|
|
|
///
|
|
|
|
/// Returns `false` if the geometry is unsuitable for tangent generation including,
|
|
|
|
/// but not limited to, lack of vertices.
|
2024-03-27 03:30:08 +00:00
|
|
|
#[allow(unsafe_code)]
|
2022-05-31 22:53:54 +00:00
|
|
|
pub fn generate_tangents<I: Geometry>(geometry: &mut I) -> bool {
|
|
|
|
unsafe { generated::genTangSpace(geometry, 180.0) }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_position<I: Geometry>(geometry: &mut I, index: usize) -> Vec3 {
|
|
|
|
let (face, vert) = index_to_face_vert(index);
|
|
|
|
geometry.position(face, vert).into()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_tex_coord<I: Geometry>(geometry: &mut I, index: usize) -> Vec3 {
|
|
|
|
let (face, vert) = index_to_face_vert(index);
|
|
|
|
let tex_coord: Vec2 = geometry.tex_coord(face, vert).into();
|
|
|
|
let val = tex_coord.extend(1.0);
|
|
|
|
val
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_normal<I: Geometry>(geometry: &mut I, index: usize) -> Vec3 {
|
|
|
|
let (face, vert) = index_to_face_vert(index);
|
|
|
|
geometry.normal(face, vert).into()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn index_to_face_vert(index: usize) -> (usize, usize) {
|
|
|
|
(index >> 2, index & 0x3)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn face_vert_to_index(face: usize, vert: usize) -> usize {
|
|
|
|
face << 2 | vert & 0x3
|
|
|
|
}
|