-
-
Notifications
You must be signed in to change notification settings - Fork 21.5k
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
Implement a system to contextualize global themes #81130
Implement a system to contextualize global themes #81130
Conversation
I suspected that this might happen so I decided to test the performance. 10 runs to open and close the editor with a specific project now takes several seconds more:
This seems to affect both startup and shutdown. I will investigate this further. |
b3ff823
to
6c88b67
Compare
Okay, I identified what was causing slow times, both on startup and shutdown. These were extra notifications trying to update the theme on many nodes in the tree. A few of those were my mistakes that I fixed, but there were also a couple of curious ones which my PR had uncovered but wasn't directly responsible for. So, the default theme has a couple of GradientTexture2Ds as icons for the color picker. These have a peculiar property of being updated with a deferred call whenever you change their properties. This means that when we set the icon in the default theme, it comes with a pending update. On the following frame the icon does update and causes the theme itself to update. Which, thanks to my changes, propagates through the entire scene tree. The rationale for this behavior is unclear to me. Even stranger is the fact that some methods have been changed from a deferred call to an immediate call (#60361). And we are calling this method too. So in the default theme we have the following code: Ref<GradientTexture2D> hue_texture;
hue_texture.instantiate();
hue_texture->set_width(800); // <-- defers an update
hue_texture->set_height(6); // <-- defers an update
hue_texture->set_gradient(hue_gradient); // <-- updates immediately This means that the texture both, updates immediately and does a deferred update. I've discussed this with @KoBeWi and he suggested a potential improvement to the entire thing (#81137). For the time being I've patched this issue in my PR by checking for the pending flag in the void GradientTexture2D::_update() {
if (!update_pending) {
return; // Might've processed already, no need to do extra work.
}
update_pending = false;
... This ensures that we don't do a deferred update, so we won't update the theme after the fact (twice, because we have two such textures!), and so we won't trigger two waves of theme changed notification propagation. This patch can be reverted when the proper change is available. Benchmarks now show little to no difference.
|
6c88b67
to
3564274
Compare
Adjusted the changes related to GradientTexture2D following #81137. |
3564274
to
3df1451
Compare
I noticed a visual bug due to the editor theme being accessed too early and never updated on theme changes in the |
Editor theme changes do not propagate to the scene.
Although it might be the same issue as with changing project theme requiring editor restart, i.e. a new Theme is created. |
Indeed, but it should be fixable. I already update theme contexts for the editor itself, it just slipped my mind to update the preview one as well. |
3df1451
to
042c5b4
Compare
Selecting theme does not close the base popup. godot.windows.editor.dev.x86_64_4qtnObGg2z.mp4 |
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.
Looks fine, aside from that minor thing with popup menu.
You could also implement switching default theme without restarting the editor. It requires a couple of lines, just like you update the editor theme, no?
For the preview at least, yeah, it should be simple. I'm concerned that there may be other places that use the project theme and won't update automatically. I'll look into it in future, but it shouldn't block this PR for now.
Hmm, didn't realize that this wasn't intended. I kind of like that you can swap them quickly like that, but I can change that — seems to be a one line change. |
If you want quick swapping then the submenu should also stay open I think. Currently it looks like a bug. |
042c5b4
to
171b74b
Compare
Fair enough. Made it disappear for now, for consistency with the grid submenu at least. I don't really expect it to require to be hot accessible, so we can see if there is any demand for quick swapping. If there is, maybe it shouldn't be in the submenu so deep in the first place. |
171b74b
to
514f305
Compare
514f305
to
886be25
Compare
Rebased following the merge of #80573. Given the nature of conflicting changes, I basically recreated most changes from #80573 in Edit: Another push to fix a leak in the tests (only relevant there because it behaves differently from the normal engine workflow). |
886be25
to
6667ca7
Compare
This commit adds the default theme context, which replaces the need to manually check the project and the default theme all the time; simplifies related code. It also adds framework for custom theme contexts, to be used by the editor. Custom contexts can be attached to any node, and not necessarily a GUI/Window node. Contexts do no break theme inheritance and only define which global themes a node uses as a fallback. Contexts propagate NOTIFICATION_THEME_CHANGED when one of their global themes changes. This ensures that global themes act just like themes assigned to individual nodes and can be previewed live in the editor.
This change defines additional theme contexts for editor branches to prevent theme leaking between the default theme, the project theme, and the editor theme. - Both editor window and EditorNode define an editor-specific context with the editor theme and the default theme. - The 2D viewport defines a project-specific context with the project theme and the default theme. - Theme editor preview tabs define the default-only context with the default theme. Additionally, the default theme context now only includes the project theme for running projects (both export and debug). This prevents the project theme from leaking into the editor. This commit also does a little clean up on the theming aspects of the EditorNode.
This commit adds a new View submenu that allows switching between the project theme (default), the editor theme, and the default theme. The last selected option is stored per project and is restored when reloading the project.
6667ca7
to
fc01e2e
Compare
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.
Looks great! Let's get this tested further ahead of dev5.
Thanks! |
This PR introduces a concept of global theme contexts.
Global themes we have now
For the purposes of this change I define as a global theme any theme that we use as a fallback when looking up items, regardless of the scene tree structure. This means that currently we have 2 global themes: the default one (
scene/theme/default_theme.cpp
, previouslyscene/resources/default_theme/default_theme.cpp
), and the custom project one (if it is defined by the user).Whenever we cannot find a theme item, we first check in the project theme, and then in the default theme. This works regardless of what other themes affect the node and what parents of the node look like (i.e. if they can affect theming or not). Thus these two are global.
In the editor we have the editor theme, but it's not global. It applies directly to Control/Window nodes, just like you would do in your project. Thus the editor theme must have an uninterrupted chain of themable nodes between its holder node and any GUI node in the editor UI that needs to respect it. If the chain breaks, or if the target node is not in a branch that has one of the themed node in it, then styling break.
At the same time, in the editor we don't need nor want to have the project theme applied as global. If there are missing definitions in the editor theme, the project theme will leak into the editor UI. Historically we've patched many such holes on the case-by-case basis. And vice versa, the editor theme can leak into the 2D preview viewport or the theme editor preview tabs, if the default theme has missing definitions.
⛔⛔⛔ THIS HAS TO STOP ⛔⛔⛔
Global themes, but contextualized
I propose a new feature, with a sidenote that it's currently not exposed to users, but it may be made public in the future. The feature is Theme Contexts (maybe a better name exists).
The idea is simple: define areas/branches in the scene tree that have their own sets of global themes. By default, in the running project you would have the same two themes as you have now, the project one and the default one. However, in the editor, we would define several areas.
The default context would only contain the default theme, and no project theme, to prevent it leaking into the editor. Then the editor window and the
EditorNode
root of all would have a custom context consisting of the editor theme and the default theme. The editor theme would not be applied to nodes directly, instead it would become one of the fallback themes for the entire editor GUI.To help users we would also add a custom context for the 2D viewport (project + default) and the theme editor tabs (just default). As a cherry on top we can also make the 2D viewport context controllable from the toolbar and add a way to switch between the project, the editor, and the default theme. The editor theme preview is very powerful for the editor plugin developers.
godot.windows.editor.dev.x86_64_2023-08-29_14-26-10.mp4
Each theme context replaces any previous context. Unlike themes themselves, contexts do not propagate and merge. They are authoritative and act as the final say in their controlled branch. This prevents theme leaks completely. To use contexts safely, you need to always add the default theme to each of them, or a theme that is equally complete. Otherwise things would turn unstyled.
Contexts can be attached to any node, not just GUI/Window nodes. This allows to put them before any themable node, and it also makes them truly global. Despite that, contexts are capable of propagating changes in their stored themes to the affected nodes (ignoring non-themable nodes). This means that with theme contexts GUI can react to global theme changes in real time. This is very useful when editing the project theme and looking at the 2D viewport to preview the changes.
godot.windows.editor.dev.x86_64_2023-08-29_15-37-14.mp4
In short (TL;DR)
This PR does the following changes:
NOTIFICATION_THEME_CHANGED
propagation when relevant global themes change.EditorNode
and non-themable nodes that also use global themes (cc @bruvzg).There are many issues that this addresses, though most of them are probably closed with fixes to themes and workarounds. Here are some that I can find:
Fixes (for real) the 4.x part of #60930 (not sure if we want to keep it open for 3.x).
Fixes #79753 (the other linked PR may still have merits).
Fixes #74762.
Fixes #73154.
Fixes #63889 (aside from the requirement to restart the editor, this can be reworked, but it's not a bug per se).
Fixes #33902 (duplicates with #63889, same note).
Fixes #81146.