Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Preserve the order of window child nodes between updates #9764

Conversation

ickshonpe
Copy link
Contributor

Objective

set_window_children doesn't preserve the order of nodes between updates because the order of the queried entities can change.

Example

use bevy::prelude::*;

fn main() {
    App::new()
        .add_plugins(DefaultPlugins)
        .add_systems(Startup, setup)
        .add_systems(Update, add_another_component)
        .run();
}

fn setup(mut commands: Commands) {
    let style = Style {
        width: Val::Percent(33.33),
        ..Default::default()
    };

    commands.spawn(Camera2dBundle::default());
    commands.spawn(NodeBundle {
        style: style.clone(),
        background_color: Color::RED.into(),
        ..Default::default()
    });

    commands.spawn(NodeBundle {
        style: style.clone(),
        background_color: Color::GREEN.into(),
        ..Default::default()
    });

    commands.spawn(NodeBundle {
        style: style.clone(),
        background_color: Color::BLUE.into(),
        ..Default::default()
    });
}

#[derive(Component)]
struct AnotherComponent;

fn add_another_component(
    mut delay: Local<f32>,
    mut commands: Commands,
    time: Res<Time>,
    query: Query<Entity, With<Style>>,
) {
    *delay += time.delta_seconds();
    if 2. < *delay {
        *delay = f32::NEG_INFINITY;
        let item = query.iter().next().unwrap();
        commands.entity(item).insert(AnotherComponent);
    }
}

The example displays red, green and blue UI nodes:
Capture-rgb

After two seconds a system adds AotherComponent to the first entity, and the order of the UI nodes changes (ymmv):
Capture-bgr

Solution

Concatenate the list of current window child nodes to the end of the previous list and then remove duplicates and nodes not present in the current list.

…updates because the order of the queried entities can change.

This commit changes `set_window_children` to preserve the order of existing window children. The list of of current window child nodes is concatenated to the end of the previous list and then duplicates and nodes not present in the current list are removed.
@rparrett
Copy link
Contributor

rparrett commented Sep 11, 2023

Just to clarify -- it seems like this is worth fixing either way -- there is no way to guarantee that root nodes spawn/query in a particular order, right?

e.g. (without the add_another_component system)

commands.spawn((
    NodeBundle {
        style: style.clone(),
        background_color: Color::RED.into(),
        ..Default::default()
    },
    AnotherComponent,
));

commands.spawn(NodeBundle {
    style: style.clone(),
    background_color: Color::GREEN.into(),
    ..Default::default()
});

commands.spawn((
    NodeBundle {
        style: style.clone(),
        background_color: Color::BLUE.into(),
        ..Default::default()
    },
    AnotherComponent,
));

Shows up as R-B-G.

@rparrett rparrett added C-Bug An unexpected or incorrect behavior A-UI Graphical user interfaces, styles, layouts, and widgets labels Sep 11, 2023
@ickshonpe
Copy link
Contributor Author

ickshonpe commented Sep 11, 2023

Just to clarify -- it seems like this is worth fixing either way -- there is no way to guarantee that root nodes spawn/query in a particular order, right?

No at the moment, as far I know.

There is a need for some sort of redesign but I think we should fix the existing problems first. Same thing with the ContentSize PR I posted yesterday,

@ickshonpe
Copy link
Contributor Author

ickshonpe commented Sep 11, 2023

It feels like there should be a better way to construct the set_window_children function, using two hashsets seems like too much. It only needs to be done once per layout update though, so it's not a huge deal. Correctness is more important here.

@rparrett
Copy link
Contributor

I just rediscovered #8574, although I guess this would still be necessary even with that.

@rparrett
Copy link
Contributor

I was hoping to find some set of variable names that makes it a bit clearer at a glance what's going on here but I find that I can't improve on what you've done.

I do think that a comment or two would be super useful.

@ickshonpe
Copy link
Contributor Author

I was hoping to find some set of variable names that makes it a bit clearer at a glance what's going on here but I find that I can't improve on what you've done.

I do think that a comment or two would be super useful.

Yeah, I guess just something to explain the intent. It was an awkward bit of a code to write.

@ickshonpe
Copy link
Contributor Author

I just rediscovered #8574, although I guess this would still be necessary even with that.

Yep the window children can still get shuffled about when the archetypes change or whatever.

I thought #8574 was okay, but no one seemed that interested. I guess it's something that might be better implemented either in Taffy or as part of the ecs.

window_children.retain(|child| child_set.contains(child) && unique.insert(*child));
window_children
} else {
new_window_children
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this branch ever execute in practice? It seems like taffy.children unconditionally returns Ok.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems like taffy.children unconditionally returns Ok

It currently does but arguably shouldn't, as it will panic if the node doesn't exist (note that if the node exists in Taffy, then the children definitely will as children is default initialized to an empty Vec). Likely Taffy's API will split/duplicated into panicking and result-returning variants at some point. See DioxusLabs/taffy#519

@rparrett
Copy link
Contributor

This could be simplified slightly by bringing in an ordered set. indexmap is all over in our deps so it seems like it shouldn't be an issue to add it?

let parent_node = *self.window_nodes.get(&parent_window).unwrap();
let existing = self.taffy.children(parent_node).unwrap();

let children = existing
    .iter()
    .copied()
    .chain(children.map(|e| *self.entity_to_taffy.get(&e).unwrap()))
    .collect::<IndexSet<_>>()
    .iter()
    .copied()
    .collect::<Vec<_>>();

self.taffy.set_children(parent_node, &children).unwrap();

@BenjaminBrienen BenjaminBrienen added D-Straightforward Simple bug fixes and API improvements, docs, test and examples S-Waiting-on-Author The author needs to make changes or address concerns before this can be merged labels Jan 23, 2025
@ickshonpe
Copy link
Contributor Author

Closing this as the design has changed and it's no longer a problem.

@ickshonpe ickshonpe closed this Feb 5, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-UI Graphical user interfaces, styles, layouts, and widgets C-Bug An unexpected or incorrect behavior D-Straightforward Simple bug fixes and API improvements, docs, test and examples S-Waiting-on-Author The author needs to make changes or address concerns before this can be merged
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants