mirror of
https://github.com/bevyengine/bevy
synced 2024-11-26 06:30:19 +00:00
118 lines
3.7 KiB
Rust
118 lines
3.7 KiB
Rust
use bevy::{
|
|
prelude::*,
|
|
render::{
|
|
mesh::shape,
|
|
pipeline::{DynamicBinding, PipelineDescriptor, PipelineSpecialization, RenderPipeline},
|
|
render_graph::{base, AssetRenderResourcesNode, RenderGraph},
|
|
renderer::RenderResources,
|
|
shader::{ShaderStage, ShaderStages},
|
|
},
|
|
};
|
|
|
|
/// This example illustrates how to create a custom material asset and a shader that uses that uses that material
|
|
fn main() {
|
|
App::build()
|
|
.add_default_plugins()
|
|
.add_asset::<MyMaterial>()
|
|
.add_startup_system(setup.system())
|
|
.run();
|
|
}
|
|
|
|
#[derive(RenderResources, Default)]
|
|
struct MyMaterial {
|
|
pub color: Color,
|
|
}
|
|
|
|
const VERTEX_SHADER: &str = r#"
|
|
#version 450
|
|
layout(location = 0) in vec3 Vertex_Position;
|
|
layout(set = 0, binding = 0) uniform Camera {
|
|
mat4 ViewProj;
|
|
};
|
|
layout(set = 1, binding = 0) uniform Transform {
|
|
mat4 Model;
|
|
};
|
|
void main() {
|
|
gl_Position = ViewProj * Model * vec4(Vertex_Position, 1.0);
|
|
}
|
|
"#;
|
|
|
|
const FRAGMENT_SHADER: &str = r#"
|
|
#version 450
|
|
layout(location = 0) out vec4 o_Target;
|
|
layout(set = 1, binding = 1) uniform MyMaterial_color {
|
|
vec4 color;
|
|
};
|
|
void main() {
|
|
o_Target = color;
|
|
}
|
|
"#;
|
|
|
|
fn setup(
|
|
mut commands: Commands,
|
|
mut pipelines: ResMut<Assets<PipelineDescriptor>>,
|
|
mut shaders: ResMut<Assets<Shader>>,
|
|
mut meshes: ResMut<Assets<Mesh>>,
|
|
mut materials: ResMut<Assets<MyMaterial>>,
|
|
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 MyMaterial resources to our shader
|
|
render_graph.add_system_node(
|
|
"my_material",
|
|
AssetRenderResourcesNode::<MyMaterial>::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", base::node::MAIN_PASS)
|
|
.unwrap();
|
|
|
|
// Create a new material
|
|
let material = materials.add(MyMaterial {
|
|
color: Color::rgb(0.0, 0.8, 0.0),
|
|
});
|
|
|
|
// Setup our world
|
|
commands
|
|
// cube
|
|
.spawn(MeshComponents {
|
|
mesh: meshes.add(Mesh::from(shape::Cube { size: 1.0 })),
|
|
render_pipelines: RenderPipelines::from_pipelines(vec![RenderPipeline::specialized(
|
|
pipeline_handle,
|
|
// NOTE: in the future you wont need to manually declare dynamic bindings
|
|
PipelineSpecialization {
|
|
dynamic_bindings: vec![
|
|
// Transform
|
|
DynamicBinding {
|
|
bind_group: 1,
|
|
binding: 0,
|
|
},
|
|
// MyMaterial_color
|
|
DynamicBinding {
|
|
bind_group: 1,
|
|
binding: 1,
|
|
},
|
|
],
|
|
..Default::default()
|
|
},
|
|
)]),
|
|
translation: Translation::new(0.0, 0.0, 0.0),
|
|
..Default::default()
|
|
})
|
|
.with(material)
|
|
// camera
|
|
.spawn(Camera3dComponents {
|
|
transform: Transform::new_sync_disabled(Mat4::face_toward(
|
|
Vec3::new(3.0, 5.0, -8.0),
|
|
Vec3::new(0.0, 0.0, 0.0),
|
|
Vec3::new(0.0, 1.0, 0.0),
|
|
)),
|
|
..Default::default()
|
|
});
|
|
}
|