mod pipeline; mod render_pass; use bevy_core_pipeline::{core_2d::Camera2d, core_3d::Camera3d}; use bevy_window::{PrimaryWindow, Window}; pub use pipeline::*; pub use render_pass::*; use crate::{prelude::UiCameraConfig, BackgroundColor, CalculatedClip, Node, UiImage, UiStack}; use bevy_app::prelude::*; use bevy_asset::{load_internal_asset, AssetEvent, Assets, Handle, HandleUntyped}; use bevy_ecs::prelude::*; use bevy_math::{Mat4, Rect, UVec4, Vec2, Vec3, Vec4Swizzles}; use bevy_reflect::TypeUuid; use bevy_render::texture::DEFAULT_IMAGE_HANDLE; use bevy_render::{ camera::Camera, color::Color, render_asset::RenderAssets, render_graph::{RenderGraph, RunGraphOnViewNode, SlotInfo, SlotType}, render_phase::{sort_phase_system, AddRenderCommand, DrawFunctions, RenderPhase}, render_resource::*, renderer::{RenderDevice, RenderQueue}, texture::Image, view::{ComputedVisibility, ExtractedView, ViewUniforms}, Extract, RenderApp, RenderStage, }; use bevy_sprite::{SpriteAssetEvents, TextureAtlas}; use bevy_text::{Text, TextLayoutInfo}; use bevy_transform::components::GlobalTransform; use bevy_utils::FloatOrd; use bevy_utils::HashMap; use bytemuck::{Pod, Zeroable}; use std::ops::Range; pub mod node { pub const UI_PASS_DRIVER: &str = "ui_pass_driver"; } pub mod draw_ui_graph { pub const NAME: &str = "draw_ui"; pub mod input { pub const VIEW_ENTITY: &str = "view_entity"; } pub mod node { pub const UI_PASS: &str = "ui_pass"; } } pub const UI_SHADER_HANDLE: HandleUntyped = HandleUntyped::weak_from_u64(Shader::TYPE_UUID, 13012847047162779583); #[derive(Debug, Hash, PartialEq, Eq, Clone, SystemLabel)] pub enum RenderUiSystem { ExtractNode, } pub fn build_ui_render(app: &mut App) { load_internal_asset!(app, UI_SHADER_HANDLE, "ui.wgsl", Shader::from_wgsl); let render_app = match app.get_sub_app_mut(RenderApp) { Ok(render_app) => render_app, Err(_) => return, }; render_app .init_resource::() .init_resource::>() .init_resource::() .init_resource::() .init_resource::() .init_resource::>() .add_render_command::() .add_system_to_stage( RenderStage::Extract, extract_default_ui_camera_view::, ) .add_system_to_stage( RenderStage::Extract, extract_default_ui_camera_view::, ) .add_system_to_stage( RenderStage::Extract, extract_uinodes.label(RenderUiSystem::ExtractNode), ) .add_system_to_stage( RenderStage::Extract, extract_text_uinodes.after(RenderUiSystem::ExtractNode), ) .add_system_to_stage(RenderStage::Prepare, prepare_uinodes) .add_system_to_stage(RenderStage::Queue, queue_uinodes) .add_system_to_stage(RenderStage::PhaseSort, sort_phase_system::); // Render graph let ui_graph_2d = get_ui_graph(render_app); let ui_graph_3d = get_ui_graph(render_app); let mut graph = render_app.world.resource_mut::(); if let Some(graph_2d) = graph.get_sub_graph_mut(bevy_core_pipeline::core_2d::graph::NAME) { graph_2d.add_sub_graph(draw_ui_graph::NAME, ui_graph_2d); graph_2d.add_node( draw_ui_graph::node::UI_PASS, RunGraphOnViewNode::new(draw_ui_graph::NAME), ); graph_2d.add_node_edge( bevy_core_pipeline::core_2d::graph::node::MAIN_PASS, draw_ui_graph::node::UI_PASS, ); graph_2d.add_slot_edge( graph_2d.input_node().id, bevy_core_pipeline::core_2d::graph::input::VIEW_ENTITY, draw_ui_graph::node::UI_PASS, RunGraphOnViewNode::IN_VIEW, ); graph_2d.add_node_edge( bevy_core_pipeline::core_2d::graph::node::END_MAIN_PASS_POST_PROCESSING, draw_ui_graph::node::UI_PASS, ); graph_2d.add_node_edge( draw_ui_graph::node::UI_PASS, bevy_core_pipeline::core_2d::graph::node::UPSCALING, ); } if let Some(graph_3d) = graph.get_sub_graph_mut(bevy_core_pipeline::core_3d::graph::NAME) { graph_3d.add_sub_graph(draw_ui_graph::NAME, ui_graph_3d); graph_3d.add_node( draw_ui_graph::node::UI_PASS, RunGraphOnViewNode::new(draw_ui_graph::NAME), ); graph_3d.add_node_edge( bevy_core_pipeline::core_3d::graph::node::MAIN_PASS, draw_ui_graph::node::UI_PASS, ); graph_3d.add_node_edge( bevy_core_pipeline::core_3d::graph::node::END_MAIN_PASS_POST_PROCESSING, draw_ui_graph::node::UI_PASS, ); graph_3d.add_node_edge( draw_ui_graph::node::UI_PASS, bevy_core_pipeline::core_3d::graph::node::UPSCALING, ); graph_3d.add_slot_edge( graph_3d.input_node().id, bevy_core_pipeline::core_3d::graph::input::VIEW_ENTITY, draw_ui_graph::node::UI_PASS, RunGraphOnViewNode::IN_VIEW, ); } } fn get_ui_graph(render_app: &mut App) -> RenderGraph { let ui_pass_node = UiPassNode::new(&mut render_app.world); let mut ui_graph = RenderGraph::default(); ui_graph.add_node(draw_ui_graph::node::UI_PASS, ui_pass_node); let input_node_id = ui_graph.set_input(vec![SlotInfo::new( draw_ui_graph::input::VIEW_ENTITY, SlotType::Entity, )]); ui_graph.add_slot_edge( input_node_id, draw_ui_graph::input::VIEW_ENTITY, draw_ui_graph::node::UI_PASS, UiPassNode::IN_VIEW, ); ui_graph } pub struct ExtractedUiNode { pub stack_index: usize, pub transform: Mat4, pub color: Color, pub rect: Rect, pub image: Handle, pub atlas_size: Option, pub clip: Option, pub flip_x: bool, pub flip_y: bool, } #[derive(Resource, Default)] pub struct ExtractedUiNodes { pub uinodes: Vec, } pub fn extract_uinodes( mut extracted_uinodes: ResMut, images: Extract>>, ui_stack: Extract>, uinode_query: Extract< Query<( &Node, &GlobalTransform, &BackgroundColor, Option<&UiImage>, &ComputedVisibility, Option<&CalculatedClip>, )>, >, ) { extracted_uinodes.uinodes.clear(); for (stack_index, entity) in ui_stack.uinodes.iter().enumerate() { if let Ok((uinode, transform, color, maybe_image, visibility, clip)) = uinode_query.get(*entity) { // Skip invisible and completely transparent nodes if !visibility.is_visible() || color.0.a() == 0.0 { continue; } let (image, flip_x, flip_y) = if let Some(image) = maybe_image { // Skip loading images if !images.contains(&image.texture) { continue; } (image.texture.clone_weak(), image.flip_x, image.flip_y) } else { (DEFAULT_IMAGE_HANDLE.typed().clone_weak(), false, false) }; extracted_uinodes.uinodes.push(ExtractedUiNode { stack_index, transform: transform.compute_matrix(), color: color.0, rect: Rect { min: Vec2::ZERO, max: uinode.calculated_size, }, image, atlas_size: None, clip: clip.map(|clip| clip.clip), flip_x, flip_y, }); } } } /// The UI camera is "moved back" by this many units (plus the [`UI_CAMERA_TRANSFORM_OFFSET`]) and also has a view /// distance of this many units. This ensures that with a left-handed projection, /// as ui elements are "stacked on top of each other", they are within the camera's view /// and have room to grow. // TODO: Consider computing this value at runtime based on the maximum z-value. const UI_CAMERA_FAR: f32 = 1000.0; // This value is subtracted from the far distance for the camera's z-position to ensure nodes at z == 0.0 are rendered // TODO: Evaluate if we still need this. const UI_CAMERA_TRANSFORM_OFFSET: f32 = -0.1; #[derive(Component)] pub struct DefaultCameraView(pub Entity); pub fn extract_default_ui_camera_view( mut commands: Commands, query: Extract), With>>, ) { for (entity, camera, camera_ui) in &query { // ignore cameras with disabled ui if matches!(camera_ui, Some(&UiCameraConfig { show_ui: false, .. })) { continue; } if let (Some(logical_size), Some((physical_origin, _)), Some(physical_size)) = ( camera.logical_viewport_size(), camera.physical_viewport_rect(), camera.physical_viewport_size(), ) { // use a projection matrix with the origin in the top left instead of the bottom left that comes with OrthographicProjection let projection_matrix = Mat4::orthographic_rh(0.0, logical_size.x, logical_size.y, 0.0, 0.0, UI_CAMERA_FAR); let default_camera_view = commands .spawn(ExtractedView { projection: projection_matrix, transform: GlobalTransform::from_xyz( 0.0, 0.0, UI_CAMERA_FAR + UI_CAMERA_TRANSFORM_OFFSET, ), view_projection: None, hdr: camera.hdr, viewport: UVec4::new( physical_origin.x, physical_origin.y, physical_size.x, physical_size.y, ), }) .id(); commands.get_or_spawn(entity).insert(( DefaultCameraView(default_camera_view), RenderPhase::::default(), )); } } } pub fn extract_text_uinodes( mut extracted_uinodes: ResMut, texture_atlases: Extract>>, windows: Extract>>, ui_stack: Extract>, uinode_query: Extract< Query<( &Node, &GlobalTransform, &Text, &TextLayoutInfo, &ComputedVisibility, Option<&CalculatedClip>, )>, >, ) { // TODO: Support window-independent UI scale: https://github.com/bevyengine/bevy/issues/5621 let scale_factor = windows .get_single() .map(|window| window.resolution.scale_factor() as f32) .unwrap_or(1.0); for (stack_index, entity) in ui_stack.uinodes.iter().enumerate() { if let Ok((uinode, global_transform, text, text_layout_info, visibility, clip)) = uinode_query.get(*entity) { if !visibility.is_visible() { continue; } // Skip if size is set to zero (e.g. when a parent is set to `Display::None`) if uinode.size() == Vec2::ZERO { continue; } let text_glyphs = &text_layout_info.glyphs; let alignment_offset = (uinode.size() / -2.0).extend(0.0); let mut color = Color::WHITE; let mut current_section = usize::MAX; for text_glyph in text_glyphs { if text_glyph.section_index != current_section { color = text.sections[text_glyph.section_index] .style .color .as_rgba_linear(); current_section = text_glyph.section_index; } let atlas = texture_atlases .get(&text_glyph.atlas_info.texture_atlas) .unwrap(); let texture = atlas.texture.clone_weak(); let index = text_glyph.atlas_info.glyph_index; let rect = atlas.textures[index]; let atlas_size = Some(atlas.size); // NOTE: Should match `bevy_text::text2d::extract_text2d_sprite` let extracted_transform = global_transform.compute_matrix() * Mat4::from_scale(Vec3::splat(scale_factor.recip())) * Mat4::from_translation( alignment_offset * scale_factor + text_glyph.position.extend(0.), ); extracted_uinodes.uinodes.push(ExtractedUiNode { stack_index, transform: extracted_transform, color, rect, image: texture, atlas_size, clip: clip.map(|clip| clip.clip), flip_x: false, flip_y: false, }); } } } } #[repr(C)] #[derive(Copy, Clone, Pod, Zeroable)] struct UiVertex { pub position: [f32; 3], pub uv: [f32; 2], pub color: [f32; 4], } #[derive(Resource)] pub struct UiMeta { vertices: BufferVec, view_bind_group: Option, } impl Default for UiMeta { fn default() -> Self { Self { vertices: BufferVec::new(BufferUsages::VERTEX), view_bind_group: None, } } } const QUAD_VERTEX_POSITIONS: [Vec3; 4] = [ Vec3::new(-0.5, -0.5, 0.0), Vec3::new(0.5, -0.5, 0.0), Vec3::new(0.5, 0.5, 0.0), Vec3::new(-0.5, 0.5, 0.0), ]; const QUAD_INDICES: [usize; 6] = [0, 2, 3, 0, 1, 2]; #[derive(Component)] pub struct UiBatch { pub range: Range, pub image: Handle, pub z: f32, } pub fn prepare_uinodes( mut commands: Commands, render_device: Res, render_queue: Res, mut ui_meta: ResMut, mut extracted_uinodes: ResMut, ) { ui_meta.vertices.clear(); // sort by ui stack index, starting from the deepest node extracted_uinodes .uinodes .sort_by_key(|node| node.stack_index); let mut start = 0; let mut end = 0; let mut current_batch_handle = Default::default(); let mut last_z = 0.0; for extracted_uinode in &extracted_uinodes.uinodes { if current_batch_handle != extracted_uinode.image { if start != end { commands.spawn(UiBatch { range: start..end, image: current_batch_handle, z: last_z, }); start = end; } current_batch_handle = extracted_uinode.image.clone_weak(); } let uinode_rect = extracted_uinode.rect; let rect_size = uinode_rect.size().extend(1.0); // Specify the corners of the node let positions = QUAD_VERTEX_POSITIONS .map(|pos| (extracted_uinode.transform * (pos * rect_size).extend(1.)).xyz()); // Calculate the effect of clipping // Note: this won't work with rotation/scaling, but that's much more complex (may need more that 2 quads) let positions_diff = if let Some(clip) = extracted_uinode.clip { [ Vec2::new( f32::max(clip.min.x - positions[0].x, 0.), f32::max(clip.min.y - positions[0].y, 0.), ), Vec2::new( f32::min(clip.max.x - positions[1].x, 0.), f32::max(clip.min.y - positions[1].y, 0.), ), Vec2::new( f32::min(clip.max.x - positions[2].x, 0.), f32::min(clip.max.y - positions[2].y, 0.), ), Vec2::new( f32::max(clip.min.x - positions[3].x, 0.), f32::min(clip.max.y - positions[3].y, 0.), ), ] } else { [Vec2::ZERO; 4] }; let positions_clipped = [ positions[0] + positions_diff[0].extend(0.), positions[1] + positions_diff[1].extend(0.), positions[2] + positions_diff[2].extend(0.), positions[3] + positions_diff[3].extend(0.), ]; let transformed_rect_size = extracted_uinode.transform.transform_vector3(rect_size); // Don't try to cull nodes that have a rotation // In a rotation around the Z-axis, this value is 0.0 for an angle of 0.0 or π // In those two cases, the culling check can proceed normally as corners will be on // horizontal / vertical lines // For all other angles, bypass the culling check // This does not properly handles all rotations on all axis if extracted_uinode.transform.x_axis[1] == 0.0 { // Cull nodes that are completely clipped if positions_diff[0].x - positions_diff[1].x >= transformed_rect_size.x || positions_diff[1].y - positions_diff[2].y >= transformed_rect_size.y { continue; } } let atlas_extent = extracted_uinode.atlas_size.unwrap_or(uinode_rect.max); let mut uvs = [ Vec2::new( uinode_rect.min.x + positions_diff[0].x, uinode_rect.min.y + positions_diff[0].y, ), Vec2::new( uinode_rect.max.x + positions_diff[1].x, uinode_rect.min.y + positions_diff[1].y, ), Vec2::new( uinode_rect.max.x + positions_diff[2].x, uinode_rect.max.y + positions_diff[2].y, ), Vec2::new( uinode_rect.min.x + positions_diff[3].x, uinode_rect.max.y + positions_diff[3].y, ), ] .map(|pos| pos / atlas_extent); if extracted_uinode.flip_x { uvs = [uvs[1], uvs[0], uvs[3], uvs[2]]; } if extracted_uinode.flip_y { uvs = [uvs[3], uvs[2], uvs[1], uvs[0]]; } let color = extracted_uinode.color.as_linear_rgba_f32(); for i in QUAD_INDICES { ui_meta.vertices.push(UiVertex { position: positions_clipped[i].into(), uv: uvs[i].into(), color, }); } last_z = extracted_uinode.transform.w_axis[2]; end += QUAD_INDICES.len() as u32; } // if start != end, there is one last batch to process if start != end { commands.spawn(UiBatch { range: start..end, image: current_batch_handle, z: last_z, }); } ui_meta.vertices.write_buffer(&render_device, &render_queue); } #[derive(Resource, Default)] pub struct UiImageBindGroups { pub values: HashMap, BindGroup>, } #[allow(clippy::too_many_arguments)] pub fn queue_uinodes( draw_functions: Res>, render_device: Res, mut ui_meta: ResMut, view_uniforms: Res, ui_pipeline: Res, mut pipelines: ResMut>, pipeline_cache: Res, mut image_bind_groups: ResMut, gpu_images: Res>, ui_batches: Query<(Entity, &UiBatch)>, mut views: Query<(&ExtractedView, &mut RenderPhase)>, events: Res, ) { // If an image has changed, the GpuImage has (probably) changed for event in &events.images { match event { AssetEvent::Created { .. } => None, AssetEvent::Modified { handle } | AssetEvent::Removed { handle } => { image_bind_groups.values.remove(handle) } }; } if let Some(view_binding) = view_uniforms.uniforms.binding() { ui_meta.view_bind_group = Some(render_device.create_bind_group(&BindGroupDescriptor { entries: &[BindGroupEntry { binding: 0, resource: view_binding, }], label: Some("ui_view_bind_group"), layout: &ui_pipeline.view_layout, })); let draw_ui_function = draw_functions.read().id::(); for (view, mut transparent_phase) in &mut views { let pipeline = pipelines.specialize( &pipeline_cache, &ui_pipeline, UiPipelineKey { hdr: view.hdr }, ); for (entity, batch) in &ui_batches { image_bind_groups .values .entry(batch.image.clone_weak()) .or_insert_with(|| { let gpu_image = gpu_images.get(&batch.image).unwrap(); render_device.create_bind_group(&BindGroupDescriptor { entries: &[ BindGroupEntry { binding: 0, resource: BindingResource::TextureView(&gpu_image.texture_view), }, BindGroupEntry { binding: 1, resource: BindingResource::Sampler(&gpu_image.sampler), }, ], label: Some("ui_material_bind_group"), layout: &ui_pipeline.image_layout, }) }); transparent_phase.add(TransparentUi { draw_function: draw_ui_function, pipeline, entity, sort_key: FloatOrd(batch.z), }); } } } }