use super::Node; use bevy_ecs::{Entity, Query, With, Without}; use bevy_transform::{ hierarchy, prelude::{Children, LocalTransform, Parent}, }; pub const UI_Z_STEP: f32 = 0.001; pub fn ui_z_system( mut root_node_query: Query>>, mut node_query: Query<(Entity, &Node, &mut LocalTransform)>, children_query: Query<&Children>, ) { let mut current_global_z = 0.0; // PERF: we can probably avoid an allocation here by making root_node_query and node_query non-overlapping let root_nodes = (&mut root_node_query.iter()) .iter() .collect::>(); for entity in root_nodes { if let Some(result) = hierarchy::run_on_hierarchy( &children_query, &mut node_query, entity, Some(current_global_z), Some(current_global_z), &mut update_node_entity, ) { current_global_z = result; } } } fn update_node_entity( node_query: &mut Query<(Entity, &Node, &mut LocalTransform)>, entity: Entity, parent_result: Option, previous_result: Option, ) -> Option { let mut transform = node_query.get_mut::(entity).ok()?; let mut z = UI_Z_STEP; let parent_global_z = parent_result.unwrap(); if let Some(previous_global_z) = previous_result { z += previous_global_z - parent_global_z; }; let global_z = z + parent_global_z; let mut position = transform.w_axis(); position.set_z(z); transform.set_w_axis(position); return Some(global_z); }