-
-
Notifications
You must be signed in to change notification settings - Fork 3.8k
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
Preserve the order of window child nodes between updates #9764
Conversation
…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.
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 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. |
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 |
It feels like there should be a better way to construct the |
I just rediscovered #8574, although I guess this would still be necessary even with that. |
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. |
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 |
There was a problem hiding this comment.
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
.
There was a problem hiding this comment.
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
This could be simplified slightly by bringing in an ordered set. 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(); |
Closing this as the design has changed and it's no longer a problem. |
Objective
set_window_children
doesn't preserve the order of nodes between updates because the order of the queried entities can change.Example
The example displays red, green and blue UI nodes:

After two seconds a system adds

AotherComponent
to the first entity, and the order of the UI nodes changes (ymmv):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.