forked from bevyengine/bevy
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Improve ui depth system (bevyengine#905)
* Add test for ui-z system * Remove generic hierarchy runner and refactor ui z-system * Remove different handling for childless nodes Having an empty children list should be the same as having no child component. * Further simplify system after change
- Loading branch information
Showing
2 changed files
with
129 additions
and
60 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,49 +1,149 @@ | ||
use super::Node; | ||
use bevy_ecs::{Entity, Query, With, Without}; | ||
use bevy_transform::{ | ||
hierarchy, | ||
prelude::{Children, Parent, Transform}, | ||
}; | ||
use bevy_transform::prelude::{Children, Parent, Transform}; | ||
|
||
pub const UI_Z_STEP: f32 = 0.001; | ||
|
||
pub fn ui_z_system( | ||
root_node_query: Query<Entity, (With<Node>, Without<Parent>)>, | ||
mut node_query: Query<(Entity, &mut Transform), With<Node>>, | ||
mut node_query: Query<&mut Transform, With<Node>>, | ||
children_query: Query<&Children>, | ||
) { | ||
let mut current_global_z = 0.0; | ||
|
||
for entity in root_node_query.iter() { | ||
if let Some(result) = hierarchy::run_on_hierarchy( | ||
current_global_z = update_hierarchy( | ||
&children_query, | ||
&mut node_query, | ||
entity, | ||
Some(current_global_z), | ||
Some(current_global_z), | ||
&mut update_node_entity, | ||
) { | ||
current_global_z = result; | ||
} | ||
current_global_z, | ||
current_global_z, | ||
); | ||
} | ||
} | ||
|
||
fn update_node_entity( | ||
node_query: &mut Query<(Entity, &mut Transform), With<Node>>, | ||
fn update_hierarchy( | ||
children_query: &Query<&Children>, | ||
node_query: &mut Query<&mut Transform, With<Node>>, | ||
entity: Entity, | ||
parent_result: Option<f32>, | ||
previous_result: Option<f32>, | ||
) -> Option<f32> { | ||
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; | ||
|
||
if let Ok(mut transform) = node_query.get_component_mut::<Transform>(entity) { | ||
transform.translation.z = z; | ||
parent_global_z: f32, | ||
mut current_global_z: f32, | ||
) -> f32 { | ||
current_global_z += UI_Z_STEP; | ||
if let Ok(mut transform) = node_query.get_mut(entity) { | ||
transform.translation.z = current_global_z - parent_global_z; | ||
} | ||
if let Ok(children) = children_query.get(entity) { | ||
let current_parent_global_z = current_global_z; | ||
for child in children.iter().cloned() { | ||
current_global_z = update_hierarchy( | ||
children_query, | ||
node_query, | ||
child, | ||
current_parent_global_z, | ||
current_global_z, | ||
); | ||
} | ||
} | ||
current_global_z | ||
} | ||
#[cfg(test)] | ||
mod tests { | ||
use bevy_ecs::{Commands, IntoSystem, Resources, Schedule, World}; | ||
use bevy_transform::{components::Transform, hierarchy::BuildChildren}; | ||
|
||
use crate::Node; | ||
|
||
use super::{ui_z_system, UI_Z_STEP}; | ||
|
||
fn node_with_transform(name: &str) -> (String, Node, Transform) { | ||
(name.to_owned(), Node::default(), Transform::default()) | ||
} | ||
|
||
Some(global_z) | ||
fn node_without_transform(name: &str) -> (String, Node) { | ||
(name.to_owned(), Node::default()) | ||
} | ||
|
||
fn get_steps(transform: &Transform) -> u32 { | ||
(transform.translation.z / UI_Z_STEP).round() as u32 | ||
} | ||
|
||
#[test] | ||
fn test_ui_z_system() { | ||
let mut world = World::default(); | ||
let mut resources = Resources::default(); | ||
let mut commands = Commands::default(); | ||
commands.set_entity_reserver(world.get_entity_reserver()); | ||
|
||
commands.spawn(node_with_transform("0")); | ||
|
||
commands | ||
.spawn(node_with_transform("1")) | ||
.with_children(|parent| { | ||
parent | ||
.spawn(node_with_transform("1-0")) | ||
.with_children(|parent| { | ||
parent.spawn(node_with_transform("1-0-0")); | ||
parent.spawn(node_without_transform("1-0-1")); | ||
parent.spawn(node_with_transform("1-0-2")); | ||
}); | ||
parent.spawn(node_with_transform("1-1")); | ||
parent | ||
.spawn(node_without_transform("1-2")) | ||
.with_children(|parent| { | ||
parent.spawn(node_with_transform("1-2-0")); | ||
parent.spawn(node_with_transform("1-2-1")); | ||
parent | ||
.spawn(node_with_transform("1-2-2")) | ||
.with_children(|_| ()); | ||
parent.spawn(node_with_transform("1-2-3")); | ||
}); | ||
parent.spawn(node_with_transform("1-3")); | ||
}); | ||
|
||
commands | ||
.spawn(node_without_transform("2")) | ||
.with_children(|parent| { | ||
parent | ||
.spawn(node_with_transform("2-0")) | ||
.with_children(|_parent| ()); | ||
parent | ||
.spawn(node_with_transform("2-1")) | ||
.with_children(|parent| { | ||
parent.spawn(node_with_transform("2-1-0")); | ||
}); | ||
}); | ||
commands.apply(&mut world, &mut resources); | ||
|
||
let mut schedule = Schedule::default(); | ||
schedule.add_stage("update"); | ||
schedule.add_system_to_stage("update", ui_z_system.system()); | ||
schedule.initialize(&mut world, &mut resources); | ||
schedule.run(&mut world, &mut resources); | ||
|
||
let mut actual_result = world | ||
.query::<(&String, &Transform)>() | ||
.map(|(name, transform)| (name.clone(), get_steps(transform))) | ||
.collect::<Vec<(String, u32)>>(); | ||
actual_result.sort_unstable_by_key(|(name, _)| name.clone()); | ||
let expected_result = vec![ | ||
("0".to_owned(), 1), | ||
("1".to_owned(), 1), | ||
("1-0".to_owned(), 1), | ||
("1-0-0".to_owned(), 1), | ||
// 1-0-1 has no transform | ||
("1-0-2".to_owned(), 3), | ||
("1-1".to_owned(), 5), | ||
// 1-2 has no transform | ||
("1-2-0".to_owned(), 1), | ||
("1-2-1".to_owned(), 2), | ||
("1-2-2".to_owned(), 3), | ||
("1-2-3".to_owned(), 4), | ||
("1-3".to_owned(), 11), | ||
// 2 has no transform | ||
("2-0".to_owned(), 1), | ||
("2-1".to_owned(), 2), | ||
("2-1-0".to_owned(), 1), | ||
]; | ||
assert_eq!(actual_result, expected_result); | ||
} | ||
} |