mirror of
https://github.com/bevyengine/bevy
synced 2024-11-22 20:53:53 +00:00
dd4a196329
Alternative to #1203 and #1611 Camera bindings have historically been "hacked in". They were _required_ in all shaders and only supported a single Mat4. PBR (#1554) requires the CameraView matrix, but adding this using the "hacked" method forced users to either include all possible camera data in a single binding (#1203) or include all possible bindings (#1611). This approach instead assigns each "active camera" its own RenderResourceBindings, which are populated by CameraNode. The PassNode then retrieves (and initializes) the relevant bind groups for all render pipelines used by visible entities. * Enables any number of camera bindings , including zero (with any set or binding number ... set 0 should still be used to avoid rebinds). * Renames Camera binding to CameraViewProj * Adds CameraView binding
147 lines
4.6 KiB
Rust
147 lines
4.6 KiB
Rust
use bevy::{
|
|
prelude::*,
|
|
reflect::TypeUuid,
|
|
render::{
|
|
mesh::{shape, VertexAttributeValues},
|
|
pipeline::{PipelineDescriptor, RenderPipeline},
|
|
render_graph::{base, AssetRenderResourcesNode, RenderGraph},
|
|
renderer::RenderResources,
|
|
shader::{ShaderStage, ShaderStages},
|
|
},
|
|
};
|
|
|
|
/// This example illustrates how to add a custom attribute to a mesh and use it in a custom shader.
|
|
fn main() {
|
|
App::build()
|
|
.add_plugins(DefaultPlugins)
|
|
.add_asset::<MyMaterialWithVertexColorSupport>()
|
|
.add_startup_system(setup.system())
|
|
.run();
|
|
}
|
|
|
|
#[derive(RenderResources, Default, TypeUuid)]
|
|
#[uuid = "0320b9b8-b3a3-4baa-8bfa-c94008177b17"]
|
|
struct MyMaterialWithVertexColorSupport;
|
|
|
|
const VERTEX_SHADER: &str = r#"
|
|
#version 450
|
|
layout(location = 0) in vec3 Vertex_Position;
|
|
layout(location = 1) in vec3 Vertex_Color;
|
|
layout(location = 0) out vec3 v_color;
|
|
|
|
layout(set = 0, binding = 0) uniform CameraViewProj {
|
|
mat4 ViewProj;
|
|
};
|
|
layout(set = 1, binding = 0) uniform Transform {
|
|
mat4 Model;
|
|
};
|
|
void main() {
|
|
gl_Position = ViewProj * Model * vec4(Vertex_Position, 1.0);
|
|
v_color = Vertex_Color;
|
|
}
|
|
"#;
|
|
|
|
const FRAGMENT_SHADER: &str = r#"
|
|
#version 450
|
|
layout(location = 0) out vec4 o_Target;
|
|
layout(location = 0) in vec3 v_color;
|
|
|
|
void main() {
|
|
o_Target = vec4(v_color, 1.0);
|
|
}
|
|
"#;
|
|
|
|
fn setup(
|
|
mut commands: Commands,
|
|
mut pipelines: ResMut<Assets<PipelineDescriptor>>,
|
|
mut shaders: ResMut<Assets<Shader>>,
|
|
mut meshes: ResMut<Assets<Mesh>>,
|
|
mut materials: ResMut<Assets<MyMaterialWithVertexColorSupport>>,
|
|
mut render_graph: ResMut<RenderGraph>,
|
|
) {
|
|
// Create a new shader pipeline
|
|
let pipeline_handle = pipelines.add(PipelineDescriptor::default_config(ShaderStages {
|
|
vertex: shaders.add(Shader::from_glsl(ShaderStage::Vertex, VERTEX_SHADER)),
|
|
fragment: Some(shaders.add(Shader::from_glsl(ShaderStage::Fragment, FRAGMENT_SHADER))),
|
|
}));
|
|
|
|
// Add an AssetRenderResourcesNode to our Render Graph. This will bind
|
|
// MyMaterialWithVertexColorSupport resources to our shader
|
|
render_graph.add_system_node(
|
|
"my_material_with_vertex_color_support",
|
|
AssetRenderResourcesNode::<MyMaterialWithVertexColorSupport>::new(true),
|
|
);
|
|
|
|
// Add a Render Graph edge connecting our new "my_material" node to the main pass node. This
|
|
// ensures "my_material" runs before the main pass
|
|
render_graph
|
|
.add_node_edge(
|
|
"my_material_with_vertex_color_support",
|
|
base::node::MAIN_PASS,
|
|
)
|
|
.unwrap();
|
|
|
|
// Create a new material
|
|
let material = materials.add(MyMaterialWithVertexColorSupport {});
|
|
|
|
// create a generic cube
|
|
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(
|
|
// name of the attribute
|
|
"Vertex_Color",
|
|
// the vertex attributes, represented by `VertexAttributeValues`
|
|
// NOTE: the attribute count has to be consistent across all attributes, otherwise bevy
|
|
// will panic.
|
|
VertexAttributeValues::from(vec![
|
|
// top
|
|
[0.79, 0.73, 0.07],
|
|
[0.74, 0.14, 0.29],
|
|
[0.08, 0.55, 0.74],
|
|
[0.20, 0.27, 0.29],
|
|
// bottom
|
|
[0.79, 0.73, 0.07],
|
|
[0.74, 0.14, 0.29],
|
|
[0.08, 0.55, 0.74],
|
|
[0.20, 0.27, 0.29],
|
|
// right
|
|
[0.79, 0.73, 0.07],
|
|
[0.74, 0.14, 0.29],
|
|
[0.08, 0.55, 0.74],
|
|
[0.20, 0.27, 0.29],
|
|
// left
|
|
[0.79, 0.73, 0.07],
|
|
[0.74, 0.14, 0.29],
|
|
[0.08, 0.55, 0.74],
|
|
[0.20, 0.27, 0.29],
|
|
// front
|
|
[0.79, 0.73, 0.07],
|
|
[0.74, 0.14, 0.29],
|
|
[0.08, 0.55, 0.74],
|
|
[0.20, 0.27, 0.29],
|
|
// back
|
|
[0.79, 0.73, 0.07],
|
|
[0.74, 0.14, 0.29],
|
|
[0.08, 0.55, 0.74],
|
|
[0.20, 0.27, 0.29],
|
|
]),
|
|
);
|
|
// Setup our world
|
|
commands
|
|
// cube
|
|
.spawn(MeshBundle {
|
|
mesh: meshes.add(cube_with_vertex_colors), // use our cube with vertex colors
|
|
render_pipelines: RenderPipelines::from_pipelines(vec![RenderPipeline::new(
|
|
pipeline_handle,
|
|
)]),
|
|
transform: Transform::from_xyz(0.0, 0.0, 0.0),
|
|
..Default::default()
|
|
})
|
|
.with(material)
|
|
// camera
|
|
.spawn(PerspectiveCameraBundle {
|
|
transform: Transform::from_xyz(3.0, 5.0, -8.0).looking_at(Vec3::ZERO, Vec3::Y),
|
|
..Default::default()
|
|
});
|
|
}
|