use std::{hash::Hash, marker::PhantomData, ops::Range}; use bevy_asset::*; use bevy_derive::{Deref, DerefMut}; use bevy_ecs::{ prelude::Component, query::ROQueryItem, storage::SparseSet, system::lifetimeless::{Read, SRes}, system::*, }; use bevy_math::{FloatOrd, Mat4, Rect, Vec2, Vec4Swizzles}; use bevy_render::{ extract_component::ExtractComponentPlugin, globals::{GlobalsBuffer, GlobalsUniform}, render_asset::RenderAssets, render_phase::*, render_resource::{binding_types::uniform_buffer, *}, renderer::{RenderDevice, RenderQueue}, texture::{BevyDefault, FallbackImage, Image}, view::*, Extract, ExtractSchedule, Render, RenderSet, }; use bevy_transform::prelude::GlobalTransform; use bevy_utils::{HashMap, HashSet}; use bevy_window::{PrimaryWindow, Window}; use bytemuck::{Pod, Zeroable}; use crate::*; pub const UI_MATERIAL_SHADER_HANDLE: Handle = Handle::weak_from_u128(10074188772096983955); const UI_VERTEX_OUTPUT_SHADER_HANDLE: Handle = Handle::weak_from_u128(10123618247720234751); /// Adds the necessary ECS resources and render logic to enable rendering entities using the given /// [`UiMaterial`] asset type (which includes [`UiMaterial`] types). pub struct UiMaterialPlugin(PhantomData); impl Default for UiMaterialPlugin { fn default() -> Self { Self(Default::default()) } } impl Plugin for UiMaterialPlugin where M::Data: PartialEq + Eq + Hash + Clone, { fn build(&self, app: &mut App) { load_internal_asset!( app, UI_VERTEX_OUTPUT_SHADER_HANDLE, "ui_vertex_output.wgsl", Shader::from_wgsl ); load_internal_asset!( app, UI_MATERIAL_SHADER_HANDLE, "ui_material.wgsl", Shader::from_wgsl ); app.init_asset::() .add_plugins(ExtractComponentPlugin::>::extract_visible()); if let Ok(render_app) = app.get_sub_app_mut(RenderApp) { render_app .add_render_command::>() .init_resource::>() .init_resource::>() .init_resource::>() .init_resource::>() .init_resource::>>() .add_systems( ExtractSchedule, ( extract_ui_materials::, extract_ui_material_nodes::.in_set(RenderUiSystem::ExtractBackgrounds), ), ) .add_systems( Render, ( prepare_ui_materials::.in_set(RenderSet::PrepareAssets), queue_ui_material_nodes::.in_set(RenderSet::Queue), prepare_uimaterial_nodes::.in_set(RenderSet::PrepareBindGroups), ), ); } } fn finish(&self, app: &mut App) { if let Ok(render_app) = app.get_sub_app_mut(RenderApp) { render_app.init_resource::>(); } } } #[derive(Resource)] pub struct UiMaterialMeta { vertices: BufferVec, view_bind_group: Option, marker: PhantomData, } impl Default for UiMaterialMeta { fn default() -> Self { Self { vertices: BufferVec::new(BufferUsages::VERTEX), view_bind_group: Default::default(), marker: PhantomData, } } } #[repr(C)] #[derive(Copy, Clone, Pod, Zeroable)] pub struct UiMaterialVertex { pub position: [f32; 3], pub uv: [f32; 2], pub size: [f32; 2], pub border_widths: [f32; 4], } // in this [`UiMaterialPipeline`] there is (currently) no batching going on. // Therefore the [`UiMaterialBatch`] is more akin to a draw call. #[derive(Component)] pub struct UiMaterialBatch { /// The range of vertices inside the [`UiMaterialMeta`] pub range: Range, pub material: AssetId, } /// Render pipeline data for a given [`UiMaterial`] #[derive(Resource)] pub struct UiMaterialPipeline { pub ui_layout: BindGroupLayout, pub view_layout: BindGroupLayout, pub vertex_shader: Option>, pub fragment_shader: Option>, marker: PhantomData, } impl SpecializedRenderPipeline for UiMaterialPipeline where M::Data: PartialEq + Eq + Hash + Clone, { type Key = UiMaterialKey; fn specialize(&self, key: Self::Key) -> RenderPipelineDescriptor { let vertex_layout = VertexBufferLayout::from_vertex_formats( VertexStepMode::Vertex, vec![ // position VertexFormat::Float32x3, // uv VertexFormat::Float32x2, // size VertexFormat::Float32x2, // border_widths VertexFormat::Float32x4, ], ); let shader_defs = Vec::new(); let mut descriptor = RenderPipelineDescriptor { vertex: VertexState { shader: UI_MATERIAL_SHADER_HANDLE, entry_point: "vertex".into(), shader_defs: shader_defs.clone(), buffers: vec![vertex_layout], }, fragment: Some(FragmentState { shader: UI_MATERIAL_SHADER_HANDLE, shader_defs, entry_point: "fragment".into(), targets: vec![Some(ColorTargetState { format: if key.hdr { ViewTarget::TEXTURE_FORMAT_HDR } else { TextureFormat::bevy_default() }, blend: Some(BlendState::ALPHA_BLENDING), write_mask: ColorWrites::ALL, })], }), layout: vec![], push_constant_ranges: Vec::new(), primitive: PrimitiveState { front_face: FrontFace::Ccw, cull_mode: None, unclipped_depth: false, polygon_mode: PolygonMode::Fill, conservative: false, topology: PrimitiveTopology::TriangleList, strip_index_format: None, }, depth_stencil: None, multisample: MultisampleState { count: 1, mask: !0, alpha_to_coverage_enabled: false, }, label: Some("ui_material_pipeline".into()), }; if let Some(vertex_shader) = &self.vertex_shader { descriptor.vertex.shader = vertex_shader.clone(); } if let Some(fragment_shader) = &self.fragment_shader { descriptor.fragment.as_mut().unwrap().shader = fragment_shader.clone(); } descriptor.layout = vec![self.view_layout.clone(), self.ui_layout.clone()]; M::specialize(&mut descriptor, key); descriptor } } impl FromWorld for UiMaterialPipeline { fn from_world(world: &mut World) -> Self { let asset_server = world.resource::(); let render_device = world.resource::(); let ui_layout = M::bind_group_layout(render_device); let view_layout = render_device.create_bind_group_layout( "ui_view_layout", &BindGroupLayoutEntries::sequential( ShaderStages::VERTEX_FRAGMENT, ( uniform_buffer::(true), uniform_buffer::(false), ), ), ); UiMaterialPipeline { ui_layout, view_layout, vertex_shader: match M::vertex_shader() { ShaderRef::Default => None, ShaderRef::Handle(handle) => Some(handle), ShaderRef::Path(path) => Some(asset_server.load(path)), }, fragment_shader: match M::fragment_shader() { ShaderRef::Default => None, ShaderRef::Handle(handle) => Some(handle), ShaderRef::Path(path) => Some(asset_server.load(path)), }, marker: PhantomData, } } } pub type DrawUiMaterial = ( SetItemPipeline, SetMatUiViewBindGroup, SetUiMaterialBindGroup, DrawUiMaterialNode, ); pub struct SetMatUiViewBindGroup(PhantomData); impl RenderCommand

for SetMatUiViewBindGroup { type Param = SRes>; type ViewQuery = Read; type ItemQuery = (); fn render<'w>( _item: &P, view_uniform: &'w ViewUniformOffset, _entity: Option<()>, ui_meta: SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { pass.set_bind_group( I, ui_meta.into_inner().view_bind_group.as_ref().unwrap(), &[view_uniform.offset], ); RenderCommandResult::Success } } pub struct SetUiMaterialBindGroup(PhantomData); impl RenderCommand

for SetUiMaterialBindGroup { type Param = SRes>; type ViewQuery = (); type ItemQuery = Read>; fn render<'w>( _item: &P, _view: (), material_handle: Option>, materials: SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { let Some(material_handle) = material_handle else { return RenderCommandResult::Failure; }; let Some(material) = materials.into_inner().get(&material_handle.material) else { return RenderCommandResult::Failure; }; pass.set_bind_group(I, &material.bind_group, &[]); RenderCommandResult::Success } } pub struct DrawUiMaterialNode(PhantomData); impl RenderCommand

for DrawUiMaterialNode { type Param = SRes>; type ViewQuery = (); type ItemQuery = Read>; #[inline] fn render<'w>( _item: &P, _view: (), batch: Option<&'w UiMaterialBatch>, ui_meta: SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { let Some(batch) = batch else { return RenderCommandResult::Failure; }; pass.set_vertex_buffer(0, ui_meta.into_inner().vertices.buffer().unwrap().slice(..)); pass.draw(batch.range.clone(), 0..1); RenderCommandResult::Success } } pub struct ExtractedUiMaterialNode { pub stack_index: usize, pub transform: Mat4, pub rect: Rect, pub border: [f32; 4], pub material: AssetId, pub clip: Option, // Camera to render this UI node to. By the time it is extracted, // it is defaulted to a single camera if only one exists. // Nodes with ambiguous camera will be ignored. pub camera_entity: Entity, } #[derive(Resource)] pub struct ExtractedUiMaterialNodes { pub uinodes: SparseSet>, } impl Default for ExtractedUiMaterialNodes { fn default() -> Self { Self { uinodes: Default::default(), } } } pub fn extract_ui_material_nodes( mut extracted_uinodes: ResMut>, materials: Extract>>, ui_stack: Extract>, default_ui_camera: Extract, uinode_query: Extract< Query< ( Entity, &Node, &Style, &GlobalTransform, &Handle, &ViewVisibility, Option<&CalculatedClip>, Option<&TargetCamera>, ), Without, >, >, windows: Extract>>, ui_scale: Extract>, ) { let ui_logical_viewport_size = windows .get_single() .map(|window| window.size()) .unwrap_or(Vec2::ZERO) // The logical window resolution returned by `Window` only takes into account the window scale factor and not `UiScale`, // so we have to divide by `UiScale` to get the size of the UI viewport. / ui_scale.0; // If there is only one camera, we use it as default let default_single_camera = default_ui_camera.get(); for (stack_index, entity) in ui_stack.uinodes.iter().enumerate() { if let Ok((entity, uinode, style, transform, handle, view_visibility, clip, camera)) = uinode_query.get(*entity) { let Some(camera_entity) = camera.map(TargetCamera::entity).or(default_single_camera) else { continue; }; // skip invisible nodes if !view_visibility.get() { continue; } // Skip loading materials if !materials.contains(handle) { continue; } // Both vertical and horizontal percentage border values are calculated based on the width of the parent node // let parent_width = uinode.size().x; let left = resolve_border_thickness(style.border.left, parent_width, ui_logical_viewport_size) / uinode.size().x; let right = resolve_border_thickness( style.border.right, parent_width, ui_logical_viewport_size, ) / uinode.size().x; let top = resolve_border_thickness(style.border.top, parent_width, ui_logical_viewport_size) / uinode.size().y; let bottom = resolve_border_thickness( style.border.bottom, parent_width, ui_logical_viewport_size, ) / uinode.size().y; extracted_uinodes.uinodes.insert( entity, ExtractedUiMaterialNode { stack_index, transform: transform.compute_matrix(), material: handle.id(), rect: Rect { min: Vec2::ZERO, max: uinode.calculated_size, }, border: [left, right, top, bottom], clip: clip.map(|clip| clip.clip), camera_entity, }, ); }; } } #[allow(clippy::too_many_arguments)] pub fn prepare_uimaterial_nodes( mut commands: Commands, render_device: Res, render_queue: Res, mut ui_meta: ResMut>, mut extracted_uinodes: ResMut>, view_uniforms: Res, globals_buffer: Res, ui_material_pipeline: Res>, mut phases: Query<&mut RenderPhase>, mut previous_len: Local, ) { if let (Some(view_binding), Some(globals_binding)) = ( view_uniforms.uniforms.binding(), globals_buffer.buffer.binding(), ) { let mut batches: Vec<(Entity, UiMaterialBatch)> = Vec::with_capacity(*previous_len); ui_meta.vertices.clear(); ui_meta.view_bind_group = Some(render_device.create_bind_group( "ui_material_view_bind_group", &ui_material_pipeline.view_layout, &BindGroupEntries::sequential((view_binding, globals_binding)), )); let mut index = 0; for mut ui_phase in &mut phases { let mut batch_item_index = 0; let mut batch_shader_handle = AssetId::invalid(); for item_index in 0..ui_phase.items.len() { let item = &mut ui_phase.items[item_index]; if let Some(extracted_uinode) = extracted_uinodes.uinodes.get(item.entity) { let mut existing_batch = batches .last_mut() .filter(|_| batch_shader_handle == extracted_uinode.material); if existing_batch.is_none() { batch_item_index = item_index; batch_shader_handle = extracted_uinode.material; let new_batch = UiMaterialBatch { range: index..index, material: extracted_uinode.material, }; batches.push((item.entity, new_batch)); existing_batch = batches.last_mut(); } let uinode_rect = extracted_uinode.rect; let rect_size = uinode_rect.size().extend(1.0); let positions = QUAD_VERTEX_POSITIONS.map(|pos| { (extracted_uinode.transform * (pos * rect_size).extend(1.0)).xyz() }); 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 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 / uinode_rect.max); for i in QUAD_INDICES { ui_meta.vertices.push(UiMaterialVertex { position: positions_clipped[i].into(), uv: uvs[i].into(), size: extracted_uinode.rect.size().into(), border_widths: extracted_uinode.border, }); } index += QUAD_INDICES.len() as u32; existing_batch.unwrap().1.range.end = index; ui_phase.items[batch_item_index].batch_range_mut().end += 1; } else { batch_shader_handle = AssetId::invalid(); } } } ui_meta.vertices.write_buffer(&render_device, &render_queue); *previous_len = batches.len(); commands.insert_or_spawn_batch(batches); } extracted_uinodes.uinodes.clear(); } #[derive(Resource, Deref, DerefMut)] pub struct RenderUiMaterials(HashMap, PreparedUiMaterial>); impl Default for RenderUiMaterials { fn default() -> Self { Self(Default::default()) } } pub struct PreparedUiMaterial { pub bindings: Vec<(u32, OwnedBindingResource)>, pub bind_group: BindGroup, pub key: T::Data, } #[derive(Resource)] pub struct ExtractedUiMaterials { extracted: Vec<(AssetId, M)>, removed: Vec>, } impl Default for ExtractedUiMaterials { fn default() -> Self { Self { extracted: Default::default(), removed: Default::default(), } } } pub fn extract_ui_materials( mut commands: Commands, mut events: Extract>>, assets: Extract>>, ) { let mut changed_assets = HashSet::default(); let mut removed = Vec::new(); for event in events.read() { #[allow(clippy::match_same_arms)] match event { AssetEvent::Added { id } | AssetEvent::Modified { id } => { changed_assets.insert(*id); } AssetEvent::Removed { id } => { changed_assets.remove(id); removed.push(*id); } AssetEvent::Unused { .. } => {} AssetEvent::LoadedWithDependencies { .. } => { // TODO: handle this } } } let mut extracted_assets = Vec::new(); for id in changed_assets.drain() { if let Some(asset) = assets.get(id) { extracted_assets.push((id, asset.clone())); } } commands.insert_resource(ExtractedUiMaterials { extracted: extracted_assets, removed, }); } pub struct PrepareNextFrameMaterials { assets: Vec<(AssetId, M)>, } impl Default for PrepareNextFrameMaterials { fn default() -> Self { Self { assets: Default::default(), } } } pub fn prepare_ui_materials( mut prepare_next_frame: Local>, mut extracted_assets: ResMut>, mut render_materials: ResMut>, render_device: Res, images: Res>, fallback_image: Res, pipeline: Res>, ) { let queued_assets = std::mem::take(&mut prepare_next_frame.assets); for (id, material) in queued_assets { if extracted_assets.removed.contains(&id) { continue; } match prepare_ui_material( &material, &render_device, &images, &fallback_image, &pipeline, ) { Ok(prepared_asset) => { render_materials.insert(id, prepared_asset); } Err(AsBindGroupError::RetryNextUpdate) => { prepare_next_frame.assets.push((id, material)); } } } for removed in std::mem::take(&mut extracted_assets.removed) { render_materials.remove(&removed); } for (handle, material) in std::mem::take(&mut extracted_assets.extracted) { match prepare_ui_material( &material, &render_device, &images, &fallback_image, &pipeline, ) { Ok(prepared_asset) => { render_materials.insert(handle, prepared_asset); } Err(AsBindGroupError::RetryNextUpdate) => { prepare_next_frame.assets.push((handle, material)); } } } } fn prepare_ui_material( material: &M, render_device: &RenderDevice, images: &RenderAssets, fallback_image: &Res, pipeline: &UiMaterialPipeline, ) -> Result, AsBindGroupError> { let prepared = material.as_bind_group(&pipeline.ui_layout, render_device, images, fallback_image)?; Ok(PreparedUiMaterial { bindings: prepared.bindings, bind_group: prepared.bind_group, key: prepared.data, }) } #[allow(clippy::too_many_arguments)] pub fn queue_ui_material_nodes( extracted_uinodes: Res>, draw_functions: Res>, ui_material_pipeline: Res>, mut pipelines: ResMut>>, pipeline_cache: Res, render_materials: Res>, mut views: Query<(&ExtractedView, &mut RenderPhase)>, ) where M::Data: PartialEq + Eq + Hash + Clone, { let draw_function = draw_functions.read().id::>(); for (entity, extracted_uinode) in extracted_uinodes.uinodes.iter() { let Some(material) = render_materials.get(&extracted_uinode.material) else { continue; }; let Ok((view, mut transparent_phase)) = views.get_mut(extracted_uinode.camera_entity) else { continue; }; let pipeline = pipelines.specialize( &pipeline_cache, &ui_material_pipeline, UiMaterialKey { hdr: view.hdr, bind_group_data: material.key.clone(), }, ); transparent_phase .items .reserve(extracted_uinodes.uinodes.len()); transparent_phase.add(TransparentUi { draw_function, pipeline, entity: *entity, sort_key: ( FloatOrd(extracted_uinode.stack_index as f32), entity.index(), ), batch_range: 0..0, dynamic_offset: None, }); } }