From d3f8badc89d152d000c96522997f91bba379aa9c Mon Sep 17 00:00:00 2001 From: Valentin Date: Tue, 17 Nov 2020 20:55:52 +0100 Subject: [PATCH 1/2] Add rectangular cuboid shape Co-authored-by: Jason Lessard --- crates/bevy_render/src/mesh/mesh.rs | 103 +++++++++++++--------- examples/3d/3d_scene.rs | 2 +- examples/3d/msaa.rs | 2 +- examples/3d/parenting.rs | 2 +- examples/3d/spawner.rs | 2 +- examples/3d/z_sort_debug.rs | 2 +- examples/android/android.rs | 2 +- examples/ios/src/lib.rs | 2 +- examples/shader/mesh_custom_attribute.rs | 2 +- examples/shader/shader_custom_material.rs | 2 +- examples/shader/shader_defs.rs | 2 +- 11 files changed, 70 insertions(+), 53 deletions(-) diff --git a/crates/bevy_render/src/mesh/mesh.rs b/crates/bevy_render/src/mesh/mesh.rs index 0f1ba85f23d18..3bd72cfb0759b 100644 --- a/crates/bevy_render/src/mesh/mesh.rs +++ b/crates/bevy_render/src/mesh/mesh.rs @@ -227,58 +227,75 @@ pub mod shape { use bevy_math::*; use hexasphere::Hexasphere; - /// A cube. - #[derive(Debug)] - pub struct Cube { - /// Half the side length of the cube. - pub size: f32, + pub struct Cuboid { + pub min_x: f32, + pub max_x: f32, + + pub min_y: f32, + pub max_y: f32, + + pub min_z: f32, + pub max_z: f32, + } + + impl Cuboid { + pub fn new(x_length: f32, y_length: f32, z_length: f32) -> Cuboid { + Cuboid { + max_x: x_length / 2.0, + min_x: -x_length / 2.0, + max_y: y_length / 2.0, + min_y: -y_length / 2.0, + max_z: z_length / 2.0, + min_z: -z_length / 2.0, + } + } } - impl Default for Cube { + impl Default for Cuboid { fn default() -> Self { - Cube { size: 1.0 } + Cuboid::new(2.0, 1.0, 1.0) } } - impl From for Mesh { - fn from(cube: Cube) -> Self { - let size = cube.size; + impl From for Mesh { + fn from(sp: Cuboid) -> Self { let vertices = &[ - // top (0., 0., size) - ([-size, -size, size], [0., 0., size], [0., 0.]), - ([size, -size, size], [0., 0., size], [size, 0.]), - ([size, size, size], [0., 0., size], [size, size]), - ([-size, size, size], [0., 0., size], [0., size]), - // bottom (0., 0., -size) - ([-size, size, -size], [0., 0., -size], [size, 0.]), - ([size, size, -size], [0., 0., -size], [0., 0.]), - ([size, -size, -size], [0., 0., -size], [0., size]), - ([-size, -size, -size], [0., 0., -size], [size, size]), - // right (size, 0., 0.) - ([size, -size, -size], [size, 0., 0.], [0., 0.]), - ([size, size, -size], [size, 0., 0.], [size, 0.]), - ([size, size, size], [size, 0., 0.], [size, size]), - ([size, -size, size], [size, 0., 0.], [0., size]), - // left (-size, 0., 0.) - ([-size, -size, size], [-size, 0., 0.], [size, 0.]), - ([-size, size, size], [-size, 0., 0.], [0., 0.]), - ([-size, size, -size], [-size, 0., 0.], [0., size]), - ([-size, -size, -size], [-size, 0., 0.], [size, size]), - // front (0., size, 0.) - ([size, size, -size], [0., size, 0.], [size, 0.]), - ([-size, size, -size], [0., size, 0.], [0., 0.]), - ([-size, size, size], [0., size, 0.], [0., size]), - ([size, size, size], [0., size, 0.], [size, size]), - // back (0., -size, 0.) - ([size, -size, size], [0., -size, 0.], [0., 0.]), - ([-size, -size, size], [0., -size, 0.], [size, 0.]), - ([-size, -size, -size], [0., -size, 0.], [size, size]), - ([size, -size, -size], [0., -size, 0.], [0., size]), + // Top + ([sp.min_x, sp.min_y, sp.max_z], [0., 0., 1.0], [0., 0.]), + ([sp.max_x, sp.min_y, sp.max_z], [0., 0., 1.0], [1.0, 0.]), + ([sp.max_x, sp.max_y, sp.max_z], [0., 0., 1.0], [1.0, 1.0]), + ([sp.min_x, sp.max_y, sp.max_z], [0., 0., 1.0], [0., 1.0]), + // Bottom + ([sp.min_x, sp.max_y, sp.min_z], [0., 0., -1.0], [1.0, 0.]), + ([sp.max_x, sp.max_y, sp.min_z], [0., 0., -1.0], [0., 0.]), + ([sp.max_x, sp.min_y, sp.min_z], [0., 0., -1.0], [0., 1.0]), + ([sp.min_x, sp.min_y, sp.min_z], [0., 0., -1.0], [1.0, 1.0]), + // Right + ([sp.max_x, sp.min_y, sp.min_z], [1.0, 0., 0.], [0., 0.]), + ([sp.max_x, sp.max_y, sp.min_z], [1.0, 0., 0.], [1.0, 0.]), + ([sp.max_x, sp.max_y, sp.max_z], [1.0, 0., 0.], [1.0, 1.0]), + ([sp.max_x, sp.min_y, sp.max_z], [1.0, 0., 0.], [0., 1.0]), + // Left + ([sp.min_x, sp.min_y, sp.max_z], [-1.0, 0., 0.], [1.0, 0.]), + ([sp.min_x, sp.max_y, sp.max_z], [-1.0, 0., 0.], [0., 0.]), + ([sp.min_x, sp.max_y, sp.min_z], [-1.0, 0., 0.], [0., 1.0]), + ([sp.min_x, sp.min_y, sp.min_z], [-1.0, 0., 0.], [1.0, 1.0]), + // Front + ([sp.max_x, sp.max_y, sp.min_z], [0., 1.0, 0.], [1.0, 0.]), + ([sp.min_x, sp.max_y, sp.min_z], [0., 1.0, 0.], [0., 0.]), + ([sp.min_x, sp.max_y, sp.max_z], [0., 1.0, 0.], [0., 1.0]), + ([sp.max_x, sp.max_y, sp.max_z], [0., 1.0, 0.], [1.0, 1.0]), + // Back + ([sp.max_x, sp.min_y, sp.max_z], [0., -1.0, 0.], [0., 0.]), + ([sp.min_x, sp.min_y, sp.max_z], [0., -1.0, 0.], [1.0, 0.]), + ([sp.min_x, sp.min_y, sp.min_z], [0., -1.0, 0.], [1.0, 1.0]), + ([sp.max_x, sp.min_y, sp.min_z], [0., -1.0, 0.], [0., 1.0]), ]; - let mut positions = Vec::new(); - let mut normals = Vec::new(); - let mut uvs = Vec::new(); + let mut positions = Vec::with_capacity(24); + let mut normals = Vec::with_capacity(24); + let mut uvs = Vec::with_capacity(24); + for (position, normal, uv) in vertices.iter() { positions.push(*position); normals.push(*normal); diff --git a/examples/3d/3d_scene.rs b/examples/3d/3d_scene.rs index fb5be88b72be8..10652be06dbd3 100644 --- a/examples/3d/3d_scene.rs +++ b/examples/3d/3d_scene.rs @@ -24,7 +24,7 @@ fn setup( }) // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), + mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), material: materials.add(Color::rgb(0.8, 0.7, 0.6).into()), transform: Transform::from_translation(Vec3::new(0.0, 1.0, 0.0)), ..Default::default() diff --git a/examples/3d/msaa.rs b/examples/3d/msaa.rs index ef3767fde14dd..d83f492a7b523 100644 --- a/examples/3d/msaa.rs +++ b/examples/3d/msaa.rs @@ -21,7 +21,7 @@ fn setup( commands // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), + mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), material: materials.add(Color::rgb(0.8, 0.7, 0.6).into()), ..Default::default() }) diff --git a/examples/3d/parenting.rs b/examples/3d/parenting.rs index 03c0c826835ec..92cf3b166ba8a 100644 --- a/examples/3d/parenting.rs +++ b/examples/3d/parenting.rs @@ -27,7 +27,7 @@ fn setup( mut meshes: ResMut>, mut materials: ResMut>, ) { - let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 1.0 })); + let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); let cube_material_handle = materials.add(StandardMaterial { albedo: Color::rgb(0.8, 0.7, 0.6), ..Default::default() diff --git a/examples/3d/spawner.rs b/examples/3d/spawner.rs index 20a7552bc43df..d16ef7af45edf 100644 --- a/examples/3d/spawner.rs +++ b/examples/3d/spawner.rs @@ -50,7 +50,7 @@ fn setup( }); let mut rng = StdRng::from_entropy(); - let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 1.0 })); + let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); for _ in 0..10000 { commands.spawn(PbrBundle { mesh: cube_handle.clone(), diff --git a/examples/3d/z_sort_debug.rs b/examples/3d/z_sort_debug.rs index 08c4cd6b762d8..2ad81c3d97d4a 100644 --- a/examples/3d/z_sort_debug.rs +++ b/examples/3d/z_sort_debug.rs @@ -46,7 +46,7 @@ fn setup( mut meshes: ResMut>, mut materials: ResMut>, ) { - let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 1.0 })); + let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); commands // parent cube .spawn(PbrBundle { diff --git a/examples/android/android.rs b/examples/android/android.rs index 15fbd04583467..05ae3f4b3e27f 100644 --- a/examples/android/android.rs +++ b/examples/android/android.rs @@ -26,7 +26,7 @@ fn setup( }) // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), + mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), material: materials.add(Color::rgb(0.8, 0.7, 0.6).into()), transform: Transform::from_translation(Vec3::new(0.0, 1.0, 0.0)), ..Default::default() diff --git a/examples/ios/src/lib.rs b/examples/ios/src/lib.rs index 56cbd1b050905..63631edfc038e 100644 --- a/examples/ios/src/lib.rs +++ b/examples/ios/src/lib.rs @@ -31,7 +31,7 @@ fn setup( }) // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), + mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), material: materials.add(Color::rgb(0.5, 0.4, 0.3).into()), transform: Transform::from_translation(Vec3::new(0.0, 1.0, 0.0)), ..Default::default() diff --git a/examples/shader/mesh_custom_attribute.rs b/examples/shader/mesh_custom_attribute.rs index 013dc9bb199b6..14648e0d70572 100644 --- a/examples/shader/mesh_custom_attribute.rs +++ b/examples/shader/mesh_custom_attribute.rs @@ -83,7 +83,7 @@ fn setup( let material = materials.add(MyMaterialWithVertexColorSupport {}); // create a generic cube - let mut cube_with_vertex_colors = Mesh::from(shape::Cube { size: 1.0 }); + let mut cube_with_vertex_colors = Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0)); // insert our custom color attribute with some nice colors! cube_with_vertex_colors.set_attribute( diff --git a/examples/shader/shader_custom_material.rs b/examples/shader/shader_custom_material.rs index 92bcbcad1128b..c1b9c7af17bd6 100644 --- a/examples/shader/shader_custom_material.rs +++ b/examples/shader/shader_custom_material.rs @@ -84,7 +84,7 @@ fn setup( commands // cube .spawn(MeshBundle { - mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), + mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), render_pipelines: RenderPipelines::from_pipelines(vec![RenderPipeline::new( pipeline_handle, )]), diff --git a/examples/shader/shader_defs.rs b/examples/shader/shader_defs.rs index 3a4abfec84f8b..ae76c2187537b 100644 --- a/examples/shader/shader_defs.rs +++ b/examples/shader/shader_defs.rs @@ -97,7 +97,7 @@ fn setup( }); // Create a cube mesh which will use our materials - let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 1.0 })); + let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); commands // cube From b95c518ba1217e06c9225c7c231c09760da2df2e Mon Sep 17 00:00:00 2001 From: Carter Anderson Date: Sat, 21 Nov 2020 12:21:54 -0800 Subject: [PATCH 2/2] Cuboid -> Box, re-add cube, move shapes to their own file --- crates/bevy_render/src/mesh/mesh.rs | 306 -------------------- crates/bevy_render/src/mesh/mod.rs | 2 + crates/bevy_render/src/mesh/shape.rs | 324 ++++++++++++++++++++++ examples/3d/3d_scene.rs | 8 +- examples/3d/msaa.rs | 2 +- examples/3d/parenting.rs | 2 +- examples/3d/spawner.rs | 2 +- examples/3d/z_sort_debug.rs | 2 +- examples/android/android.rs | 8 +- examples/ios/src/lib.rs | 8 +- examples/shader/mesh_custom_attribute.rs | 2 +- examples/shader/shader_custom_material.rs | 2 +- examples/shader/shader_defs.rs | 2 +- 13 files changed, 345 insertions(+), 325 deletions(-) create mode 100644 crates/bevy_render/src/mesh/shape.rs diff --git a/crates/bevy_render/src/mesh/mesh.rs b/crates/bevy_render/src/mesh/mesh.rs index 3bd72cfb0759b..ceb001bffc053 100644 --- a/crates/bevy_render/src/mesh/mesh.rs +++ b/crates/bevy_render/src/mesh/mesh.rs @@ -220,312 +220,6 @@ impl Mesh { } } -/// Generation for some primitive shape meshes. -pub mod shape { - use super::{Indices, Mesh}; - use crate::pipeline::PrimitiveTopology; - use bevy_math::*; - use hexasphere::Hexasphere; - - pub struct Cuboid { - pub min_x: f32, - pub max_x: f32, - - pub min_y: f32, - pub max_y: f32, - - pub min_z: f32, - pub max_z: f32, - } - - impl Cuboid { - pub fn new(x_length: f32, y_length: f32, z_length: f32) -> Cuboid { - Cuboid { - max_x: x_length / 2.0, - min_x: -x_length / 2.0, - max_y: y_length / 2.0, - min_y: -y_length / 2.0, - max_z: z_length / 2.0, - min_z: -z_length / 2.0, - } - } - } - - impl Default for Cuboid { - fn default() -> Self { - Cuboid::new(2.0, 1.0, 1.0) - } - } - - impl From for Mesh { - fn from(sp: Cuboid) -> Self { - let vertices = &[ - // Top - ([sp.min_x, sp.min_y, sp.max_z], [0., 0., 1.0], [0., 0.]), - ([sp.max_x, sp.min_y, sp.max_z], [0., 0., 1.0], [1.0, 0.]), - ([sp.max_x, sp.max_y, sp.max_z], [0., 0., 1.0], [1.0, 1.0]), - ([sp.min_x, sp.max_y, sp.max_z], [0., 0., 1.0], [0., 1.0]), - // Bottom - ([sp.min_x, sp.max_y, sp.min_z], [0., 0., -1.0], [1.0, 0.]), - ([sp.max_x, sp.max_y, sp.min_z], [0., 0., -1.0], [0., 0.]), - ([sp.max_x, sp.min_y, sp.min_z], [0., 0., -1.0], [0., 1.0]), - ([sp.min_x, sp.min_y, sp.min_z], [0., 0., -1.0], [1.0, 1.0]), - // Right - ([sp.max_x, sp.min_y, sp.min_z], [1.0, 0., 0.], [0., 0.]), - ([sp.max_x, sp.max_y, sp.min_z], [1.0, 0., 0.], [1.0, 0.]), - ([sp.max_x, sp.max_y, sp.max_z], [1.0, 0., 0.], [1.0, 1.0]), - ([sp.max_x, sp.min_y, sp.max_z], [1.0, 0., 0.], [0., 1.0]), - // Left - ([sp.min_x, sp.min_y, sp.max_z], [-1.0, 0., 0.], [1.0, 0.]), - ([sp.min_x, sp.max_y, sp.max_z], [-1.0, 0., 0.], [0., 0.]), - ([sp.min_x, sp.max_y, sp.min_z], [-1.0, 0., 0.], [0., 1.0]), - ([sp.min_x, sp.min_y, sp.min_z], [-1.0, 0., 0.], [1.0, 1.0]), - // Front - ([sp.max_x, sp.max_y, sp.min_z], [0., 1.0, 0.], [1.0, 0.]), - ([sp.min_x, sp.max_y, sp.min_z], [0., 1.0, 0.], [0., 0.]), - ([sp.min_x, sp.max_y, sp.max_z], [0., 1.0, 0.], [0., 1.0]), - ([sp.max_x, sp.max_y, sp.max_z], [0., 1.0, 0.], [1.0, 1.0]), - // Back - ([sp.max_x, sp.min_y, sp.max_z], [0., -1.0, 0.], [0., 0.]), - ([sp.min_x, sp.min_y, sp.max_z], [0., -1.0, 0.], [1.0, 0.]), - ([sp.min_x, sp.min_y, sp.min_z], [0., -1.0, 0.], [1.0, 1.0]), - ([sp.max_x, sp.min_y, sp.min_z], [0., -1.0, 0.], [0., 1.0]), - ]; - - let mut positions = Vec::with_capacity(24); - let mut normals = Vec::with_capacity(24); - let mut uvs = Vec::with_capacity(24); - - for (position, normal, uv) in vertices.iter() { - positions.push(*position); - normals.push(*normal); - uvs.push(*uv); - } - - let indices = Indices::U32(vec![ - 0, 1, 2, 2, 3, 0, // top - 4, 5, 6, 6, 7, 4, // bottom - 8, 9, 10, 10, 11, 8, // right - 12, 13, 14, 14, 15, 12, // left - 16, 17, 18, 18, 19, 16, // front - 20, 21, 22, 22, 23, 20, // back - ]); - - let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); - mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, positions); - mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); - mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); - mesh.set_indices(Some(indices)); - mesh - } - } - - /// A rectangle on the XY plane. - #[derive(Debug)] - pub struct Quad { - /// Full width and height of the rectangle. - pub size: Vec2, - /// Flips the texture coords of the resulting vertices. - pub flip: bool, - } - - impl Quad { - pub fn new(size: Vec2) -> Self { - Self { size, flip: false } - } - - pub fn flipped(size: Vec2) -> Self { - Self { size, flip: true } - } - } - - impl From for Mesh { - fn from(quad: Quad) -> Self { - let extent_x = quad.size.x / 2.0; - let extent_y = quad.size.y / 2.0; - - let north_west = vec2(-extent_x, extent_y); - let north_east = vec2(extent_x, extent_y); - let south_west = vec2(-extent_x, -extent_y); - let south_east = vec2(extent_x, -extent_y); - let vertices = if quad.flip { - [ - ( - [south_east.x, south_east.y, 0.0], - [0.0, 0.0, 1.0], - [1.0, 1.0], - ), - ( - [north_east.x, north_east.y, 0.0], - [0.0, 0.0, 1.0], - [1.0, 0.0], - ), - ( - [north_west.x, north_west.y, 0.0], - [0.0, 0.0, 1.0], - [0.0, 0.0], - ), - ( - [south_west.x, south_west.y, 0.0], - [0.0, 0.0, 1.0], - [0.0, 1.0], - ), - ] - } else { - [ - ( - [south_west.x, south_west.y, 0.0], - [0.0, 0.0, 1.0], - [0.0, 1.0], - ), - ( - [north_west.x, north_west.y, 0.0], - [0.0, 0.0, 1.0], - [0.0, 0.0], - ), - ( - [north_east.x, north_east.y, 0.0], - [0.0, 0.0, 1.0], - [1.0, 0.0], - ), - ( - [south_east.x, south_east.y, 0.0], - [0.0, 0.0, 1.0], - [1.0, 1.0], - ), - ] - }; - - let indices = Indices::U32(vec![0, 2, 1, 0, 3, 2]); - - let mut positions = Vec::<[f32; 3]>::new(); - let mut normals = Vec::<[f32; 3]>::new(); - let mut uvs = Vec::<[f32; 2]>::new(); - for (position, normal, uv) in vertices.iter() { - positions.push(*position); - normals.push(*normal); - uvs.push(*uv); - } - - let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); - mesh.set_indices(Some(indices)); - mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, positions); - mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); - mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); - mesh - } - } - - /// A square on the XZ plane. - #[derive(Debug)] - pub struct Plane { - /// The total side length of the square. - pub size: f32, - } - - impl From for Mesh { - fn from(plane: Plane) -> Self { - let extent = plane.size / 2.0; - - let vertices = [ - ([extent, 0.0, -extent], [0.0, 1.0, 0.0], [1.0, 1.0]), - ([extent, 0.0, extent], [0.0, 1.0, 0.0], [1.0, 0.0]), - ([-extent, 0.0, extent], [0.0, 1.0, 0.0], [0.0, 0.0]), - ([-extent, 0.0, -extent], [0.0, 1.0, 0.0], [0.0, 1.0]), - ]; - - let indices = Indices::U32(vec![0, 2, 1, 0, 3, 2]); - - let mut positions = Vec::new(); - let mut normals = Vec::new(); - let mut uvs = Vec::new(); - for (position, normal, uv) in vertices.iter() { - positions.push(*position); - normals.push(*normal); - uvs.push(*uv); - } - - let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); - mesh.set_indices(Some(indices)); - mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, positions); - mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); - mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); - mesh - } - } - - /// A sphere made from a subdivided Icosahedron. - #[derive(Debug)] - pub struct Icosphere { - /// The radius of the sphere. - pub radius: f32, - /// The number of subdivisions applied. - pub subdivisions: usize, - } - - impl Default for Icosphere { - fn default() -> Self { - Self { - radius: 1.0, - subdivisions: 5, - } - } - } - - impl From for Mesh { - fn from(sphere: Icosphere) -> Self { - if sphere.subdivisions >= 80 { - let temp_sphere = Hexasphere::new(sphere.subdivisions, |_| ()); - - panic!( - "Cannot create an icosphere of {} subdivisions due to there being too many vertices being generated: {} (Limited to 65535 vertices or 79 subdivisions)", - sphere.subdivisions, - temp_sphere.raw_points().len() - ); - } - let hexasphere = Hexasphere::new(sphere.subdivisions, |point| { - let inclination = point.z.acos(); - let azumith = point.y.atan2(point.x); - - let norm_inclination = 1.0 - (inclination / std::f32::consts::PI); - let norm_azumith = (azumith / std::f32::consts::PI) * 0.5; - - [norm_inclination, norm_azumith] - }); - - let raw_points = hexasphere.raw_points(); - - let points = raw_points - .iter() - .map(|&p| (p * sphere.radius).into()) - .collect::>(); - - let normals = raw_points - .iter() - .copied() - .map(Into::into) - .collect::>(); - - let uvs = hexasphere.raw_data().to_owned(); - - let mut indices = Vec::with_capacity(hexasphere.indices_per_main_triangle() * 20); - - for i in 0..20 { - hexasphere.get_indices(i, &mut indices); - } - - let indices = Indices::U32(indices); - - let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); - mesh.set_indices(Some(indices)); - mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, points); - mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); - mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); - mesh - } - } -} - fn remove_resource_save( render_resource_context: &dyn RenderResourceContext, handle: &Handle, diff --git a/crates/bevy_render/src/mesh/mod.rs b/crates/bevy_render/src/mesh/mod.rs index 1db18c486545b..4ed2fc54ef70b 100644 --- a/crates/bevy_render/src/mesh/mod.rs +++ b/crates/bevy_render/src/mesh/mod.rs @@ -1,4 +1,6 @@ #[allow(clippy::module_inception)] mod mesh; +/// Generation for some primitive shape meshes. +pub mod shape; pub use mesh::*; diff --git a/crates/bevy_render/src/mesh/shape.rs b/crates/bevy_render/src/mesh/shape.rs new file mode 100644 index 0000000000000..c29323d9979f8 --- /dev/null +++ b/crates/bevy_render/src/mesh/shape.rs @@ -0,0 +1,324 @@ +use super::{Indices, Mesh}; +use crate::pipeline::PrimitiveTopology; +use bevy_math::*; +use hexasphere::Hexasphere; + +pub struct Cube { + pub size: f32, +} + +impl Cube { + pub fn new(size: f32) -> Cube { + Cube { size } + } +} + +impl Default for Cube { + fn default() -> Self { + Cube { size: 1.0 } + } +} + +impl From for Mesh { + fn from(cube: Cube) -> Self { + Box::new(cube.size, cube.size, cube.size).into() + } +} + +pub struct Box { + pub min_x: f32, + pub max_x: f32, + + pub min_y: f32, + pub max_y: f32, + + pub min_z: f32, + pub max_z: f32, +} + +impl Box { + pub fn new(x_length: f32, y_length: f32, z_length: f32) -> Box { + Box { + max_x: x_length / 2.0, + min_x: -x_length / 2.0, + max_y: y_length / 2.0, + min_y: -y_length / 2.0, + max_z: z_length / 2.0, + min_z: -z_length / 2.0, + } + } +} + +impl Default for Box { + fn default() -> Self { + Box::new(2.0, 1.0, 1.0) + } +} + +impl From for Mesh { + fn from(sp: Box) -> Self { + let vertices = &[ + // Top + ([sp.min_x, sp.min_y, sp.max_z], [0., 0., 1.0], [0., 0.]), + ([sp.max_x, sp.min_y, sp.max_z], [0., 0., 1.0], [1.0, 0.]), + ([sp.max_x, sp.max_y, sp.max_z], [0., 0., 1.0], [1.0, 1.0]), + ([sp.min_x, sp.max_y, sp.max_z], [0., 0., 1.0], [0., 1.0]), + // Bottom + ([sp.min_x, sp.max_y, sp.min_z], [0., 0., -1.0], [1.0, 0.]), + ([sp.max_x, sp.max_y, sp.min_z], [0., 0., -1.0], [0., 0.]), + ([sp.max_x, sp.min_y, sp.min_z], [0., 0., -1.0], [0., 1.0]), + ([sp.min_x, sp.min_y, sp.min_z], [0., 0., -1.0], [1.0, 1.0]), + // Right + ([sp.max_x, sp.min_y, sp.min_z], [1.0, 0., 0.], [0., 0.]), + ([sp.max_x, sp.max_y, sp.min_z], [1.0, 0., 0.], [1.0, 0.]), + ([sp.max_x, sp.max_y, sp.max_z], [1.0, 0., 0.], [1.0, 1.0]), + ([sp.max_x, sp.min_y, sp.max_z], [1.0, 0., 0.], [0., 1.0]), + // Left + ([sp.min_x, sp.min_y, sp.max_z], [-1.0, 0., 0.], [1.0, 0.]), + ([sp.min_x, sp.max_y, sp.max_z], [-1.0, 0., 0.], [0., 0.]), + ([sp.min_x, sp.max_y, sp.min_z], [-1.0, 0., 0.], [0., 1.0]), + ([sp.min_x, sp.min_y, sp.min_z], [-1.0, 0., 0.], [1.0, 1.0]), + // Front + ([sp.max_x, sp.max_y, sp.min_z], [0., 1.0, 0.], [1.0, 0.]), + ([sp.min_x, sp.max_y, sp.min_z], [0., 1.0, 0.], [0., 0.]), + ([sp.min_x, sp.max_y, sp.max_z], [0., 1.0, 0.], [0., 1.0]), + ([sp.max_x, sp.max_y, sp.max_z], [0., 1.0, 0.], [1.0, 1.0]), + // Back + ([sp.max_x, sp.min_y, sp.max_z], [0., -1.0, 0.], [0., 0.]), + ([sp.min_x, sp.min_y, sp.max_z], [0., -1.0, 0.], [1.0, 0.]), + ([sp.min_x, sp.min_y, sp.min_z], [0., -1.0, 0.], [1.0, 1.0]), + ([sp.max_x, sp.min_y, sp.min_z], [0., -1.0, 0.], [0., 1.0]), + ]; + + let mut positions = Vec::with_capacity(24); + let mut normals = Vec::with_capacity(24); + let mut uvs = Vec::with_capacity(24); + + for (position, normal, uv) in vertices.iter() { + positions.push(*position); + normals.push(*normal); + uvs.push(*uv); + } + + let indices = Indices::U32(vec![ + 0, 1, 2, 2, 3, 0, // top + 4, 5, 6, 6, 7, 4, // bottom + 8, 9, 10, 10, 11, 8, // right + 12, 13, 14, 14, 15, 12, // left + 16, 17, 18, 18, 19, 16, // front + 20, 21, 22, 22, 23, 20, // back + ]); + + let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); + mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, positions); + mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); + mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); + mesh.set_indices(Some(indices)); + mesh + } +} + +/// A rectangle on the XY plane. +#[derive(Debug)] +pub struct Quad { + /// Full width and height of the rectangle. + pub size: Vec2, + /// Flips the texture coords of the resulting vertices. + pub flip: bool, +} + +impl Quad { + pub fn new(size: Vec2) -> Self { + Self { size, flip: false } + } + + pub fn flipped(size: Vec2) -> Self { + Self { size, flip: true } + } +} + +impl From for Mesh { + fn from(quad: Quad) -> Self { + let extent_x = quad.size.x / 2.0; + let extent_y = quad.size.y / 2.0; + + let north_west = vec2(-extent_x, extent_y); + let north_east = vec2(extent_x, extent_y); + let south_west = vec2(-extent_x, -extent_y); + let south_east = vec2(extent_x, -extent_y); + let vertices = if quad.flip { + [ + ( + [south_east.x, south_east.y, 0.0], + [0.0, 0.0, 1.0], + [1.0, 1.0], + ), + ( + [north_east.x, north_east.y, 0.0], + [0.0, 0.0, 1.0], + [1.0, 0.0], + ), + ( + [north_west.x, north_west.y, 0.0], + [0.0, 0.0, 1.0], + [0.0, 0.0], + ), + ( + [south_west.x, south_west.y, 0.0], + [0.0, 0.0, 1.0], + [0.0, 1.0], + ), + ] + } else { + [ + ( + [south_west.x, south_west.y, 0.0], + [0.0, 0.0, 1.0], + [0.0, 1.0], + ), + ( + [north_west.x, north_west.y, 0.0], + [0.0, 0.0, 1.0], + [0.0, 0.0], + ), + ( + [north_east.x, north_east.y, 0.0], + [0.0, 0.0, 1.0], + [1.0, 0.0], + ), + ( + [south_east.x, south_east.y, 0.0], + [0.0, 0.0, 1.0], + [1.0, 1.0], + ), + ] + }; + + let indices = Indices::U32(vec![0, 2, 1, 0, 3, 2]); + + let mut positions = Vec::<[f32; 3]>::new(); + let mut normals = Vec::<[f32; 3]>::new(); + let mut uvs = Vec::<[f32; 2]>::new(); + for (position, normal, uv) in vertices.iter() { + positions.push(*position); + normals.push(*normal); + uvs.push(*uv); + } + + let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); + mesh.set_indices(Some(indices)); + mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, positions); + mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); + mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); + mesh + } +} + +/// A square on the XZ plane. +#[derive(Debug)] +pub struct Plane { + /// The total side length of the square. + pub size: f32, +} + +impl From for Mesh { + fn from(plane: Plane) -> Self { + let extent = plane.size / 2.0; + + let vertices = [ + ([extent, 0.0, -extent], [0.0, 1.0, 0.0], [1.0, 1.0]), + ([extent, 0.0, extent], [0.0, 1.0, 0.0], [1.0, 0.0]), + ([-extent, 0.0, extent], [0.0, 1.0, 0.0], [0.0, 0.0]), + ([-extent, 0.0, -extent], [0.0, 1.0, 0.0], [0.0, 1.0]), + ]; + + let indices = Indices::U32(vec![0, 2, 1, 0, 3, 2]); + + let mut positions = Vec::new(); + let mut normals = Vec::new(); + let mut uvs = Vec::new(); + for (position, normal, uv) in vertices.iter() { + positions.push(*position); + normals.push(*normal); + uvs.push(*uv); + } + + let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); + mesh.set_indices(Some(indices)); + mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, positions); + mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); + mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); + mesh + } +} + +/// A sphere made from a subdivided Icosahedron. +#[derive(Debug)] +pub struct Icosphere { + /// The radius of the sphere. + pub radius: f32, + /// The number of subdivisions applied. + pub subdivisions: usize, +} + +impl Default for Icosphere { + fn default() -> Self { + Self { + radius: 1.0, + subdivisions: 5, + } + } +} + +impl From for Mesh { + fn from(sphere: Icosphere) -> Self { + if sphere.subdivisions >= 80 { + let temp_sphere = Hexasphere::new(sphere.subdivisions, |_| ()); + + panic!( + "Cannot create an icosphere of {} subdivisions due to there being too many vertices being generated: {} (Limited to 65535 vertices or 79 subdivisions)", + sphere.subdivisions, + temp_sphere.raw_points().len() + ); + } + let hexasphere = Hexasphere::new(sphere.subdivisions, |point| { + let inclination = point.z.acos(); + let azumith = point.y.atan2(point.x); + + let norm_inclination = 1.0 - (inclination / std::f32::consts::PI); + let norm_azumith = (azumith / std::f32::consts::PI) * 0.5; + + [norm_inclination, norm_azumith] + }); + + let raw_points = hexasphere.raw_points(); + + let points = raw_points + .iter() + .map(|&p| (p * sphere.radius).into()) + .collect::>(); + + let normals = raw_points + .iter() + .copied() + .map(Into::into) + .collect::>(); + + let uvs = hexasphere.raw_data().to_owned(); + + let mut indices = Vec::with_capacity(hexasphere.indices_per_main_triangle() * 20); + + for i in 0..20 { + hexasphere.get_indices(i, &mut indices); + } + + let indices = Indices::U32(indices); + + let mut mesh = Mesh::new(PrimitiveTopology::TriangleList); + mesh.set_indices(Some(indices)); + mesh.set_attribute(Mesh::ATTRIBUTE_POSITION, points); + mesh.set_attribute(Mesh::ATTRIBUTE_NORMAL, normals); + mesh.set_attribute(Mesh::ATTRIBUTE_UV_0, uvs); + mesh + } +} diff --git a/examples/3d/3d_scene.rs b/examples/3d/3d_scene.rs index 10652be06dbd3..3ea3492f6bfb5 100644 --- a/examples/3d/3d_scene.rs +++ b/examples/3d/3d_scene.rs @@ -18,15 +18,15 @@ fn setup( commands // plane .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Plane { size: 10.0 })), + mesh: meshes.add(Mesh::from(shape::Plane { size: 5.0 })), material: materials.add(Color::rgb(0.3, 0.5, 0.3).into()), ..Default::default() }) // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), + mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), material: materials.add(Color::rgb(0.8, 0.7, 0.6).into()), - transform: Transform::from_translation(Vec3::new(0.0, 1.0, 0.0)), + transform: Transform::from_translation(Vec3::new(0.0, 0.5, 0.0)), ..Default::default() }) // light @@ -36,7 +36,7 @@ fn setup( }) // camera .spawn(Camera3dBundle { - transform: Transform::from_translation(Vec3::new(-3.0, 5.0, 8.0)) + transform: Transform::from_translation(Vec3::new(-2.0, 2.5, 5.0)) .looking_at(Vec3::default(), Vec3::unit_y()), ..Default::default() }); diff --git a/examples/3d/msaa.rs b/examples/3d/msaa.rs index d83f492a7b523..d6dbea34b8e84 100644 --- a/examples/3d/msaa.rs +++ b/examples/3d/msaa.rs @@ -21,7 +21,7 @@ fn setup( commands // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), + mesh: meshes.add(Mesh::from(shape::Cube { size: 2.0 })), material: materials.add(Color::rgb(0.8, 0.7, 0.6).into()), ..Default::default() }) diff --git a/examples/3d/parenting.rs b/examples/3d/parenting.rs index 92cf3b166ba8a..a1125e5762df5 100644 --- a/examples/3d/parenting.rs +++ b/examples/3d/parenting.rs @@ -27,7 +27,7 @@ fn setup( mut meshes: ResMut>, mut materials: ResMut>, ) { - let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); + let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 2.0 })); let cube_material_handle = materials.add(StandardMaterial { albedo: Color::rgb(0.8, 0.7, 0.6), ..Default::default() diff --git a/examples/3d/spawner.rs b/examples/3d/spawner.rs index d16ef7af45edf..20a7552bc43df 100644 --- a/examples/3d/spawner.rs +++ b/examples/3d/spawner.rs @@ -50,7 +50,7 @@ fn setup( }); let mut rng = StdRng::from_entropy(); - let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); + let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 1.0 })); for _ in 0..10000 { commands.spawn(PbrBundle { mesh: cube_handle.clone(), diff --git a/examples/3d/z_sort_debug.rs b/examples/3d/z_sort_debug.rs index 2ad81c3d97d4a..73cc3c4ea5ec6 100644 --- a/examples/3d/z_sort_debug.rs +++ b/examples/3d/z_sort_debug.rs @@ -46,7 +46,7 @@ fn setup( mut meshes: ResMut>, mut materials: ResMut>, ) { - let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); + let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 2.0 })); commands // parent cube .spawn(PbrBundle { diff --git a/examples/android/android.rs b/examples/android/android.rs index 05ae3f4b3e27f..0fa98018fda59 100644 --- a/examples/android/android.rs +++ b/examples/android/android.rs @@ -20,15 +20,15 @@ fn setup( commands // plane .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Plane { size: 10.0 })), + mesh: meshes.add(Mesh::from(shape::Plane { size: 5.0 })), material: materials.add(Color::rgb(0.3, 0.5, 0.3).into()), ..Default::default() }) // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), + mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), material: materials.add(Color::rgb(0.8, 0.7, 0.6).into()), - transform: Transform::from_translation(Vec3::new(0.0, 1.0, 0.0)), + transform: Transform::from_translation(Vec3::new(0.0, 0.5, 0.0)), ..Default::default() }) // light @@ -38,7 +38,7 @@ fn setup( }) // camera .spawn(Camera3dBundle { - transform: Transform::from_translation(Vec3::new(-3.0, 5.0, 8.0)) + transform: Transform::from_translation(Vec3::new(-2.0, 2.5, 5.0)) .looking_at(Vec3::default(), Vec3::unit_y()), ..Default::default() }); diff --git a/examples/ios/src/lib.rs b/examples/ios/src/lib.rs index 63631edfc038e..803752e0d6672 100644 --- a/examples/ios/src/lib.rs +++ b/examples/ios/src/lib.rs @@ -25,15 +25,15 @@ fn setup( commands // plane .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Plane { size: 10.0 })), + mesh: meshes.add(Mesh::from(shape::Plane { size: 5.0 })), material: materials.add(Color::rgb(0.1, 0.2, 0.1).into()), ..Default::default() }) // cube .spawn(PbrBundle { - mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), + mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })), material: materials.add(Color::rgb(0.5, 0.4, 0.3).into()), - transform: Transform::from_translation(Vec3::new(0.0, 1.0, 0.0)), + transform: Transform::from_translation(Vec3::new(0.0, 0.5, 0.0)), ..Default::default() }) // sphere @@ -53,7 +53,7 @@ fn setup( }) // camera .spawn(Camera3dBundle { - transform: Transform::from_translation(Vec3::new(-3.0, 5.0, 8.0)) + transform: Transform::from_translation(Vec3::new(-2.0, 2.5, 5.0)) .looking_at(Vec3::default(), Vec3::unit_y()), ..Default::default() }); diff --git a/examples/shader/mesh_custom_attribute.rs b/examples/shader/mesh_custom_attribute.rs index 14648e0d70572..98ea547636459 100644 --- a/examples/shader/mesh_custom_attribute.rs +++ b/examples/shader/mesh_custom_attribute.rs @@ -83,7 +83,7 @@ fn setup( let material = materials.add(MyMaterialWithVertexColorSupport {}); // create a generic cube - let mut cube_with_vertex_colors = Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0)); + let mut cube_with_vertex_colors = Mesh::from(shape::Cube { size: 2.0 }); // insert our custom color attribute with some nice colors! cube_with_vertex_colors.set_attribute( diff --git a/examples/shader/shader_custom_material.rs b/examples/shader/shader_custom_material.rs index c1b9c7af17bd6..ddaae959ac848 100644 --- a/examples/shader/shader_custom_material.rs +++ b/examples/shader/shader_custom_material.rs @@ -84,7 +84,7 @@ fn setup( commands // cube .spawn(MeshBundle { - mesh: meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))), + mesh: meshes.add(Mesh::from(shape::Cube { size: 2.0 })), render_pipelines: RenderPipelines::from_pipelines(vec![RenderPipeline::new( pipeline_handle, )]), diff --git a/examples/shader/shader_defs.rs b/examples/shader/shader_defs.rs index ae76c2187537b..f3426160513bd 100644 --- a/examples/shader/shader_defs.rs +++ b/examples/shader/shader_defs.rs @@ -97,7 +97,7 @@ fn setup( }); // Create a cube mesh which will use our materials - let cube_handle = meshes.add(Mesh::from(shape::Cuboid::new(1.0, 1.0, 1.0))); + let cube_handle = meshes.add(Mesh::from(shape::Cube { size: 2.0 })); commands // cube