bevy/crates/bevy_ui/src/render/mod.rs

156 lines
5.3 KiB
Rust
Raw Normal View History

2020-07-10 08:37:06 +00:00
use crate::Node;
2020-05-13 23:42:27 +00:00
use bevy_asset::{Assets, Handle};
2020-07-10 08:37:06 +00:00
use bevy_ecs::Resources;
2020-05-03 19:35:07 +00:00
use bevy_render::{
camera::ActiveCameras,
pass::{
LoadOp, Operations, PassDescriptor, RenderPassDepthStencilAttachmentDescriptor,
TextureAttachment,
},
pipeline::*,
2020-07-30 01:15:15 +00:00
prelude::Msaa,
render_graph::{
base, CameraNode, PassNode, RenderGraph, RenderResourcesNode, WindowSwapChainNode,
WindowTextureNode,
},
2020-05-03 19:35:07 +00:00
shader::{Shader, ShaderStage, ShaderStages},
texture::TextureFormat,
2020-05-03 19:35:07 +00:00
};
2020-05-03 19:35:07 +00:00
pub const UI_PIPELINE_HANDLE: Handle<PipelineDescriptor> =
2020-05-04 18:20:12 +00:00
Handle::from_u128(323432002226399387835192542539754486265);
2020-05-03 19:35:07 +00:00
2020-05-13 23:42:27 +00:00
pub fn build_ui_pipeline(shaders: &mut Assets<Shader>) -> PipelineDescriptor {
2020-05-03 19:35:07 +00:00
PipelineDescriptor {
rasterization_state: Some(RasterizationStateDescriptor {
front_face: FrontFace::Ccw,
cull_mode: CullMode::Back,
2020-05-03 19:35:07 +00:00
depth_bias: 0,
depth_bias_slope_scale: 0.0,
depth_bias_clamp: 0.0,
}),
depth_stencil_state: Some(DepthStencilStateDescriptor {
format: TextureFormat::Depth32Float,
depth_write_enabled: true,
depth_compare: CompareFunction::Less,
2020-05-03 19:35:07 +00:00
stencil_front: StencilStateFaceDescriptor::IGNORE,
stencil_back: StencilStateFaceDescriptor::IGNORE,
stencil_read_mask: 0,
stencil_write_mask: 0,
}),
color_states: vec![ColorStateDescriptor {
format: TextureFormat::Bgra8UnormSrgb,
color_blend: BlendDescriptor {
src_factor: BlendFactor::SrcAlpha,
dst_factor: BlendFactor::OneMinusSrcAlpha,
operation: BlendOperation::Add,
},
alpha_blend: BlendDescriptor {
src_factor: BlendFactor::One,
dst_factor: BlendFactor::One,
operation: BlendOperation::Add,
},
write_mask: ColorWrite::ALL,
}],
..PipelineDescriptor::new(ShaderStages {
vertex: shaders.add(Shader::from_glsl(
ShaderStage::Vertex,
include_str!("ui.vert"),
)),
fragment: Some(shaders.add(Shader::from_glsl(
ShaderStage::Fragment,
include_str!("ui.frag"),
))),
})
}
}
pub mod node {
pub const UI_CAMERA: &'static str = "ui_camera";
pub const NODE: &'static str = "node";
pub const UI_PASS: &'static str = "ui_pass";
}
pub mod camera {
pub const UI_CAMERA: &'static str = "UiCamera";
}
2020-05-03 19:35:07 +00:00
pub trait UiRenderGraphBuilder {
fn add_ui_graph(&mut self, resources: &Resources) -> &mut Self;
}
impl UiRenderGraphBuilder for RenderGraph {
fn add_ui_graph(&mut self, resources: &Resources) -> &mut Self {
2020-05-14 01:05:18 +00:00
let mut pipelines = resources.get_mut::<Assets<PipelineDescriptor>>().unwrap();
2020-05-13 23:42:27 +00:00
let mut shaders = resources.get_mut::<Assets<Shader>>().unwrap();
2020-07-30 01:15:15 +00:00
let msaa = resources.get::<Msaa>().unwrap();
2020-05-16 07:21:04 +00:00
pipelines.set(UI_PIPELINE_HANDLE, build_ui_pipeline(&mut shaders));
let mut ui_pass_node = PassNode::<&Node>::new(PassDescriptor {
color_attachments: vec![msaa.color_attachment_descriptor(
TextureAttachment::Input("color_attachment".to_string()),
TextureAttachment::Input("color_resolve_target".to_string()),
Operations {
load: LoadOp::Load,
store: true,
},
)],
depth_stencil_attachment: Some(RenderPassDepthStencilAttachmentDescriptor {
attachment: TextureAttachment::Input("depth".to_string()),
depth_ops: Some(Operations {
load: LoadOp::Clear(1.0),
store: true,
}),
stencil_ops: None,
}),
2020-07-30 01:15:15 +00:00
sample_count: msaa.samples,
});
ui_pass_node.add_camera(camera::UI_CAMERA);
self.add_node(node::UI_PASS, ui_pass_node);
self.add_slot_edge(
base::node::PRIMARY_SWAP_CHAIN,
WindowSwapChainNode::OUT_TEXTURE,
node::UI_PASS,
2020-07-30 01:15:15 +00:00
if msaa.samples > 1 {
"color_resolve_target"
} else {
"color_attachment"
},
)
.unwrap();
self.add_slot_edge(
base::node::MAIN_DEPTH_TEXTURE,
WindowTextureNode::OUT_TEXTURE,
node::UI_PASS,
"depth",
)
.unwrap();
2020-07-30 01:15:15 +00:00
if msaa.samples > 1 {
self.add_slot_edge(
base::node::MAIN_SAMPLED_COLOR_ATTACHMENT,
WindowSwapChainNode::OUT_TEXTURE,
node::UI_PASS,
"color_attachment",
)
.unwrap();
}
// ensure ui pass runs after main pass
self.add_node_edge(base::node::MAIN_PASS, node::UI_PASS)
.unwrap();
// setup ui camera
self.add_system_node(node::UI_CAMERA, CameraNode::new(camera::UI_CAMERA));
self.add_node_edge(node::UI_CAMERA, node::UI_PASS).unwrap();
self.add_system_node(node::NODE, RenderResourcesNode::<Node>::new(true));
self.add_node_edge(node::NODE, node::UI_PASS).unwrap();
let mut active_cameras = resources.get_mut::<ActiveCameras>().unwrap();
active_cameras.add(camera::UI_CAMERA);
2020-05-03 19:35:07 +00:00
self
}
2020-05-04 18:20:12 +00:00
}